Use modifiers and shortcuts#

Traditionally, to modify a component of a pre-defined model, one has to rewrite the code. The approach is error-prone and not scalable. mmodel uses DAG to create modular models that allow easy modification of nodes and models post-definition using modifiers. A modifier is a decorator that can modify the function of a node or the model itself. To go step further, mmodel also provides shortcuts, that is designed to apply directly to a model. The shortcuts applies modifiers to the model or nodes in the model.

available modifiers and shortcuts#

Available modifiers#

Modifier

Module

Description

loop_input

mmodel.modifier

Modify function to iterate one given parameter.

zip_loop_inputs

mmodel.modifier

Modify function to iterate the parameters pairwise.

profile_time

mmodel.modifier

Profile the execution time of a function.

print_inputs

mmodel.modifier

Print the inputs of the function with units.

print_output

mmodel.modifier

Print the outputs of the function with units.

loop_shortcut

mmodel.shortcut

Loop over a parameter during the experiment execution.

print_shortcut

mmodel.shortcut

Apply print_inputs and print_output shortcuts to individual nodes that print out intermediate variable values during node execution.

apply a modifier to a node or model#

For example, to add a loop modifier to the node “add”:

from mmodel import Graph, Node
G = Graph()


def add(a, b):
    """The sum of a and b."""
    return a + b


def squared(c):
    """The squared value of c."""
    return c**2


from mmodel.modifier import loop_input

G.add_edge("add", "squared")
# set object without modifiers
G.set_node_object(Node("squared", squared, "d"))

# set object with modifier
G.set_node_object(Node("add", add, "c", modifiers=[loop_input(parameter='b')]))

# post modification
# a new copy of the graph is created

H = G.edit_node("add", modifiers=[loop_input('b')])

Similarly, use the modifiers argument to define model modifiers.

shortcut usage#

loop shortcut#

The loop shortcut works by locating the first dependency of the parameter in the graph. It then creates a subgraph that contains all the nodes that depend on the parameter. A new experiment is created with the subgraph as a single node, and the node function is an experiment model. When multiple parameters need to be looped, the user needs to inspect the order of the parameter appearance in the graph to achieve an optimal result.

For example, a graph of \(G=\{V=\{A, B, C\}, E=\{(A, B), (B, C)\}\}\):

A -> B -> C

A(a, b)
B(c, d)
C(e, f)

The optimal way to loop c and e is to define the loop of parameter e in node C first and then define the loop of parameter c in node B second. If the order given is reversed, both parameters c and e are looped at node B level. The reason for the behavior is that when loop c is created, the graph:

A -> BC

A(a, b)
BC(c, d, e, f)

As a result, the subsequent loop definition only recognizes the subgraph node BC and loop the node instead.

Note

For a two-loop system, the optimal order can always be resolved. However, looping more than three parameters, the optimal order may not be resolved. Therefore, the design decision is made for the user to define the loop order.

modifier and shortcut chaining#

Because the modifiers are decorators, they can be chained. The modifiers in the list are applied in the order of the modifiers argument. The output of a shortcut is a model, therefore, we can apply multiple shortcuts to a single model.

See modifier and shortcut API reference for creating custom modifiers and shortcuts. See modifier reference and shortcut reference for all available modifiers.