2017-10-17 2 views
1

Verwenden von TestInfra mit dem Ansible-Backend für Testzwecke. Alles geht in Ordnung, außer mit ansible selbst während Tests laufenVerwenden von Ansible-Variablen in Testinfra

test.py

import pytest 
def test_zabbix_agent_package(host): 
    package = host.package("zabbix-agent") 
    assert package.is_installed 
    package_version = host.ansible("debug", "msg={{ zabbix_agent_version }}")["msg"] 
    (...) 

wo zabbix_agent_version eine ansible Variable aus group_vars. Es kann durch Ausführen dieses Textbuch

- hosts: all 
    become: true 
    tasks: 
    - name: debug 
    debug: msg={{ zabbix_agent_version }} 

Befehl ausführen Tests

pytest --connection=ansible --ansible-inventory=inventory --hosts=$hosts -v test.py 

ansible.cfg

[defaults] 
timeout = 10 
host_key_checking = False 
library=library/ 
retry_files_enabled = False 
roles_path=roles/ 
pipelining=true 
ConnectTimeout=60 
remote_user=deploy 
private_key_file=/opt/jenkins/.ssh/deploy 

der Ausgang I ist

self = <ansible>, module_name = 'debug', module_args = 'msg={{ zabbix_agent_version }}', check = True, kwargs = {} 
result = {'failed': True, 'msg': "the field 'args' has an invalid value, which appears to include a variable that is undefined. The error was: 'zabbix_agent_version' is undefined"} 

    def __call__(self, module_name, module_args=None, check=True, **kwargs): 
     if not self._host.backend.HAS_RUN_ANSIBLE: 
      raise RuntimeError((
       "Ansible module is only available with ansible " 
       "connection backend")) 
     result = self._host.backend.run_ansible(
      module_name, module_args, check=check, **kwargs) 
     if result.get("failed", False) is True: 
>   raise AnsibleException(result) 
E   AnsibleException: Unexpected error: {'failed': True, 
E   'msg': u"the field 'args' has an invalid value, which appears to include a variable that is undefined. The error was: 'zabbix_agent_version' is undefined"} 

/usr/lib/python2.7/site-packages/testinfra/modules/ansible.py:70: AnsibleException 

Jede Idee, warum bekommen, erhalten werden Ansible kann diese Variable nicht sehen, wenn das Ansible-Modul von Testinfra laufen lassen, während es Ansible alleine laufen lässt?

+0

ansible Version 2.2.1.0 Python Version 2.7.5 Testinfra Version 1.6.4 Pytest Version 3.1.3 – FRC

+0

Wo ist 'definiert zabbix_agent_version' wenn Sie laufen manuell ansible?Kommt dies von einer benutzerdefinierten Tatsache auf dem Remote-Host oder wird diese in einer lokalen Variablendatei festgelegt? – larsks

+0

zabbix_agent_version ist in group_vars definiert. Wenn Sie ein Playbook ausführen, das diese Tatsache überprüft (debug: msg = {{zabbix_agent_version}}), kann es von Ansible bezogen werden. – FRC

Antwort

1

Wenn zabbix_agent_version ein variabler Satz ist group_vars verwendet wird, dann scheint es, als ob Sie es host.ansible.get_variables() mit Zugriff werden sollte, anstatt laufen debug Aufgabe. In jedem Fall sollten beide funktionieren. Wenn ich in meinem aktuellen Verzeichnis:

test_myvar.py 
group_vars/ 
    all.yml 

Und in group_vars/all.yml ich habe:

myvar: value 

Und in test_myvar.py ich habe:

def test_myvar_using_get_variables(host): 
    all_variables = host.ansible.get_variables() 
    assert 'myvar' in all_variables 
    assert all_variables['myvar'] == 'myvalue' 


def test_myvar_using_debug_var(host): 
    result = host.ansible("debug", "var=myvar") 
    assert 'myvar' in result 
    assert result['myvar'] == 'myvalue' 


def test_myvar_using_debug_msg(host): 
    result = host.ansible("debug", "msg={{ myvar }}") 
    assert 'msg' in result 
    assert result['msg'] == 'myvalue' 

Dann werden alle Tests bestanden:

$ py.test --connection=ansible --ansible-inventory=hosts -v 
test_myvar.py 
============================= test session starts ============================== 
platform linux2 -- Python 2.7.13, pytest-3.2.3, py-1.4.34, pluggy-0.4.0 -- /home/lars/env/common/bin/python2 
cachedir: .cache 
rootdir: /home/lars/tmp/testinfra, inifile: 
plugins: testinfra-1.8.1.dev2 
collected 3 items                

test_myvar.py::test_myvar_using_get_variables[ansible://localhost] PASSED 
test_myvar.py::test_myvar_using_debug_var[ansible://localhost] PASSED 
test_myvar.py::test_myvar_using_debug_msg[ansible://localhost] PASSED 

=========================== 3 passed in 1.77 seconds =========================== 

Können Sie bestätigen, dass das Layout unserer Dateien (insbesondere der Speicherort Ihres Verzeichnisses group_vars relativ zu Ihren Tests) dem entspricht, was ich hier gezeigt habe?

+0

Ja, ich habe das gleiche Layout wie das, das Sie vorgestellt haben. Bei weiteren Tests ist es seltsam, dass Testinfra, wenn ich versuche, etwas von Ansible eingebaut zu bekommen (wie ansible_ssh_host), es ohne Probleme finden kann, aber wenn ich etwas versuche, das von meinen group_vars eingerichtet wird, kann es nicht finden es zwar beim Ausführen von Debug Ansible kann es sehen. – FRC

0

Ich jagte eine Antwort auf diese Tage. Hier ist, was endlich für mich funktioniert hat. Im Wesentlichen verwenden Sie das Ansible-Modul von Testinfra, um auf die Funktion include_vars von Ansible zuzugreifen.

import pytest 

@pytest.fixture() 
def AnsibleVars(host): 
ansible_vars = host.ansible(
    "include_vars", "file=./group_vars/all/vars.yml") 
return ansible_vars["ansible_facts"] 

Dann in meinen Tests, inklusive ich die Funktion als Parameter:

def test_something(host, AnsibleVars): 

Diese Lösung teilweise von https://github.com/metacloud/molecule/issues/151

Ich hatte ein interessantes Thema gemacht, wo ich versuchte, die einschließen Variablen aus meinem Haupt-Playbook und ich erhielt einen Fehler von "muss als ein Wörterbuch/Hash gespeichert werden", wenn Sie die playbook.yml-Datei enthalten. Die Trennung der Variablen in die Datei group_vars/all/vars.yml löste diesen Fehler.