ansible/docs/docsite/rst/user_guide/playbooks_python_version.rst
Toshio Kuratomi 4273e84962
Upstream jinja has decided not to rename from jinja2. (#66880)
So we're going to keep our intersphinx labels as jinja2 to match what
upstream is doing
2020-01-30 10:57:50 -05:00

2.3 KiB

Python3 in templates

Ansible uses Jinja2 to leverage Python data types and standard functions in templates and variables. You can use these data types and standard functions to perform a rich set of operations on your data. However, if you use templates, you must be aware of differences between Python versions.

These topics help you design templates that work on both Python2 and Python3. They might also help if you are upgrading from Python2 to Python3. Upgrading within Python2 or Python3 does not usually introduce changes that affect Jinja2 templates.

Dictionary views

In Python2, the dict.keys, dict.values, and dict.items methods return a list. Jinja2 returns that to Ansible via a string representation that Ansible can turn back into a list.

In Python3, those methods return a dictionary view <python3:dict-views> object. The string representation that Jinja2 returns for dictionary views cannot be parsed back into a list by Ansible. It is, however, easy to make this portable by using the list <jinja2:list> filter whenever using dict.keys, dict.values, or dict.items:

vars:
  hosts:
    testhost1: 127.0.0.2
    testhost2: 127.0.0.3
tasks:
  - debug:
      msg: '{{ item }}'
    # Only works with Python 2
    #loop: "{{ hosts.keys() }}"
    # Works with both Python 2 and Python 3
    loop: "{{ hosts.keys() | list }}"

dict.iteritems()

Python2 dictionaries have ~dict.iterkeys, ~dict.itervalues, and ~dict.iteritems methods.

Python3 dictionaries do not have these methods. Use dict.keys, dict.values, and dict.items to make your playbooks and templates compatible with both Python2 and Python3:

vars:
  hosts:
    testhost1: 127.0.0.2
    testhost2: 127.0.0.3
tasks:
  - debug:
      msg: '{{ item }}'
    # Only works with Python 2
    #loop: "{{ hosts.iteritems() }}"
    # Works with both Python 2 and Python 3
    loop: "{{ hosts.items() | list }}"

* The pb-py-compat-dict-views entry for information on why the list filter <jinja2:list> is necessary here.