pastafarian v1.2.1
pastafarian
A tiny event emitter-based finite state machine
(https://badge-size.herokuapp.com/orbitbot/pastafarian/master/pastafarian.min.js?color=yellow&label=minfied size)
Grab a lightweight event emitter implementation, add some logic to track states and Voilà! A tiny finite state machine implementation at little less than 550 bytes minfied and gzipped. pastafarian is implemented as a UMD module, so it should run in most javascript setups.

Features
- probably the smallest FSM on the block in javascript-land
- simple but powerful API
- no external dependencies
- synchronous state transitions only (async transitions are actually waiting states... but have a look at hendersonfor an almost identical approach with promises)
- well below 100 LOC, small enough to read and understand immediately
Example
var state = new StateMachine({
  initial : 'start',
  states  : {
    start : ['end', 'start'],
    end   : ['start']
  }
});
state.on('*', function(prev, next) {
  console.log('State changed from ' + prev + ' to ' + next);
});
state
  .on('before:start', function(prev, param) {
    console.log('Reset with param === "foo": ' + param === 'foo');
  })
  .on('after:start', function(next) {
    console.log('Going to ' + next);
  })
  .on('end', function(prev, param) {
    console.log('Now at end, 2 + 2 = ' + param);
  });
state.go('end', 2 + 2);
state.reset = state.go.bind(state, 'start');
state.reset('foo');Installation
Right click to save or use the URLs in your script tags
or use
$ npm install pastafarian
$ bower install pastafarianIf you're using pastafarian in a browser environment, the constructor is attached to the StateMachine global.
Usage
The StateMachine global or the pastafarian module is a constructor for a finite-state machine. The constructor expects a single configuration object:
| field | type | functionality | 
|---|---|---|
| initial | string | the starting state of the state machine | 
| states | object | keys are state names, values are arrays of valid states to transition to <state name> : ['<state>', '...'] | 
| error | function | optional, function that handles errors in state transition callbacks or illegal state transitions | 
A simple state machine that describes a traffic light might be defined as
var StateMachine = require('pastafarian');
var trafficLight = new StateMachine({
  initial : 'red',
  states  : {
    green  : ['yellow'],
    yellow : ['green', 'red'],
    red    : ['red'],
  },
  error : console.error.bind(console, 'Error: ')
});... which will create a state machine like this diagram:

State machine API
A state machine var fsm = new StateMachine(config) will have
Methods:
fsm.bind(eventName, callback or [callbacks]) ⇒ fsm
Attaches a single callback or an array of [callbacks] to be called whenever eventName is triggered by a state transition. See the Event callback API for all possible events for a single transition.
fsm.unbind(eventName, callback) ⇒ fsm
De-registers callback so it will not be triggered for eventName. Previously registered callbacks must be named values for this to have an effect, if a callback was defined as an anonymous function this method will silently fail.
fsm.on(eventName, callback or [callbacks]) ⇒ fsm
Synonym for fsm.bind.
fsm.go(state /* ...args */) ⇒ fsm
Transitions the state machine to state and causes any registered callbacks for this transition (including before:, after: and wildcard callbacks) to be triggered. All parameters after state are passed on to each callback along with the states involved in the transition, see the Event callback API for the exact signatures.
All methods as well as the constructor return the state machine itself, and are therefore chainable.
Fields:
fsm.transitions : object
An object where the keys are state names, and the values of each key is an array of the states that can be transitioned to from this state, as defined by config.states.
fsm.current : string
Tracks the current state, the starting value is config.initial. The value changes during state transitions, see Event callback API.
fsm.error : function
The if defined, the function from config.error, see Error handling.
If you need to change the functionality or state without going through transitions, these fields can be edited as required. See the section on extending below for some ideas.
Event callback API
Callbacks triggered on state transitions can be registered with fsm.on or fsm.bind:
fsm.on(eventName, function() {
  // do something
});Every call to fsm.go will trigger all callbacks registered for the states involved in the transition according to the following semantics:
Assuming that
- fsmis in state- PREVIOUS_STATE, and
- fsmcan transition from- PREVIOUS_STATEto- NEXT_STATE,
- a call fsm.go(NEXT_STATE, /* ...args */)
- will trigger all callbacks registered for eventwith
| event | signature | fsm.current | 
|---|---|---|
| after:PREVIOUS_STATE | function(next /* ...args */) {} | PREVIOUS_STATE | 
| before:NEXT_STATE | function(previous /* ...args */) {} | PREVIOUS_STATE | 
| NEXT_STATE | function(previous /* ...args */) {} | NEXT_STATE | 
| * | function(previous, next /* ...args */) {} | NEXT_STATE | 
- in all callback signatures, nextisNEXT_STATEandpreviousisPREVIOUS_STATE
- before:NEXT_STATEand- NEXT_STATEdiffer only in that- fsm.currenthas changed when the callback is being executed
- the wildcard callback (*) is triggered on every successful transition, but not if a transition between states is not possible
Also note that
- all arguments to fsm.goafter the first are always passed to the callbacks, according to the above signatures
- any number of callbacks can be registered for any event
- callbacks will be triggered in the order registered
- fsm.onand- fsm.bindwill accept any valid object key as the first parameter and will perform no checks to ensure a matching state is defined, so watch out for typos
- fsm.errorsignificantly affects how- pastafarianworks in the case of thrown exceptions in callbacks, see Differences in functionality if- fsm.erroris defined or not
So, given a basic state machine:
var state = new StateMachine({
  initial : 'start',
  states  : {
    start : ['end', 'start'],
    end   : ['start']
  }
});... the following callbacks may be triggered as the state changes
state.on('*',            function() { });
state.on('before:start', function() { });
state.on('start',        function() { });
state.on('after:start',  function() { });
state.on('before:end',   function() { });
state.on('end',          function() { });
state.on('after:end',    function() { });Error handling
If defined, the fsm.error function will be called in two separate cases:
- when trying to perform an invalid state transition
- when a callback defined for a transition throws an error
The signature of this function is
function errorHandler(error, prev, next /* ...params */) {
  if (error.name === 'IllegalTransitionException') {
    console.log(error.message);
    // prev is fsm.current
    // next is the transition attempted, eg. fsm.go(next, ...)
    // params are any other parameters to fsm.go, eg. fsm.go(next, param1, param2 ...)
  } else {
    // error is whatever was thrown in the transition callback that caused the error
    // prev, next and other arguments will be undefined
  }
}If the error handler function is not defined, any calls to fsm.go may throw errors or exceptions for the above reasons and can be caught similarly using try/catch blocks.
Differences in functionality if fsm.error is defined or not
The existance of fsm.error has a significant impact on functionality:
- if fsm.erroris not defined, an uncaught exception in a callback will stop execution and subsequent callbacks will not be triggered, potentially leaving your application in an undefined state if you are relying on the side-effects of a certain callback being applied.fsm.currentmay also still be in the previous state, depending on which events the callbacks were registered to
- if fsm.erroris defined, an uncaught exception in a callback will trigger the error handler and stop further execution of the code inside that callback, but all other callbacks will be triggered and the state transition will be completed, which may also cause cause problems with unfinished side-effects
IllegalTransitionException
pastafarian defines a custom exception which is generated when the transitions array of the current state doesn't contain the state passed to fsm.go:
- name: IllegalTransitionException
- message: Transition from <current> to <next> is not allowed
- prev : <current>
- attempt : <next>
The exception is generated inside the library, but in modern environments it should contain a stacktrace that allows you to track which line caused the exception.
Extending
pastafarian omits most safety checks and a larger API in favor of size, but can be extended in different ways to support different usage patterns and semantics.
If you find yourself often needing to check the current state or valid transitions, these helpers might provide a nicer interface:
// is parameter state a valid transition from the current state?
fsm.can = function(state) {
  return fsm.transitions[fsm.current].indexOf(state) > -1;
};
// is parameter state an invalid transition from the current state?
fsm.cannot = function(state) {
  return fsm.transitions[fsm.current].indexOf(state) === -1;
};
// shorthand to check if parameter state is the current one
fsm.is = function(state) {
  return fsm.current === state;
};
// return a list of the valid states to enter from the current state
fsm.allowed = function() {
  return fsm.transitions[fsm.current];
};A "fire once" callback can be implemented with
fsm.once = function(evt, fn) {
  fsm.on(evt, function onceCb() {
    fn.apply(fn, Array.prototype.slice.call(arguments));
    fsm.unbind(evt, onceCb);
  });
  return fsm;
};If you need to add or remove states after the state machine has been initialized, something like the following might serve:
fsm.add = function(state, from, to) {
  fsm.transitions[state] = to;
  from.forEach(function(elem) {
    fsm.transitions[elem].push(state);
  });
};
fsm.remove = function(obsolete) {
  delete fsm.transitions[obsolete];
  for (var state in fsm.transitions) {
    if (fsm.transitions.hasOwnProperty(state)) {
      var index = fsm.transitions[state].indexOf(obsolete);
      if (index > -1)
        fsm.transitions[state].splice(index, 1);
    }
  }
  // probably should also set or check fsm.current to see we're still in a valid state
};Transitions between states can be removed in a similar fashion.
If you wish to apply some common sanity checks before state transitions, one way to add these would be by patching the .go method:
fsm.origo = fsm.go;
fsm.go = function() {
  // put state validation, parameter checks, anything you might need here
  return fsm.origo.apply(this, Array.prototype.slice.call(arguments));
};Alternatives
Too basic? Not quite what you were looking for? Some other alternatives for state machines in javascript are
Searching on bower or npm will probably also find some other takes on the subject.
Colophon
The event emitter pattern that pastafarian uses at its core is based on microevent.js.
License
pastafarian is ISC licensed.
Development
A basic development workflow is defined using npm run scripts. Get started with
$ git clone https://github.com/orbitbot/pastafarian
$ npm install
$ npm run developBugfixes and improvements are welcome, however, please open an Issue to discuss any larger changes beforehand, and consider if functionality can be implemented with a simple monkey-patching extension script. Useful extensions are more than welcome!
Possible future
- a transition that throws an error can be canceled, ie. intelligent rollback?