Подтвердить что ты не робот

Невозможный результат регистрации нескольких команд

Мне была предоставлена ​​задача проверить некоторые записи маршрутизации для всего Linux-сервера, и вот как я это сделал с помощью загружаемой книги Ansible

---
  - hosts: Linux
    serial: 1

    tasks:
      - name: Check first
        command: /sbin/ip route list xxx.xxx.xxx.xxx/24
        register: result
        changed_when: false

      - debug: msg="{{result.stdout}}"

      - name: Check second
        command: /sbin/ip route list xxx.xxx.xxx.xxx/24
        register: result
        changed_when: false

      - debug: msg="{{result.stdout}}"

Вы можете видеть, что мне нужно повторить одну и ту же задачу для каждой записи маршрутизации, и я считаю, что я должен быть в состоянии избежать этого. Я попытался использовать цикл with_items, но получил следующее сообщение об ошибке

One or more undefined variables: 'dict object' has no attribute 'stdout'

Есть ли способ зарегистрировать переменную для каждой команды и цикл по ней один за другим?

4b9b3361

Ответ 1

Начиная с Ansible 1.6.1, результаты, зарегистрированные с несколькими элементами, сохраняются в result.results как массив. Таким образом, вы можете использовать result.results[0].stdout и т.д.

Тестирование playbook:

---
- hosts: localhost
  gather_facts: no
  tasks:
    - command: "echo {{item}}"
      register: result
      with_items: [1, 2]
    - debug:
        var: result

Результат:

$ ansible-playbook -i localhost, test.yml

PLAY [localhost] ************************************************************** 

TASK: [command echo {{item}}] ************************************************* 
changed: [localhost] => (item=1)
changed: [localhost] => (item=2)

TASK: [debug ] **************************************************************** 
ok: [localhost] => {
    "var": {
        "result": {
            "changed": true, 
            "msg": "All items completed", 
            "results": [
                {
                    "changed": true, 
                    "cmd": [
                        "echo", 
                        "1"
                    ], 
                    "delta": "0:00:00.002502", 
                    "end": "2015-08-07 16:44:08.901313", 
                    "invocation": {
                        "module_args": "echo 1", 
                        "module_name": "command"
                    }, 
                    "item": 1, 
                    "rc": 0, 
                    "start": "2015-08-07 16:44:08.898811", 
                    "stderr": "", 
                    "stdout": "1", 
                    "stdout_lines": [
                        "1"
                    ], 
                    "warnings": []
                }, 
                {
                    "changed": true, 
                    "cmd": [
                        "echo", 
                        "2"
                    ], 
                    "delta": "0:00:00.002516", 
                    "end": "2015-08-07 16:44:09.038458", 
                    "invocation": {
                        "module_args": "echo 2", 
                        "module_name": "command"
                    }, 
                    "item": 2, 
                    "rc": 0, 
                    "start": "2015-08-07 16:44:09.035942", 
                    "stderr": "", 
                    "stdout": "2", 
                    "stdout_lines": [
                        "2"
                    ], 
                    "warnings": []
                }
            ]
        }
    }
}

PLAY RECAP ******************************************************************** 
localhost                  : ok=2    changed=1    unreachable=0    failed=0   

Ответ 2

Несколько другая ситуация, которая потребовала времени, чтобы разобраться. Если вы хотите использовать результаты нескольких элементов, но для changed_when, то переменная регистра не будет иметь var.results! Вместо этого changed_when оценивается для каждого элемента, и вы можете просто использовать регистр var.

Простой пример, который приведет к изменению: false:

- action: command echo {{item}}
  register: out
  changed_when: "'z' in out.stdout"
  with_items:
    - hello
    - foo
    - bye

Другой пример:

- name: Create fulltext index for faster text searches.
  mysql_db: name={{SO_database}} state=import target=/tmp/fulltext-{{item.tableName}}-{{item.columnName}}.sql
  with_items: 
    - {tableName: Posts,  columnName: Title}
    - {tableName: Posts,  columnName: Body}
    - {tableName: Posts,  columnName: Tags}
    - {tableName: Comments, columnName: Text}
  register: createfulltextcmd
  changed_when: createindexcmd.msg.find('already exists') == -1

Наконец, когда вы хотите перебирать результаты в других контекстах, для программного доступа к индексу может показаться немного сложным, поскольку это не отображается. Я нашел этот пример, который может быть многообещающим:

- name: add hosts to known_hosts
shell: 'ssh-keyscan -H {{item.host}}>> /home/testuser/known_hosts'
with_items:
  - { index: 0, host: testhost1.test.dom }
  - { index: 1, host: testhost2.test.dom }
  - { index: 2, host: 192.168.202.100 }
when: ssh_known_hosts.results[{{item.index}}].rc == 1

Ответ 3

Если вам нужно зарегистрировать вывод двух команд отдельно, используйте разные имена переменных.

---
- hosts: Linux
  serial: 1
  tasks:
  - name: Check first
    command: /sbin/ip route list xxx.xxx.xxx.xxx/24
    register: result0
    changed_when: false

  - debug: msg="{{result0.stdout}}"

  - name: Check second
    command: /sbin/ip route list xxx.xxx.xxx.xxx/24
    register: result1
    changed_when: false

  - debug: msg="{{result1.stdout}}"