feat: Consider inclue_role as normal role instead of task (#82)

This commit is contained in:
Mohamed El Mouctar Haidara 2021-09-15 22:29:19 +02:00 committed by GitHub
parent 783d77593c
commit dd0c4e5db2
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
4 changed files with 80 additions and 67 deletions

View file

@ -39,6 +39,14 @@ class CompositeNode(Node):
# The dict will contain the different types of composition. # The dict will contain the different types of composition.
self._compositions = defaultdict(list) # type: Dict[str, List] self._compositions = defaultdict(list) # type: Dict[str, List]
@property
def total_length(self) -> int:
"""
Return the total length of elements in this Composite node
:return:
"""
return sum([len(val) for val in self._compositions.values()])
def add_node(self, target_composition: str, node: Node): def add_node(self, target_composition: str, node: Node):
""" """
Add a node in the target composition Add a node in the target composition
@ -50,8 +58,8 @@ class CompositeNode(Node):
def links_structure(self) -> Dict[Node, List[Node]]: def links_structure(self) -> Dict[Node, List[Node]]:
""" """
Return a representation of the composite node where each key of a dictionary is the node ID and the values is a Return a representation of the composite node where each key of the dictionary is the node ID and the values is
list of linked nodes a list of the linked nodes
:return: :return:
""" """
links = defaultdict(list) links = defaultdict(list)

View file

@ -68,7 +68,7 @@ class BaseParser(ABC):
:return: True if the task has been included, false otherwise :return: True if the task has been included, false otherwise
""" """
self.display.vv(f"Adding {node_type}: '{task.get_name()}' to the graph") self.display.vv(f"Adding {node_type} '{task.get_name()}' to the graph with counter {loop_counter}")
if not task.evaluate_tags(only_tags=self.tags, skip_tags=self.skip_tags, all_vars=task_vars): if not task.evaluate_tags(only_tags=self.tags, skip_tags=self.skip_tags, all_vars=task_vars):
self.display.vv(f"The task '{task.get_name()}' is skipped due to the tags.") self.display.vv(f"The task '{task.get_name()}' is skipped due to the tags.")
@ -117,12 +117,12 @@ class PlaybookParser(BaseParser):
The graph is drawn following this order (https://docs.ansible.com/ansible/2.4/playbooks_reuse_roles.html#using-roles) The graph is drawn following this order (https://docs.ansible.com/ansible/2.4/playbooks_reuse_roles.html#using-roles)
for each play: for each play:
draw pre_tasks add pre_tasks
draw roles add roles
if include_role_tasks if include_role_tasks
draw role_tasks add role_tasks
draw tasks add tasks
draw post_tasks add post_tasks
:return: :return:
""" """
@ -148,20 +148,14 @@ class PlaybookParser(BaseParser):
# loop through the pre_tasks # loop through the pre_tasks
self.display.v("Parsing pre_tasks...") self.display.v("Parsing pre_tasks...")
nb_pre_tasks = 0
for pre_task_block in play.pre_tasks: for pre_task_block in play.pre_tasks:
nb_pre_tasks = self._include_tasks_in_blocks(current_play=play, parent_node=play_node, self._include_tasks_in_blocks(current_play=play, parent_nodes=[play_node], block=pre_task_block,
block=pre_task_block, current_counter=nb_pre_tasks, play_vars=play_vars, node_type="pre_task")
play_vars=play_vars, node_type="pre_task")
# global_tasks_counter will hold the number of pre_tasks + tasks + and post_tasks
global_tasks_counter = nb_pre_tasks
self.display.v(f"{global_tasks_counter} pre_task(s) added to the graph.")
# loop through the roles # loop through the roles
self.display.v("Parsing roles...") self.display.v("Parsing roles...")
role_number = 0
for role in play.get_roles(): for role_counter, role in enumerate(play.get_roles(), 1):
# Don't insert tasks from ``import/include_role``, preventing duplicate graphing # Don't insert tasks from ``import/include_role``, preventing duplicate graphing
if role.from_include: if role.from_include:
continue continue
@ -173,43 +167,35 @@ class PlaybookParser(BaseParser):
# Go to the next role # Go to the next role
continue continue
role_number += 1
role_node = RoleNode("[role] " + clean_name(role.get_name())) role_node = RoleNode("[role] " + clean_name(role.get_name()))
# edge from play to role # edge from play to role
play_node.add_node("roles", EdgeNode(str(role_number + global_tasks_counter), play_node, role_node)) play_node.add_node("roles",
EdgeNode(str(role_counter + len(play_node.pre_tasks)), play_node, role_node))
# loop through the tasks of the roles
if self.include_role_tasks: if self.include_role_tasks:
role_tasks_counter = 0 # the role tasks start a 0 # loop through the tasks of the roles
for block in role.compile(play): for block in role.compile(play):
role_tasks_counter = self._include_tasks_in_blocks(current_play=play, parent_node=role_node, self._include_tasks_in_blocks(current_play=play, parent_nodes=[role_node], block=block,
block=block, play_vars=play_vars, play_vars=play_vars, node_type="task")
current_counter=role_tasks_counter,
node_type="task")
role_tasks_counter += 1
# end of roles loop # end of roles loop
self.display.v(f"{role_number} role(s) added to the graph")
# loop through the tasks # loop through the tasks
self.display.v("Parsing tasks...") self.display.v("Parsing tasks...")
for task_block in play.tasks: for task_block in play.tasks:
global_tasks_counter = self._include_tasks_in_blocks(current_play=play, parent_node=play_node, self._include_tasks_in_blocks(current_play=play, parent_nodes=[play_node], block=task_block,
block=task_block, play_vars=play_vars, play_vars=play_vars, node_type="task")
current_counter=role_number + global_tasks_counter,
node_type="task")
nb_tasks = global_tasks_counter - role_number - nb_pre_tasks
self.display.v(f"{nb_tasks} task(s) added to the graph.")
# loop through the post_tasks # loop through the post_tasks
self.display.v("Parsing post_tasks...") self.display.v("Parsing post_tasks...")
for post_task_block in play.post_tasks: for post_task_block in play.post_tasks:
global_tasks_counter = self._include_tasks_in_blocks(current_play=play, parent_node=play_node, self._include_tasks_in_blocks(current_play=play, parent_nodes=[play_node], block=post_task_block,
block=post_task_block, play_vars=play_vars, play_vars=play_vars, node_type="post_task")
current_counter=global_tasks_counter, # Summary
node_type="post_task") self.display.display("") # just an empty line
nb_post_tasks = global_tasks_counter - nb_tasks - role_number - nb_pre_tasks self.display.v(f"{len(play_node.pre_tasks)} pre_task(s) added to the graph.")
self.display.v(f"{nb_post_tasks} post_task(s) added to the graph.") self.display.v(f"{len(play_node.roles)} role(s) added to the play")
self.display.v(f"{len(play_node.tasks)} task(s) added to the play")
self.display.v(f"{len(play_node.post_tasks)} post_task(s) added to the play")
self.display.banner(f"Done parsing {play_name}") self.display.banner(f"Done parsing {play_name}")
self.display.display("") # just an empty line self.display.display("") # just an empty line
@ -217,25 +203,24 @@ class PlaybookParser(BaseParser):
return self.playbook_root_node return self.playbook_root_node
def _include_tasks_in_blocks(self, current_play: Play, parent_node: CompositeNode, block: Union[Block, TaskInclude], def _include_tasks_in_blocks(self, current_play: Play, parent_nodes: List[CompositeNode],
current_counter: int, play_vars: Dict = None, node_type: str = "") -> int: block: Union[Block, TaskInclude], node_type: str, play_vars: Dict = None):
""" """
Recursively read all the tasks of the block and add it to the graph Recursively read all the tasks of the block and add it to the graph
FIXME: This function needs some refactoring :param parent_nodes: This a list of parent nodes. Each time, we see an include_role, the corresponding node is
added to this list
:param current_play: :param current_play:
:param block: :param block:
:param current_counter:
:param play_vars: :param play_vars:
:param node_type: :param node_type:
:return: :return:
""" """
# loop through the tasks # loop through the tasks
for counter, task_or_block in enumerate(block.block, 1): for task_or_block in block.block:
if isinstance(task_or_block, Block): if isinstance(task_or_block, Block):
current_counter = self._include_tasks_in_blocks(current_play=current_play, parent_node=parent_node, self._include_tasks_in_blocks(current_play=current_play, parent_nodes=parent_nodes, block=task_or_block,
block=task_or_block, node_type=node_type, node_type=node_type, play_vars=play_vars)
current_counter=current_counter, play_vars=play_vars)
elif isinstance(task_or_block, TaskInclude): # include, include_tasks, include_role are dynamic elif isinstance(task_or_block, TaskInclude): # include, include_tasks, include_role are dynamic
# So we need to process them explicitly because Ansible does it during the execution of the playbook # So we need to process them explicitly because Ansible does it during the execution of the playbook
@ -245,8 +230,19 @@ class PlaybookParser(BaseParser):
# Here we have an 'include_role'. The class IncludeRole is a subclass of TaskInclude. # Here we have an 'include_role'. The class IncludeRole is a subclass of TaskInclude.
# We do this because the management of an 'include_role' is different. # We do this because the management of an 'include_role' is different.
# See :func:`~ansible.playbook.included_file.IncludedFile.process_include_results` from line 155 # See :func:`~ansible.playbook.included_file.IncludedFile.process_include_results` from line 155
self.display.v(
f"An 'include_role' found. Including tasks from the role '{task_or_block.args['name']}'")
role_node = RoleNode(task_or_block.args['name'])
parent_nodes[-1].add_node("roles",
EdgeNode(str(parent_nodes[-1].total_length + 1), parent_nodes[-1],
role_node))
if self.include_role_tasks:
# If we have an include_role and we want to include role tasks, the parent node now becomes
# the role.
parent_nodes.append(role_node)
self.display.v(f"An 'include_role' found. Including tasks from '{task_or_block.args['name']}'")
block_list, _ = task_or_block.get_block_list(play=current_play, loader=self.data_loader, block_list, _ = task_or_block.get_block_list(play=current_play, loader=self.data_loader,
variable_manager=self.variable_manager) variable_manager=self.variable_manager)
else: else:
@ -261,9 +257,8 @@ class PlaybookParser(BaseParser):
self.display.warning( self.display.warning(
f"Unable to translate the include task '{task_or_block.get_name()}' due to an undefined variable: {str(e)}. " f"Unable to translate the include task '{task_or_block.get_name()}' due to an undefined variable: {str(e)}. "
"Some variables are available only during the execution of the playbook.") "Some variables are available only during the execution of the playbook.")
current_counter += 1 self._add_task(task=task_or_block, loop_counter=parent_nodes[-1].total_length + 1,
self._add_task(task=task_or_block, loop_counter=current_counter, task_vars=task_vars, task_vars=task_vars, node_type=node_type, parent_node=parent_nodes[-1])
node_type=node_type, parent_node=parent_node)
continue continue
data = self.data_loader.load_from_file(include_file) data = self.data_loader.load_from_file(include_file)
@ -279,10 +274,14 @@ class PlaybookParser(BaseParser):
parent_block=task_or_block) parent_block=task_or_block)
for b in block_list: # loop through the blocks inside the included tasks or role for b in block_list: # loop through the blocks inside the included tasks or role
current_counter = self._include_tasks_in_blocks(current_play=current_play, parent_node=parent_node, self._include_tasks_in_blocks(current_play=current_play, parent_nodes=parent_nodes, block=b,
block=b, play_vars=task_vars, node_type=node_type, play_vars=task_vars, node_type=node_type)
current_counter=current_counter)
else: else:
if len(parent_nodes) > 1 and not has_role_parent(task_or_block):
# We add a new parent node only if we found an include_role. If an include_role is not found, and we
# have a task that is not from an include_role, we remove the last RoleNode we have added.
parent_nodes.pop()
# check if this task comes from a role, and we don't want to include tasks of the role # check if this task comes from a role, and we don't want to include tasks of the role
if has_role_parent(task_or_block) and not self.include_role_tasks: if has_role_parent(task_or_block) and not self.include_role_tasks:
# skip role's task # skip role's task
@ -292,10 +291,5 @@ class PlaybookParser(BaseParser):
# skipping # skipping
continue continue
is_task_included = self._add_task(task=task_or_block, loop_counter=current_counter + 1, self._add_task(task=task_or_block, loop_counter=parent_nodes[-1].total_length + 1, task_vars=play_vars,
task_vars=play_vars, node_type=node_type, parent_node=parent_node) node_type=node_type, parent_node=parent_nodes[-1])
if is_task_included:
# only increment the counter if task has been successfully included.
current_counter += 1
return current_counter

