Inject attempts into result earlier for retry logic (#34148)
* Add integration test for missing attempts in until with failed/changed_when * Extend tests to validate additional known behaviors of do-until loops * Inject attempts into result earlier
This commit is contained in:
parent
6d3fafa2d5
commit
3d0da8f093
2 changed files with 51 additions and 3 deletions
|
@ -535,7 +535,7 @@ class TaskExecutor:
|
||||||
# update the local copy of vars with the registered value, if specified,
|
# update the local copy of vars with the registered value, if specified,
|
||||||
# or any facts which may have been generated by the module execution
|
# or any facts which may have been generated by the module execution
|
||||||
if self._task.register:
|
if self._task.register:
|
||||||
vars_copy[self._task.register] = wrap_var(result.copy())
|
vars_copy[self._task.register] = wrap_var(result)
|
||||||
|
|
||||||
if self._task.async_val > 0:
|
if self._task.async_val > 0:
|
||||||
if self._task.poll > 0 and not result.get('skipped') and not result.get('failed'):
|
if self._task.poll > 0 and not result.get('skipped') and not result.get('failed'):
|
||||||
|
@ -575,10 +575,20 @@ class TaskExecutor:
|
||||||
else:
|
else:
|
||||||
result['failed'] = False
|
result['failed'] = False
|
||||||
|
|
||||||
|
# Make attempts and retries available early to allow their use in changed/failed_when
|
||||||
|
result['attempts'] = attempt
|
||||||
|
|
||||||
# set the changed property if it was missing.
|
# set the changed property if it was missing.
|
||||||
if 'changed' not in result:
|
if 'changed' not in result:
|
||||||
result['changed'] = False
|
result['changed'] = False
|
||||||
|
|
||||||
|
# re-update the local copy of vars with the registered value, if specified,
|
||||||
|
# or any facts which may have been generated by the module execution
|
||||||
|
# This gives changed/failed_when access to additional recently modified
|
||||||
|
# attributes of result
|
||||||
|
if self._task.register:
|
||||||
|
vars_copy[self._task.register] = wrap_var(result)
|
||||||
|
|
||||||
# if we didn't skip this task, use the helpers to evaluate the changed/
|
# if we didn't skip this task, use the helpers to evaluate the changed/
|
||||||
# failed_when properties
|
# failed_when properties
|
||||||
if 'skipped' not in result:
|
if 'skipped' not in result:
|
||||||
|
@ -588,7 +598,6 @@ class TaskExecutor:
|
||||||
if retries > 1:
|
if retries > 1:
|
||||||
cond = Conditional(loader=self._loader)
|
cond = Conditional(loader=self._loader)
|
||||||
cond.when = self._task.until
|
cond.when = self._task.until
|
||||||
result['attempts'] = attempt
|
|
||||||
if cond.evaluate_conditional(templar, vars_copy):
|
if cond.evaluate_conditional(templar, vars_copy):
|
||||||
break
|
break
|
||||||
else:
|
else:
|
||||||
|
|
|
@ -29,4 +29,43 @@
|
||||||
- assert:
|
- assert:
|
||||||
that: runcount.stdout | int == 6 # initial + 5 retries
|
that: runcount.stdout | int == 6 # initial + 5 retries
|
||||||
|
|
||||||
- file: path="{{ until_tempfile_path }}" state=absent
|
- file:
|
||||||
|
path: "{{ until_tempfile_path }}"
|
||||||
|
state: absent
|
||||||
|
|
||||||
|
- name: Test failed_when impacting until
|
||||||
|
shell: 'true'
|
||||||
|
register: failed_when_until
|
||||||
|
failed_when: True
|
||||||
|
until: failed_when_until is successful
|
||||||
|
retries: 3
|
||||||
|
delay: 0.5
|
||||||
|
ignore_errors: True
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- failed_when_until.attempts == 3
|
||||||
|
|
||||||
|
- name: Test changed_when impacting until
|
||||||
|
shell: 'true'
|
||||||
|
register: changed_when_until
|
||||||
|
changed_when: False
|
||||||
|
until: changed_when_until is changed
|
||||||
|
retries: 3
|
||||||
|
delay: 0.5
|
||||||
|
ignore_errors: True
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- changed_when_until.attempts == 3
|
||||||
|
|
||||||
|
# This task shouldn't fail, previously .attempts was not available to changed_when/failed_when
|
||||||
|
# and would cause the conditional to fail due to ``'dict object' has no attribute 'attempts'``
|
||||||
|
# https://github.com/ansible/ansible/issues/34139
|
||||||
|
- name: Test access to attempts in changed_when/failed_when
|
||||||
|
shell: 'true'
|
||||||
|
register: changed_when_attempts
|
||||||
|
until: 1 == 0
|
||||||
|
retries: 5
|
||||||
|
delay: 0.5
|
||||||
|
failed_when: changed_when_attempts.attempts > 6
|
||||||
|
|
Loading…
Reference in a new issue