2013-09-30 01:03:51 +02:00
Conditionals
============
2013-12-26 20:32:01 +01:00
.. contents :: Topics
2017-01-18 03:55:03 +01:00
Often the result of a play may depend on the value of a variable, fact (something learned about the remote system), or previous task result.
In some cases, the values of variables may depend on other variables.
2017-11-22 05:14:27 +01:00
Further, additional groups can be created to manage hosts based on whether the hosts match other criteria. This topic covers how conditionals are used in playbooks.
.. note :: There are many options to control execution flow in Ansible. More examples of supported conditionals can be located here: http://jinja.pocoo.org/docs/dev/templates/#comparisons.
2013-09-30 01:03:51 +02:00
2015-03-06 22:35:49 +01:00
.. _the_when_statement:
2013-09-30 01:03:51 +02:00
The When Statement
`` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ``
2017-01-18 03:55:03 +01:00
Sometimes you will want to skip a particular step on a particular host.
This could be something as simple as not installing a certain package if the operating system is a particular version,
2013-09-30 01:03:51 +02:00
or it could be something like performing some cleanup steps if a filesystem is getting full.
2016-06-28 01:34:14 +02:00
This is easy to do in Ansible with the `when` clause, which contains a raw Jinja2 expression without double curly braces (see :doc: `playbooks_variables` ).
2013-10-05 01:06:20 +02:00
It's actually pretty simple::
2013-09-30 01:03:51 +02:00
tasks:
2016-08-16 21:02:25 +02:00
- name: "shut down Debian flavored systems"
2013-09-30 01:03:51 +02:00
command: /sbin/shutdown -t now
when: ansible_os_family == "Debian"
2016-06-28 01:34:14 +02:00
# note that Ansible facts and vars like ansible_os_family can be used
# directly in conditionals without double curly braces
2013-09-30 01:03:51 +02:00
2014-10-05 19:54:31 +02:00
You can also use parentheses to group conditions::
tasks:
2016-08-16 21:02:25 +02:00
- name: "shut down CentOS 6 and Debian 7 systems"
2014-10-05 19:54:31 +02:00
command: /sbin/shutdown -t now
2015-11-01 22:31:23 +01:00
when: (ansible_distribution == "CentOS" and ansible_distribution_major_version == "6") or
(ansible_distribution == "Debian" and ansible_distribution_major_version == "7")
2014-10-05 19:54:31 +02:00
2016-08-25 18:52:52 +02:00
Multiple conditions that all need to be true (a logical 'and') can also be specified as a list::
tasks:
- name: "shut down CentOS 6 systems"
command: /sbin/shutdown -t now
when:
- ansible_distribution == "CentOS"
- ansible_distribution_major_version == "6"
2013-09-30 01:03:51 +02:00
A number of Jinja2 "filters" can also be used in when statements, some of which are unique
and provided by Ansible. Suppose we want to ignore the error of one statement and then
decide to do something conditionally based on success or failure::
tasks:
- command: /bin/false
register: result
ignore_errors: True
2016-06-01 17:06:47 +02:00
2013-09-30 01:03:51 +02:00
- command: /bin/something
when: result|failed
2016-06-01 17:06:47 +02:00
2016-06-30 23:55:40 +02:00
# In older versions of ansible use |success, now both are valid but succeeded uses the correct tense.
2013-09-30 01:03:51 +02:00
- command: /bin/something_else
2016-01-14 18:28:34 +01:00
when: result|succeeded
2016-06-01 17:06:47 +02:00
2013-09-30 01:03:51 +02:00
- command: /bin/still/something_else
when: result|skipped
2016-06-01 17:06:47 +02:00
2017-11-22 05:14:27 +01:00
.. note :: both `success` and `succeeded` work (`fail` /`failed` , etc).
2016-06-01 17:06:47 +02:00
2013-09-30 01:03:51 +02:00
2017-11-22 05:14:27 +01:00
As a reminder, to see what facts are available on a particular system, you can do the following::
2013-09-30 01:03:51 +02:00
ansible hostname.example.com -m setup
Tip: Sometimes you'll get back a variable that's a string and you'll want to do a math operation comparison on it. You can do this like so::
tasks:
- shell: echo "only on Red Hat 6, derivatives, and later"
when: ansible_os_family == "RedHat" and ansible_lsb.major_release|int >= 6
.. note :: the above example requires the lsb_release package on the target host in order to return the ansible_lsb.major_release fact.
Variables defined in the playbooks or inventory can also be used. An example may be the execution of a task based on a variable's boolean value::
vars:
epic: true
Then a conditional execution might look like::
tasks:
- shell: echo "This certainly is epic!"
when: epic
or::
2017-01-18 03:55:03 +01:00
2013-09-30 01:03:51 +02:00
tasks:
- shell: echo "This certainly isn't epic!"
when: not epic
2017-01-18 03:55:03 +01:00
If a required variable has not been set, you can skip or fail using Jinja2's `defined` test. For example::
2013-09-30 01:03:51 +02:00
tasks:
- shell: echo "I've got '{{ foo }}' and am not afraid to use it!"
when: foo is defined
2014-01-17 14:09:52 +01:00
- fail: msg="Bailing out. this play requires 'bar'"
2015-07-25 14:05:27 +02:00
when: bar is undefined
2013-09-30 01:03:51 +02:00
2017-01-18 03:55:03 +01:00
This is especially useful in combination with the conditional import of vars files (see below).
As the examples show, you don't need to use `{{ }}` to use variables inside conditionals, as these are already implied.
2013-09-30 01:03:51 +02:00
2016-04-21 15:42:00 +02:00
.. _loops_and_conditionals:
Loops and Conditionals
`` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ``
move from with_<lookup>: to loop:
- old functionality is still available direct lookup use, the following are equivalent
with_nested: [[1,2,3], ['a','b','c']]
loop: "{{lookup('nested', [1,2,3], ['a','b','c'])}}"
- avoid squashing with 'loop:'
- fixed test to use new intenal attributes
- removed most of 'lookup docs' as these now reside in the plugins
2017-09-17 05:32:34 +02:00
Combining `when` with loops (see :doc: `playbooks_loops` ), be aware that the `when` statement is processed separately for each item. This is by design::
2013-09-30 01:03:51 +02:00
tasks:
- command: echo {{ item }}
move from with_<lookup>: to loop:
- old functionality is still available direct lookup use, the following are equivalent
with_nested: [[1,2,3], ['a','b','c']]
loop: "{{lookup('nested', [1,2,3], ['a','b','c'])}}"
- avoid squashing with 'loop:'
- fixed test to use new intenal attributes
- removed most of 'lookup docs' as these now reside in the plugins
2017-09-17 05:32:34 +02:00
loop: [ 0, 2, 4, 6, 8, 10 ]
2013-09-30 01:03:51 +02:00
when: item > 5
2016-04-21 15:42:00 +02:00
If you need to skip the whole task depending on the loop variable being defined, used the `|default` filter to provide an empty iterator::
- command: echo {{ item }}
move from with_<lookup>: to loop:
- old functionality is still available direct lookup use, the following are equivalent
with_nested: [[1,2,3], ['a','b','c']]
loop: "{{lookup('nested', [1,2,3], ['a','b','c'])}}"
- avoid squashing with 'loop:'
- fixed test to use new intenal attributes
- removed most of 'lookup docs' as these now reside in the plugins
2017-09-17 05:32:34 +02:00
loop: "{{ mylist|default([]) }}"
2016-04-21 15:42:00 +02:00
when: item > 5
move from with_<lookup>: to loop:
- old functionality is still available direct lookup use, the following are equivalent
with_nested: [[1,2,3], ['a','b','c']]
loop: "{{lookup('nested', [1,2,3], ['a','b','c'])}}"
- avoid squashing with 'loop:'
- fixed test to use new intenal attributes
- removed most of 'lookup docs' as these now reside in the plugins
2017-09-17 05:32:34 +02:00
If using a dict in a loop::
2016-04-21 15:42:00 +02:00
- command: echo {{ item.key }}
move from with_<lookup>: to loop:
- old functionality is still available direct lookup use, the following are equivalent
with_nested: [[1,2,3], ['a','b','c']]
loop: "{{lookup('nested', [1,2,3], ['a','b','c'])}}"
- avoid squashing with 'loop:'
- fixed test to use new intenal attributes
- removed most of 'lookup docs' as these now reside in the plugins
2017-09-17 05:32:34 +02:00
loop: "{{ lookup('dict', mydict|default({})) }}"
2016-04-21 15:42:00 +02:00
when: item.value > 5
.. _loading_in_custom_facts:
2013-09-30 01:03:51 +02:00
Loading in Custom Facts
`` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ``
2017-01-07 20:38:52 +01:00
It's also easy to provide your own facts if you want, which is covered in :doc: `dev_guide/developing_modules` . To run them, just
2013-09-30 01:03:51 +02:00
make a call to your own custom fact gathering module at the top of your list of tasks, and variables returned
there will be accessible to future tasks::
tasks:
- name: gather site specific fact data
action: site_facts
- command: /usr/bin/thingy
2013-10-05 01:06:20 +02:00
when: my_custom_fact_just_retrieved_from_the_remote_system == '1234'
2016-04-21 15:42:00 +02:00
.. _when_roles_and_includes:
2017-09-17 20:02:46 +02:00
Applying 'when' to roles, imports, and includes
`` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ``
2013-09-30 01:03:51 +02:00
Note that if you have several tasks that all share the same conditional statement, you can affix the conditional
2015-08-04 22:13:11 +02:00
to a task include statement as below. All the tasks get evaluated, but the conditional is applied to each and every task::
2013-09-30 01:03:51 +02:00
2017-09-17 20:02:46 +02:00
- import_tasks: tasks/sometasks.yml
2013-09-30 01:03:51 +02:00
when: "'reticulating splines' in output"
2015-08-04 22:13:11 +02:00
.. note :: In versions prior to 2.0 this worked with task includes but not playbook includes. 2.0 allows it to work with both.
2013-09-30 01:03:51 +02:00
Or with a role::
- hosts: webservers
roles:
- { role: debian_stock_config, when: ansible_os_family == 'Debian' }
You will note a lot of 'skipped' output by default in Ansible when using this approach on systems that don't match the criteria.
2013-12-21 08:06:38 +01:00
Read up on the 'group_by' module in the :doc: `modules` docs for a more streamlined way to accomplish the same thing.
2013-09-30 01:03:51 +02:00
2017-09-17 20:02:46 +02:00
When used with `include_*` tasks instead of imports, the conditional is applied _only_ to the include task itself and not any other
tasks within the included file(s). A common situation where this distinction is important is as follows::
# include a file to define a variable when it is not already defined
# main.yml
- include_tasks: other_tasks.yml
when: x is not defined
# other_tasks.yml
- set_fact:
x: foo
- debug:
var: x
In the above example, if `` import_tasks `` had been used instead both included tasks would have also been skipped. With `` include_tasks ``
instead, the tasks are executed as expected because the conditional is not applied to them.
2016-04-21 15:42:00 +02:00
.. _conditional_imports:
2013-09-30 01:03:51 +02:00
Conditional Imports
`` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ``
2017-11-22 05:14:27 +01:00
.. note :: This is an advanced topic that is infrequently used.
2013-09-30 01:03:51 +02:00
Sometimes you will want to do certain things differently in a playbook based on certain criteria.
Having one playbook that works on multiple platforms and OS versions is a good example.
As an example, the name of the Apache package may be different between CentOS and Debian,
but it is easily handled with a minimum of syntax in an Ansible Playbook::
---
- hosts: all
remote_user: root
vars_files:
- "vars/common.yml"
- [ "vars/{{ ansible_os_family }}.yml", "vars/os_defaults.yml" ]
tasks:
2017-02-14 11:39:27 +01:00
- name: make sure apache is started
service: name={{ apache }} state=started
2013-09-30 01:03:51 +02:00
.. note ::
The variable 'ansible_os_family' is being interpolated into
the list of filenames being defined for vars_files.
As a reminder, the various YAML files contain just keys and values::
---
# for vars/CentOS.yml
apache: httpd
somethingelse: 42
How does this work? If the operating system was 'CentOS', the first file Ansible would try to import
would be 'vars/CentOS.yml', followed by '/vars/os_defaults.yml' if that file
did not exist. If no files in the list were found, an error would be raised.
On Debian, it would instead first look towards 'vars/Debian.yml' instead of 'vars/CentOS.yml', before
falling back on 'vars/os_defaults.yml'. Pretty simple.
To use this conditional import feature, you'll need facter or ohai installed prior to running the playbook, but
you can of course push this out with Ansible if you like::
# for facter
2014-12-22 05:42:01 +01:00
ansible -m yum -a "pkg=facter state=present"
ansible -m yum -a "pkg=ruby-json state=present"
2013-09-30 01:03:51 +02:00
# for ohai
2014-12-22 05:42:01 +01:00
ansible -m yum -a "pkg=ohai state=present"
2013-09-30 01:03:51 +02:00
Ansible's approach to configuration -- separating variables from tasks, keeps your playbooks
from turning into arbitrary code with ugly nested ifs, conditionals, and so on - and results
in more streamlined & auditable configuration rules -- especially because there are a
minimum of decision points to track.
Selecting Files And Templates Based On Variables
`` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ``
.. note :: This is an advanced topic that is infrequently used. You can probably skip this section.
Sometimes a configuration file you want to copy, or a template you will use may depend on a variable.
The following construct selects the first available file appropriate for the variables of a given host, which is often much cleaner than putting a lot of if conditionals in a template.
The following example shows how to template out a configuration file that was very different between, say, CentOS and Debian::
- name: template a file
template: src={{ item }} dest=/etc/myapp/foo.conf
move from with_<lookup>: to loop:
- old functionality is still available direct lookup use, the following are equivalent
with_nested: [[1,2,3], ['a','b','c']]
loop: "{{lookup('nested', [1,2,3], ['a','b','c'])}}"
- avoid squashing with 'loop:'
- fixed test to use new intenal attributes
- removed most of 'lookup docs' as these now reside in the plugins
2017-09-17 05:32:34 +02:00
loop: "{{lookup('first_found', { 'files': myfiles, 'paths': mypaths})}}"
vars:
myfiles:
- "{{ansible_distribution}}.conf"
- default.conf
mypaths: ['search_location_one/somedir/', '/opt/other_location/somedir/']
2013-09-30 01:03:51 +02:00
Register Variables
`` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ``
Often in a playbook it may be useful to store the result of a given command in a variable and access
it later. Use of the command module in this way can in many ways eliminate the need to write site specific facts, for
instance, you could test for the existence of a particular program.
The 'register' keyword decides what variable to save a result in. The resulting variables can be used in templates, action lines, or *when* statements. It looks like this (in an obviously trivial example)::
- name: test play
hosts: all
tasks:
- shell: cat /etc/motd
register: motd_contents
- shell: echo "motd contains the word hi"
when: motd_contents.stdout.find('hi') != -1
As shown previously, the registered variable's string contents are accessible with the 'stdout' value.
move from with_<lookup>: to loop:
- old functionality is still available direct lookup use, the following are equivalent
with_nested: [[1,2,3], ['a','b','c']]
loop: "{{lookup('nested', [1,2,3], ['a','b','c'])}}"
- avoid squashing with 'loop:'
- fixed test to use new intenal attributes
- removed most of 'lookup docs' as these now reside in the plugins
2017-09-17 05:32:34 +02:00
The registered result can be used in the loop of a task if it is converted into
2013-09-30 01:03:51 +02:00
a list (or already is a list) as shown below. "stdout_lines" is already available on the object as
well though you could also call "home_dirs.stdout.split()" if you wanted, and could split by other
fields::
move from with_<lookup>: to loop:
- old functionality is still available direct lookup use, the following are equivalent
with_nested: [[1,2,3], ['a','b','c']]
loop: "{{lookup('nested', [1,2,3], ['a','b','c'])}}"
- avoid squashing with 'loop:'
- fixed test to use new intenal attributes
- removed most of 'lookup docs' as these now reside in the plugins
2017-09-17 05:32:34 +02:00
- name: registered variable usage as a loop list
2013-09-30 01:03:51 +02:00
hosts: all
tasks:
- name: retrieve the list of home directories
command: ls /home
register: home_dirs
- name: add home dirs to the backup spooler
file: path=/mnt/bkspool/{{ item }} src=/home/{{ item }} state=link
move from with_<lookup>: to loop:
- old functionality is still available direct lookup use, the following are equivalent
with_nested: [[1,2,3], ['a','b','c']]
loop: "{{lookup('nested', [1,2,3], ['a','b','c'])}}"
- avoid squashing with 'loop:'
- fixed test to use new intenal attributes
- removed most of 'lookup docs' as these now reside in the plugins
2017-09-17 05:32:34 +02:00
loop: "{{ home_dirs.stdout_lines }}"
# same as loop: "{{ home_dirs.stdout.split() }}"
2013-09-30 01:03:51 +02:00
2016-04-22 22:23:17 +02:00
As shown previously, the registered variable's string contents are accessible with the 'stdout' value.
You may check the registered variable's string contents for emptiness::
- name: check registered variable for emptiness
hosts: all
tasks:
- name: list contents of directory
command: ls mydir
register: contents
- name: check contents for emptiness
debug: msg="Directory is empty"
when: contents.stdout == ""
2013-09-30 01:03:51 +02:00
2013-10-05 18:31:16 +02:00
.. seealso ::
:doc: `playbooks`
An introduction to playbooks
2017-06-06 23:39:48 +02:00
:doc: `playbooks_reuse_roles`
2013-10-05 18:31:16 +02:00
Playbook organization by roles
:doc: `playbooks_best_practices`
Best practices in playbooks
:doc: `playbooks_variables`
All about variables
`User Mailing List <http://groups.google.com/group/ansible-devel> `_
Have a question? Stop by the google group!
`irc.freenode.net <http://irc.freenode.net> `_
#ansible IRC chat channel