::\

A state machine builder for creating deterministic state machines

crates.io Docs

Automafish is a state machine builder for turning [nondeterministic finite automata] into [deterministic finite automata]. Its primary use case is driving the selector logic of [metamorfish] to enable mutation of [protofish] values in [proxide].

```rust use std::iter::FromIterator; use automafish::{Builder, State, Transition, Criteria, Condition};

let mut builder : Builder, &mut dyn FnMut(&mut char)> = Builder::new();

// Set up an automata that capitalizes first character of a word. let mut uppercase = |mut c: &mut char| { c.makeasciiuppercase(); }; let waitnotspace = builder.createinitialstate(); let firstnotspace = builder.addstate(State::withaction(&mut uppercase)); let waitforspace = builder.add_state(State::new());

builder.addtransition(Transition::new( waitnotspace, Condition::Is(vec![' ']), waitnotspace)); builder.addtransition(Transition::new( waitnotspace, Condition::Not(vec![' ']), firstnotspace)); builder.addtransition(Transition::new( firstnotspace, Condition::Not(vec![' ']), waitforspace)); builder.addtransition(Transition::new( firstnotspace, Condition::Is(vec![' ']), waitnotspace)); builder.addtransition(Transition::new( waitforspace, Condition::Not(vec![' ']), waitforspace)); builder.addtransition(Transition::new( waitforspace, Condition::Is(vec![' ']), waitnotspace));

// Set up an automata that counts all exclamation marks. // This automata modifies a value outside the state machine. let mut exclamations = 0; let mut exclamationcounter = |: &mut char| { exclamations += 1; }; let waitexclamation = builder.createinitialstate(); let exclamation = builder.addstate(State::withaction(&mut exclamationcounter));

builder.addtransition(Transition::new( waitexclamation, Condition::Any, waitexclamation)); builder.addtransition(Transition::new( wait_exclamation, Condition::Is(vec!['!']), exclamation));

// Build the machine. let mut machine = builder.build();

// Execute the machine on an input string. let mut currentstate = machine.start(); let mut input : Vec = "hello world! this is rust!".chars().collect(); for i in &mut input { currentstate = machine.stepandexecutemut(currentstate, i); }

let output : String = String::from_iter(input);

asserteq!("Hello World! This Is Rust!", output); asserteq!(2, exclamations); ```

Related crates

In addition there seem to be other generic state machine crates that I discovered while looking into options. If you ended up here looking for one, you might be interested in the following crates as well. Note that I haven't personally tested any of these crates.

Goals

The only current goal for Automafish is providing a somewhat efficient state machine implementation for Metamorfish.

Use cases outside of those required by Metamorfish are definitely not out of scope, but currently it's unlikely I'll be adding new features to Automafish that Metamorfish doesn't need on my own. Feature and pull requests are definitely welcome though! It's just that without any other priorities, Metamorfish and Proxide itself are more interesting to me. :)

Maybe in the future

Defining custom Criteria is somewhat painful currently with the need to resolve intersections and differences for such criteria. I would simplify the process of handling Criteria, but currently I don't really know how to build the powersets without (at the very least) the intersection handling.

Then again, I haven't really looked into available algorithms for such use cases.