Apparat

An ergomonic, flexible and light weight behavioral state machine

Notable features:

Architecture and usage

Types you provide

* The data within the state types is exclusively accessible in the respective state. It get's dropped on transition and should be rather small and cheap to move. If it's not, consider putting it in a Box for better performance and memory efficiency.

Note: All provided types must implement the Debug trait.

Entities that are generated

Traits

ApparatState<StateWrapper>

This trait must be implemented for all state types. The handle method is the only one that doesn't provide a default impl and must be written manually. There are two other methods in the trait that form an initialization mechanism: After initializing the Apparat using the new methods or after handling any event, init is called on the new state, until its is_init method returns false. This way multiple transitions can be triggered by a single event. This happens in a while loop without stressing the recursion limit. If a state doesn't need that initialization, the methods can be ignored so their default implementation is getting used.

The handle method returns a Handled<StateWrapper> struct where StateWrapper is the state wrapper enum that apparat generated for us. Handled<StateWrapper> just combines this enum with the provided output type. If this output type implements the Default trait, the StateWrapper can be turned into a Handled<StateWrapper> with the default output value using into(). This is demonstrated and commented in the example below.

TransitionFrom<OtherState, ContextData>

The TransitionFrom trait can be used to define specific transitions between states. The TransitionTo trait is then automatically implemented, just to be able to call the transition method using the turbofish syntax. This mechanism is similar to From and Into in the rust standard library. The difference to std::convert::From is that TransitionFrom can also mutate the provided context as a side effect. The usage of these traits is optional but recommended.

Wrap

The Wrap<StateWrapper> trait provides a wrap method for convenience to turn an individual state into a StateWrapper. This is implemented automatically for your states using a blanket impl.

Minimal example

For a slightly more complete example, have a look at counter.rs in the examples directory.

```rust use apparat::prelude::*;

// Define the necessary types // --------------------------

// States

[derive(Debug, Default)]

pub struct StateA;

[derive(Debug, Default)]

pub struct StateB { events: usize, // just an example of a state holding exclusive data }

// Context

// Data that survives state transitions and can be accessed in all states

[derive(Debug, Default)]

pub struct ContextData { toggled: usize, }

// Auto-generate the state wrapper and auto-implement traits // ---------------------------------------------------------

// Since we are only handling one kind of event in this example and we don't // care about values being returned when events are handled, we are just using // the unit type for event and output build_wrapper! { states: [StateA, StateB], wrapper: MyStateWrapper, // this is just an identifier we can pick context: ContextData, event: (), output: (), }

// Define transitions // ------------------

impl TransitionFrom for StateA { fn transitionfrom(prev: StateB, ctx: &mut ContextData) -> Self { println!("B -> A | toggled: {}", ctx.toggled); StateA::default() } }

impl TransitionFrom for StateB { fn transitionfrom(prev: StateA, ctx: &mut ContextData) -> Self { println!("A -> B | toggled: {}", ctx.toggled); StateB::default() } }

// Implement the ApparateState trait for all states // --------------------------------------------------

impl ApparatState for StateA { fn handle(self, _event: (), ctx: &mut ContextData) -> Handled { println!("A handles event | toggled: {}", ctx.toggled); // increase toggled value ctx.toggled += 1; self.transition::(ctx) .wrap() // turn the StateB into a MyStateWrapper .into() // turn it into a Handled<MyStateWrapper> // Using into assumes you want to use the default value // of your output type. It only works if your output type // implements the Default trait in the first place. } }

impl ApparatState for StateB { fn handle(mut self, _event: (), ctx: &mut ContextData) -> Handled { println!("B handles event | toggled: {}", ctx.toggled); self.events += 1; if self.events > 2 { self.transition::(ctx).wrap().into() } else { self.wrap().into() } } }

// Run the machine // ---------------

fn main() { let mut apparat = Apparat::new(StateA::default().wrap(), ContextData::default());

// Handle some events
for _ in 0..10 {
    apparat.handle(());
}

} ```