使用Ansible中的JINJA2从JSON中提取带有条件的嵌套dict值

时间:2019-05-23 02:17:51

标签: python json ansible jinja2

我正在尝试从Ansible的cloudfront_facts中提取特定别名的CDN域名,并提供以下(摘要)输出:

{
  "cdn_facts": {
   "ansible_facts": {
    "cloudfront": {
      "summary": {
       "distributions": [
        {
          "Aliases": [
           "media.example.com"
          ],
          "DomainName": "a1b2c3d4e5f6g.cloudfront.net"
        },
        {
          "Aliases": [
           "example.com"
          ],
          "DomainName": "g7f6e5d4c3b2a.cloudfront.net"
        }
      ]
     }
    }
  }
}

换句话说,对于别名example.com,我想设置一个值为g7f6e5d4c3b2a.cloudfront.net的事实。

我已经尝试了以下方法来尝试打印出该值,但它只是产生一个空列表。

- debug:
    msg: "{{ cdn_facts.ansible_facts.cloudfront.summary.distributions | selectattr('Aliases[0]', 'equalto', 'example.com') | map(attribute='DomainName') | list }}"

我不知道如何调试它,因为cdn_facts.ansible_facts.cloudfront.summary.distributions产生了一系列的字典,但是selectattrmap却什么也没产生。

1 个答案:

答案 0 :(得分:1)

数据中没有一个名为Aliases[0]的属性,所以这就是selectattr过滤器失败的原因。

要从深度嵌套的字典中提取数据,与尝试链接本机Jinja过滤器相比,json_query过滤器通常是更好的选择。例如,给定别名example.com,以下表达式将提取相应的DomainName属性:

cdn_facts.ansible_facts.cloudfront.summary.distributions[?Aliases[0] == `example.com`]|[0].DomainName

我们可以在这样的剧本中使用它:

---
- hosts: localhost
  gather_facts: false
  tasks:
    - include_vars:
        file: data.json
        name: data

    - debug:
        msg: >-
          {{ item }} ->
          {{
          (data|json_query('cdn_facts.ansible_facts.cloudfront.summary.distributions[?Aliases[0] == `{}`]|[0].DomainName'.format(item)))
          }}
      loop:
        - media.example.com
        - example.com

哪个会输出:

TASK [debug] **********************************************************************************************************************************************************************************
ok: [localhost] => (item=media.example.com) => {
    "msg": "media.example.com -> a1b2c3d4e5f6g.cloudfront.net"
}
ok: [localhost] => (item=example.com) => {
    "msg": "example.com -> g7f6e5d4c3b2a.cloudfront.net"
}

json_query模块使用jmespath搜索语法。您可以通过将数据粘贴到https://jmespath.org的文本框中,然后在其上方的字段中输入搜索表达式来试验jmespath表达式。或者,您可以使用jpterm命令行工具。