Directed Acyclic Graph implementation for Django & Postgresql
Project description
Django & Postgresql-based Directed Acyclic Graphs
The main distinguishing factor for this project is that it can retrieve entire sections of a graph in a single query. The trade off is portability: it uses Postgres Common Table Expressions (CTE) to achieve this and is therefore not compatible with other databases.
NOTE: Not all methods which would benefit from CTEs use them yet. NOTE: This project is a work in progress. While functional, it is not optimized.
Currently provides numerous methods for retrieving nodes, and a few for retrieving edges within the graph.
Example:
models.py
from django.db import models
from django_postgresql_dag.models import node_factory, edge_factory
class GroupedEdgeSet(models.Model):
"""Set of NetworkEdges belonging to a specific Groupe
Serves as the primary 'Grouped' object
This can be thought of as a "complex edge"
"""
name = models.CharField(max_length=100)
class GroupedType(models.TextChoices):
GROUPED_TYPE_A = "A", _("A")
GROUPED_TYPE_B = "B", _("B")
GROUPED_TYPE_C = "C", _("C")
grouped_type = models.CharField(
_("Grouped Type"),
choices=GroupedType.choices,
default=GroupedType.GROUPED_TYPE_A,
max_length=20,
help_text=_("What type of grouping is this?"),
)
class NetworkEdge(edge_factory("NetworkNode", concrete=False)):
name = models.CharField(max_length=100)
grouped_edge_set = models.ForeignKey(
GroupedEdgeSet,
on_delete=models.CASCADE,
null=True,
blank=True,
related_name="grouped_network_edges",
)
def __str__(self):
return self.name
def save(self, *args, **kwargs):
self.name = f"{self.parent.name} {self.child.name}"
super().save(*args, **kwargs)
class NetworkNode(node_factory(NetworkEdge)):
name = models.CharField(max_length=100)
def __str__(self):
return self.name
Shell
from myapp.models import GroupedEdgeSet, NetworkNode, NetworkEdge
root = NetworkNode.objects.create(name="root")
a1 = NetworkNode.objects.create(name="a1")
a2 = NetworkNode.objects.create(name="a2")
a3 = NetworkNode.objects.create(name="a3")
b1 = NetworkNode.objects.create(name="b1")
b2 = NetworkNode.objects.create(name="b2")
b3 = NetworkNode.objects.create(name="b3")
b4 = NetworkNode.objects.create(name="b4")
c1 = NetworkNode.objects.create(name="c1")
c2 = NetworkNode.objects.create(name="c2")
root.add_child(a1)
root.add_child(a2)
a3.add_parent(root) # Nodes can be added in either direction
b1.add_parent(a1)
a1.add_child(b2)
a2.add_child(b2)
a3.add_child(b3)
a3.add_child(b4)
b3.add_child(c2)
b3.add_child(c1)
b4.add_child(c1)
# Get a couple of the automatically generated edges to work with below
e1 = NetworkEdge.objects.first()
e2 = NetworkEdge.objects.last()
# Work with the graph
# Descendant methods which return ids
root.descendant_ids()
root.self_and_descendant_ids()
root.descendants_and_self_ids()
# Descendant methods which return a queryset
root.descendants()
root.self_and_descendants()
root.descendants_and_self()
# Ancestor methods which return ids
c1.ancestor_ids()
c1.ancestor_and_self_ids()
c1.self_and_ancestor_ids()
# Ancestor methods which return a queryset
c1.ancestors()
c1.ancestors_and_self()
c1.self_and_ancestors()
# Get the node's clan (all ancestors, self, and all descendants)
b3.clan_ids()
b3.clan()
# Get all roots or leaves associated with the node
b3.get_roots()
b3.get_leaves()
# Get the nodes at the start or end of an edge
e1.parent
e1.child
e2.parent
e2.child
# Edge-specific Manager methods
NetworkEdge.objects.descendants(b3)
NetworkEdge.objects.ancestors(b3)
NetworkEdge.objects.clan(b3)
NetworkEdge.objects.path(root, c1)
Credits:
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file django-postgresql-dag-0.0.5.tar.gz.
File metadata
- Download URL: django-postgresql-dag-0.0.5.tar.gz
- Upload date:
- Size: 10.2 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/3.2.0 pkginfo/1.5.0.1 requests/2.23.0 setuptools/50.3.1 requests-toolbelt/0.9.1 tqdm/4.48.0 CPython/3.8.2
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
54231a0d477afb626d75dd4ca4baf89947a7d33fada1796ff1c15614c239ed51
|
|
| MD5 |
08f854c58c076b7df6921d5f747b5704
|
|
| BLAKE2b-256 |
975e306249540860baade29b03d3185fe6fd9ae9ea9bc097df39dff80e8f188b
|
File details
Details for the file django_postgresql_dag-0.0.5-py3-none-any.whl.
File metadata
- Download URL: django_postgresql_dag-0.0.5-py3-none-any.whl
- Upload date:
- Size: 10.3 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/3.2.0 pkginfo/1.5.0.1 requests/2.23.0 setuptools/50.3.1 requests-toolbelt/0.9.1 tqdm/4.48.0 CPython/3.8.2
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
e5610257e27449fac79e0c0bcaf7f71185cdf4dbadf32fd160dd20981b6e5327
|
|
| MD5 |
4edc75edd99014a5587a8f459e67098f
|
|
| BLAKE2b-256 |
16949d6d6b62f7b4640a33ecc87cd7ce46ec62cc3036831d4a366249064a9865
|