Ansible-core 2.14 Porting Guide
This section discusses the behavioral changes between ansible-core 2.13 and ansible-core 2.14.
It is intended to assist in updating your playbooks, plugins and other parts of your Ansible infrastructure so they will work with this version of Ansible.
We suggest you read this page along with ansible-core Changelog for 2.14 to understand what updates you may need to make.
This document is part of a collection on porting. The complete list of porting guides can be found at porting guides.
Playbook
- Conditionals - due to mitigation of security issue CVE-2023-5764 in ansible-core 2.14.12, conditional expressions with embedded template blocks can fail with the message “ - Conditional is marked as unsafe, and cannot be evaluated.” when an embedded template consults data from untrusted sources like module results or vars marked- !unsafe. Conditionals with embedded templates can be a source of malicious template injection when referencing untrusted data, and can nearly always be rewritten without embedded templates. Playbook task conditional keywords such as- whenand- untilhave long displayed warnings discouraging use of embedded templates in conditionals; this warning has been expanded to non-task conditionals as well, such as the- assertaction.- - name: task with a module result (always untrusted by Ansible) shell: echo "hi mom" register: untrusted_result # don't do it this way... # - name: insecure conditional with embedded template consulting untrusted data # assert: # that: '"hi mom" is in {{ untrusted_result.stdout }}' - name: securely access untrusted values directly as Jinja variables instead assert: that: '"hi mom" is in untrusted_result.stdout' 
- Variables are now evaluated lazily; only when they are actually used. For example, in ansible-core 2.14 an expression - {{ defined_variable or undefined_variable }}does not fail on- undefined_variableif the first part of- oris evaluated to- Trueas it is not needed to evaluate the second part. One particular case of a change in behavior to note is the task below which uses the- undefinedtest. Prior to version 2.14 this would result in a fatal error trying to access the undefined value in the dictionary. In 2.14 the assertion passes as the dictionary is evaluated as undefined through one of its undefined values:
- assert: that: - some_defined_dict_with_undefined_values is undefined vars: dict_value: 1 some_defined_dict_with_undefined_values: key1: value1 key2: '{{ dict_value }}' key3: '{{ undefined_dict_value }}'
Command Line
- Python 3.9 on the controller node is a hard requirement for this release. 
- At startup the filesystem encoding and locale are checked to verify they are UTF-8. If not, the process exits with an error reporting the errant encoding. If you were previously using the - Cor- POSIXlocale, you may be able to use- C.UTF-8. If you were previously using a locale such as- en_US.ISO-8859-1, you may be able to use- en_US.UTF-8. For simplicity it may be easiest to export the appropriate locale using the- LC_ALLenvironment variable. An alternative to modifying your system locale is to run Python in UTF-8 mode; See the Python documentation for more information.
Deprecated
No notable changes
Modules
No notable changes
Modules removed
The following modules no longer exist:
- No notable changes 
Deprecation notices
No notable changes
Noteworthy module changes
No notable changes
Plugins
No notable changes
Porting custom scripts
No notable changes
Networking
No notable changes
