Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add an 'extract' filter #13345

Merged
merged 1 commit into from
Dec 3, 2015
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Jump to
Jump to file
Failed to load files.
Diff view
Diff view
33 changes: 33 additions & 0 deletions docsite/rst/playbooks_filters.rst
Original file line number Diff line number Diff line change
Expand Up @@ -352,6 +352,39 @@ override those in `b`, and so on.
This behaviour does not depend on the value of the `hash_behaviour`
setting in `ansible.cfg`.

.. _extract_filter:

Extracting values from containers
---------------------------------

.. versionadded:: 2.0

The `extract` filter is used to map from a list of indices to a list of
values from a container (hash or array)::

{{ [0,2]|map('extract', ['x','y','z'])|list }}
{{ ['x','y']|map('extract', {'x': 42, 'y': 31})|list }}

The results of the above expressions would be::

['x', 'z']
[42, 31]

The filter can take another argument::

{{ groups['x']|map('extract', hostvars, 'ec2_ip_address')|list }}

This takes the list of hosts in group 'x', looks them up in `hostvars`,
and then looks up the `ec2_ip_address` of the result. The final result
is a list of IP addresses for the hosts in group 'x'.

The third argument to the filter can also be a list, for a recursive
lookup inside the container::

{{ ['a']|map('extract', b, ['x','y'])|list }}

This would return a list containing the value of `b['a']['x']['y']`.

.. _comment_filter:

Comment Filter
Expand Down
15 changes: 15 additions & 0 deletions lib/ansible/plugins/filter/core.py
Original file line number Diff line number Diff line change
Expand Up @@ -339,6 +339,18 @@ def comment(text, style='plain', **kw):
str_postfix,
str_end)

def extract(item, container, morekeys=None):
from jinja2.runtime import Undefined

value = container[item]

if value is not Undefined and morekeys is not None:
if not isinstance(morekeys, list):
morekeys = [morekeys]

value = reduce(lambda d, k: d[k], morekeys, value)

return value

class FilterModule(object):
''' Ansible core jinja2 filters '''
Expand Down Expand Up @@ -415,4 +427,7 @@ def filters(self):

# comment-style decoration
'comment': comment,

# array and dict lookups
'extract': extract,
}
10 changes: 10 additions & 0 deletions test/integration/roles/test_filters/tasks/main.yml
Original file line number Diff line number Diff line change
Expand Up @@ -68,3 +68,13 @@
- '"0.10 GB" == 102400000|human_readable(unit="G")'
- '"0.10 Gb" == 102400000|human_readable(isbits=True, unit="G")'

- name: Container lookups with extract
assert:
that:
- "'x' == [0]|map('extract',['x','y'])|list|first"
- "'y' == [1]|map('extract',['x','y'])|list|first"
- "42 == ['x']|map('extract',{'x':42,'y':31})|list|first"
- "31 == ['x','y']|map('extract',{'x':42,'y':31})|list|last"
- "'local' == ['localhost']|map('extract',hostvars,'ansible_connection')|list|first"
- "'local' == ['localhost']|map('extract',hostvars,['ansible_connection'])|list|first"
- "'ungrouped' == ['localhost']|map('extract',hostvars,['vars','group_names',0])|list|first"