From 4a518881a83e4f94da37cd22f880bd312ba79d90 Mon Sep 17 00:00:00 2001 From: Gytha Ogg Date: Wed, 13 Nov 2024 23:50:58 +0100 Subject: [PATCH] feat(relations): graph visualisation of relations Adds a Graph tab to the tabbed relations template. Graph tab shows a graphical representation of the all relations that include the current object. Nodes represent objects, each entity type deterministically gets a unique colour. All edges are directed from the object to the related objects. --- .../templates/relations/graph_relations.html | 10 +++ .../list_relations_include_tabs.html | 4 ++ apis_core/relations/templatetags/relations.py | 71 +++++++++++++++++++ 3 files changed, 85 insertions(+) create mode 100644 apis_core/relations/templates/relations/graph_relations.html diff --git a/apis_core/relations/templates/relations/graph_relations.html b/apis_core/relations/templates/relations/graph_relations.html new file mode 100644 index 000000000..db0f863ca --- /dev/null +++ b/apis_core/relations/templates/relations/graph_relations.html @@ -0,0 +1,10 @@ +{% load relations %} +{% relations_graph relations target as graph %} +{% if graph.error %}{% endif %} +{% if graph.svg %} +
+
{{ graph.svg|safe }}
+
+{% endif %} diff --git a/apis_core/relations/templates/relations/list_relations_include_tabs.html b/apis_core/relations/templates/relations/list_relations_include_tabs.html index 9afc63de2..37935c5bd 100644 --- a/apis_core/relations/templates/relations/list_relations_include_tabs.html +++ b/apis_core/relations/templates/relations/list_relations_include_tabs.html @@ -17,6 +17,9 @@ href="#reltab_{{ object.id }}_{{ target.name }}">{{ target.name | title }} {% endfor %} +
@@ -25,6 +28,7 @@ {% for target in possible_targets %}
{% include "relations/list_relations.html" %}
{% endfor %} +
{% include "relations/graph_relations.html" %}
diff --git a/apis_core/relations/templatetags/relations.py b/apis_core/relations/templatetags/relations.py index f8a14576a..d7230a32d 100644 --- a/apis_core/relations/templatetags/relations.py +++ b/apis_core/relations/templatetags/relations.py @@ -91,3 +91,74 @@ def relations_verbose_name_listview_url(): for relation in relation_classes } return sorted(ret.items()) + + +@register.simple_tag(takes_context=True) +def relations_graph(context, relations, target=None): + + from django.conf import settings + + if "apis_core.relations" not in settings.INSTALLED_APPS: + return { + "error": "This has currently been configured only for apis_core.relations." + } + + import hashlib + + from pydot import Dot, Edge, Node + + def get_colour(entity_type): + # TODO: Could this be better, by choosing from a proper matplotlib palette? + hash_value = int(hashlib.md5(entity_type.encode("utf-8")).hexdigest(), 16) + r = (hash_value >> 16) & 0xFF + g = (hash_value >> 8) & 0xFF + b = hash_value & 0xFF + + # Format it into a hex color string + return f"#{r:02x}{g:02x}{b:02x}" + + def get_node(obj): + node = Node( + obj.pk, + label="", # use tooltip instead + shape="circle", + style="filled", + splines=False, + fillcolor=get_colour(obj.__class__.__name__), + tooltip=str(rel.obj), + ) + node.set_tooltip(str(obj)) + node.set_URL(obj.get_absolute_url()) + return node + + graph = Dot( + graph_type="digraph", + layout="sfdp", + splines=True, + center=True, + # repulsiveforce=2.0, # + # K=1, + ) + + for rel in relations: + graph.add_node(get_node(rel.subj)) + graph.add_node(get_node(rel.obj)) + + src = rel.subj.pk if rel.forward else rel.obj.pk + dest = rel.obj.pk if rel.forward else rel.subj.pk + edge_label = rel.name() if rel.forward else rel.reverse_name() + + e = Edge( + src, + dest, + label=edge_label, + ) + graph.add_edge(e) + + graph_data = {} + try: + graph_data["dot"] = graph.to_string() + graph_data["svg"] = graph.create_svg().decode() + except Exception as e: + graph_data["error"] = f"{str(e)} occured while creating graph." + return graph_data