[docs]defrecursive_conversion(true_func,false_func):"""Recursively apply true_func to the nodes and false_func to the rest of the objects in a container of nodes. Container of nodes are tuple, list, dict, set, and NodeContainer. Args: true_func (callable): the function to be applied to the nodes. false_func (callable): the function to be applied to the rest of the objects. """deffunc(obj):ifisinstance(obj,Node):# base casereturntrue_func(obj)elifisinstance(obj,tuple):returntuple(func(x)forxinobj)elifisinstance(obj,list):return[func(x)forxinobj]elifisinstance(obj,dict):return{k:func(v)fork,vinobj.items()}elifisinstance(obj,set):return{func(x)forxinobj}elifisinstance(obj,NodeContainer):output=copy.copy(obj)fork,vinobj.__dict__.items():setattr(output,k,func(v))returnoutputelse:returnfalse_func(obj)returnfunc
# TODO to test it and clean up the code
[docs]defapply_op(op,output,*args,**kwargs):"""A broadcasting operation that applies an op to container of Nodes. Args: op (callable): the operator to be applied. output (Any): the container to be updated. *args (Any): the positional inputs of the operator. **kwargs (Any): the keyword inputs of the operator. """inputs=list(args)+list(kwargs.values())containers=[xforxininputsifnotisinstance(x,Node)]iflen(containers)==0:# all inputs are Nodes, we just apply opreturnop(*args,**kwargs)# # there is at least one container# output = copy.deepcopy(containers[0]) # this would be used as the template of the outputdefadmissible_type(x,base):returntype(x)istype(base)orisinstance(x,Node)assertall(admissible_type(x,output)forxininputs)# All inputs are either Nodes or the same type as outputifisinstance(output,list)orisinstance(output,tuple):assertall(isinstance(x,Node)orlen(output)==len(x)forxininputs),f"output {output} and inputs {inputs} are of different lengths."forkinrange(len(output)):_args=[xifisinstance(x,Node)elsex[k]forxinargs]_kwargs={kk:vvifisinstance(vv,Node)elsevv[k]forkk,vvinkwargs.items()}output[k]=apply_op(op,output[k],*_args,**_kwargs)ifisinstance(output,tuple):output=tuple(output)elifisinstance(output,dict):fork,vinoutput.items():_args=[xifisinstance(x,Node)elsex[k]forxinargs]_kwargs={kk:vvifisinstance(vv,Node)elsevv[k]forkk,vvinkwargs.items()}output[k]=apply_op(op,output[k],*_args,**_kwargs)elifisinstance(output,NodeContainer):# this is a NodeContainer object instancefork,vinoutput.__dict__.items():_args=[xifisinstance(x,Node)elsegetattr(x,k)forxinargs]_kwargs={kk:vvifisinstance(v,Node)elsegetattr(vv,k)forkk,vvinkwargs.items()}new_v=apply_op(op,v,*_args,**_kwargs)setattr(output,k,new_v)else:passreturnoutput