Ansible - 如何有條件地更改清單中的項目

Ansible - 如何有條件地更改清單中的項目

我有兩個清單:

        "combo_all": [
        [
            null,
            "3",
            "210
        ],
        [
            "r0004",
            "65",
            "212"
        ]

        "zone_name_list": [
        {
            "dns_zone_id": 10423,
            "name": "212.18.172.in-addr.arpa."
        },
        {
            "dns_zone_id": 10424,
            "name": ""210.18.172.in-addr.arpa."."
        }

每當“name”中的第一個八位元組與第一個列表中的第三個項目(210, 212)匹配時,如何用“dns_zone_id”數字(來自第二個列表)替換“combo_all” (第一個清單)中的第三個項目?

答案1

創建字典八位元組1_id

  dns_zone_id: "{{ zone_name_list|map(attribute='dns_zone_id') }}"
  octet1: "{{ zone_name_list|map(attribute='name')
                            |map('split', '.')
                            |map('first') }}"
  octet1_id: "{{ dict(octet1|zip(dns_zone_id)) }}"

給出

  octet1_id:
    '210': 10424
    '212': 10423

然後創建一個列表

    - set_fact:
        combo_id: "{{ combo_id|d([]) + [item[:-1] + [octet1_id[item[-1]]]] }}"
      loop: "{{ combo_all }}"

給出

  combo_id:
    - [null, '3', 10424]
    - [r0004, '65', 10423]

用於測試的完整劇本範例

- hosts: all

  vars:

    combo_all:
      - [ null, "3", "210"]
      - [ "r0004", "65", "212"]
    zone_name_list:
      - {dns_zone_id: 10423, name: 212.18.172.in-addr.arpa.}
      - {dns_zone_id: 10424, name: 210.18.172.in-addr.arpa.}

    dns_zone_id: "{{ zone_name_list|map(attribute='dns_zone_id') }}"
    octet1: "{{ zone_name_list|map(attribute='name')
                              |map('split', '.')
                              |map('first') }}"
    octet1_id: "{{ dict(octet1|zip(dns_zone_id)) }}"

  tasks:

    - debug:
        var: dns_zone_id
    - debug:
        var: octet1
    - debug:
        var: octet1_id

    - set_fact:
        combo_id: "{{ combo_id|d([]) + [item[:-1] + [octet1_id[item[-1]]]] }}"
      loop: "{{ combo_all }}"
    - debug:
        var: combo_id|to_yaml

答案2

一般來說,Ansible 並不是一個很好的工具修改資料結構。您可以透過創建一個接近您想要的新的透過執行以下操作,使用您想要的內容變數:

- hosts: localhost
  gather_facts: false
  vars:
    "combo_all": [
      [ null, "3", "210" ],
      [ "r0004", "65", "212" ]
      ]
    "zone_name_list": [
        {
            "dns_zone_id": 10423,
            "name": "212.18.172.in-addr.arpa."
        },
        {
            "dns_zone_id": 10424,
            "name": "210.18.172.in-addr.arpa."
        }
      ]

  tasks:
    - vars:
        new_combo_all: []
        prefix: "{{ item[1]['name'].split('.')[0] }}"
      set_fact:
        new_combo_all: >
          {{ new_combo_all + [[item[0][0], item[0][1], item[1]['dns_zone_id']]] }}
      loop: "{{ combo_all | product(zone_name_list) }}"
      when: prefix == item[0][2]

    - debug:
        var: new_combo_all

這將設定new_combo_all為:

ok: [localhost] => {
    "new_combo_all": [
        [
            null,
            "3",
            10424
        ],
        [
            "r0004",
            "65",
            10423
        ]
    ]
}

combo_all它的工作原理是迭代 和的笛卡爾積zone_name_list,僅選擇name前綴八位組等於 的子列表的第三個成員的那些對combo_all,並使用它來產生new_combo_all變數。

相關內容