在Ansible / Jinja2中dictsort生成什么数据类型?

时间:2017-11-01 11:21:38

标签: ansible jinja2 ansible-template

解释

假设我将字典mydict设置为{ "key1": "value1" }

  • Ansible中dictsort过滤器(mydict|dictsort)的结果似乎是包含其他列表的列表:

    [
        [
            "key1",
            "value1"
        ]
    ]
    
  • 但是,当直接在Jinja2模板(mydict|dictsort)[0]中访问此列表的第一个元素时,它呈现出奇怪的样子:

    (u'key1', u'value1')
    
  • 然后,如果我设置一个值为(mydict|dictsort)的事实,它的行为就像一个常规列表 - 使用[0]访问第一个元素会导致:

    [
        "key1",
        "value1"
    ]
    

    访问其[0]元素会返回key1

  • 但如果我设置一个值为(mydict|dictsort)[0]的事实,它的行为就像一个字符串 - 访问[0]元素会返回第一个字符,即(

  • 另一方面,如果我直接访问子元素,例如(mydict|dictsort)[0][0],它的行为就像一个列表,即返回key1

问题

  • 什么是(u'key1', u'value1')dictsort产生什么样的物体?

  • 如何以一致,可靠的方式访问dictsort结果?

完整的剧本:

---
- hosts: localhost
  gather_facts: no
  connection: local
  vars:
    mydict:
      key1: value1
  tasks:   
    - name: show dict
      debug: 
        msg: "{{ mydict }}"

    - name: show mydict|dictsort
      debug: 
        msg: "{{ mydict|dictsort }}"

    - set_fact:
        mydict_dictsorted: "{{ mydict|dictsort }}"

    - name: show (mydict|dictsort)[0]
      debug:
        msg: "{{ (mydict|dictsort)[0] }}"

    - name: show mydict_dictsorted[0]
      debug:
        msg: "{{ mydict_dictsorted[0] }}"

    - name: show (mydict|dictsort|list)[0]
      debug:
        msg: "{{ (mydict|dictsort|list)[0] }}"

    - name: show (mydict_dictsorted|list)[0]
      debug:
        msg: "{{ (mydict_dictsorted|list)[0] }}"

    - set_fact:
        mydict_dictsorted_element: "{{ (mydict|dictsort)[0] }}"

    - name: mydict_dictsorted_element
      debug:
        msg: "{{ mydict_dictsorted_element }}"

    - name: mydict_dictsorted_element[0]
      debug:
        msg: "{{ mydict_dictsorted_element[0] }}"

    - name: (mydict|dictsort)[0][0]
      debug:
        msg: "{{ (mydict|dictsort)[0][0] }}"

完整的成绩单:

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

TASK [show dict] ********************************************************************************************
ok: [localhost] => {
    "msg": {
        "key1": "value1"
    }
}

TASK [show mydict|dictsort] *********************************************************************************
ok: [localhost] => {
    "msg": [
        [
            "key1",
            "value1"
        ]
    ]
}

TASK [set_fact] *********************************************************************************************
ok: [localhost]

TASK [show (mydict|dictsort)[0]] ****************************************************************************
ok: [localhost] => {
    "msg": "(u'key1', u'value1')"
}

TASK [show mydict_dictsorted[0]] ****************************************************************************
ok: [localhost] => {
    "msg": [
        "key1",
        "value1"
    ]
}

TASK [show (mydict|dictsort|list)[0]] ***********************************************************************
ok: [localhost] => {
    "msg": "(u'key1', u'value1')"
}

TASK [show (mydict_dictsorted|list)[0]] *********************************************************************
ok: [localhost] => {
    "msg": [
        "key1",
        "value1"
    ]
}

TASK [set_fact] *********************************************************************************************
ok: [localhost]

TASK [mydict_dictsorted_element] ****************************************************************************
ok: [localhost] => {
    "msg": "(u'key1', u'value1')"
}

TASK [mydict_dictsorted_element[0]] *************************************************************************
ok: [localhost] => {
    "msg": "("
}

TASK [(mydict|dictsort)[0][0]] ******************************************************************************
ok: [localhost] => {
    "msg": "key1"

我使用copy / content检查了这些值,它们与debug相同(缩进除外),因此发布debug结果是为了清晰

1 个答案:

答案 0 :(得分:4)

dictsort生成tuples列表。它使用了dict.items()

因此,当您以(mydict|dictsort)[0]访问它时,您将访问Python的元组 如果你在模板化之后访问它,你会得到通用列表,因为JSON在元组和列表之间没有区别,它只有列表。

更新:如何测试 - 将print插入_dump_results here,如下所示:

    print("Unaltered: {}".format(abridged_result))
    return json.dumps(abridged_result, indent=indent, ensure_ascii=False, sort_keys=sort_keys)

并将此视为输出:

TASK [show mydict|dictsort] ***************************
Unaltered: {'msg': [(u'key1', u'value1')]}
ok: [localhost] => {
    "msg": [
        [
            "key1",
            "value1"
        ]
    ]
}

Update2 :为什么元组列表会成为列表列表,但是单个元组会变成字符串repr?

这是因为{{...}}中的Jinja2表达式只能生成字符串作为其输出,而且还有一些Ansible template magic尝试将其类型转换回某些复杂的类型。但这种魔法仅适用于看起来像dictslists而不是tuples的字符串。因此,如果你有里面的元组或元组列表的dict,你将得到它的评估,但如果你有一个元组,它将保持一个字符串。这是以下演示:

- name: results in a string
  debug:
    msg: "{{ test_str }}"
  vars:
    test_str: "(u'a', u'b')"

- name: results in a list of tuples/lists
  debug:
    msg: "{{ test_str }}"
  vars:
    test_str: "[(u'a', u'b')]"

输出:

TASK [results in a string] ******************************************
ok: [localhost] => {
    "msg": "(u'a', u'b')"
}

TASK [results in a list of tuples/lists] ****************************
ok: [localhost] => {
    "msg": [
        [
            "a",
            "b"
        ]
    ]

}