diff options
Diffstat (limited to 'lib/automata.py')
-rwxr-xr-x | lib/automata.py | 43 |
1 files changed, 41 insertions, 2 deletions
diff --git a/lib/automata.py b/lib/automata.py index 35be2fd..2c21389 100755 --- a/lib/automata.py +++ b/lib/automata.py @@ -5,6 +5,7 @@ from utils import is_numeric import itertools import numpy as np import json +import queue import yaml @@ -783,11 +784,49 @@ class PTA: ret.append((transition, arguments, param)) yield ret + def bfs(self, depth: int = 10, orig_state: str = 'UNINITIALIZED', transition_filter=None, state_filter=None): + """ + Return a generator object for breadth-first search of traces starting at orig_state. + + Each trace consists of a list of + tuples describing the corresponding transition and (if enabled) + arguments and parameters. When both with_arguments and with_parameters + are True, each transition is a (Transition object, argument list, parameter dict) tuple. + + Note that the parameter dict refers to parameter values _after_ + passing the corresponding transition. Although this may seem odd at + first, it is useful when analyzing measurements: Properties of + the state following this transition may be affected by the parameters + set by the transition, so it is useful to have those readily available. + + A trace is (at the moment) a list of alternating states and transition, both starting and ending with a state. + Does not yield the no-operation trace consisting only of `orig_state`. If `orig_state` has no outgoing transitions, the output is empty. + + :param orig_state: initial state for breadth-first search + :param depth: search depth, default 10 + :param transition_filter: If set, only follow a transition if transition_filter(transition object) returns true. Default None. + :param state_iflter: If set, only follow a state if state_filter(state_object) returns true. Default None. + """ + state_queue = queue.Queue() + state_queue.put((list(), self.state[orig_state])) + + while not state_queue.empty(): + trace, state = state_queue.get() + if len(trace) > depth: + return + if state_filter is None or state_filter(state): + for transition in state.outgoing_transitions.values(): + if transition_filter is None or transition_filter(transition): + new_trace = trace.copy() + new_trace.append((transition,)) + yield new_trace + state_queue.put((new_trace, transition.destination)) + def dfs(self, depth: int = 10, orig_state: str = 'UNINITIALIZED', param_dict: dict = None, with_parameters: bool = False, **kwargs): """ Return a generator object for depth-first search starting at orig_state. - :param depth: search depth + :param depth: search depth, default 10 :param orig_state: initial state for depth-first search :param param_dict: initial parameter values :param with_arguments: perform dfs with argument values @@ -795,7 +834,7 @@ class PTA: :param trace_filter: list of lists. Each sub-list is a trace. Only traces matching one of the provided sub-lists are returned. :param sleep: sleep duration between states in us. If None or 0, no sleep pseudo-transitions will be included in the trace. - The returned generator emits traces. Each trace consts of a list of + Each trace consists of a list of tuples describing the corresponding transition and (if enabled) arguments and parameters. When both with_arguments and with_parameters are True, each transition is a (Transition object, argument list, parameter dict) tuple. |