View file

@ -3,6 +3,16 @@
tags: tags:
- play2 - play2
tasks: tasks:
- name: Include role - name: (1) Debug
debug:
msg: "Debug 1"
when: ansible_os == "ubuntu"
- name: (2) Include role
include_role: include_role:
name: fake_role_2 name: fake_role
- name: (3) Debug 2
debug:
msg: "Debug 2"
- name: (4) Include role 2
include_role:
name: fake_role_2

View file

@ -135,7 +135,7 @@ def test_with_roles(request, include_role_tasks_option, expected_tasks_number):
@pytest.mark.parametrize(["include_role_tasks_option", "expected_tasks_number"], @pytest.mark.parametrize(["include_role_tasks_option", "expected_tasks_number"],
[("--", 0), ("--include-role-tasks", 3)], [("--", 2), ("--include-role-tasks", 8)],
ids=["no_include_role_tasks_option", "include_role_tasks_option"]) ids=["no_include_role_tasks_option", "include_role_tasks_option"])
def test_include_role(request, include_role_tasks_option, expected_tasks_number): def test_include_role(request, include_role_tasks_option, expected_tasks_number):
""" """
@ -144,7 +144,8 @@ def test_include_role(request, include_role_tasks_option, expected_tasks_number)
svg_path, playbook_path = run_grapher("include_role.yml", output_filename=request.node.name, svg_path, playbook_path = run_grapher("include_role.yml", output_filename=request.node.name,
additional_args=[include_role_tasks_option]) additional_args=[include_role_tasks_option])
_common_tests(svg_path=svg_path, playbook_path=playbook_path, plays_number=1, tasks_number=expected_tasks_number) _common_tests(svg_path=svg_path, playbook_path=playbook_path, plays_number=1, tasks_number=expected_tasks_number,
roles_number=2)
def test_with_block(request): def test_with_block(request):