Transition is a Golang state machine implementation.
NOTE: This fork removes all GORM/database backed functionality including dependencies. The goal here is to have a clean and useful FSM implementation, and nothing more.
NOTE Jan/24/2023: This new fork adds support for Generics and go.mod
Embed transition.Transition
into your struct, it will enable the state machine feature for the struct:
import "github.com/daegalus/transition"
type Order struct {
ID uint
transition.Transition
}
var OrderStateMachine = transition.New(&Order{})
// Define initial state
OrderStateMachine.Initial("draft")
// Define a State
OrderStateMachine.State("checkout")
// Define another State and what to do when entering and exiting that state.
OrderStateMachine.State("paid").Enter(func(order *Order) error {
// To get order object use 'order.(*Order)'
// business logic here
return
}).Exit(func(order *Order) error {
// business logic here
return
})
// Define more States
OrderStateMachine.State("cancelled")
OrderStateMachine.State("paid_cancelled")
// Define an Event
OrderStateMachine.Event("checkout").To("checkout").From("draft")
// Define another event and what to do before and after performing the transition.
OrderStateMachine.Event("paid").To("paid").From("checkout").Before(func(order *Order) error {
// business logic here
return nil
}).After(func(order *Order) error {
// business logic here
return nil
})
// Different state transitions for one event
cancellEvent := OrderStateMachine.Event("cancel")
cancellEvent.To("cancelled").From("draft", "checkout")
cancellEvent.To("paid_cancelled").From("paid").After(func(order *Order) error {
// Refund
}})
// func (*StateMachine) Trigger(name string, value Stater) error
OrderStatemachine.Trigger("paid", &order)
OrderStatemachine.Trigger("cancel", &order)
// order's state will be changed to cancelled if current state is "draft"
// order's state will be changed to paid_cancelled if current state is "paid"
var order Order
// Get Current State
order.GetState()
// Set State
order.SetState("finished") // this will only update order's state
Released under the ISC License.