Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

fix: Roles usages and do not use multiple edges for role tasks #120

Merged
merged 4 commits into from
Aug 17, 2022
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
103 changes: 77 additions & 26 deletions ansibleplaybookgrapher/graph.py
Original file line number Diff line number Diff line change
Expand Up @@ -14,8 +14,7 @@
# along with this program. If not, see <https://www.gnu.org/licenses/>.
import os
from collections import defaultdict
from typing import Dict, List, ItemsView, Set
from collections import defaultdict
from typing import Dict, List, ItemsView, Set, Type

from ansibleplaybookgrapher.utils import generate_id

Expand All @@ -25,16 +24,25 @@ class Node:
A node in the graph. Everything of the final graph is a node: playbook, plays, edges, tasks and roles.
"""

def __init__(self, node_name: str, node_id: str, when: str = "", raw_object=None):
def __init__(
self,
node_name: str,
node_id: str,
when: str = "",
raw_object=None,
parent: "Node" = None,
):
"""

:param node_name: The name of the node
:param node_id: An identifier for this node
:param when: The conditional attached to the node
:param raw_object: The raw ansible object matching this node in the graph. Will be None if there is no match on
Ansible side
:param parent: The parent of this node
"""
self.name = node_name
self.parent = parent
self.id = node_id
self.when = when
self.raw_object = raw_object
Expand All @@ -50,8 +58,22 @@ def retrieve_position(self):
if self.raw_object and self.raw_object.get_ds():
self.path, self.line, self.column = self.raw_object.get_ds().ansible_pos

def get_first_parent_matching_type(self, node_type: Type) -> "Type":
"""
Get the first parent of this node matching the given type
:return:
"""
current_parent = self.parent

while current_parent is not None:
if isinstance(current_parent, node_type):
return current_parent
current_parent = current_parent.parent

raise ValueError(f"No parent of type {node_type} found for {self}")

def __repr__(self):
return f"{type(self).__name__}(name='{self.name}',id='{self.id}')"
return f"{type(self).__name__}(name='{self.name}', id='{self.id}')"

def __eq__(self, other):
return self.id == other.id
Expand All @@ -74,6 +96,7 @@ def __init__(
node_id: str,
when: str = "",
raw_object=None,
parent: "Node" = None,
supported_compositions: List[str] = None,
):
"""
Expand All @@ -84,7 +107,7 @@ def __init__(
Ansible side
:param supported_compositions:
"""
super().__init__(node_name, node_id, when, raw_object)
super().__init__(node_name, node_id, when, raw_object, parent)
self._supported_compositions = supported_compositions or []
# The dict will contain the different types of composition.
self._compositions = defaultdict(list) # type: Dict[str, List]
Expand Down Expand Up @@ -133,17 +156,17 @@ def _get_all_tasks_nodes(self, task_acc: List["Node"]):
elif isinstance(node, CompositeNode):
node._get_all_tasks_nodes(task_acc)

def links_structure(self) -> Dict[str, List[Node]]:
def links_structure(self) -> Dict[Node, List[Node]]:
"""
Return a representation of the composite node where each key of the dictionary is the node id and the
value is the list of the linked nodes
:return:
"""
links: Dict[str, List[Node]] = defaultdict(list)
links: Dict[Node, List[Node]] = defaultdict(list)
self._get_all_links(links)
return links

def _get_all_links(self, links: Dict[str, List[Node]]):
def _get_all_links(self, links: Dict[Node, List[Node]]):
"""
Recursively get the node links
:return:
Expand All @@ -152,16 +175,25 @@ def _get_all_links(self, links: Dict[str, List[Node]]):
for node in nodes:
if isinstance(node, CompositeNode):
node._get_all_links(links)
links[self.id].append(node)
links[self].append(node)


class CompositeTasksNode(CompositeNode):
"""
A special composite node which only support adding "tasks". Useful for block and role
"""

def __init__(self, node_name: str, node_id: str, when: str = "", raw_object=None):
super().__init__(node_name, node_id, when=when, raw_object=raw_object)
def __init__(
self,
node_name: str,
node_id: str,
when: str = "",
raw_object=None,
parent: "Node" = None,
):
super().__init__(
node_name, node_id, when=when, raw_object=raw_object, parent=parent
)
self._supported_compositions = ["tasks"]

def add_node(self, target_composition: str, node: Node):
Expand Down Expand Up @@ -216,7 +248,7 @@ def plays(self) -> List["PlayNode"]:
"""
return self._compositions["plays"]

def roles_usage(self) -> Dict["RoleNode", List[str]]:
def roles_usage(self) -> Dict["RoleNode", List[Node]]:
"""
For each role in the graph, return the plays that reference the role
FIXME: Review this implementation. It may not be the most efficient way, but it's ok for the moment
Expand All @@ -226,18 +258,15 @@ def roles_usage(self) -> Dict["RoleNode", List[str]]:
usages = defaultdict(list)
links = self.links_structure()

for node_id, linked_nodes in links.items():
for node, linked_nodes in links.items():
for linked_node in linked_nodes:
if isinstance(linked_node, RoleNode):
usages[linked_node].append(node_id)

# In case a role is used by another role, replace it by the play associated with using role (transitivity)
for usages_set in usages.values():
for node_id in usages_set.copy():
for r in usages:
if node_id == r.id:
usages_set.remove(node_id)
usages_set.extend(usages[r])
if isinstance(node, PlayNode):
usages[linked_node].append(node)
else:
usages[linked_node].append(
node.get_first_parent_matching_type(PlayNode)
)

return usages

Expand All @@ -257,6 +286,7 @@ def __init__(
node_id: str = None,
when: str = "",
raw_object=None,
parent: "Node" = None,
hosts: List[str] = None,
):
"""
Expand All @@ -269,6 +299,7 @@ def __init__(
node_id or generate_id("play_"),
when=when,
raw_object=raw_object,
parent=parent,
supported_compositions=["pre_tasks", "roles", "tasks", "post_tasks"],
)
self.hosts = hosts or []
Expand Down Expand Up @@ -296,13 +327,19 @@ class BlockNode(CompositeTasksNode):
"""

def __init__(
self, node_name: str, node_id: str = None, when: str = "", raw_object=None
self,
node_name: str,
node_id: str = None,
when: str = "",
raw_object=None,
parent: "Node" = None,
):
super().__init__(
node_name,
node_id or generate_id("block_"),
when=when,
raw_object=raw_object,
parent=parent,
)


Expand All @@ -312,7 +349,12 @@ class TaskNode(Node):
"""

def __init__(
self, node_name: str, node_id: str = None, when: str = "", raw_object=None
self,
node_name: str,
node_id: str = None,
when: str = "",
raw_object=None,
parent: "Node" = None,
):
"""

Expand All @@ -321,7 +363,11 @@ def __init__(
:param raw_object:
"""
super().__init__(
node_name, node_id or generate_id("task_"), when=when, raw_object=raw_object
node_name,
node_id or generate_id("task_"),
when=when,
raw_object=raw_object,
parent=parent,
)


Expand All @@ -336,6 +382,7 @@ def __init__(
node_id: str = None,
when: str = "",
raw_object=None,
parent: "Node" = None,
include_role: bool = False,
):
"""
Expand All @@ -346,7 +393,11 @@ def __init__(
"""
self.include_role = include_role
super().__init__(
node_name, node_id or generate_id("role_"), when=when, raw_object=raw_object
node_name,
node_id or generate_id("role_"),
when=when,
raw_object=raw_object,
parent=parent,
)

def retrieve_position(self):
Expand Down
23 changes: 13 additions & 10 deletions ansibleplaybookgrapher/graphbuilder.py
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,7 @@
#
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <https://www.gnu.org/licenses/>.
from typing import Dict, Optional, Tuple, List
from typing import Dict, Optional, Tuple, List, Set

from ansible.utils.display import Display
from graphviz import Digraph
Expand All @@ -23,6 +23,7 @@
RoleNode,
BlockNode,
Node,
PlayNode,
)
from ansibleplaybookgrapher.utils import get_play_colors, merge_dicts

Expand Down Expand Up @@ -86,7 +87,7 @@ def parse(
for play in playbook_node.plays:
# TODO: find a way to create visual distance between the generated colors
# https://stackoverflow.com/questions/9018016/how-to-compare-two-colors-for-similarity-difference
self.plays_color[play.id] = get_play_colors(play.id)
self.plays_color[play] = get_play_colors(play.id)

# Update the usage of the roles
self.roles_usage = merge_dicts(
Expand Down Expand Up @@ -142,11 +143,11 @@ class GraphvizGraphBuilder:

def __init__(
self,
playbook_node: "PlaybookNode",
playbook_node: PlaybookNode,
open_protocol_handler: str,
digraph: Digraph,
play_colors: Dict[str, Tuple[str, str]],
roles_usage: Dict["RoleNode", List[str]] = None,
play_colors: Dict[PlayNode, Tuple[str, str]],
roles_usage: Dict[RoleNode, List[Node]] = None,
roles_built: Dict = None,
open_protocol_custom_formats: Dict[str, str] = None,
):
Expand Down Expand Up @@ -337,9 +338,13 @@ def build_role(
if role_to_render is None:
# Merge the colors for each play where this role is used
role_plays = self.roles_usage[destination]
colors = list(map(self.play_colors.get, role_plays))
# Graphviz support providing multiple colors separated by :
role_color = ":".join([c[0] for c in colors])
if len(role_plays) > 1:
# If the role is used in multiple plays, we take black as the default color
role_color = "black"
else:
colors = list(map(self.play_colors.get, role_plays))[0]
role_color = colors[0]

self.roles_built[destination.name] = destination

Expand All @@ -361,8 +366,6 @@ def build_role(
counter=role_task_counter,
color=role_color,
)
else:
print("here")

def build_graphviz_graph(self):
"""
Expand All @@ -380,7 +383,7 @@ def build_graphviz_graph(self):

for play_counter, play in enumerate(self.playbook_node.plays, 1):
with self.digraph.subgraph(name=play.name) as play_subgraph:
color, play_font_color = self.play_colors[play.id]
color, play_font_color = self.play_colors[play]
play_tooltip = (
",".join(play.hosts) if len(play.hosts) > 0 else play.name
)
Expand Down
8 changes: 7 additions & 1 deletion ansibleplaybookgrapher/parser.py
Original file line number Diff line number Diff line change
Expand Up @@ -118,6 +118,7 @@ def _add_task(
generate_id(f"{node_type}_"),
when=convert_when_to_str(task.when),
raw_object=task,
parent=parent_node,
),
)

Expand Down Expand Up @@ -229,6 +230,7 @@ def parse(self, *args, **kwargs) -> PlaybookNode:
clean_name(role.get_name()),
node_id="role_" + hash_value(role.get_name()),
raw_object=role,
parent=play_node,
)
# edge from play to role
play_node.add_node("roles", role_node)
Expand Down Expand Up @@ -297,7 +299,10 @@ def _include_tasks_in_blocks(
if not block._implicit and block._role is None:
# Here we have an explicit block. Ansible internally converts all normal tasks to Block
block_node = BlockNode(
str(block.name), when=convert_when_to_str(block.when), raw_object=block
str(block.name),
when=convert_when_to_str(block.when),
raw_object=block,
parent=parent_nodes[-1],
)
parent_nodes[-1].add_node(f"{node_type}s", block_node)
parent_nodes.append(block_node)
Expand Down Expand Up @@ -341,6 +346,7 @@ def _include_tasks_in_blocks(
node_id="role_" + hash_value(task_or_block._role_name),
when=convert_when_to_str(task_or_block.when),
raw_object=task_or_block,
parent=parent_nodes[-1],
include_role=True,
)
parent_nodes[-1].add_node(
Expand Down
6 changes: 3 additions & 3 deletions ansibleplaybookgrapher/postprocessor.py
Original file line number Diff line number Diff line change
Expand Up @@ -137,10 +137,10 @@ def _insert_links(self, playbook_node: PlaybookNode):
display.vv(f"Inserting links structure for the playbook '{playbook_node.name}'")
links_structure = playbook_node.links_structure()

for node_id, node_links in links_structure.items():
for node, node_links in links_structure.items():
# Find the group g with the specified id
xpath_result = self.root.xpath(
f"ns:g/*[@id='{node_id}']", namespaces={"ns": SVG_NAMESPACE}
f"ns:g/*[@id='{node.id}']", namespaces={"ns": SVG_NAMESPACE}
)
if xpath_result:
element = xpath_result[0]
Expand All @@ -151,7 +151,7 @@ def _insert_links(self, playbook_node: PlaybookNode):
"link",
attrib={
"target": link.id,
"edge": f"edge_{counter}_{node_id}_{link.id}",
"edge": f"edge_{counter}_{node.id}_{link.id}",
},
)
)
Expand Down
2 changes: 1 addition & 1 deletion ansibleplaybookgrapher/utils.py
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,7 @@
from collections import defaultdict
from itertools import chain
from operator import methodcaller
from typing import Tuple, List, Dict, Any
from typing import Tuple, List, Dict, Any, Set

from ansible.errors import AnsibleError
from ansible.module_utils._text import to_text
Expand Down
Loading