mirror of
https://github.com/ansible-collections/community.general.git
synced 2025-06-28 11:10:21 -07:00
Merge branch 'combine_vars' of git://github.com/laggyluke/ansible into exp
Conflicts: lib/ansible/inventory/vars_plugins/group_vars.py lib/ansible/runner/__init__.py lib/ansible/utils/__init__.py test/TestPlayBook.py
This commit is contained in:
commit
7c6341718e
9 changed files with 111 additions and 30 deletions
|
@ -51,12 +51,7 @@ class VarsModule(object):
|
||||||
data = utils.parse_yaml_from_file(path)
|
data = utils.parse_yaml_from_file(path)
|
||||||
if type(data) != dict:
|
if type(data) != dict:
|
||||||
raise errors.AnsibleError("%s must be stored as a dictionary/hash" % path)
|
raise errors.AnsibleError("%s must be stored as a dictionary/hash" % path)
|
||||||
if C.DEFAULT_HASH_BEHAVIOUR == "merge":
|
results = utils.combine_vars(results, data);
|
||||||
# let data content override results if needed
|
|
||||||
results = utils.merge_hash(results, data)
|
|
||||||
else:
|
|
||||||
results.update(data)
|
|
||||||
break
|
|
||||||
|
|
||||||
# load vars in inventory_dir/hosts_vars/name_of_host
|
# load vars in inventory_dir/hosts_vars/name_of_host
|
||||||
p = os.path.join(basedir, "host_vars/%s" % host.name)
|
p = os.path.join(basedir, "host_vars/%s" % host.name)
|
||||||
|
@ -66,11 +61,7 @@ class VarsModule(object):
|
||||||
data = utils.parse_yaml_from_file(path)
|
data = utils.parse_yaml_from_file(path)
|
||||||
if type(data) != dict:
|
if type(data) != dict:
|
||||||
raise errors.AnsibleError("%s must be stored as a dictionary/hash" % path)
|
raise errors.AnsibleError("%s must be stored as a dictionary/hash" % path)
|
||||||
if C.DEFAULT_HASH_BEHAVIOUR == "merge":
|
results = utils.combine_vars(results, data);
|
||||||
# let data content override results if needed
|
|
||||||
results = utils.merge_hash(results, data)
|
|
||||||
else:
|
|
||||||
results.update(data)
|
|
||||||
break
|
|
||||||
return results
|
return results
|
||||||
|
|
||||||
|
|
|
@ -438,8 +438,9 @@ class Play(object):
|
||||||
if host is not None and self._has_vars_in(filename2) and not self._has_vars_in(filename3):
|
if host is not None and self._has_vars_in(filename2) and not self._has_vars_in(filename3):
|
||||||
# running a host specific pass and has host specific variables
|
# running a host specific pass and has host specific variables
|
||||||
# load into setup cache
|
# load into setup cache
|
||||||
self.playbook.SETUP_CACHE[host].update(new_vars)
|
self.playbook.SETUP_CACHE[host] = utils.combine_vars(
|
||||||
|
self.playbook.SETUP_CACHE[host], new_vars)
|
||||||
self.playbook.callbacks.on_import_for_host(host, filename4)
|
self.playbook.callbacks.on_import_for_host(host, filename4)
|
||||||
elif host is None:
|
elif host is None:
|
||||||
# running a non-host specific pass and we can update the global vars instead
|
# running a non-host specific pass and we can update the global vars instead
|
||||||
self.vars.update(new_vars)
|
self.vars = utils.combine_vars(self.vars, new_vars)
|
||||||
|
|
|
@ -182,7 +182,6 @@ class Runner(object):
|
||||||
|
|
||||||
# ensure we are using unique tmp paths
|
# ensure we are using unique tmp paths
|
||||||
random.seed()
|
random.seed()
|
||||||
|
|
||||||
# *****************************************************
|
# *****************************************************
|
||||||
|
|
||||||
def _complex_args_hack(self, complex_args, module_args):
|
def _complex_args_hack(self, complex_args, module_args):
|
||||||
|
@ -350,11 +349,10 @@ class Runner(object):
|
||||||
port = self.remote_port
|
port = self.remote_port
|
||||||
|
|
||||||
inject = {}
|
inject = {}
|
||||||
inject.update(host_variables)
|
inject = utils.combine_vars(inject, host_variables)
|
||||||
inject.update(self.module_vars)
|
inject = utils.combine_vars(inject, self.module_vars)
|
||||||
inject.update(self.setup_cache[host])
|
inject = utils.combine_vars(inject, self.setup_cache[host])
|
||||||
|
inject['hostvars'] = HostVars(self.setup_cache, self.inventory)
|
||||||
inject['hostvars'] = HostVars(self.setup_cache, self.inventory)
|
|
||||||
inject['group_names'] = host_variables.get('group_names', [])
|
inject['group_names'] = host_variables.get('group_names', [])
|
||||||
inject['groups'] = self.inventory.groups_list()
|
inject['groups'] = self.inventory.groups_list()
|
||||||
inject['vars'] = self.module_vars
|
inject['vars'] = self.module_vars
|
||||||
|
@ -522,7 +520,7 @@ class Runner(object):
|
||||||
# all modules get a tempdir, action plugins get one unless they have NEEDS_TMPPATH set to False
|
# all modules get a tempdir, action plugins get one unless they have NEEDS_TMPPATH set to False
|
||||||
if getattr(handler, 'NEEDS_TMPPATH', True):
|
if getattr(handler, 'NEEDS_TMPPATH', True):
|
||||||
tmp = self._make_tmp_path(conn)
|
tmp = self._make_tmp_path(conn)
|
||||||
|
|
||||||
result = handler.run(conn, tmp, module_name, module_args, inject, complex_args)
|
result = handler.run(conn, tmp, module_name, module_args, inject, complex_args)
|
||||||
|
|
||||||
conn.close()
|
conn.close()
|
||||||
|
@ -655,8 +653,8 @@ class Runner(object):
|
||||||
module_data = f.read()
|
module_data = f.read()
|
||||||
if module_common.REPLACER in module_data:
|
if module_common.REPLACER in module_data:
|
||||||
is_new_style=True
|
is_new_style=True
|
||||||
|
|
||||||
complex_args_json = utils.jsonify(complex_args)
|
complex_args_json = utils.jsonify(complex_args)
|
||||||
encoded_args = "\"\"\"%s\"\"\"" % module_args.replace("\"","\\\"")
|
encoded_args = "\"\"\"%s\"\"\"" % module_args.replace("\"","\\\"")
|
||||||
encoded_lang = "\"\"\"%s\"\"\"" % C.DEFAULT_MODULE_LANG
|
encoded_lang = "\"\"\"%s\"\"\"" % C.DEFAULT_MODULE_LANG
|
||||||
encoded_complex = "\"\"\"%s\"\"\"" % complex_args_json.replace("\\", "\\\\")
|
encoded_complex = "\"\"\"%s\"\"\"" % complex_args_json.replace("\\", "\\\\")
|
||||||
|
@ -665,7 +663,7 @@ class Runner(object):
|
||||||
module_data = module_data.replace(module_common.REPLACER_ARGS, encoded_args)
|
module_data = module_data.replace(module_common.REPLACER_ARGS, encoded_args)
|
||||||
module_data = module_data.replace(module_common.REPLACER_LANG, encoded_lang)
|
module_data = module_data.replace(module_common.REPLACER_LANG, encoded_lang)
|
||||||
module_data = module_data.replace(module_common.REPLACER_COMPLEX, encoded_complex)
|
module_data = module_data.replace(module_common.REPLACER_COMPLEX, encoded_complex)
|
||||||
|
|
||||||
if is_new_style:
|
if is_new_style:
|
||||||
facility = C.DEFAULT_SYSLOG_FACILITY
|
facility = C.DEFAULT_SYSLOG_FACILITY
|
||||||
if 'ansible_syslog_facility' in inject:
|
if 'ansible_syslog_facility' in inject:
|
||||||
|
@ -767,7 +765,7 @@ class Runner(object):
|
||||||
# run once per hostgroup, rather than pausing once per each
|
# run once per hostgroup, rather than pausing once per each
|
||||||
# host.
|
# host.
|
||||||
p = utils.plugins.action_loader.get(self.module_name, self)
|
p = utils.plugins.action_loader.get(self.module_name, self)
|
||||||
|
|
||||||
if p and getattr(p, 'BYPASS_HOST_LOOP', None):
|
if p and getattr(p, 'BYPASS_HOST_LOOP', None):
|
||||||
|
|
||||||
# Expose the current hostgroup to the bypassing plugins
|
# Expose the current hostgroup to the bypassing plugins
|
||||||
|
|
|
@ -318,7 +318,7 @@ def merge_hash(a, b):
|
||||||
for k, v in b.iteritems():
|
for k, v in b.iteritems():
|
||||||
if k in a and isinstance(a[k], dict):
|
if k in a and isinstance(a[k], dict):
|
||||||
# if this key is a hash and exists in a
|
# if this key is a hash and exists in a
|
||||||
# we recursively call ourselves with
|
# we recursively call ourselves with
|
||||||
# the key value of b
|
# the key value of b
|
||||||
a[k] = merge_hash(a[k], v)
|
a[k] = merge_hash(a[k], v)
|
||||||
else:
|
else:
|
||||||
|
@ -743,9 +743,10 @@ def listify_lookup_plugin_terms(terms, basedir, inject):
|
||||||
|
|
||||||
return terms
|
return terms
|
||||||
|
|
||||||
|
def combine_vars(a, b):
|
||||||
|
if C.DEFAULT_HASH_BEHAVIOUR == "merge":
|
||||||
|
return merge_hash(a, b)
|
||||||
|
else:
|
||||||
|
return dict(a.items() + b.items())
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -10,6 +10,7 @@ import ansible.utils as utils
|
||||||
import ansible.callbacks as ans_callbacks
|
import ansible.callbacks as ans_callbacks
|
||||||
import os
|
import os
|
||||||
import shutil
|
import shutil
|
||||||
|
import ansible.constants as C
|
||||||
|
|
||||||
EVENTS = []
|
EVENTS = []
|
||||||
|
|
||||||
|
@ -93,6 +94,8 @@ class TestPlaybook(unittest.TestCase):
|
||||||
os.unlink('/tmp/ansible_test_data_copy.out')
|
os.unlink('/tmp/ansible_test_data_copy.out')
|
||||||
if os.path.exists('/tmp/ansible_test_data_template.out'):
|
if os.path.exists('/tmp/ansible_test_data_template.out'):
|
||||||
os.unlink('/tmp/ansible_test_data_template.out')
|
os.unlink('/tmp/ansible_test_data_template.out')
|
||||||
|
if os.path.exists('/tmp/ansible_test_messages.out'):
|
||||||
|
os.unlink('/tmp/ansible_test_messages.out')
|
||||||
|
|
||||||
def _prepare_stage_dir(self):
|
def _prepare_stage_dir(self):
|
||||||
stage_path = os.path.join(self.test_dir, 'test_data')
|
stage_path = os.path.join(self.test_dir, 'test_data')
|
||||||
|
@ -285,3 +288,70 @@ class TestPlaybook(unittest.TestCase):
|
||||||
print utils.jsonify(expected, format=True)
|
print utils.jsonify(expected, format=True)
|
||||||
|
|
||||||
assert utils.jsonify(expected, format=True) == utils.jsonify(actual,format=True)
|
assert utils.jsonify(expected, format=True) == utils.jsonify(actual,format=True)
|
||||||
|
|
||||||
|
def test_playbook_hash_replace(self):
|
||||||
|
# save default hash behavior so we can restore it in the end of the test
|
||||||
|
saved_hash_behavior = C.DEFAULT_HASH_BEHAVIOUR
|
||||||
|
C.DEFAULT_HASH_BEHAVIOUR = "replace"
|
||||||
|
|
||||||
|
test_callbacks = TestCallbacks()
|
||||||
|
playbook = ansible.playbook.PlayBook(
|
||||||
|
playbook=os.path.join(self.test_dir, 'test_hash_behavior', 'playbook.yml'),
|
||||||
|
host_list='test/ansible_hosts',
|
||||||
|
stats=ans_callbacks.AggregateStats(),
|
||||||
|
callbacks=test_callbacks,
|
||||||
|
runner_callbacks=test_callbacks
|
||||||
|
)
|
||||||
|
playbook.run()
|
||||||
|
|
||||||
|
with open('/tmp/ansible_test_messages.out') as f:
|
||||||
|
actual = [l.strip() for l in f.readlines()]
|
||||||
|
|
||||||
|
print "**ACTUAL**"
|
||||||
|
print actual
|
||||||
|
|
||||||
|
expected = [
|
||||||
|
"goodbye: Goodbye World!"
|
||||||
|
]
|
||||||
|
|
||||||
|
print "**EXPECTED**"
|
||||||
|
print expected
|
||||||
|
|
||||||
|
assert actual == expected
|
||||||
|
|
||||||
|
# restore default hash behavior
|
||||||
|
C.DEFAULT_HASH_BEHAVIOUR = saved_hash_behavior
|
||||||
|
|
||||||
|
def test_playbook_hash_merge(self):
|
||||||
|
# save default hash behavior so we can restore it in the end of the test
|
||||||
|
saved_hash_behavior = C.DEFAULT_HASH_BEHAVIOUR
|
||||||
|
C.DEFAULT_HASH_BEHAVIOUR = "merge"
|
||||||
|
|
||||||
|
test_callbacks = TestCallbacks()
|
||||||
|
playbook = ansible.playbook.PlayBook(
|
||||||
|
playbook=os.path.join(self.test_dir, 'test_hash_behavior', 'playbook.yml'),
|
||||||
|
host_list='test/ansible_hosts',
|
||||||
|
stats=ans_callbacks.AggregateStats(),
|
||||||
|
callbacks=test_callbacks,
|
||||||
|
runner_callbacks=test_callbacks
|
||||||
|
)
|
||||||
|
playbook.run()
|
||||||
|
|
||||||
|
with open('/tmp/ansible_test_messages.out') as f:
|
||||||
|
actual = [l.strip() for l in f.readlines()]
|
||||||
|
|
||||||
|
print "**ACTUAL**"
|
||||||
|
print actual
|
||||||
|
|
||||||
|
expected = [
|
||||||
|
"hello: Hello World!",
|
||||||
|
"goodbye: Goodbye World!"
|
||||||
|
]
|
||||||
|
|
||||||
|
print "**EXPECTED**"
|
||||||
|
print expected
|
||||||
|
|
||||||
|
assert actual == expected
|
||||||
|
|
||||||
|
# restore default hash behavior
|
||||||
|
C.DEFAULT_HASH_BEHAVIOUR = saved_hash_behavior
|
||||||
|
|
3
test/test_hash_behavior/goodbye.yml
Normal file
3
test/test_hash_behavior/goodbye.yml
Normal file
|
@ -0,0 +1,3 @@
|
||||||
|
---
|
||||||
|
messages:
|
||||||
|
goodbye: "Goodbye World!"
|
3
test/test_hash_behavior/hello.yml
Normal file
3
test/test_hash_behavior/hello.yml
Normal file
|
@ -0,0 +1,3 @@
|
||||||
|
---
|
||||||
|
messages:
|
||||||
|
hello: "Hello World!"
|
3
test/test_hash_behavior/message.j2
Normal file
3
test/test_hash_behavior/message.j2
Normal file
|
@ -0,0 +1,3 @@
|
||||||
|
{% for k, v in messages.iteritems() %}
|
||||||
|
{{ k }}: {{ v }}
|
||||||
|
{% endfor %}
|
11
test/test_hash_behavior/playbook.yml
Normal file
11
test/test_hash_behavior/playbook.yml
Normal file
|
@ -0,0 +1,11 @@
|
||||||
|
---
|
||||||
|
- hosts: all
|
||||||
|
connection: local
|
||||||
|
|
||||||
|
vars_files:
|
||||||
|
- hello.yml
|
||||||
|
- goodbye.yml
|
||||||
|
|
||||||
|
tasks:
|
||||||
|
- name: generate messages
|
||||||
|
action: template src=message.j2 dest=/tmp/ansible_test_messages.out
|
Loading…
Add table
Add a link
Reference in a new issue