From 3b0e64127dceb467b04005b3c2abc2b272a03548 Mon Sep 17 00:00:00 2001 From: James Cammarata Date: Tue, 28 Oct 2014 14:35:29 -0500 Subject: [PATCH] Refactoring role spec stuff into a dedicated parsing class Also reworking tests to cut down on the number of patches required by sub-classing the DataLoader() class and reworking the base object's structure a bit to allow its use --- v2/ansible/parsing/yaml/__init__.py | 11 +- v2/ansible/playbook/base.py | 20 +- v2/ansible/playbook/block.py | 6 +- v2/ansible/playbook/role.py | 399 ------------------------ v2/ansible/playbook/role/__init__.py | 205 ++++++++++++ v2/ansible/playbook/role/definition.py | 153 +++++++++ v2/ansible/playbook/role/include.py | 52 +++ v2/ansible/playbook/role/metadata.py | 91 ++++++ v2/ansible/playbook/role/requirement.py | 166 ++++++++++ v2/ansible/playbook/task.py | 10 +- v2/test/mock/__init__.py | 20 ++ v2/test/mock/loader.py | 80 +++++ v2/test/playbook/test_role.py | 265 +++++++--------- 13 files changed, 912 insertions(+), 566 deletions(-) delete mode 100644 v2/ansible/playbook/role.py create mode 100644 v2/ansible/playbook/role/__init__.py create mode 100644 v2/ansible/playbook/role/definition.py create mode 100644 v2/ansible/playbook/role/include.py create mode 100644 v2/ansible/playbook/role/metadata.py create mode 100644 v2/ansible/playbook/role/requirement.py create mode 100644 v2/test/mock/__init__.py create mode 100644 v2/test/mock/loader.py diff --git a/v2/ansible/parsing/yaml/__init__.py b/v2/ansible/parsing/yaml/__init__.py index c3822823985fa5..969fd2a3b555ea 100644 --- a/v2/ansible/parsing/yaml/__init__.py +++ b/v2/ansible/parsing/yaml/__init__.py @@ -91,6 +91,15 @@ def load_from_file(self, file_name): return parsed_data + def path_exists(self, path): + return os.path.exists(path) + + def is_directory(self, path): + return os.path.isdir(path) + + def is_file(self, path): + return os.path.isfile(path) + def _safe_load(self, stream): ''' Implements yaml.safe_load(), except using our custom loader class. ''' return load(stream, AnsibleLoader) @@ -100,7 +109,7 @@ def _get_file_contents(self, file_name): Reads the file contents from the given file name, and will decrypt them if they are found to be vault-encrypted. ''' - if not os.path.exists(file_name) or not os.path.isfile(file_name): + if not self.path_exists(file_name) or not self.is_file(file_name): raise AnsibleParserError("the file_name '%s' does not exist, or is not readable" % file_name) show_content = True diff --git a/v2/ansible/playbook/base.py b/v2/ansible/playbook/base.py index ce0e2a199c037d..e2b96c8cc25ce4 100644 --- a/v2/ansible/playbook/base.py +++ b/v2/ansible/playbook/base.py @@ -29,13 +29,11 @@ class Base: - _tags = FieldAttribute(isa='list') - _when = FieldAttribute(isa='list') + def __init__(self): - def __init__(self, loader=DataLoader): - - # the data loader class is used to parse data from strings and files - self._loader = loader() + # initialize the data loader, this will be provided later + # when the object is actually loaded + self._loader = None # each class knows attributes set upon it, see Task.py for example self._attributes = dict() @@ -61,11 +59,17 @@ def munge(self, ds): return ds - def load_data(self, ds): + def load_data(self, ds, loader=None): ''' walk the input datastructure and assign any values ''' assert ds is not None + # the data loader class is used to parse data from strings and files + if loader is not None: + self._loader = loader + else: + self._loader = DataLoader() + if isinstance(ds, string_types) or isinstance(ds, FileIO): ds = self._loader.load(ds) @@ -89,6 +93,8 @@ def load_data(self, ds): self.validate() return self + def get_loader(self): + return self._loader def validate(self): ''' validation that is done at parse time, not load time ''' diff --git a/v2/ansible/playbook/block.py b/v2/ansible/playbook/block.py index 5e4826d119da06..5f21cdaf606f4e 100644 --- a/v2/ansible/playbook/block.py +++ b/v2/ansible/playbook/block.py @@ -28,6 +28,8 @@ class Block(Base): _block = FieldAttribute(isa='list') _rescue = FieldAttribute(isa='list') _always = FieldAttribute(isa='list') + _tags = FieldAttribute(isa='list', default=[]) + _when = FieldAttribute(isa='list', default=[]) # for future consideration? this would be functionally # similar to the 'else' clause for exceptions @@ -43,9 +45,9 @@ def get_variables(self): return dict() @staticmethod - def load(data, role=None): + def load(data, role=None, loader=None): b = Block(role=role) - return b.load_data(data) + return b.load_data(data, loader=loader) def munge(self, ds): ''' diff --git a/v2/ansible/playbook/role.py b/v2/ansible/playbook/role.py deleted file mode 100644 index b4b7eed012a734..00000000000000 --- a/v2/ansible/playbook/role.py +++ /dev/null @@ -1,399 +0,0 @@ -# (c) 2012-2014, Michael DeHaan -# -# This file is part of Ansible -# -# Ansible is free software: you can redistribute it and/or modify -# it under the terms of the GNU General Public License as published by -# the Free Software Foundation, either version 3 of the License, or -# (at your option) any later version. -# -# Ansible is distributed in the hope that it will be useful, -# but WITHOUT ANY WARRANTY; without even the implied warranty of -# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -# GNU General Public License for more details. -# -# You should have received a copy of the GNU General Public License -# along with Ansible. If not, see . - -# Make coding more python3-ish -from __future__ import (absolute_import, division, print_function) -__metaclass__ = type - -from six import iteritems, string_types - -import os - -from hashlib import md5 - -from ansible.errors import AnsibleError, AnsibleParserError -from ansible.parsing.yaml import DataLoader -from ansible.playbook.attribute import FieldAttribute -from ansible.playbook.base import Base -from ansible.playbook.block import Block - -from ansible.parsing.yaml.objects import AnsibleBaseYAMLObject, AnsibleMapping - -__all__ = ['Role'] - -# The role cache is used to prevent re-loading roles, which -# may already exist. Keys into this cache are the MD5 hash -# of the role definition (for dictionary definitions, this -# will be based on the repr() of the dictionary object) -_ROLE_CACHE = dict() - -# The valid metadata keys for meta/main.yml files -_VALID_METADATA_KEYS = [ - 'dependencies', - 'allow_duplicates', - 'galaxy_info', -] - -class Role(Base): - - _role_name = FieldAttribute(isa='string') - _role_path = FieldAttribute(isa='string') - _src = FieldAttribute(isa='string') - _scm = FieldAttribute(isa='string') - _version = FieldAttribute(isa='string') - _task_blocks = FieldAttribute(isa='list', default=[]) - _handler_blocks = FieldAttribute(isa='list', default=[]) - _params = FieldAttribute(isa='dict', default=dict()) - _default_vars = FieldAttribute(isa='dict', default=dict()) - _role_vars = FieldAttribute(isa='dict', default=dict()) - - # Attributes based on values in metadata. These MUST line up - # with the values stored in _VALID_METADATA_KEYS - _dependencies = FieldAttribute(isa='list', default=[]) - _allow_duplicates = FieldAttribute(isa='bool', default=False) - _galaxy_info = FieldAttribute(isa='dict', default=dict()) - - def __init__(self, loader=DataLoader): - self._role_path = None - self._parents = [] - - super(Role, self).__init__(loader=loader) - - def __repr__(self): - return self.get_name() - - def get_name(self): - return self._attributes['role_name'] - - @staticmethod - def load(data, parent_role=None): - assert isinstance(data, string_types) or isinstance(data, dict) - - # Check to see if this role has been loaded already, based on the - # role definition, partially to save loading time and also to make - # sure that roles are run a single time unless specifically allowed - # to run more than once - - # FIXME: the tags and conditionals, if specified in the role def, - # should not figure into the resulting hash - cache_key = md5(repr(data)) - if cache_key in _ROLE_CACHE: - r = _ROLE_CACHE[cache_key] - else: - try: - # load the role - r = Role() - r.load_data(data) - # and cache it for next time - _ROLE_CACHE[cache_key] = r - except RuntimeError: - raise AnsibleError("A recursive loop was detected while loading your roles", obj=data) - - # now add the parent to the (new) role - if parent_role: - r.add_parent(parent_role) - - return r - - #------------------------------------------------------------------------------ - # munge, and other functions used for loading the ds - - def munge(self, ds): - # create the new ds as an AnsibleMapping, so we can preserve any line/column - # data from the parser, and copy that info from the old ds (if applicable) - new_ds = AnsibleMapping() - if isinstance(ds, AnsibleBaseYAMLObject): - new_ds.copy_position_info(ds) - - # Role definitions can be strings or dicts, so we fix things up here. - # Anything that is not a role name, tag, or conditional will also be - # added to the params sub-dictionary for loading later - if isinstance(ds, string_types): - new_ds['role_name'] = ds - else: - # munge the role ds here to correctly fill in the various fields which - # may be used to define the role, like: role, src, scm, etc. - ds = self._munge_role(ds) - - # now we split any random role params off from the role spec and store - # them in a dictionary of params for parsing later - params = dict() - attr_names = [attr_name for (attr_name, attr_value) in self._get_base_attributes().iteritems()] - for (key, value) in iteritems(ds): - if key not in attr_names and key != 'role': - # this key does not match a field attribute, so it must be a role param - params[key] = value - else: - # this is a field attribute, so copy it over directly - new_ds[key] = value - new_ds['params'] = params - - # Set the role name and path, based on the role definition - (role_name, role_path) = self._get_role_path(new_ds.get('role_name')) - new_ds['role_name'] = role_name - new_ds['role_path'] = role_path - - # load the role's files, if they exist - new_ds['task_blocks'] = self._load_role_yaml(role_path, 'tasks') - new_ds['handler_blocks'] = self._load_role_yaml(role_path, 'handlers') - new_ds['default_vars'] = self._load_role_yaml(role_path, 'defaults') - new_ds['role_vars'] = self._load_role_yaml(role_path, 'vars') - - # we treat metadata slightly differently: we instead pull out the - # valid metadata keys and munge them directly into new_ds - metadata_ds = self._munge_metadata(role_name, role_path) - new_ds.update(metadata_ds) - - # and return the newly munged ds - return new_ds - - def _load_role_yaml(self, role_path, subdir): - file_path = os.path.join(role_path, subdir) - if os.path.exists(file_path) and os.path.isdir(file_path): - main_file = self._resolve_main(file_path) - if os.path.exists(main_file): - return self._loader.load_from_file(main_file) - return None - - def _resolve_main(self, basepath): - ''' flexibly handle variations in main filenames ''' - possible_mains = ( - os.path.join(basepath, 'main'), - os.path.join(basepath, 'main.yml'), - os.path.join(basepath, 'main.yaml'), - os.path.join(basepath, 'main.json'), - ) - - if sum([os.path.isfile(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 os.path.isfile(m): - return m # exactly one main file - return possible_mains[0] # zero mains (we still need to return something) - - def _get_role_path(self, role): - ''' - the 'role', as specified in the ds (or as a bare string), can either - be a simple name or a full path. If it is a full path, we use the - basename as the role name, otherwise we take the name as-given and - append it to the default role path - ''' - - # FIXME: this should use unfrackpath once the utils code has been sorted out - role_path = os.path.normpath(role) - if os.path.exists(role_path): - role_name = os.path.basename(role) - return (role_name, role_path) - else: - for path in ('./roles', '/etc/ansible/roles'): - role_path = os.path.join(path, role) - if os.path.exists(role_path): - return (role, role_path) - - # FIXME: make the parser smart about list/string entries - # in the yaml so the error line/file can be reported - # here - raise AnsibleError("the role '%s' was not found" % role, obj=role) - - def _repo_url_to_role_name(self, repo_url): - # gets the role name out of a repo like - # http://git.example.com/repos/repo.git" => "repo" - - if '://' not in repo_url and '@' not in repo_url: - return repo_url - trailing_path = repo_url.split('/')[-1] - if trailing_path.endswith('.git'): - trailing_path = trailing_path[:-4] - if trailing_path.endswith('.tar.gz'): - trailing_path = trailing_path[:-7] - if ',' in trailing_path: - trailing_path = trailing_path.split(',')[0] - return trailing_path - - def _role_spec_parse(self, role_spec): - # takes a repo and a version like - # git+http://git.example.com/repos/repo.git,v1.0 - # and returns a list of properties such as: - # { - # 'scm': 'git', - # 'src': 'http://git.example.com/repos/repo.git', - # 'version': 'v1.0', - # 'name': 'repo' - # } - - default_role_versions = dict(git='master', hg='tip') - - role_spec = role_spec.strip() - role_version = '' - if role_spec == "" or role_spec.startswith("#"): - return (None, None, None, None) - - tokens = [s.strip() for s in role_spec.split(',')] - - # assume https://github.com URLs are git+https:// URLs and not - # tarballs unless they end in '.zip' - if 'github.com/' in tokens[0] and not tokens[0].startswith("git+") and not tokens[0].endswith('.tar.gz'): - tokens[0] = 'git+' + tokens[0] - - if '+' in tokens[0]: - (scm, role_url) = tokens[0].split('+') - else: - scm = None - role_url = tokens[0] - - if len(tokens) >= 2: - role_version = tokens[1] - - if len(tokens) == 3: - role_name = tokens[2] - else: - role_name = self._repo_url_to_role_name(tokens[0]) - - if scm and not role_version: - role_version = default_role_versions.get(scm, '') - - return dict(scm=scm, src=role_url, version=role_version, role_name=role_name) - - def _munge_role(self, ds): - if 'role' in ds: - # Old style: {role: "galaxy.role,version,name", other_vars: "here" } - role_info = self._role_spec_parse(ds['role']) - if isinstance(role_info, dict): - # Warning: Slight change in behaviour here. name may be being - # overloaded. Previously, name was only a parameter to the role. - # Now it is both a parameter to the role and the name that - # ansible-galaxy will install under on the local system. - if 'name' in ds and 'name' in role_info: - del role_info['name'] - ds.update(role_info) - else: - # New style: { src: 'galaxy.role,version,name', other_vars: "here" } - if 'github.com' in ds["src"] and 'http' in ds["src"] and '+' not in ds["src"] and not ds["src"].endswith('.tar.gz'): - ds["src"] = "git+" + ds["src"] - - if '+' in ds["src"]: - (scm, src) = ds["src"].split('+') - ds["scm"] = scm - ds["src"] = src - - if 'name' in role: - ds["role"] = ds["name"] - del ds["name"] - else: - ds["role"] = self._repo_url_to_role_name(ds["src"]) - - # set some values to a default value, if none were specified - ds.setdefault('version', '') - ds.setdefault('scm', None) - - return ds - - def _munge_metadata(self, role_name, role_path): - ''' - loads the metadata main.yml (if it exists) and creates a clean - datastructure we can merge into the newly munged ds - ''' - - meta_ds = dict() - - metadata = self._load_role_yaml(role_path, 'meta') - if metadata: - if not isinstance(metadata, dict): - raise AnsibleParserError("The metadata for role '%s' should be a dictionary, instead it is a %s" % (role_name, type(metadata)), obj=metadata) - - for key in metadata: - if key in _VALID_METADATA_KEYS: - if isinstance(metadata[key], dict): - meta_ds[key] = metadata[key].copy() - elif isinstance(metadata[key], list): - meta_ds[key] = metadata[key][:] - else: - meta_ds[key] = metadata[key] - else: - raise AnsibleParserError("%s is not a valid metadata key for role '%s'" % (key, role_name), obj=metadata) - - return meta_ds - - #------------------------------------------------------------------------------ - # attribute loading defs - - def _load_list_of_blocks(self, ds): - assert type(ds) == list - block_list = [] - for block in ds: - b = Block(block) - block_list.append(b) - return block_list - - def _load_task_blocks(self, attr, ds): - if ds is None: - return [] - return self._load_list_of_blocks(ds) - - def _load_handler_blocks(self, attr, ds): - if ds is None: - return [] - return self._load_list_of_blocks(ds) - - def _load_dependencies(self, attr, ds): - assert type(ds) in (list, type(None)) - - deps = [] - if ds: - for role_def in ds: - r = Role.load(role_def, parent_role=self) - deps.append(r) - return deps - - #------------------------------------------------------------------------------ - # other functions - - def add_parent(self, parent_role): - ''' adds a role to the list of this roles parents ''' - assert isinstance(parent_role, Role) - - if parent_role not in self._parents: - self._parents.append(parent_role) - - def get_parents(self): - return self._parents - - # FIXME: not yet used - #def get_variables(self): - # # returns the merged variables for this role, including - # # recursively merging those of all child roles - # return dict() - - def get_direct_dependencies(self): - return self._attributes['dependencies'][:] - - def get_all_dependencies(self): - # returns a list built recursively, of all deps from - # all child dependencies - - child_deps = [] - direct_deps = self.get_direct_dependencies() - - for dep in direct_deps: - dep_deps = dep.get_all_dependencies() - for dep_dep in dep_deps: - if dep_dep not in child_deps: - child_deps.append(dep_dep) - - return direct_deps + child_deps - diff --git a/v2/ansible/playbook/role/__init__.py b/v2/ansible/playbook/role/__init__.py new file mode 100644 index 00000000000000..ed7355f921421b --- /dev/null +++ b/v2/ansible/playbook/role/__init__.py @@ -0,0 +1,205 @@ +# (c) 2012-2014, Michael DeHaan +# +# This file is part of Ansible +# +# Ansible is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# Ansible is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with Ansible. If not, see . + +# Make coding more python3-ish +from __future__ import (absolute_import, division, print_function) +__metaclass__ = type + +from six import iteritems, string_types + +import os + +from hashlib import md5 +from types import NoneType + +from ansible.errors import AnsibleError, AnsibleParserError +from ansible.parsing.yaml import DataLoader +from ansible.playbook.attribute import FieldAttribute +from ansible.playbook.base import Base +from ansible.playbook.block import Block +from ansible.playbook.role.include import RoleInclude +from ansible.playbook.role.metadata import RoleMetadata + + +__all__ = ['Role', 'ROLE_CACHE'] + + +# The role cache is used to prevent re-loading roles, which +# may already exist. Keys into this cache are the MD5 hash +# of the role definition (for dictionary definitions, this +# will be based on the repr() of the dictionary object) +ROLE_CACHE = dict() + + +class Role: + + def __init__(self): + self._role_name = None + self._role_path = None + self._role_params = dict() + self._loader = None + + self._metadata = None + self._parents = [] + self._dependencies = [] + self._task_blocks = [] + self._handler_blocks = [] + self._default_vars = dict() + self._role_vars = dict() + + def __repr__(self): + return self.get_name() + + def get_name(self): + return self._role_name + + @staticmethod + def load(role_include, parent_role=None): + # FIXME: add back in the role caching support + try: + r = Role() + r._load_role_data(role_include, parent_role=parent_role) + except RuntimeError: + # FIXME: needs a better way to access the ds in the role include + raise AnsibleError("A recursion loop was detected with the roles specified. Make sure child roles do not have dependencies on parent roles", obj=role_include._ds) + return r + + def _load_role_data(self, role_include, parent_role=None): + self._role_name = role_include.role + self._role_path = role_include.get_role_path() + self._role_params = role_include.get_role_params() + self._loader = role_include.get_loader() + + if parent_role: + self.add_parent(parent_role) + + # load the role's files, if they exist + metadata = self._load_role_yaml('meta') + if metadata: + self._metadata = RoleMetadata.load(metadata, owner=self, loader=self._loader) + self._dependencies = self._load_dependencies() + + task_data = self._load_role_yaml('tasks') + if task_data: + self._task_blocks = self._load_list_of_blocks(task_data) + + handler_data = self._load_role_yaml('handlers') + if handler_data: + self._handler_blocks = self._load_list_of_blocks(handler_data) + + # vars and default vars are regular dictionaries + self._role_vars = self._load_role_yaml('vars') + if not isinstance(self._role_vars, (dict, NoneType)): + raise AnsibleParserError("The vars/main.yml file for role '%s' must contain a dictionary of variables" % self._role_name, obj=ds) + + self._default_vars = self._load_role_yaml('defaults') + if not isinstance(self._default_vars, (dict, NoneType)): + raise AnsibleParserError("The default/main.yml file for role '%s' must contain a dictionary of variables" % self._role_name, obj=ds) + + def _load_role_yaml(self, subdir): + file_path = os.path.join(self._role_path, subdir) + if self._loader.path_exists(file_path) and self._loader.is_directory(file_path): + main_file = self._resolve_main(file_path) + if self._loader.path_exists(main_file): + return self._loader.load_from_file(main_file) + return None + + def _resolve_main(self, basepath): + ''' flexibly handle variations in main filenames ''' + possible_mains = ( + os.path.join(basepath, 'main.yml'), + os.path.join(basepath, 'main.yaml'), + os.path.join(basepath, 'main.json'), + os.path.join(basepath, 'main'), + ) + + 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_list_of_blocks(self, ds): + ''' + Given a list of mixed task/block data (parsed from YAML), + return a list of Block() objects, where implicit blocks + are created for each bare Task. + ''' + + assert type(ds) in (list, NoneType) + + block_list = [] + if ds: + for block in ds: + b = Block(block) + block_list.append(b) + + return block_list + + def _load_dependencies(self): + ''' + Recursively loads role dependencies from the metadata list of + dependencies, if it exists + ''' + + deps = [] + if self._metadata: + for role_include in self._metadata.dependencies: + r = Role.load(role_include, parent_role=self) + deps.append(r) + + return deps + + #------------------------------------------------------------------------------ + # other functions + + def add_parent(self, parent_role): + ''' adds a role to the list of this roles parents ''' + assert isinstance(parent_role, Role) + + if parent_role not in self._parents: + self._parents.append(parent_role) + + def get_parents(self): + return self._parents + + # FIXME: not yet used + #def get_variables(self): + # # returns the merged variables for this role, including + # # recursively merging those of all child roles + # return dict() + + def get_direct_dependencies(self): + return self._dependencies[:] + + def get_all_dependencies(self): + # returns a list built recursively, of all deps from + # all child dependencies + + child_deps = [] + direct_deps = self.get_direct_dependencies() + + for dep in direct_deps: + dep_deps = dep.get_all_dependencies() + for dep_dep in dep_deps: + if dep_dep not in child_deps: + child_deps.append(dep_dep) + + return direct_deps + child_deps + diff --git a/v2/ansible/playbook/role/definition.py b/v2/ansible/playbook/role/definition.py new file mode 100644 index 00000000000000..08d62afbe4b739 --- /dev/null +++ b/v2/ansible/playbook/role/definition.py @@ -0,0 +1,153 @@ +# (c) 2014 Michael DeHaan, +# +# This file is part of Ansible +# +# Ansible is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# Ansible is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with Ansible. If not, see . + +# Make coding more python3-ish +from __future__ import (absolute_import, division, print_function) +__metaclass__ = type + +from six import iteritems, string_types + +import os + +from ansible.errors import AnsibleError +from ansible.parsing.yaml.objects import AnsibleBaseYAMLObject, AnsibleMapping +from ansible.playbook.attribute import Attribute, FieldAttribute +from ansible.playbook.base import Base + + +__all__ = ['RoleDefinition'] + + +class RoleDefinition(Base): + + _role = FieldAttribute(isa='string') + + def __init__(self): + self._role_path = None + self._role_params = dict() + super(RoleDefinition, self).__init__() + + def __repr__(self): + return 'ROLEDEF: ' + self._attributes.get('role', '') + + @staticmethod + def load(data, loader=None): + raise AnsibleError("not implemented") + + def munge(self, ds): + + assert isinstance(ds, dict) or isinstance(ds, string_types) + + # we create a new data structure here, using the same + # object used internally by the YAML parsing code so we + # can preserve file:line:column information if it exists + new_ds = AnsibleMapping() + if isinstance(ds, AnsibleBaseYAMLObject): + new_ds.copy_position_info(ds) + + # first we pull the role name out of the data structure, + # and then use that to determine the role path (which may + # result in a new role name, if it was a file path) + role_name = self._load_role_name(ds) + (role_name, role_path) = self._load_role_path(role_name) + + # next, we split the role params out from the valid role + # attributes and update the new datastructure with that + # result and the role name + if isinstance(ds, dict): + (new_role_def, role_params) = self._split_role_params(ds) + new_ds.update(new_role_def) + self._role_params = role_params + + # set the role name in the new ds + new_ds['role'] = role_name + + # we store the role path internally + self._role_path = role_path + + # save the original ds for use later + self._ds = ds + + # and return the cleaned-up data structure + return new_ds + + def _load_role_name(self, ds): + ''' + Returns the role name (either the role: or name: field) from + the role definition, or (when the role definition is a simple + string), just that string + ''' + + if isinstance(ds, string_types): + return ds + + role_name = ds.get('role', ds.get('name')) + if not role_name: + raise AnsibleError('role definitions must contain a role name', obj=ds) + + return role_name + + def _load_role_path(self, role_name): + ''' + the 'role', as specified in the ds (or as a bare string), can either + be a simple name or a full path. If it is a full path, we use the + basename as the role name, otherwise we take the name as-given and + append it to the default role path + ''' + + # FIXME: this should use unfrackpath once the utils code has been sorted out + role_path = os.path.normpath(role_name) + if self._loader.path_exists(role_path): + role_name = os.path.basename(role_name) + return (role_name, role_path) + else: + # FIXME: this should search in the configured roles path + for path in ('./roles', '/etc/ansible/roles'): + role_path = os.path.join(path, role_name) + if self._loader.path_exists(role_path): + return (role_name, role_path) + + # FIXME: make the parser smart about list/string entries + # in the yaml so the error line/file can be reported + # here + raise AnsibleError("the role '%s' was not found" % role_name) + + def _split_role_params(self, ds): + ''' + Splits any random role params off from the role spec and store + them in a dictionary of params for parsing later + ''' + + role_def = dict() + role_params = dict() + for (key, value) in iteritems(ds): + # use the list of FieldAttribute values to determine what is and is not + # an extra parameter for this role (or sub-class of this role) + if key not in [attr_name for (attr_name, attr_value) in self._get_base_attributes().iteritems()]: + # this key does not match a field attribute, so it must be a role param + role_params[key] = value + else: + # this is a field attribute, so copy it over directly + role_def[key] = value + + return (role_def, role_params) + + def get_role_params(self): + return self._role_params.copy() + + def get_role_path(self): + return self._role_path diff --git a/v2/ansible/playbook/role/include.py b/v2/ansible/playbook/role/include.py new file mode 100644 index 00000000000000..d36b0a93970c41 --- /dev/null +++ b/v2/ansible/playbook/role/include.py @@ -0,0 +1,52 @@ +# (c) 2014 Michael DeHaan, +# +# This file is part of Ansible +# +# Ansible is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# Ansible is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with Ansible. If not, see . + +# Make coding more python3-ish +from __future__ import (absolute_import, division, print_function) +__metaclass__ = type + +from six import iteritems, string_types + +import os + +from ansible.errors import AnsibleError, AnsibleParserError +from ansible.playbook.attribute import Attribute, FieldAttribute +from ansible.playbook.role.definition import RoleDefinition + + +__all__ = ['RoleInclude'] + + +class RoleInclude(RoleDefinition): + + """ + FIXME: docstring + """ + + _tags = FieldAttribute(isa='list', default=[]) + _when = FieldAttribute(isa='list', default=[]) + + def __init__(self): + super(RoleInclude, self).__init__() + + @staticmethod + def load(data, parent_role=None, loader=None): + assert isinstance(data, string_types) or isinstance(data, dict) + + ri = RoleInclude() + return ri.load_data(data, loader=loader) + diff --git a/v2/ansible/playbook/role/metadata.py b/v2/ansible/playbook/role/metadata.py new file mode 100644 index 00000000000000..9e732d6eeaabd4 --- /dev/null +++ b/v2/ansible/playbook/role/metadata.py @@ -0,0 +1,91 @@ +# (c) 2014 Michael DeHaan, +# +# This file is part of Ansible +# +# Ansible is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# Ansible is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with Ansible. If not, see . + +# Make coding more python3-ish +from __future__ import (absolute_import, division, print_function) +__metaclass__ = type + +from six import iteritems, string_types + +from ansible.errors import AnsibleError, AnsibleParserError +from ansible.playbook.attribute import Attribute, FieldAttribute +from ansible.playbook.base import Base +from ansible.playbook.role.include import RoleInclude + + +__all__ = ['RoleMetadata'] + + +class RoleMetadata(Base): + ''' + This class wraps the parsing and validation of the optional metadata + within each Role (meta/main.yml). + ''' + + _allow_duplicates = FieldAttribute(isa='bool', default=False) + _dependencies = FieldAttribute(isa='list', default=[]) + _galaxy_info = FieldAttribute(isa='GalaxyInfo') + + def __init__(self): + self._owner = None + super(RoleMetadata, self).__init__() + + @staticmethod + def load(data, owner, loader=None): + ''' + Returns a new RoleMetadata object based on the datastructure passed in. + ''' + + if not isinstance(data, dict): + raise AnsibleParserError("the 'meta/main.yml' for role %s is not a dictionary" % owner.get_name()) + + m = RoleMetadata().load_data(data, loader=loader) + return m + + def munge(self, ds): + # make sure there are no keys in the datastructure which + # do not map to attributes for this object + valid_attrs = [name for (name, attribute) in iteritems(self._get_base_attributes())] + for name in ds: + if name not in valid_attrs: + print("'%s' is not a valid attribute" % name) + raise AnsibleParserError("'%s' is not a valid attribute" % name, obj=ds) + return ds + + def _load_dependencies(self, attr, ds): + ''' + This is a helper loading function for the dependencis list, + which returns a list of RoleInclude objects + ''' + + assert isinstance(ds, list) + + deps = [] + for role_def in ds: + i = RoleInclude.load(role_def, loader=self._loader) + deps.append(i) + + return deps + + def _load_galaxy_info(self, attr, ds): + ''' + This is a helper loading function for the galaxy info entry + in the metadata, which returns a GalaxyInfo object rather than + a simple dictionary. + ''' + + return ds diff --git a/v2/ansible/playbook/role/requirement.py b/v2/ansible/playbook/role/requirement.py new file mode 100644 index 00000000000000..d321f6e17dfb32 --- /dev/null +++ b/v2/ansible/playbook/role/requirement.py @@ -0,0 +1,166 @@ +# (c) 2014 Michael DeHaan, +# +# This file is part of Ansible +# +# Ansible is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# Ansible is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with Ansible. If not, see . + +# Make coding more python3-ish +from __future__ import (absolute_import, division, print_function) +__metaclass__ = type + +from six import iteritems, string_types + +import os + +from ansible.errors import AnsibleError, AnsibleParserError +from ansible.playbook.role.definition import RoleDefinition + +__all__ = ['RoleRequirement'] + + +class RoleRequirement(RoleDefinition): + + """ + FIXME: document various ways role specs can be specified + """ + + def __init__(self): + pass + + def _get_valid_spec_keys(self): + return ( + 'name', + 'role', + 'scm', + 'src', + 'version', + ) + + def parse(self, ds): + ''' + FIXME: docstring + ''' + + assert type(ds) == dict or isinstance(ds, string_types) + + role_name = '' + role_params = dict() + new_ds = dict() + + if isinstance(ds, string_types): + role_name = ds + else: + ds = self._munge_role_spec(ds) + (new_ds, role_params) = self._split_role_params(ds) + + # pull the role name out of the ds + role_name = new_ds.get('role_name') + del ds['role_name'] + + return (new_ds, role_name, role_params) + + def _munge_role_spec(self, ds): + if 'role' in ds: + # Old style: {role: "galaxy.role,version,name", other_vars: "here" } + role_info = self._role_spec_parse(ds['role']) + if isinstance(role_info, dict): + # Warning: Slight change in behaviour here. name may be being + # overloaded. Previously, name was only a parameter to the role. + # Now it is both a parameter to the role and the name that + # ansible-galaxy will install under on the local system. + if 'name' in ds and 'name' in role_info: + del role_info['name'] + ds.update(role_info) + else: + # New style: { src: 'galaxy.role,version,name', other_vars: "here" } + if 'github.com' in ds["src"] and 'http' in ds["src"] and '+' not in ds["src"] and not ds["src"].endswith('.tar.gz'): + ds["src"] = "git+" + ds["src"] + + if '+' in ds["src"]: + (scm, src) = ds["src"].split('+') + ds["scm"] = scm + ds["src"] = src + + if 'name' in role: + ds["role"] = ds["name"] + del ds["name"] + else: + ds["role"] = self._repo_url_to_role_name(ds["src"]) + + # set some values to a default value, if none were specified + ds.setdefault('version', '') + ds.setdefault('scm', None) + + return ds + + def _repo_url_to_role_name(self, repo_url): + # gets the role name out of a repo like + # http://git.example.com/repos/repo.git" => "repo" + + if '://' not in repo_url and '@' not in repo_url: + return repo_url + trailing_path = repo_url.split('/')[-1] + if trailing_path.endswith('.git'): + trailing_path = trailing_path[:-4] + if trailing_path.endswith('.tar.gz'): + trailing_path = trailing_path[:-7] + if ',' in trailing_path: + trailing_path = trailing_path.split(',')[0] + return trailing_path + + def _role_spec_parse(self, role_spec): + # takes a repo and a version like + # git+http://git.example.com/repos/repo.git,v1.0 + # and returns a list of properties such as: + # { + # 'scm': 'git', + # 'src': 'http://git.example.com/repos/repo.git', + # 'version': 'v1.0', + # 'name': 'repo' + # } + + default_role_versions = dict(git='master', hg='tip') + + role_spec = role_spec.strip() + role_version = '' + if role_spec == "" or role_spec.startswith("#"): + return (None, None, None, None) + + tokens = [s.strip() for s in role_spec.split(',')] + + # assume https://github.com URLs are git+https:// URLs and not + # tarballs unless they end in '.zip' + if 'github.com/' in tokens[0] and not tokens[0].startswith("git+") and not tokens[0].endswith('.tar.gz'): + tokens[0] = 'git+' + tokens[0] + + if '+' in tokens[0]: + (scm, role_url) = tokens[0].split('+') + else: + scm = None + role_url = tokens[0] + + if len(tokens) >= 2: + role_version = tokens[1] + + if len(tokens) == 3: + role_name = tokens[2] + else: + role_name = self._repo_url_to_role_name(tokens[0]) + + if scm and not role_version: + role_version = default_role_versions.get(scm, '') + + return dict(scm=scm, src=role_url, version=role_version, role_name=role_name) + + diff --git a/v2/ansible/playbook/task.py b/v2/ansible/playbook/task.py index aa79d49410457c..422668148bafb0 100644 --- a/v2/ansible/playbook/task.py +++ b/v2/ansible/playbook/task.py @@ -83,14 +83,16 @@ class Task(Base): _sudo = FieldAttribute(isa='bool') _sudo_user = FieldAttribute(isa='string') _sudo_pass = FieldAttribute(isa='string') + _tags = FieldAttribute(isa='list', default=[]) _transport = FieldAttribute(isa='string') _until = FieldAttribute(isa='list') # ? + _when = FieldAttribute(isa='list', default=[]) - def __init__(self, block=None, role=None, loader=DataLoader): + def __init__(self, block=None, role=None): ''' constructors a task, without the Task.load classmethod, it will be pretty blank ''' self._block = block self._role = role - super(Task, self).__init__(loader) + super(Task, self).__init__() def get_name(self): ''' return the name of the task ''' @@ -118,9 +120,9 @@ def _merge_kv(self, ds): return buf @staticmethod - def load(data, block=None, role=None): + def load(data, block=None, role=None, loader=None): t = Task(block=block, role=role) - return t.load_data(data) + return t.load_data(data, loader=loader) def __repr__(self): ''' returns a human readable representation of the task ''' diff --git a/v2/test/mock/__init__.py b/v2/test/mock/__init__.py new file mode 100644 index 00000000000000..ae8ccff5952585 --- /dev/null +++ b/v2/test/mock/__init__.py @@ -0,0 +1,20 @@ +# (c) 2012-2014, Michael DeHaan +# +# This file is part of Ansible +# +# Ansible is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# Ansible is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with Ansible. If not, see . + +# Make coding more python3-ish +from __future__ import (absolute_import, division, print_function) +__metaclass__ = type diff --git a/v2/test/mock/loader.py b/v2/test/mock/loader.py new file mode 100644 index 00000000000000..89dbfeea622bbf --- /dev/null +++ b/v2/test/mock/loader.py @@ -0,0 +1,80 @@ +# (c) 2012-2014, Michael DeHaan +# +# This file is part of Ansible +# +# Ansible is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# Ansible is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with Ansible. If not, see . + +# Make coding more python3-ish +from __future__ import (absolute_import, division, print_function) +__metaclass__ = type + +import os + +from ansible.parsing.yaml import DataLoader + +class DictDataLoader(DataLoader): + + def __init__(self, file_mapping=dict()): + assert type(file_mapping) == dict + + self._file_mapping = file_mapping + self._build_known_directories() + + super(DictDataLoader, self).__init__() + + def load_from_file(self, path): + if path in self._file_mapping: + return self.load(self._file_mapping[path], path) + return None + + def path_exists(self, path): + return path in self._file_mapping or path in self._known_directories + + def is_file(self, path): + return path in self._file_mapping + + def is_directory(self, path): + return path in self._known_directories + + def _add_known_directory(self, directory): + if directory not in self._known_directories: + self._known_directories.append(directory) + + def _build_known_directories(self): + self._known_directories = [] + for path in self._file_mapping: + dirname = os.path.dirname(path) + while dirname not in ('/', ''): + self._add_known_directory(dirname) + dirname = os.path.dirname(dirname) + + def push(self, path, content): + rebuild_dirs = False + if path not in self._file_mapping: + rebuild_dirs = True + + self._file_mapping[path] = content + + if rebuild_dirs: + self._build_known_directories() + + def pop(self, path): + if path in self._file_mapping: + del self._file_mapping[path] + self._build_known_directories() + + def clear(self): + self._file_mapping = dict() + self._known_directories = [] + diff --git a/v2/test/playbook/test_role.py b/v2/test/playbook/test_role.py index 2c1ca6c959d44b..d0f3708898d157 100644 --- a/v2/test/playbook/test_role.py +++ b/v2/test/playbook/test_role.py @@ -25,9 +25,10 @@ from ansible.errors import AnsibleError, AnsibleParserError from ansible.playbook.block import Block from ansible.playbook.role import Role +from ansible.playbook.role.include import RoleInclude from ansible.playbook.task import Task -from ansible.parsing.yaml import DataLoader +from test.mock.loader import DictDataLoader class TestRole(unittest.TestCase): @@ -37,172 +38,130 @@ def setUp(self): def tearDown(self): pass - def test_construct_empty_block(self): - r = Role() - - @patch.object(DataLoader, 'load_from_file') - def test__load_role_yaml(self, _load_from_file): - _load_from_file.return_value = dict(foo='bar') - r = Role() - with patch('os.path.exists', return_value=True): - with patch('os.path.isdir', return_value=True): - res = r._load_role_yaml('/fake/path', 'some_subdir') - self.assertEqual(res, dict(foo='bar')) - - def test_role__load_list_of_blocks(self): - task = dict(action='test') - r = Role() - self.assertEqual(r._load_list_of_blocks([]), []) - res = r._load_list_of_blocks([task]) - self.assertEqual(len(res), 1) - assert isinstance(res[0], Block) - res = r._load_list_of_blocks([task,task,task]) - self.assertEqual(len(res), 3) - - @patch.object(Role, '_get_role_path') - @patch.object(Role, '_load_role_yaml') - def test_load_role_with_tasks(self, _load_role_yaml, _get_role_path): - - _get_role_path.return_value = ('foo', '/etc/ansible/roles/foo') - - def fake_load_role_yaml(role_path, subdir): - if role_path == '/etc/ansible/roles/foo': - if subdir == 'tasks': - return [dict(shell='echo "hello world"')] - return None - - _load_role_yaml.side_effect = fake_load_role_yaml - - r = Role.load('foo') - self.assertEqual(len(r.task_blocks), 1) - assert isinstance(r.task_blocks[0], Block) - - @patch.object(Role, '_get_role_path') - @patch.object(Role, '_load_role_yaml') - def test_load_role_with_handlers(self, _load_role_yaml, _get_role_path): - - _get_role_path.return_value = ('foo', '/etc/ansible/roles/foo') - - def fake_load_role_yaml(role_path, subdir): - if role_path == '/etc/ansible/roles/foo': - if subdir == 'handlers': - return [dict(name='test handler', shell='echo "hello world"')] - return None - - _load_role_yaml.side_effect = fake_load_role_yaml - - r = Role.load('foo') - self.assertEqual(len(r.handler_blocks), 1) - assert isinstance(r.handler_blocks[0], Block) - - @patch.object(Role, '_get_role_path') - @patch.object(Role, '_load_role_yaml') - def test_load_role_with_vars(self, _load_role_yaml, _get_role_path): - - _get_role_path.return_value = ('foo', '/etc/ansible/roles/foo') - - def fake_load_role_yaml(role_path, subdir): - if role_path == '/etc/ansible/roles/foo': - if subdir == 'defaults': - return dict(foo='bar') - elif subdir == 'vars': - return dict(foo='bam') - return None - - _load_role_yaml.side_effect = fake_load_role_yaml - - r = Role.load('foo') - self.assertEqual(r.default_vars, dict(foo='bar')) - self.assertEqual(r.role_vars, dict(foo='bam')) - - @patch.object(Role, '_get_role_path') - @patch.object(Role, '_load_role_yaml') - def test_load_role_with_metadata(self, _load_role_yaml, _get_role_path): - - def fake_get_role_path(role): - if role == 'foo': - return ('foo', '/etc/ansible/roles/foo') - elif role == 'bar': - return ('bar', '/etc/ansible/roles/bar') - elif role == 'baz': - return ('baz', '/etc/ansible/roles/baz') - elif role == 'bam': - return ('bam', '/etc/ansible/roles/bam') - elif role == 'bad1': - return ('bad1', '/etc/ansible/roles/bad1') - elif role == 'bad2': - return ('bad2', '/etc/ansible/roles/bad2') - elif role == 'recursive1': - return ('recursive1', '/etc/ansible/roles/recursive1') - elif role == 'recursive2': - return ('recursive2', '/etc/ansible/roles/recursive2') - - def fake_load_role_yaml(role_path, subdir): - if role_path == '/etc/ansible/roles/foo': - if subdir == 'meta': - return dict(dependencies=['bar'], allow_duplicates=True, galaxy_info=dict(a='1', b='2', c='3')) - elif role_path == '/etc/ansible/roles/bar': - if subdir == 'meta': - return dict(dependencies=['baz']) - elif role_path == '/etc/ansible/roles/baz': - if subdir == 'meta': - return dict(dependencies=['bam']) - elif role_path == '/etc/ansible/roles/bam': - if subdir == 'meta': - return dict() - elif role_path == '/etc/ansible/roles/bad1': - if subdir == 'meta': - return 1 - elif role_path == '/etc/ansible/roles/bad2': - if subdir == 'meta': - return dict(foo='bar') - elif role_path == '/etc/ansible/roles/recursive1': - if subdir == 'meta': - return dict(dependencies=['recursive2']) - elif role_path == '/etc/ansible/roles/recursive2': - if subdir == 'meta': - return dict(dependencies=['recursive1']) - return None - - _get_role_path.side_effect = fake_get_role_path - _load_role_yaml.side_effect = fake_load_role_yaml - - r = Role.load('foo') + def test_load_role_with_tasks(self): + + fake_loader = DictDataLoader({ + "/etc/ansible/roles/foo/tasks/main.yml": """ + - shell: echo 'hello world' + """, + }) + + i = RoleInclude.load('foo', loader=fake_loader) + r = Role.load(i) + + self.assertEqual(str(r), 'foo') + self.assertEqual(len(r._task_blocks), 1) + assert isinstance(r._task_blocks[0], Block) + + def test_load_role_with_handlers(self): + + fake_loader = DictDataLoader({ + "/etc/ansible/roles/foo/handlers/main.yml": """ + - name: test handler + shell: echo 'hello world' + """, + }) + + i = RoleInclude.load('foo', loader=fake_loader) + r = Role.load(i) + + self.assertEqual(len(r._handler_blocks), 1) + assert isinstance(r._handler_blocks[0], Block) + + def test_load_role_with_vars(self): + + fake_loader = DictDataLoader({ + "/etc/ansible/roles/foo/defaults/main.yml": """ + foo: bar + """, + "/etc/ansible/roles/foo/vars/main.yml": """ + foo: bam + """, + }) + + i = RoleInclude.load('foo', loader=fake_loader) + r = Role.load(i) + + self.assertEqual(r._default_vars, dict(foo='bar')) + self.assertEqual(r._role_vars, dict(foo='bam')) + + def test_load_role_with_metadata(self): + + fake_loader = DictDataLoader({ + '/etc/ansible/roles/foo/meta/main.yml': """ + allow_duplicates: true + dependencies: + - bar + galaxy_info: + a: 1 + b: 2 + c: 3 + """, + '/etc/ansible/roles/bar/meta/main.yml': """ + dependencies: + - baz + """, + '/etc/ansible/roles/baz/meta/main.yml': """ + dependencies: + - bam + """, + '/etc/ansible/roles/bam/meta/main.yml': """ + dependencies: [] + """, + '/etc/ansible/roles/bad1/meta/main.yml': """ + 1 + """, + '/etc/ansible/roles/bad2/meta/main.yml': """ + foo: bar + """, + '/etc/ansible/roles/recursive1/meta/main.yml': """ + dependencies: ['recursive2'] + """, + '/etc/ansible/roles/recursive2/meta/main.yml': """ + dependencies: ['recursive1'] + """, + }) + + i = RoleInclude.load('foo', loader=fake_loader) + r = Role.load(i) + role_deps = r.get_direct_dependencies() self.assertEqual(len(role_deps), 1) self.assertEqual(type(role_deps[0]), Role) self.assertEqual(len(role_deps[0].get_parents()), 1) self.assertEqual(role_deps[0].get_parents()[0], r) - self.assertEqual(r.allow_duplicates, True) - self.assertEqual(r.galaxy_info, dict(a='1', b='2', c='3')) + self.assertEqual(r._metadata.allow_duplicates, True) + self.assertEqual(r._metadata.galaxy_info, dict(a=1, b=2, c=3)) all_deps = r.get_all_dependencies() self.assertEqual(len(all_deps), 3) - self.assertEqual(all_deps[0].role_name, 'bar') - self.assertEqual(all_deps[1].role_name, 'baz') - self.assertEqual(all_deps[2].role_name, 'bam') + self.assertEqual(all_deps[0].get_name(), 'bar') + self.assertEqual(all_deps[1].get_name(), 'baz') + self.assertEqual(all_deps[2].get_name(), 'bam') + + i = RoleInclude.load('bad1', loader=fake_loader) + self.assertRaises(AnsibleParserError, Role.load, i) - self.assertRaises(AnsibleParserError, Role.load, 'bad1') - self.assertRaises(AnsibleParserError, Role.load, 'bad2') - self.assertRaises(AnsibleError, Role.load, 'recursive1') + i = RoleInclude.load('bad2', loader=fake_loader) + self.assertRaises(AnsibleParserError, Role.load, i) - @patch.object(Role, '_get_role_path') - @patch.object(Role, '_load_role_yaml') - def test_load_role_complex(self, _load_role_yaml, _get_role_path): + i = RoleInclude.load('recursive1', loader=fake_loader) + self.assertRaises(AnsibleError, Role.load, i) - _get_role_path.return_value = ('foo', '/etc/ansible/roles/foo') + def test_load_role_complex(self): - def fake_load_role_yaml(role_path, subdir): - if role_path == '/etc/ansible/roles/foo': - if subdir == 'tasks': - return [dict(shell='echo "hello world"')] - return None + # FIXME: add tests for the more complex uses of + # params and tags/when statements - _load_role_yaml.side_effect = fake_load_role_yaml + fake_loader = DictDataLoader({ + "/etc/ansible/roles/foo/tasks/main.yml": """ + - shell: echo 'hello world' + """, + }) - r = Role.load(dict(role='foo')) + i = RoleInclude.load(dict(role='foo'), loader=fake_loader) + r = Role.load(i) - # FIXME: add tests for the more complex url-type - # constructions and tags/when statements + self.assertEqual(r.get_name(), "foo")