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
|
||||
# a dummy array for the later code so it doesn't fail
|
||||
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:
|
||||
raise AnsibleError("Failed to find the lookup named '%s' in the available lookup plugins" % task.loop_with)
|
||||
elif task.loop is not None:
|
||||
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:
|
||||
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-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