Cache items when task.loop/with_items is evaluated to set delegate_to vars (#41969)

* If we evaluate task.loop/with_items when calculating delegate_to vars, cache the items. Fixes #28231

* Add comments about caching loop items

* Add test for delegate_to+loop+random

* Be more careful about where we update task.loop
This commit is contained in:
Matt Martz 2018-06-26 15:10:04 -05:00 committed by GitHub
parent 11ce954226
commit 1401fa74cc
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
3 changed files with 69 additions and 0 deletions

View file

@ -498,10 +498,19 @@ class VariableManager:
# This task will be skipped later due to this, so we just setup # This task will be skipped later due to this, so we just setup
# a dummy array for the later code so it doesn't fail # a dummy array for the later code so it doesn't fail
items = [None] items = [None]
# Update task.loop with templated items, this ensures that delegate_to+loop
# doesn't produce different restuls than TaskExecutor which may reprocess the loop
# Set loop_with to None, so we don't do extra unexpected processing on the cached items later
# in TaskExecutor
task.loop_with = None
task.loop = items
else: else:
raise AnsibleError("Failed to find the lookup named '%s' in the available lookup plugins" % task.loop_with) raise AnsibleError("Failed to find the lookup named '%s' in the available lookup plugins" % task.loop_with)
elif task.loop is not None: elif task.loop is not None:
items = templar.template(task.loop) items = templar.template(task.loop)
# Update task.loop with templated items, this ensures that delegate_to+loop
# doesn't produce different restuls than TaskExecutor which may reprocess the loop
task.loop = items
else: else:
items = [None] items = [None]

View file

@ -6,3 +6,5 @@ ANSIBLE_SSH_ARGS='-C -o ControlMaster=auto -o ControlPersist=60s -o UserKnownHos
ANSIBLE_HOST_KEY_CHECKING=false ansible-playbook test_delegate_to.yml -i ../../inventory -v "$@" ANSIBLE_HOST_KEY_CHECKING=false ansible-playbook test_delegate_to.yml -i ../../inventory -v "$@"
ansible-playbook test_loop_control.yml -v "$@" ansible-playbook test_loop_control.yml -v "$@"
ansible-playbook test_delegate_to_loop_randomness.yml -v "$@"

View file

@ -0,0 +1,58 @@
---
- name: Integration tests for #28231
hosts: localhost
gather_facts: false
tasks:
- name: Add some test hosts
add_host:
name: "foo{{item}}"
groups: foo
loop: "{{ range(10)|list }}"
# We expect all of the next 3 runs to succeeed
# this is done multiple times to increase randomness
- assert:
that:
- item in ansible_delegated_vars
delegate_to: "{{ item }}"
loop:
- "{{ groups.foo|random }}"
ignore_errors: true
register: result1
- assert:
that:
- item in ansible_delegated_vars
delegate_to: "{{ item }}"
loop:
- "{{ groups.foo|random }}"
ignore_errors: true
register: result2
- assert:
that:
- item in ansible_delegated_vars
delegate_to: "{{ item }}"
loop:
- "{{ groups.foo|random }}"
ignore_errors: true
register: result3
- debug:
var: result1
- debug:
var: result2
- debug:
var: result3
- name: Ensure all of the 3 asserts were successful
assert:
that:
- results is all
vars:
results:
- "{{ (result1.results|first) is successful }}"
- "{{ (result2.results|first) is successful }}"
- "{{ (result3.results|first) is successful }}"