mirror of
				https://github.com/ansible-collections/community.general.git
				synced 2025-10-25 05:23:58 -07:00 
			
		
		
		
	Parse async response in async action. (#16534)
* Parse async response in async action. * Add async test for non-JSON data before module output. * Fix existing async unit test. Resolves #16156
This commit is contained in:
		
					parent
					
						
							
								f86c527736
							
						
					
				
			
			
				commit
				
					
						292785ff2b
					
				
			
		
					 5 changed files with 29 additions and 12 deletions
				
			
		|  | @ -458,15 +458,6 @@ class TaskExecutor: | ||||||
|                 vars_copy[self._task.register] = wrap_var(result.copy()) |                 vars_copy[self._task.register] = wrap_var(result.copy()) | ||||||
| 
 | 
 | ||||||
|             if self._task.async > 0: |             if self._task.async > 0: | ||||||
|                 # the async_wrapper module returns dumped JSON via its stdout |  | ||||||
|                 # response, so we parse it here and replace the result |  | ||||||
|                 try: |  | ||||||
|                     if 'skipped' in result and result['skipped'] or 'failed' in result and result['failed']: |  | ||||||
|                         return result |  | ||||||
|                     result = json.loads(result.get('stdout')) |  | ||||||
|                 except (TypeError, ValueError) as e: |  | ||||||
|                     return dict(failed=True, msg=u"The async task did not return valid JSON: %s" % to_unicode(e)) |  | ||||||
| 
 |  | ||||||
|                 if self._task.poll > 0: |                 if self._task.poll > 0: | ||||||
|                     result = self._poll_async_result(result=result, templar=templar) |                     result = self._poll_async_result(result=result, templar=templar) | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -99,4 +99,11 @@ class ActionModule(ActionBase): | ||||||
| 
 | 
 | ||||||
|         result['changed'] = True |         result['changed'] = True | ||||||
| 
 | 
 | ||||||
|  |         if 'skipped' in result and result['skipped'] or 'failed' in result and result['failed']: | ||||||
|  |             return result | ||||||
|  | 
 | ||||||
|  |         # the async_wrapper module returns dumped JSON via its stdout | ||||||
|  |         # response, so we parse it here and replace the result | ||||||
|  |         result = self._parse_returned_data(result) | ||||||
|  | 
 | ||||||
|         return result |         return result | ||||||
|  |  | ||||||
|  | @ -27,7 +27,7 @@ UNAME := $(shell uname | tr '[:upper:]' '[:lower:]') | ||||||
| 
 | 
 | ||||||
| all: setup other non_destructive destructive | all: setup other non_destructive destructive | ||||||
| 
 | 
 | ||||||
| other: test_test_infra parsing test_var_precedence unicode test_templating_settings environment test_connection includes blocks pull check_mode test_hash test_handlers test_group_by test_vault test_tags test_lookup_paths no_log test_gathering_facts test_binary_modules | other: test_test_infra parsing test_var_precedence unicode test_templating_settings environment test_connection includes blocks pull check_mode test_hash test_handlers test_group_by test_vault test_tags test_lookup_paths no_log test_gathering_facts test_binary_modules test_async | ||||||
| 
 | 
 | ||||||
| test_test_infra: | test_test_infra: | ||||||
| 	# ensure fail/assert work locally and can stop execution with non-zero exit code | 	# ensure fail/assert work locally and can stop execution with non-zero exit code | ||||||
|  | @ -302,3 +302,10 @@ test_binary_modules: | ||||||
| 	cd ..; \
 | 	cd ..; \
 | ||||||
| 	rm -rf $$mytmpdir; \
 | 	rm -rf $$mytmpdir; \
 | ||||||
| 	ANSIBLE_HOST_KEY_CHECKING=false ansible-playbook test_binary_modules.yml -i $(INVENTORY) -v $(TEST_FLAGS) | 	ANSIBLE_HOST_KEY_CHECKING=false ansible-playbook test_binary_modules.yml -i $(INVENTORY) -v $(TEST_FLAGS) | ||||||
|  | 
 | ||||||
|  | test_async: | ||||||
|  | 	# Verify that extra data before module JSON output during async call is ignored. | ||||||
|  | 	LC_ALL=bogus ansible-playbook test_async.yml -i $(INVENTORY) -e outputdir=$(TEST_DIR) -v $(TEST_FLAGS) | ||||||
|  | 	# Verify that the warning exists by examining debug output. | ||||||
|  | 	ANSIBLE_DEBUG=1 LC_ALL=bogus ansible-playbook test_async.yml -i $(INVENTORY) -e outputdir=$(TEST_DIR) -v $(TEST_FLAGS) \
 | ||||||
|  | 	| grep -q 'bash: warning: setlocale: LC_ALL: cannot change locale (bogus)' | ||||||
|  |  | ||||||
							
								
								
									
										10
									
								
								test/integration/test_async.yml
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								test/integration/test_async.yml
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,10 @@ | ||||||
|  | - hosts: testhost3 | ||||||
|  |   gather_facts: false | ||||||
|  |   tasks: | ||||||
|  |   # make sure non-JSON data before module output is ignored | ||||||
|  |   - name: async ping with invalid locale via ssh | ||||||
|  |     ping: | ||||||
|  |     async: 3 | ||||||
|  |     poll: 1 | ||||||
|  |     register: result | ||||||
|  |   - debug: var=result | ||||||
|  | @ -26,6 +26,7 @@ from ansible.errors import AnsibleError, AnsibleParserError | ||||||
| from ansible.executor.task_executor import TaskExecutor | from ansible.executor.task_executor import TaskExecutor | ||||||
| from ansible.playbook.play_context import PlayContext | from ansible.playbook.play_context import PlayContext | ||||||
| from ansible.plugins import action_loader, lookup_loader | from ansible.plugins import action_loader, lookup_loader | ||||||
|  | from ansible.parsing.yaml.objects import AnsibleUnicode | ||||||
| 
 | 
 | ||||||
| from units.mock.loader import DictDataLoader | from units.mock.loader import DictDataLoader | ||||||
| 
 | 
 | ||||||
|  | @ -375,6 +376,7 @@ class TestTaskExecutor(unittest.TestCase): | ||||||
|         # here: on Python 2 comparing MagicMock() > 0 returns True, and the |         # here: on Python 2 comparing MagicMock() > 0 returns True, and the | ||||||
|         # other reason is that if I specify 0 here, the test fails. ;) |         # other reason is that if I specify 0 here, the test fails. ;) | ||||||
|         mock_task.async = 1 |         mock_task.async = 1 | ||||||
|  |         mock_task.poll = 0 | ||||||
| 
 | 
 | ||||||
|         mock_play_context = MagicMock() |         mock_play_context = MagicMock() | ||||||
|         mock_play_context.post_validate.return_value = None |         mock_play_context.post_validate.return_value = None | ||||||
|  | @ -408,11 +410,11 @@ class TestTaskExecutor(unittest.TestCase): | ||||||
|         mock_action.run.return_value = dict(ansible_facts=dict()) |         mock_action.run.return_value = dict(ansible_facts=dict()) | ||||||
|         res = te._execute() |         res = te._execute() | ||||||
| 
 | 
 | ||||||
|         mock_task.changed_when = "1 == 1" |         mock_task.changed_when = MagicMock(return_value=AnsibleUnicode("1 == 1")) | ||||||
|         res = te._execute() |         res = te._execute() | ||||||
| 
 | 
 | ||||||
|         mock_task.changed_when = None |         mock_task.changed_when = None | ||||||
|         mock_task.failed_when = "1 == 1" |         mock_task.failed_when = MagicMock(return_value=AnsibleUnicode("1 == 1")) | ||||||
|         res = te._execute() |         res = te._execute() | ||||||
| 
 | 
 | ||||||
|         mock_task.failed_when = None |         mock_task.failed_when = None | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue