2013-09-29 23:02:53 +02:00
Loops
=====
2013-09-25 07:18:43 +02:00
2013-10-05 19:25:05 +02:00
.. contents ::
:depth: 2
2013-10-05 19:50:53 +02:00
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.
2013-09-25 06:26:14 +02:00
2013-10-05 00:34:39 +02:00
.. _standard_loops:
2013-09-25 06:26:14 +02:00
2013-09-29 23:02:53 +02:00
Standard Loops
`` ` ` ` ` ` ` ` ` ` ` ``
2012-05-13 17:00:02 +02:00
To save some typing, repeated tasks can be written in short-hand like so::
2013-03-02 17:47:43 +01:00
- name: add several users
2013-09-18 19:22:19 +02:00
user: name={{ item }} state=present groups=wheel
2012-05-13 17:00:02 +02:00
with_items:
- testuser1
- testuser2
2012-08-01 04:19:04 +02:00
If you have defined a YAML list in a variables file, or the 'vars' section, you can also do::
2013-04-13 00:21:09 +02:00
with_items: somelist
2012-08-01 04:19:04 +02:00
2012-05-13 17:00:02 +02:00
The above would be the equivalent of::
- name: add user testuser1
2013-09-18 19:22:19 +02:00
user: name=testuser1 state=present groups=wheel
2012-05-13 17:00:02 +02:00
- name: add user testuser2
2013-09-18 19:22:19 +02:00
user: name=testuser2 state=present groups=wheel
2012-05-13 17:00:02 +02:00
2012-10-08 13:38:21 +02:00
The yum and apt modules use with_items to execute fewer package manager transactions.
2012-05-13 17:00:02 +02:00
2012-10-17 01:12:31 +02:00
Note that the types of items you iterate over with 'with_items' do not have to be simple lists of strings.
2012-10-17 00:58:31 +02:00
If you have a list of hashes, you can reference subkeys using things like::
2013-05-30 00:05:14 +02:00
- name: add several users
2013-09-18 19:22:19 +02:00
user: name={{ item.name }} state=present groups={{ item.groups }}
2013-05-30 00:05:14 +02:00
with_items:
- { name: 'testuser1', groups: 'wheel' }
- { name: 'testuser2', groups: 'root' }
2012-10-17 00:58:31 +02:00
2013-10-05 00:34:39 +02:00
.. _nested_loops:
2013-07-15 06:28:02 +02:00
Nested Loops
`` ` ` ` ` ` ` ` ` ``
2013-07-19 15:31:50 +02:00
Loops can be nested as well::
2013-07-15 06:28:02 +02:00
2013-07-19 15:13:30 +02:00
- name: give users access to multiple databases
2013-09-22 14:17:28 +02:00
mysql_user: name={{ item[0] }} priv={{ item[1] }}.*:ALL password=foo
2013-07-15 06:28:02 +02:00
with_nested:
- [ 'alice', 'bob', 'eve' ]
- [ 'clientdb', 'employeedb', 'providerdb' ]
2013-09-22 14:17:28 +02:00
As with the case of 'with_items' above, you can use previously defined variables. Just specify the variable's name without templating it with '{{ }}'::
2013-07-15 06:28:02 +02:00
- name: here, 'users' contains the above list of employees
2013-09-22 14:17:28 +02:00
mysql_user: name={{ item[0] }} priv={{ item[1] }}.*:ALL password=foo
2013-07-15 06:28:02 +02:00
with_nested:
- users
- [ 'clientdb', 'employeedb', 'providerdb' ]
2013-10-05 00:34:39 +02:00
.. _looping_over_fileglobs:
2013-09-29 23:02:53 +02:00
Looping over Fileglobs
`` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ``
2012-10-17 00:51:10 +02:00
2013-04-05 19:00:23 +02:00
`` with_fileglob `` matches all files in a single directory, non-recursively, that match a pattern. It can
2012-10-17 00:51:10 +02:00
be used like this::
2013-05-31 10:54:13 +02:00
---
2012-10-17 00:51:10 +02:00
- hosts: all
tasks:
# first ensure our target directory exists
2013-07-15 19:50:48 +02:00
- file: dest=/etc/fooapp state=directory
2012-10-17 00:51:10 +02:00
# copy each file over that matches the given pattern
2013-07-15 19:50:48 +02:00
- copy: src={{ item }} dest=/etc/fooapp/ owner=root mode=600
2013-02-17 18:47:51 +01:00
with_fileglob:
2013-02-02 17:51:25 +01:00
- /playbooks/files/fooapp/*
2013-09-29 23:02:53 +02:00
Looping over Parallel Sets of Data
`` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ``
2013-06-14 17:13:38 +02:00
2013-10-05 00:34:39 +02:00
.. note :: This is an uncommon thing to want to do, but we're documenting it for completeness. You won't be reaching for this one often.
Suppose you have the following variable data was loaded in via somewhere::
---
alpha: [ 'a', 'b', 'c', 'd' ]
numbers: [ 1, 2, 3, 4 ]
And you want the set of '(a, 1)' and '(b, 2)' and so on. Use 'with_together' to get this::
tasks:
- debug: msg="{{ item.0 }} and {{ item.1 }}"
with_together:
- alpha
- numbers
2013-02-02 17:51:25 +01:00
2013-09-29 23:02:53 +02:00
Looping over Subelements
`` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ``
2013-02-02 18:19:21 +01:00
2013-10-05 00:34:39 +02:00
Suppose you want to do something like loop over a list of users, creating them, and allowing them to login by a certain set of
SSH keys.
How might that be accomplished? Let's assume you had the following defined and loaded in via "vars_files" or maybe a "group_vars/all" file::
---
users:
- name: alice
authorized:
- /tmp/alice/onekey.pub
- /tmp/alice/twokey.pub
- name: bob
authorized:
- /tmp/bob/id_rsa.pub
It might happen like so::
- user: name={{ item.name }} state=present generate_ssh_key=yes
with_items: users
2013-02-02 18:56:56 +01:00
2013-10-05 00:34:39 +02:00
- authorized_key: user={{ item.0.name }} key={{ lookup('file', item.1) }}
with_subelements:
- users
- authorized
Subelements walks a list of hashes (aka dictionaries) and then traverses a list with a given key inside of those
records.
The authorized_key pattern is exactly where it comes up most.
.. _looping_over_integer_sequences:
2013-02-02 18:19:21 +01:00
2013-09-29 23:02:53 +02:00
Looping over Integer Sequences
`` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ` ``
2013-01-08 21:13:25 +01:00
2013-04-05 19:00:23 +02:00
`` with_sequence `` generates a sequence of items in ascending numerical order. You
2013-02-02 17:51:25 +01:00
can specify a start, end, and an optional step value.
2013-01-08 21:13:25 +01:00
2013-04-13 00:21:09 +02:00
Arguments should be specified in key=value pairs. If supplied, the 'format' is a printf style string.
2013-02-02 17:51:25 +01:00
2013-02-17 18:47:51 +01:00
Numerical values can be specified in decimal, hexadecimal (0x3f8) or octal (0600).
2013-02-02 17:51:25 +01:00
Negative numbers are not supported. This works as follows::
---
2013-01-08 21:13:25 +01:00
- hosts: all
tasks:
# create groups
- group: name=evens state=present
- group: name=odds state=present
2013-04-13 00:21:09 +02:00
# create some test users
2013-09-18 19:22:19 +02:00
- user: name={{ item }} state=present groups=evens
2013-04-17 02:41:42 +02:00
with_sequence: start=0 end=32 format=testuser%02x
2013-01-08 21:13:25 +01:00
2013-04-13 00:21:09 +02:00
# create a series of directories with even numbers for some reason
- file: dest=/var/stuff/{{ item }} state=directory
with_sequence: start=4 end=16 stride=2
2013-01-08 21:13:25 +01:00
2013-02-02 17:51:25 +01:00
# a simpler way to use the sequence plugin
2013-01-08 21:13:25 +01:00
# create 4 groups
2013-04-13 00:21:09 +02:00
- group: name=group{{ item }} state=present
2013-01-08 21:13:25 +01:00
with_sequence: count=4
2013-10-05 00:34:39 +02:00
.. _random_choice:
Random Choices
`` ` ` ` ` ` ` ` ` ` ` ``
The 'random_choice' feature can be used to pick something at random. While it's not a load balancer, it can
somewhat be used as a poor man's loadbalancer in a MacGyver like situation::
- debug: msg={{ item }}
with_random_choice:
- "go through the door"
- "drink from the goblet"
- "press the red button"
- "do nothing"
One of the provided strings will be selected at random.
At a more basic level, they can be used to add chaos and excitement to otherwise predictable automation environments.
.. _do_until_loops:
2013-09-29 23:02:53 +02:00
Do-Until Loops
`` ` ` ` ` ` ` ` ` ` ` ``
2013-09-03 06:08:00 +02:00
2013-09-29 23:02:53 +02:00
Sometimes you would want to retry a task until a certain condition is met. Here's an example::
- action: shell /usr/bin/foo
register: result
2013-10-03 22:28:29 +02:00
until: result.stdout.find("all systems go") != -1
2013-09-29 23:02:53 +02:00
retries: 5
delay: 10
2013-08-12 15:49:34 +02:00
2013-09-29 23:02:53 +02:00
The above example run the shell module recursively till the module's result has "all systems go" in it's 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.
2012-08-01 04:19:04 +02:00
2013-09-29 23:02:53 +02:00
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.
2012-08-01 04:19:04 +02:00
2013-09-29 23:02:53 +02:00
The Do/Until feature does not take decision on whether to fail or pass the play when the maximum retries are completed, the user can
can do that in the next task as follows::
- action: shell /usr/bin/foo
register: result
2013-10-03 22:28:29 +02:00
until: result.stdout.find("all systems go") != -1
2013-09-29 23:02:53 +02:00
retries: 5
delay: 10
failed_when: result.attempts == 5
2012-05-13 17:56:09 +02:00
2013-10-05 18:31:16 +02:00
.. seealso ::
:doc: `playbooks`
An introduction to playbooks
:doc: `playbooks_roles`
Playbook organization by roles
:doc: `playbooks_best_practices`
Best practices in playbooks
:doc: `playbooks_conditionals`
Conditional statements 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
2012-05-13 17:00:02 +02:00