9.9 KiB
Loops
Often you'll want to do many things in one task, such as create a lot of users, install a lot of packages, or repeat a polling step until a certain result is reached.
This chapter is all about how to use loops in playbooks.
Topics
Standard Loops
To save some typing, repeated tasks can be written in short-hand like so:
- name: add several users
user:
name: "{{ item }}"
state: present
groups: "wheel"
loop:
- testuser1
- testuser2
If you have defined a YAML list in a variables file, or the 'vars' section, you can also do:
loop: "{{ somelist }}"
The above would be the equivalent of:
- name: add user testuser1
user:
name: "testuser1"
state: present
groups: "wheel"
- name: add user testuser2
user:
name: "testuser2"
state: present
groups: "wheel"
Note
Before 2.5 Ansible mainly used the with_<lookup>
keywords to create loops, the loop
keyword is basically analogous to with_list
.
Some plugins like, the yum and apt modules can take lists directly to their options, this is more optimal than looping over the task. See each action's documentation for details, for now here is an example:
- name: optimal yum
yum: name={{list_of_packages}} state=present
- name: non optimal yum, not only slower but might cause issues with interdependencies
yum: name={{item}} state=present
loop: "{{list_of_packages}}"
Note that the types of items you iterate over do not have to be simple lists of strings. If you have a list of hashes, you can reference subkeys using things like:
- name: add several users
user:
name: "{{ item.name }}"
state: present
groups: "{{ item.groups }}"
loop:
- { name: 'testuser1', groups: 'wheel' }
- { name: 'testuser2', groups: 'root' }
Also be aware that when combining when: playbooks_conditionals
with a loop, the
when:
statement is processed separately for each item. See
the_when_statement
for
an example.
Complex loops
Sometimes you need more than what a simple list provides, you can use Jinja2 expressions to create complex lists: For example, using the 'nested' lookup, you can combine lists:
- name: give users access to multiple databases
mysql_user:
name: "{{ item[0] }}"
priv: "{{ item[1] }}.*:ALL"
append_privs: yes
password: "foo"
loop: "{{ lookup('nested', [ 'alice', 'bob' ], [ 'clientdb', 'employeedb', 'providerdb' ]) }}"
Jinja2 lookups playbooks_lookups
, filters playbooks_filters
and
tests playbooks_tests
make for some powerful data generation and manipulation.
Note
with_
loops are actually a combination of things
with_
+ lookup()
, even items
is a
lookup. loop
can be used in the same way as shown
above.
Do-Until Loops
1.4
Sometimes you would want to retry a task until a certain condition is met. Here's an example:
- shell: /usr/bin/foo
register: result
until: result.stdout.find("all systems go") != -1
retries: 5
delay: 10
The above example run the shell module recursively till the module's result has "all systems go" in its stdout or the task has been retried for 5 times with a delay of 10 seconds. The default value for "retries" is 3 and "delay" is 5.
The task returns the results returned by the last task run. The results of individual retries can be viewed by -vv option. The registered variable will also have a new key "attempts" which will have the number of the retries for the task.
Note
If the until
parameter isn't defined, the value for the
retries
parameter is forced to 1.
Using register with a loop
After using register
with a loop, the data structure
placed in the variable will contain a results
attribute
that is a list of all responses from the module.
Here is an example of using register
with
with_items
:
- shell: "echo {{ item }}"
loop:
- "one"
- "two"
register: echo
This differs from the data structure returned when using
register
without a loop:
{
"changed": true,
"msg": "All items completed",
"results": [
{
"changed": true,
"cmd": "echo \"one\" ",
"delta": "0:00:00.003110",
"end": "2013-12-19 12:00:05.187153",
"invocation": {
"module_args": "echo \"one\"",
"module_name": "shell"
},
"item": "one",
"rc": 0,
"start": "2013-12-19 12:00:05.184043",
"stderr": "",
"stdout": "one"
},
{
"changed": true,
"cmd": "echo \"two\" ",
"delta": "0:00:00.002920",
"end": "2013-12-19 12:00:05.245502",
"invocation": {
"module_args": "echo \"two\"",
"module_name": "shell"
},
"item": "two",
"rc": 0,
"start": "2013-12-19 12:00:05.242582",
"stderr": "",
"stdout": "two"
}
]
}
Subsequent loops over the registered variable to inspect the results may look like:
- name: Fail if return code is not 0
fail:
msg: "The command ({{ item.cmd }}) did not have a 0 return code"
when: item.rc != 0
loop: "{{ echo.results }}"
During iteration, the result of the current item will be placed in the variable:
- shell: echo "{{ item }}"
loop:
- one
- two
register: echo
changed_when: echo.stdout != "one"
Looping over the inventory
If you wish to loop over the inventory, or just a subset of it, there
is multiple ways. One can use a regular loop
with the
ansible_play_batch
or groups
variables, like
this:
# show all the hosts in the inventory
- debug:
msg: "{{ item }}"
loop:
- "{{ groups['all'] }}"
# show all the hosts in the current play
- debug:
msg: "{{ item }}"
loop:
- "{{ ansible_play_batch }}"
There is also a specific lookup plugin
inventory_hostnames
that can be used like this:
# show all the hosts in the inventory
- debug:
msg: "{{ item }}"
loop: "{{lookup('inventory_hostnames', 'all'}}"
# show all the hosts matching the pattern, ie all but the group www
- debug:
msg: "{{ item }}"
loop: "{{lookup('inventory_hostnames', 'all!www'}}"
More information on the patterns can be found on intro_patterns
Loop Control
2.1
In 2.0 you are again able to use loops and task includes (but not
playbook includes). This adds the ability to loop over the set of tasks
in one shot. Ansible by default sets the loop variable item
for each loop, which causes these nested loops to overwrite the value of
item
from the "outer" loops. As of Ansible 2.1, the
loop_control
option can be used to specify the name of the
variable to be used for the loop:
# main.yml
- include: inner.yml
- include_tasks: inner.yml
loop:
- 1
- 2
- 3
loop_control:
loop_var: outer_item
# inner.yml
- debug:
msg: "outer item={{ outer_item }} inner item={{ item }}"
loop:
- a
- b
- c
Note
If Ansible detects that the current loop is using a variable which has already been defined, it will raise an error to fail the task.
2.2
When using complex data structures for looping the display might get
a bit too "busy", this is where the label
directive comes
to help:
- name: create servers
digital_ocean:
name: "{{ item.name }}"
state: present
loop:
- name: server1
disks: 3gb
ram: 15Gb
network:
nic01: 100Gb
nic02: 10Gb
...
loop_control:
label: "{{ item.name }}"
This will now display just the label
field instead of
the whole structure per item
, it defaults to
{{ item }}
to display things as usual.
2.2
Another option to loop control is pause
, which allows
you to control the time (in seconds) between execution of items in a
task loop.:
# main.yml
- name: create servers, pause 3s before creating next
digital_ocean:
name: "{{ item }}"
state: present
loop:
- server1
- server2
loop_control:
pause: 3
Loops and Includes in 2.0
Because loop_control
is not available in Ansible 2.0,
when using an include with a loop you should use set_fact
to save the "outer" loops value for item
:
# main.yml
- include_tasks: inner.yml
loop:
- 1
- 2
- 3
# inner.yml
- set_fact:
outer_item: "{{ item }}"
- debug:
msg: "outer item={{ outer_item }} inner item={{ item }}"
loop:
- a
- b
- c
Note
include is deprecated, you should be using include_tasks, import_tasks, import_play instead.
playbooks
-
An introduction to playbooks
playbooks_reuse_roles
-
Playbook organization by roles
playbooks_best_practices
-
Best practices in playbooks
playbooks_conditionals
-
Conditional statements in playbooks
playbooks_variables
-
All about variables
- User Mailing List
-
Have a question? Stop by the google group!
- irc.freenode.net
-
#ansible IRC chat channel