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:
parent
11ce954226
commit
1401fa74cc
3 changed files with 69 additions and 0 deletions
|
@ -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]
|
||||||
|
|
||||||
|
|
|
@ -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 "$@"
|
||||||
|
|
|
@ -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 }}"
|
Loading…
Reference in a new issue