From d3a1aea7c5ee37b11f9798618d9ae83beb568d83 Mon Sep 17 00:00:00 2001 From: Toshio Kuratomi Date: Tue, 28 Mar 2017 11:38:37 -0700 Subject: [PATCH] Fix a couple issues in synchronize with docker (#23047) * Fix a couple issues in synchronize with docker * Make the rsync_opts parse as a list using the same criteria as module_utils argumentspec parsing * Do not quote arguments in the action plugin. The module will quote as it knows whether it will invoke rsync with a shell or via exec. Fixes #23046 --- lib/ansible/plugins/action/synchronize.py | 26 +++++++++++++++-------- 1 file changed, 17 insertions(+), 9 deletions(-) diff --git a/lib/ansible/plugins/action/synchronize.py b/lib/ansible/plugins/action/synchronize.py index 7283fedd39..df1bee5767 100644 --- a/lib/ansible/plugins/action/synchronize.py +++ b/lib/ansible/plugins/action/synchronize.py @@ -18,11 +18,14 @@ from __future__ import (absolute_import, division, print_function) __metaclass__ = type import os.path +from collections import MutableSequence +from ansible import constants as C +from ansible.module_utils.six import string_types +from ansible.module_utils._text import to_text from ansible.playbook.play_context import MAGIC_VARIABLE_MAPPING from ansible.plugins.action import ActionBase from ansible.plugins import connection_loader -from ansible import constants as C boolean = C.mk_boolean @@ -364,9 +367,7 @@ class ActionModule(ActionBase): # to. self._play_context.become = False - # make sure rsync path is quoted. - if rsync_path: - _tmp_args['rsync_path'] = '"%s"' % rsync_path + _tmp_args['rsync_path'] = rsync_path if use_ssh_args: ssh_args = [ @@ -379,16 +380,23 @@ class ActionModule(ActionBase): # If launching synchronize against docker container # use rsync_opts to support container to override rsh options if self._remote_transport in [ 'docker' ]: - if not isinstance(_tmp_args.get('rsync_opts'), list): - _tmp_args['rsync_opts'] = self._task.args.get('rsync_opts', '').split(' ') + # Replicate what we do in the module argumentspec handling for lists + if not isinstance(_tmp_args.get('rsync_opts'), MutableSequence): + tmp_rsync_opts = _tmp_args.get('rsync_opts', []) + if isinstance(tmp_rsync_opts, string_types): + tmp_rsync_opts = tmp_rsync_opts.split(',') + elif isinstance(tmp_rsync_opts, (int, float)): + tmp_rsync_opts = [to_text(tmp_rsync_opts)] + _tmp_args['rsync_opts'] = tmp_rsync_opts + if '--blocking-io' not in _tmp_args['rsync_opts']: _tmp_args['rsync_opts'].append('--blocking-io') if become and self._play_context.become_user: - _tmp_args['rsync_opts'].append("--rsh='%s exec -u %s -i'" % (self._docker_cmd, self._play_context.become_user)) + _tmp_args['rsync_opts'].append("--rsh=%s exec -u %s -i" % (self._docker_cmd, self._play_context.become_user)) elif user is not None: - _tmp_args['rsync_opts'].append("--rsh='%s exec -u %s -i'" % (self._docker_cmd, user)) + _tmp_args['rsync_opts'].append("--rsh=%s exec -u %s -i" % (self._docker_cmd, user)) else: - _tmp_args['rsync_opts'].append("--rsh='%s exec -i'" % self._docker_cmd) + _tmp_args['rsync_opts'].append("--rsh=%s exec -i" % self._docker_cmd) # run the module and store the result result.update(self._execute_module('synchronize', module_args=_tmp_args, task_vars=task_vars))