Skip to content

Latest commit

 

History

History
97 lines (81 loc) · 3.97 KB

README.md

File metadata and controls

97 lines (81 loc) · 3.97 KB

Finite State Machine (FSM) in Luau

A feature rich and fully typed implementation of a Finite State Machine in Luau.

This project is licensed under the terms of the MIT license. See LICENSE.md for details.

What's a finite state machine?

A Finite State Machine (FSM) provides a way to enforce specific logical flow among a set of States. Given an Event, the FSM responds by looking up the corresponding Transition for that Event in its current State. A Transition is a callback function that is invoked when an Event is given to the FSM, and it returns the next State for the FSM to move to.

The FSM enforces that Events can only be called when a Transition is defined for that Event in its current State by erroring if an Event is called in an invalid state (a state with no Transition defined for that Event).

Included features

The FSM provides 5 signals that can be listened to during handling of an event. These signals, transition callbacks, and state changes are processed in the following order:

SequenceDiagram

  1. Fire beforeEvent signal
    • with arguments eventName, beforeState
  2. Call transition.beforeAsync() (required, returns next state)
    • with the VarArgs from :handle(eventName, transitionArgs...)
  3. Fire leavingState signal
    • with arguments beforeState, afterState
  4. Update _currentState to next state
  5. Fire stateEntered signal
    • with arguments afterState, beforeState
  6. Call transition.afterAsync() (if specified)
    • with the VarArgs from :handle(eventName, transitionArgs...)
  7. Fire afterEvent signal
    • with arguments eventName, afterState, beforeState
  8. Fire finished signal if next state from beforeAsync() was nil
    • with argument beforeState

Transitions can be asynchronous, which is supported by queuing each Event submitted via :handle() and processing them in First-In-First-Out (FIFO) order. The next Event starts processing immediately after the previous Event's handler fires afterEvent.

The FSM can Finish if a Transition does not return a "next state" during an event marked as "canBeFinal". In such a case, the FSM will fire a finished event and will error if any further Events are handled. A nil state means the FSM has Finished.

Example usage

A simple state machine diagram for a light switch may look like this, where

  • States are represented as rectangles, and squares indicate the State can be Final
  • Events are represented as capsules

Screen Shot 2023-12-14 at 18 04 33

local LightState = {
	On = "On",
	Off = "Off",
}

local Event = {
	SwitchOn = "SwitchOn",
	SwitchOff = "SwitchOff",
}

local light = StateMachine.new(LightState.On, {
	[Event.SwitchOn] = {
		canBeFinal = true,
		from = {
			[LightState.Off] = { -- From state
				beforeAsync = function() -- Transition
					print("Light is transitioning to On")
					for i = 1, 100 do
						-- do some action to increase brightness of a light here
						task.wait()
					end
					return LightState.On -- To next state
				end,
				afterAsync = function()
					print("Light is now On")
				end,
			}
		},
	},
	[Event.SwitchOff] = {
		canBeFinal = true,
		from = {
			[LightState.On] = {
				beforeAsync = function()
					print("Light is transitioning to Off")
					return LightState.Off
				end,
			}
		},
	},
})

light:handle(Event.SwitchOff) -- prints "Light is transitioning to Off"
light:handle(Event.SwitchOn) -- prints "Light is transitioning to On", increases brightness over time, and then prints "Light is now On"
light:handle(Event.SwitchOn) -- warns "Illegal event `SwitchOn` called during state `On`" with a stack trace

Detailed system flowchart

Flowchart