使用 Jinja 过滤器从嵌套对象中获取唯一键

Get only key from nested object with Jinja filter

我在 ansible 中使用 Jinja 过滤器以正确的格式提取我需要的值来处理它。

这是 JSON 格式的数据(我缩短了输出,通常每个项目有更多变量,并不是所有项目都有 IPv4 变量等等):

"interfaces": {
    "GigabitEthernet0": {
        "arp_timeout": "00:20:00",
        "arp_type": "arpa",
        "auto_negotiate": true,
        "bandwidth": 1000000
    },
    "GigabitEthernet0/0/0": {
        "arp_timeout": "00:20:00",
        "arp_type": "arpa",
        "auto_negotiate": true,
        "bandwidth": 10000
    },
    "GigabitEthernet0/0/0.3": {
        "arp_timeout": "04:00:00",
        "arp_type": "arpa",
        "bandwidth": 10000,
        "delay": 10,
        "description": "Private1 MPLS",
        "enabled": true,
        "encapsulations": {
            "encapsulation": "dot1q",
            "first_dot1q": "3"
        },
        "ipv4": {
            "10.10.84.2/30": {
                "ip": "10.10.84.2",
                "prefix_length": "30"
            }

然后我使用那个简单的 Jinja 文件管理器来提取我需要的信息,例如接口名称和 IPv4:

[
{% for interface in interfaces if interfaces[interface]['ipv4'] is defined %}
{
"name": "{{ interface }}",
{% if interfaces[interface]['ipv4'] is defined %}
"prefix": "{{ interfaces[interface]['ipv4'] }}",
{% endif %}
"hostname": "{{ hostname }}"
}{{ ", " if not loop.last else "" }}
{% endfor %}
]

我现在的问题是解析数据如下所示:

{
"name": "GigabitEthernet0/0/0.3",
"prefix": "{'10.10.84.2/30': {'ip': '10.10.84.2', 'prefix_length': '30'}}",
"hostname": "Horst1"
},

但我只想像这样嵌套字典中的键:

{
"name": "GigabitEthernet0/0/0.3",
"prefix": "10.10.84.2/30",
"hostname": "Horst1"
},

Jinja 中没有一个简单的方法来从嵌套对象中获取键吗?

这是一个可能更简单的模板,使用 for key, value in dict.items() 构造:

[
{% for name, interface in interfaces.items() if interface.ipv4 is defined %}
  {
    "name": "{{ name }}",
    "prefix": "{{ (interface.ipv4.keys() | list).0 }}",
    "hostname": "{{ hostname }}"
  }{{ ", " if not loop.last }}
{% endfor %}
]

keys() 方法是 Python 中的方法,return 表示该字典的键列表的视图。将其转换回列表并获取其中的第一个元素,您应该可以开始了。

另一种选择是使用 dict2items,再次获取生成列表的第一个元素并获取其键:

    "prefix": "{{ (interface.ipv4 | dict2items).0.key }}",

Jinja 和迭代都不需要。下面的表达式

name_prefix: "{{ interfaces|
                 dict2items|
                 selectattr('value.ipv4', 'defined')|
                 json_query('[].{name: key,
                                 prefix: value.ipv4.keys(@)|[0]}') }}"

创建词典列表

name_prefix:
  - name: GigabitEthernet0/0/0.3
    prefix: 10.10.84.2/30

备注

  • 属性ipv4可能定义在更多接口中,因此结果是一个列表。喜欢就拿第一个吧

  • 您可以根据需要合并词典。例如

horst_default:
  hostname: Horst1
  domain: foo.bar
horst: "{{ name_prefix|map('combine', horst_default)|list }}"

给予

horst:
  - domain: foo.bar
    hostname: Horst1
    name: GigabitEthernet0/0/0.3
    prefix: 10.10.84.2/30
  • 如果你愿意,你可以使用 Jinja 中的变量。例如
    - debug:
        msg: |
          {{ horst|first|to_nice_json }}

给予

  msg: |-
    {
        "domain": "foo.bar",
        "hostname": "Horst1",
        "name": "GigabitEthernet0/0/0.3",
        "prefix": "10.10.84.2/30"
    }