使用 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"
}
我在 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"
}