Build a model graph#
A directed acyclic graph (DAG) is a directed graph without any cycles. The model graphs in mmodel are based on the DAG, where each node represents an execution step, and each edge represents the data flow from one callable to another. DAG structure allows us to create model graphs with nonlinear nodes. dDefine a graph ————–
The Graph
class is the main graph class to establish a model graph.
The class inherits from networkx.DiGraph
, which is compatible with all
NetworkX operations
(see documentation).
To create and modify the graph,
see the documentation for adding
nodes
and adding edges.
Aside from the NetworkX operations,
mmodel provides add_grouped_edge
and add_grouped_edges_from
to add edges.
from mmodel import Graph, Node
G = Graph()
G.add_grouped_edge(['a', 'b'], 'c')
# equivalent to
# G.add_edge('a', 'b')
# G.add_edge('a', 'c')
Similarly, with multiple grouped edges
grouped_edges = [
(['a', 'b'], 'c'),
('c', ['d', 'e']),
]
G = Graph()
G.add_grouped_edges_from(grouped_edges)
>>> G # no name is given
<mmodel.graph.Graph>
>>> print(G)
Graph with 5 nodes and 4 edges
For linking the node object to the node, two methods are provided:
set_node_object
and set_node_objects_from
.
The latter accepts a list of node objects.
def add(x, y):
"""The sum of x and y."""
return x + y
def subtract(x, y):
"""The difference between x and y."""
return x - y
def multiply(x, y):
"""The product of x and y."""
return x * y
G = Graph()
G.add_grouped_edge(['a', 'b'], 'c')
G.set_node_object(Node(name="a", func=add, output="z"))
The node object can be accessed using the get_node_object
method.
node_a = G.get_node_object("a")
>>> print(node_a)
a
add(x, y)
return: z
functype: function
The sum of x and y.
# or with multiple node objects
# both nodes add input values but outputs in different parameter
node_objects = [
Node("a", add, output="z"),
Node("b", subtract, output="m"),
Node("c", multiply, output="n", inputs=["z", "m"]),
]
G.set_node_objects_from(node_objects)
>>> node_b = G.get_node_object("b")
>>> print(node_b)
b
subtract(x, y)
return: m
functype: function
The difference between x and y.
The object is stored as a node attribute, and the function signature (inspect.Signature) is stored. The parameter values are converted to signature objects.
graph Methods#
visualization#
The graph can be visualized or saved using the visualize
method.
G.visualize()
# or with a filename
G.visualize(outfile="graph.png")
name and docstring#
The name and graph string behaves as the networkx graphs. To add the name to the graph:
# during graph definition
G = Graph(name="Graph Example")
# after definition
# G.graph['name'] = 'ModelGraph Example'
>>> G
<mmodel.graph.Graph 'Graph Example'>
>>> print(G)
Graph named 'Graph Example' with 0 nodes and 0 edges
mutability#
The graph object is mutable. A shallow or deepcopy might be needed to create a copy of the graph.
G.copy() # shallow copy
G.deepcopy() # deep copy
For more ways to interact with Graph
and networkx.graph
see
graph reference.