Allow loading dirs from role defaults/vars (#36357)
This commit moves code to look for vars files/dirs to a common place and uses it for loading role defaults/vars. This allows things such as 'defaults/main' or 'vars/main' being a directory in a role, allowing splitting of defaults/vars into multiple files. This commit also fixes the role loading unit tests for py3 when bytestrings are used for paths instead of utf8 strings. This fixes #14248 and #11639.
This commit is contained in:
parent
cc250156c4
commit
95ce00ff00
5 changed files with 151 additions and 84 deletions
|
@ -12,6 +12,7 @@ import os.path
|
||||||
import re
|
import re
|
||||||
import tempfile
|
import tempfile
|
||||||
|
|
||||||
|
from ansible import constants as C
|
||||||
from ansible.errors import AnsibleFileNotFound, AnsibleParserError
|
from ansible.errors import AnsibleFileNotFound, AnsibleParserError
|
||||||
from ansible.module_utils.basic import is_executable
|
from ansible.module_utils.basic import is_executable
|
||||||
from ansible.module_utils.six import binary_type, text_type
|
from ansible.module_utils.six import binary_type, text_type
|
||||||
|
@ -393,3 +394,53 @@ class DataLoader:
|
||||||
self.cleanup_tmp_file(f)
|
self.cleanup_tmp_file(f)
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
display.warning("Unable to cleanup temp files: %s" % to_native(e))
|
display.warning("Unable to cleanup temp files: %s" % to_native(e))
|
||||||
|
|
||||||
|
def find_vars_files(self, path, name, extensions=None, allow_dir=True):
|
||||||
|
"""
|
||||||
|
Find vars files in a given path with specified name. This will find
|
||||||
|
files in a dir named <name>/ or a file called <name> ending in known
|
||||||
|
extensions.
|
||||||
|
"""
|
||||||
|
|
||||||
|
b_path = to_bytes(os.path.join(path, name))
|
||||||
|
found = []
|
||||||
|
|
||||||
|
if extensions is None:
|
||||||
|
# Look for file with no extension first to find dir before file
|
||||||
|
extensions = [''] + C.YAML_FILENAME_EXTENSIONS
|
||||||
|
# add valid extensions to name
|
||||||
|
for ext in extensions:
|
||||||
|
|
||||||
|
if '.' in ext:
|
||||||
|
full_path = b_path + to_bytes(ext)
|
||||||
|
elif ext:
|
||||||
|
full_path = b'.'.join([b_path, to_bytes(ext)])
|
||||||
|
else:
|
||||||
|
full_path = b_path
|
||||||
|
|
||||||
|
if self.path_exists(full_path):
|
||||||
|
if self.is_directory(full_path):
|
||||||
|
if allow_dir:
|
||||||
|
found.extend(self._get_dir_vars_files(to_text(full_path), extensions))
|
||||||
|
else:
|
||||||
|
next
|
||||||
|
else:
|
||||||
|
found.append(full_path)
|
||||||
|
break
|
||||||
|
return found
|
||||||
|
|
||||||
|
def _get_dir_vars_files(self, path, extensions):
|
||||||
|
found = []
|
||||||
|
for spath in sorted(self.list_directory(path)):
|
||||||
|
if not spath.startswith(u'.') and not spath.endswith(u'~'): # skip hidden and backups
|
||||||
|
|
||||||
|
ext = os.path.splitext(spath)[-1]
|
||||||
|
full_spath = os.path.join(path, spath)
|
||||||
|
|
||||||
|
if self.is_directory(full_spath) and not ext: # recursive search if dir
|
||||||
|
found.extend(self._get_dir_vars_files(full_spath, extensions))
|
||||||
|
elif self.is_file(full_spath) and (not ext or to_text(ext) in extensions):
|
||||||
|
# only consider files with valid extensions or no extension
|
||||||
|
found.append(full_spath)
|
||||||
|
|
||||||
|
return found
|
||||||
|
|
|
@ -223,58 +223,46 @@ class Role(Base, Become, Conditional, Taggable):
|
||||||
obj=handler_data, orig_exc=e)
|
obj=handler_data, orig_exc=e)
|
||||||
|
|
||||||
# vars and default vars are regular dictionaries
|
# vars and default vars are regular dictionaries
|
||||||
self._role_vars = self._load_role_yaml('vars', main=self._from_files.get('vars'))
|
self._role_vars = self._load_role_yaml('vars', main=self._from_files.get('vars'), allow_dir=True)
|
||||||
if self._role_vars is None:
|
if self._role_vars is None:
|
||||||
self._role_vars = dict()
|
self._role_vars = dict()
|
||||||
elif not isinstance(self._role_vars, dict):
|
elif not isinstance(self._role_vars, dict):
|
||||||
raise AnsibleParserError("The vars/main.yml file for role '%s' must contain a dictionary of variables" % self._role_name)
|
raise AnsibleParserError("The vars/main.yml file for role '%s' must contain a dictionary of variables" % self._role_name)
|
||||||
|
|
||||||
self._default_vars = self._load_role_yaml('defaults', main=self._from_files.get('defaults'))
|
self._default_vars = self._load_role_yaml('defaults', main=self._from_files.get('defaults'), allow_dir=True)
|
||||||
if self._default_vars is None:
|
if self._default_vars is None:
|
||||||
self._default_vars = dict()
|
self._default_vars = dict()
|
||||||
elif not isinstance(self._default_vars, dict):
|
elif not isinstance(self._default_vars, dict):
|
||||||
raise AnsibleParserError("The defaults/main.yml file for role '%s' must contain a dictionary of variables" % self._role_name)
|
raise AnsibleParserError("The defaults/main.yml file for role '%s' must contain a dictionary of variables" % self._role_name)
|
||||||
|
|
||||||
def _load_role_yaml(self, subdir, main=None):
|
def _load_role_yaml(self, subdir, main=None, allow_dir=False):
|
||||||
file_path = os.path.join(self._role_path, subdir)
|
file_path = os.path.join(self._role_path, subdir)
|
||||||
if self._loader.path_exists(file_path) and self._loader.is_directory(file_path):
|
if self._loader.path_exists(file_path) and self._loader.is_directory(file_path):
|
||||||
main_file = self._resolve_main(file_path, main)
|
# Valid extensions and ordering for roles is hard-coded to maintain
|
||||||
if self._loader.path_exists(main_file):
|
# role portability
|
||||||
return self._loader.load_from_file(main_file)
|
extensions = ['.yml', '.yaml', '.json']
|
||||||
|
# If no <main> is specified by the user, look for files with
|
||||||
|
# extensions before bare name. Otherwise, look for bare name first.
|
||||||
|
if main is None:
|
||||||
|
_main = 'main'
|
||||||
|
extensions.append('')
|
||||||
|
else:
|
||||||
|
_main = main
|
||||||
|
extensions.insert(0, '')
|
||||||
|
found_files = self._loader.find_vars_files(file_path, _main, extensions, allow_dir)
|
||||||
|
if found_files:
|
||||||
|
data = {}
|
||||||
|
for found in found_files:
|
||||||
|
new_data = self._loader.load_from_file(found)
|
||||||
|
if new_data and allow_dir:
|
||||||
|
data = combine_vars(data, new_data)
|
||||||
|
else:
|
||||||
|
data = new_data
|
||||||
|
return data
|
||||||
elif main is not None:
|
elif main is not None:
|
||||||
raise AnsibleParserError("Could not find specified file in role: %s/%s" % (subdir, main))
|
raise AnsibleParserError("Could not find specified file in role: %s/%s" % (subdir, main))
|
||||||
return None
|
return None
|
||||||
|
|
||||||
def _resolve_main(self, basepath, main=None):
|
|
||||||
''' flexibly handle variations in main filenames '''
|
|
||||||
|
|
||||||
post = False
|
|
||||||
# allow override if set, otherwise use default
|
|
||||||
if main is None:
|
|
||||||
main = 'main'
|
|
||||||
post = True
|
|
||||||
|
|
||||||
bare_main = os.path.join(basepath, main)
|
|
||||||
|
|
||||||
possible_mains = (
|
|
||||||
os.path.join(basepath, '%s.yml' % main),
|
|
||||||
os.path.join(basepath, '%s.yaml' % main),
|
|
||||||
os.path.join(basepath, '%s.json' % main),
|
|
||||||
)
|
|
||||||
|
|
||||||
if post:
|
|
||||||
possible_mains = possible_mains + (bare_main,)
|
|
||||||
else:
|
|
||||||
possible_mains = (bare_main,) + possible_mains
|
|
||||||
|
|
||||||
if sum([self._loader.is_file(x) for x in possible_mains]) > 1:
|
|
||||||
raise AnsibleError("found multiple main files at %s, only one allowed" % (basepath))
|
|
||||||
else:
|
|
||||||
for m in possible_mains:
|
|
||||||
if self._loader.is_file(m):
|
|
||||||
return m # exactly one main file
|
|
||||||
return possible_mains[0] # zero mains (we still need to return something)
|
|
||||||
|
|
||||||
def _load_dependencies(self):
|
def _load_dependencies(self):
|
||||||
'''
|
'''
|
||||||
Recursively loads role dependencies from the metadata list of
|
Recursively loads role dependencies from the metadata list of
|
||||||
|
|
|
@ -89,7 +89,7 @@ class VarsModule(BaseVarsPlugin):
|
||||||
if os.path.exists(b_opath):
|
if os.path.exists(b_opath):
|
||||||
if os.path.isdir(b_opath):
|
if os.path.isdir(b_opath):
|
||||||
self._display.debug("\tprocessing dir %s" % opath)
|
self._display.debug("\tprocessing dir %s" % opath)
|
||||||
found_files = self._find_vars_files(opath, entity.name)
|
found_files = loader.find_vars_files(opath, entity.name)
|
||||||
FOUND[key] = found_files
|
FOUND[key] = found_files
|
||||||
else:
|
else:
|
||||||
self._display.warning("Found %s that is not a directory, skipping: %s" % (subdir, opath))
|
self._display.warning("Found %s that is not a directory, skipping: %s" % (subdir, opath))
|
||||||
|
@ -102,48 +102,3 @@ class VarsModule(BaseVarsPlugin):
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
raise AnsibleParserError(to_native(e))
|
raise AnsibleParserError(to_native(e))
|
||||||
return data
|
return data
|
||||||
|
|
||||||
def _find_vars_files(self, path, name):
|
|
||||||
""" Find {group,host}_vars files """
|
|
||||||
|
|
||||||
b_path = to_bytes(os.path.join(path, name))
|
|
||||||
found = []
|
|
||||||
|
|
||||||
# first look for w/o extensions
|
|
||||||
if os.path.exists(b_path):
|
|
||||||
if os.path.isdir(b_path):
|
|
||||||
found.extend(self._get_dir_files(to_text(b_path)))
|
|
||||||
else:
|
|
||||||
found.append(b_path)
|
|
||||||
else:
|
|
||||||
# add valid extensions to name
|
|
||||||
for ext in C.YAML_FILENAME_EXTENSIONS:
|
|
||||||
|
|
||||||
if '.' in ext:
|
|
||||||
full_path = b_path + to_bytes(ext)
|
|
||||||
elif ext:
|
|
||||||
full_path = b'.'.join([b_path, to_bytes(ext)])
|
|
||||||
else:
|
|
||||||
full_path = b_path
|
|
||||||
|
|
||||||
if os.path.exists(full_path) and os.path.isfile(full_path):
|
|
||||||
found.append(full_path)
|
|
||||||
break
|
|
||||||
return found
|
|
||||||
|
|
||||||
def _get_dir_files(self, path):
|
|
||||||
|
|
||||||
found = []
|
|
||||||
for spath in sorted(os.listdir(path)):
|
|
||||||
if not spath.startswith(u'.') and not spath.endswith(u'~'): # skip hidden and backups
|
|
||||||
|
|
||||||
ext = os.path.splitext(spath)[-1]
|
|
||||||
full_spath = os.path.join(path, spath)
|
|
||||||
|
|
||||||
if os.path.isdir(full_spath) and not ext: # recursive search if dir
|
|
||||||
found.extend(self._get_dir_files(full_spath))
|
|
||||||
elif os.path.isfile(full_spath) and (not ext or to_text(ext) in C.YAML_FILENAME_EXTENSIONS):
|
|
||||||
# only consider files with valid extensions or no extension
|
|
||||||
found.append(full_spath)
|
|
||||||
|
|
||||||
return found
|
|
||||||
|
|
|
@ -23,7 +23,7 @@ import os
|
||||||
|
|
||||||
from ansible.errors import AnsibleParserError
|
from ansible.errors import AnsibleParserError
|
||||||
from ansible.parsing.dataloader import DataLoader
|
from ansible.parsing.dataloader import DataLoader
|
||||||
from ansible.module_utils._text import to_bytes
|
from ansible.module_utils._text import to_bytes, to_text
|
||||||
|
|
||||||
|
|
||||||
class DictDataLoader(DataLoader):
|
class DictDataLoader(DataLoader):
|
||||||
|
@ -39,6 +39,7 @@ class DictDataLoader(DataLoader):
|
||||||
self._vault_secrets = None
|
self._vault_secrets = None
|
||||||
|
|
||||||
def load_from_file(self, path, unsafe=False):
|
def load_from_file(self, path, unsafe=False):
|
||||||
|
path = to_text(path)
|
||||||
if path in self._file_mapping:
|
if path in self._file_mapping:
|
||||||
return self.load(self._file_mapping[path], path)
|
return self.load(self._file_mapping[path], path)
|
||||||
return None
|
return None
|
||||||
|
@ -46,22 +47,32 @@ class DictDataLoader(DataLoader):
|
||||||
# TODO: the real _get_file_contents returns a bytestring, so we actually convert the
|
# TODO: the real _get_file_contents returns a bytestring, so we actually convert the
|
||||||
# unicode/text it's created with to utf-8
|
# unicode/text it's created with to utf-8
|
||||||
def _get_file_contents(self, path):
|
def _get_file_contents(self, path):
|
||||||
|
path = to_text(path)
|
||||||
if path in self._file_mapping:
|
if path in self._file_mapping:
|
||||||
return (to_bytes(self._file_mapping[path]), False)
|
return (to_bytes(self._file_mapping[path]), False)
|
||||||
else:
|
else:
|
||||||
raise AnsibleParserError("file not found: %s" % path)
|
raise AnsibleParserError("file not found: %s" % path)
|
||||||
|
|
||||||
def path_exists(self, path):
|
def path_exists(self, path):
|
||||||
|
path = to_text(path)
|
||||||
return path in self._file_mapping or path in self._known_directories
|
return path in self._file_mapping or path in self._known_directories
|
||||||
|
|
||||||
def is_file(self, path):
|
def is_file(self, path):
|
||||||
|
path = to_text(path)
|
||||||
return path in self._file_mapping
|
return path in self._file_mapping
|
||||||
|
|
||||||
def is_directory(self, path):
|
def is_directory(self, path):
|
||||||
|
path = to_text(path)
|
||||||
return path in self._known_directories
|
return path in self._known_directories
|
||||||
|
|
||||||
def list_directory(self, path):
|
def list_directory(self, path):
|
||||||
return [x for x in self._known_directories]
|
ret = []
|
||||||
|
path = to_text(path)
|
||||||
|
for x in (list(self._file_mapping.keys()) + self._known_directories):
|
||||||
|
if x.startswith(path):
|
||||||
|
if os.path.dirname(x) == path:
|
||||||
|
ret.append(os.path.basename(x))
|
||||||
|
return ret
|
||||||
|
|
||||||
def is_executable(self, path):
|
def is_executable(self, path):
|
||||||
# FIXME: figure out a way to make paths return true for this
|
# FIXME: figure out a way to make paths return true for this
|
||||||
|
|
|
@ -187,6 +187,68 @@ class TestRole(unittest.TestCase):
|
||||||
self.assertEqual(r._default_vars, dict(foo='bar'))
|
self.assertEqual(r._default_vars, dict(foo='bar'))
|
||||||
self.assertEqual(r._role_vars, dict(foo='bam'))
|
self.assertEqual(r._role_vars, dict(foo='bam'))
|
||||||
|
|
||||||
|
@patch('ansible.playbook.role.definition.unfrackpath', mock_unfrackpath_noop)
|
||||||
|
def test_load_role_with_vars_dirs(self):
|
||||||
|
|
||||||
|
fake_loader = DictDataLoader({
|
||||||
|
"/etc/ansible/roles/foo_vars/defaults/main/foo.yml": """
|
||||||
|
foo: bar
|
||||||
|
""",
|
||||||
|
"/etc/ansible/roles/foo_vars/vars/main/bar.yml": """
|
||||||
|
foo: bam
|
||||||
|
""",
|
||||||
|
})
|
||||||
|
|
||||||
|
mock_play = MagicMock()
|
||||||
|
mock_play.ROLE_CACHE = {}
|
||||||
|
|
||||||
|
i = RoleInclude.load('foo_vars', play=mock_play, loader=fake_loader)
|
||||||
|
r = Role.load(i, play=mock_play)
|
||||||
|
|
||||||
|
self.assertEqual(r._default_vars, dict(foo='bar'))
|
||||||
|
self.assertEqual(r._role_vars, dict(foo='bam'))
|
||||||
|
|
||||||
|
@patch('ansible.playbook.role.definition.unfrackpath', mock_unfrackpath_noop)
|
||||||
|
def test_load_role_with_vars_nested_dirs(self):
|
||||||
|
|
||||||
|
fake_loader = DictDataLoader({
|
||||||
|
"/etc/ansible/roles/foo_vars/defaults/main/foo/bar.yml": """
|
||||||
|
foo: bar
|
||||||
|
""",
|
||||||
|
"/etc/ansible/roles/foo_vars/vars/main/bar/foo.yml": """
|
||||||
|
foo: bam
|
||||||
|
""",
|
||||||
|
})
|
||||||
|
|
||||||
|
mock_play = MagicMock()
|
||||||
|
mock_play.ROLE_CACHE = {}
|
||||||
|
|
||||||
|
i = RoleInclude.load('foo_vars', play=mock_play, loader=fake_loader)
|
||||||
|
r = Role.load(i, play=mock_play)
|
||||||
|
|
||||||
|
self.assertEqual(r._default_vars, dict(foo='bar'))
|
||||||
|
self.assertEqual(r._role_vars, dict(foo='bam'))
|
||||||
|
|
||||||
|
@patch('ansible.playbook.role.definition.unfrackpath', mock_unfrackpath_noop)
|
||||||
|
def test_load_role_with_vars_dir_vs_file(self):
|
||||||
|
|
||||||
|
fake_loader = DictDataLoader({
|
||||||
|
"/etc/ansible/roles/foo_vars/vars/main/foo.yml": """
|
||||||
|
foo: bar
|
||||||
|
""",
|
||||||
|
"/etc/ansible/roles/foo_vars/vars/main.yml": """
|
||||||
|
foo: bam
|
||||||
|
""",
|
||||||
|
})
|
||||||
|
|
||||||
|
mock_play = MagicMock()
|
||||||
|
mock_play.ROLE_CACHE = {}
|
||||||
|
|
||||||
|
i = RoleInclude.load('foo_vars', play=mock_play, loader=fake_loader)
|
||||||
|
r = Role.load(i, play=mock_play)
|
||||||
|
|
||||||
|
self.assertEqual(r._role_vars, dict(foo='bam'))
|
||||||
|
|
||||||
@patch('ansible.playbook.role.definition.unfrackpath', mock_unfrackpath_noop)
|
@patch('ansible.playbook.role.definition.unfrackpath', mock_unfrackpath_noop)
|
||||||
def test_load_role_with_metadata(self):
|
def test_load_role_with_metadata(self):
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue