Source code for mmodel.node
from mmodel.signature import convert_func, get_node_signature, get_parameters
from mmodel.metadata import nodeformatter
from mmodel.utility import (
modify_func,
parse_functype,
EditMixin,
ReprMixin,
)
from inspect import signature
[docs]
class Node(EditMixin, ReprMixin):
"""A node class that formats node function and metadata."""
def __init__(
self,
name,
func,
inputs=None,
output=None,
modifiers=None,
**kwargs,
):
# static
self.name = self.__name__ = name
self.output = output
self._inputs = inputs or []
self._modifiers = modifiers or []
self.func = func
self.functype = parse_functype(func)
self.doc = func.__doc__
self._base_func = self.convert_func(func, self._inputs)
self._node_func = modify_func(self._base_func, self._modifiers)
# kwargs can overwrite values like doc, functype, etc.
for key, value in kwargs.items():
setattr(self, key, value)
@property
def node_func(self):
"""Return node function, the node function cannot be reset."""
return self._node_func
@property
def __signature__(self):
"""Node signature for inspection."""
return signature(self.node_func)
@property
def signature(self):
"""Return signature."""
return self.__signature__
[docs]
def convert_func(self, func, inputs_list):
"""Convert function to a node function.
To replace the positional or positional-or-keyword parameters,
the list arglist needs to be defined. The user can select
desired keyword-only parameters after "*" in the input. To replace the
keyword-only parameters, a custom function needs to be defined
to replace the function.
For functions that already fit the criteria and have no argument
list, we still wrap the function. The overhead is minimal.
"""
base_params = get_parameters(func)
node_sig = get_node_signature(base_params, inputs_list)
return convert_func(func, node_sig)
@property
def inputs(self):
"""Return a copy of inputs."""
return self._inputs.copy()
@property
def modifiers(self):
"""Return a copy of modifiers."""
return self._modifiers.copy()
[docs]
def edit(self, **kwargs):
"""Edit node. A new node object is created."""
# if the the function is updated, the inputs are reset
if "func" in kwargs:
kwargs["inputs"] = kwargs.get("inputs", None)
edit_dict = self.edit_dict
edit_dict.update(kwargs)
return self.__class__(**edit_dict)
def __call__(self, *args, **kwargs):
"""Node function callable.
The ``node_func`` method is used internally. The ``__call__`` method
is used for external calls.
"""
bound = self.signature.bind(*args, **kwargs)
return self.node_func(**bound.arguments)
def __str__(self):
return nodeformatter(self)