123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810 |
- // Copyright (c) 2013 - Max Persson <max@looplab.se>
- //
- // Licensed under the Apache License, Version 2.0 (the "License");
- // you may not use this file except in compliance with the License.
- // You may obtain a copy of the License at
- //
- // http://www.apache.org/licenses/LICENSE-2.0
- //
- // Unless required by applicable law or agreed to in writing, software
- // distributed under the License is distributed on an "AS IS" BASIS,
- // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- // See the License for the specific language governing permissions and
- // limitations under the License.
- package fsm
- import (
- "fmt"
- "sort"
- "sync"
- "testing"
- "time"
- )
- type fakeTransitionerObj struct {
- }
- func (t fakeTransitionerObj) transition(f *FSM) error {
- return &InternalError{}
- }
- func TestSameState(t *testing.T) {
- fsm := NewFSM(
- "start",
- Events{
- {Name: "run", Src: []string{"start"}, Dst: "start"},
- },
- Callbacks{},
- )
- fsm.Event("run")
- if fsm.Current() != "start" {
- t.Error("expected state to be 'start'")
- }
- }
- func TestSetState(t *testing.T) {
- fsm := NewFSM(
- "walking",
- Events{
- {Name: "walk", Src: []string{"start"}, Dst: "walking"},
- },
- Callbacks{},
- )
- fsm.SetState("start")
- if fsm.Current() != "start" {
- t.Error("expected state to be 'walking'")
- }
- err := fsm.Event("walk")
- if err != nil {
- t.Error("transition is expected no error")
- }
- }
- func TestBadTransition(t *testing.T) {
- fsm := NewFSM(
- "start",
- Events{
- {Name: "run", Src: []string{"start"}, Dst: "running"},
- },
- Callbacks{},
- )
- fsm.transitionerObj = new(fakeTransitionerObj)
- err := fsm.Event("run")
- if err == nil {
- t.Error("bad transition should give an error")
- }
- }
- func TestInappropriateEvent(t *testing.T) {
- fsm := NewFSM(
- "closed",
- Events{
- {Name: "open", Src: []string{"closed"}, Dst: "open"},
- {Name: "close", Src: []string{"open"}, Dst: "closed"},
- },
- Callbacks{},
- )
- err := fsm.Event("close")
- if e, ok := err.(InvalidEventError); !ok && e.Event != "close" && e.State != "closed" {
- t.Error("expected 'InvalidEventError' with correct state and event")
- }
- }
- func TestInvalidEvent(t *testing.T) {
- fsm := NewFSM(
- "closed",
- Events{
- {Name: "open", Src: []string{"closed"}, Dst: "open"},
- {Name: "close", Src: []string{"open"}, Dst: "closed"},
- },
- Callbacks{},
- )
- err := fsm.Event("lock")
- if e, ok := err.(UnknownEventError); !ok && e.Event != "close" {
- t.Error("expected 'UnknownEventError' with correct event")
- }
- }
- func TestMultipleSources(t *testing.T) {
- fsm := NewFSM(
- "one",
- Events{
- {Name: "first", Src: []string{"one"}, Dst: "two"},
- {Name: "second", Src: []string{"two"}, Dst: "three"},
- {Name: "reset", Src: []string{"one", "two", "three"}, Dst: "one"},
- },
- Callbacks{},
- )
- fsm.Event("first")
- if fsm.Current() != "two" {
- t.Error("expected state to be 'two'")
- }
- fsm.Event("reset")
- if fsm.Current() != "one" {
- t.Error("expected state to be 'one'")
- }
- fsm.Event("first")
- fsm.Event("second")
- if fsm.Current() != "three" {
- t.Error("expected state to be 'three'")
- }
- fsm.Event("reset")
- if fsm.Current() != "one" {
- t.Error("expected state to be 'one'")
- }
- }
- func TestMultipleEvents(t *testing.T) {
- fsm := NewFSM(
- "start",
- Events{
- {Name: "first", Src: []string{"start"}, Dst: "one"},
- {Name: "second", Src: []string{"start"}, Dst: "two"},
- {Name: "reset", Src: []string{"one"}, Dst: "reset_one"},
- {Name: "reset", Src: []string{"two"}, Dst: "reset_two"},
- {Name: "reset", Src: []string{"reset_one", "reset_two"}, Dst: "start"},
- },
- Callbacks{},
- )
- fsm.Event("first")
- fsm.Event("reset")
- if fsm.Current() != "reset_one" {
- t.Error("expected state to be 'reset_one'")
- }
- fsm.Event("reset")
- if fsm.Current() != "start" {
- t.Error("expected state to be 'start'")
- }
- fsm.Event("second")
- fsm.Event("reset")
- if fsm.Current() != "reset_two" {
- t.Error("expected state to be 'reset_two'")
- }
- fsm.Event("reset")
- if fsm.Current() != "start" {
- t.Error("expected state to be 'start'")
- }
- }
- func TestGenericCallbacks(t *testing.T) {
- beforeEvent := false
- leaveState := false
- enterState := false
- afterEvent := false
- fsm := NewFSM(
- "start",
- Events{
- {Name: "run", Src: []string{"start"}, Dst: "end"},
- },
- Callbacks{
- "before_event": func(e *Event) {
- beforeEvent = true
- },
- "leave_state": func(e *Event) {
- leaveState = true
- },
- "enter_state": func(e *Event) {
- enterState = true
- },
- "after_event": func(e *Event) {
- afterEvent = true
- },
- },
- )
- fsm.Event("run")
- if !(beforeEvent && leaveState && enterState && afterEvent) {
- t.Error("expected all callbacks to be called")
- }
- }
- func TestSpecificCallbacks(t *testing.T) {
- beforeEvent := false
- leaveState := false
- enterState := false
- afterEvent := false
- fsm := NewFSM(
- "start",
- Events{
- {Name: "run", Src: []string{"start"}, Dst: "end"},
- },
- Callbacks{
- "before_run": func(e *Event) {
- beforeEvent = true
- },
- "leave_start": func(e *Event) {
- leaveState = true
- },
- "enter_end": func(e *Event) {
- enterState = true
- },
- "after_run": func(e *Event) {
- afterEvent = true
- },
- },
- )
- fsm.Event("run")
- if !(beforeEvent && leaveState && enterState && afterEvent) {
- t.Error("expected all callbacks to be called")
- }
- }
- func TestSpecificCallbacksShortform(t *testing.T) {
- enterState := false
- afterEvent := false
- fsm := NewFSM(
- "start",
- Events{
- {Name: "run", Src: []string{"start"}, Dst: "end"},
- },
- Callbacks{
- "end": func(e *Event) {
- enterState = true
- },
- "run": func(e *Event) {
- afterEvent = true
- },
- },
- )
- fsm.Event("run")
- if !(enterState && afterEvent) {
- t.Error("expected all callbacks to be called")
- }
- }
- func TestBeforeEventWithoutTransition(t *testing.T) {
- beforeEvent := true
- fsm := NewFSM(
- "start",
- Events{
- {Name: "dontrun", Src: []string{"start"}, Dst: "start"},
- },
- Callbacks{
- "before_event": func(e *Event) {
- beforeEvent = true
- },
- },
- )
- err := fsm.Event("dontrun")
- if e, ok := err.(NoTransitionError); !ok && e.Err != nil {
- t.Error("expected 'NoTransitionError' without custom error")
- }
- if fsm.Current() != "start" {
- t.Error("expected state to be 'start'")
- }
- if !beforeEvent {
- t.Error("expected callback to be called")
- }
- }
- func TestCancelBeforeGenericEvent(t *testing.T) {
- fsm := NewFSM(
- "start",
- Events{
- {Name: "run", Src: []string{"start"}, Dst: "end"},
- },
- Callbacks{
- "before_event": func(e *Event) {
- e.Cancel()
- },
- },
- )
- fsm.Event("run")
- if fsm.Current() != "start" {
- t.Error("expected state to be 'start'")
- }
- }
- func TestCancelBeforeSpecificEvent(t *testing.T) {
- fsm := NewFSM(
- "start",
- Events{
- {Name: "run", Src: []string{"start"}, Dst: "end"},
- },
- Callbacks{
- "before_run": func(e *Event) {
- e.Cancel()
- },
- },
- )
- fsm.Event("run")
- if fsm.Current() != "start" {
- t.Error("expected state to be 'start'")
- }
- }
- func TestCancelLeaveGenericState(t *testing.T) {
- fsm := NewFSM(
- "start",
- Events{
- {Name: "run", Src: []string{"start"}, Dst: "end"},
- },
- Callbacks{
- "leave_state": func(e *Event) {
- e.Cancel()
- },
- },
- )
- fsm.Event("run")
- if fsm.Current() != "start" {
- t.Error("expected state to be 'start'")
- }
- }
- func TestCancelLeaveSpecificState(t *testing.T) {
- fsm := NewFSM(
- "start",
- Events{
- {Name: "run", Src: []string{"start"}, Dst: "end"},
- },
- Callbacks{
- "leave_start": func(e *Event) {
- e.Cancel()
- },
- },
- )
- fsm.Event("run")
- if fsm.Current() != "start" {
- t.Error("expected state to be 'start'")
- }
- }
- func TestCancelWithError(t *testing.T) {
- fsm := NewFSM(
- "start",
- Events{
- {Name: "run", Src: []string{"start"}, Dst: "end"},
- },
- Callbacks{
- "before_event": func(e *Event) {
- e.Cancel(fmt.Errorf("error"))
- },
- },
- )
- err := fsm.Event("run")
- if _, ok := err.(CanceledError); !ok {
- t.Error("expected only 'CanceledError'")
- }
- if e, ok := err.(CanceledError); ok && e.Err.Error() != "error" {
- t.Error("expected 'CanceledError' with correct custom error")
- }
- if fsm.Current() != "start" {
- t.Error("expected state to be 'start'")
- }
- }
- func TestAsyncTransitionGenericState(t *testing.T) {
- fsm := NewFSM(
- "start",
- Events{
- {Name: "run", Src: []string{"start"}, Dst: "end"},
- },
- Callbacks{
- "leave_state": func(e *Event) {
- e.Async()
- },
- },
- )
- fsm.Event("run")
- if fsm.Current() != "start" {
- t.Error("expected state to be 'start'")
- }
- fsm.Transition()
- if fsm.Current() != "end" {
- t.Error("expected state to be 'end'")
- }
- }
- func TestAsyncTransitionSpecificState(t *testing.T) {
- fsm := NewFSM(
- "start",
- Events{
- {Name: "run", Src: []string{"start"}, Dst: "end"},
- },
- Callbacks{
- "leave_start": func(e *Event) {
- e.Async()
- },
- },
- )
- fsm.Event("run")
- if fsm.Current() != "start" {
- t.Error("expected state to be 'start'")
- }
- fsm.Transition()
- if fsm.Current() != "end" {
- t.Error("expected state to be 'end'")
- }
- }
- func TestAsyncTransitionInProgress(t *testing.T) {
- fsm := NewFSM(
- "start",
- Events{
- {Name: "run", Src: []string{"start"}, Dst: "end"},
- {Name: "reset", Src: []string{"end"}, Dst: "start"},
- },
- Callbacks{
- "leave_start": func(e *Event) {
- e.Async()
- },
- },
- )
- fsm.Event("run")
- err := fsm.Event("reset")
- if e, ok := err.(InTransitionError); !ok && e.Event != "reset" {
- t.Error("expected 'InTransitionError' with correct state")
- }
- fsm.Transition()
- fsm.Event("reset")
- if fsm.Current() != "start" {
- t.Error("expected state to be 'start'")
- }
- }
- func TestAsyncTransitionNotInProgress(t *testing.T) {
- fsm := NewFSM(
- "start",
- Events{
- {Name: "run", Src: []string{"start"}, Dst: "end"},
- {Name: "reset", Src: []string{"end"}, Dst: "start"},
- },
- Callbacks{},
- )
- err := fsm.Transition()
- if _, ok := err.(NotInTransitionError); !ok {
- t.Error("expected 'NotInTransitionError'")
- }
- }
- func TestCallbackNoError(t *testing.T) {
- fsm := NewFSM(
- "start",
- Events{
- {Name: "run", Src: []string{"start"}, Dst: "end"},
- },
- Callbacks{
- "run": func(e *Event) {
- },
- },
- )
- e := fsm.Event("run")
- if e != nil {
- t.Error("expected no error")
- }
- }
- func TestCallbackError(t *testing.T) {
- fsm := NewFSM(
- "start",
- Events{
- {Name: "run", Src: []string{"start"}, Dst: "end"},
- },
- Callbacks{
- "run": func(e *Event) {
- e.Err = fmt.Errorf("error")
- },
- },
- )
- e := fsm.Event("run")
- if e.Error() != "error" {
- t.Error("expected error to be 'error'")
- }
- }
- func TestCallbackArgs(t *testing.T) {
- fsm := NewFSM(
- "start",
- Events{
- {Name: "run", Src: []string{"start"}, Dst: "end"},
- },
- Callbacks{
- "run": func(e *Event) {
- if len(e.Args) != 1 {
- t.Error("too few arguments")
- }
- arg, ok := e.Args[0].(string)
- if !ok {
- t.Error("not a string argument")
- }
- if arg != "test" {
- t.Error("incorrect argument")
- }
- },
- },
- )
- fsm.Event("run", "test")
- }
- func TestNoDeadLock(t *testing.T) {
- var fsm *FSM
- fsm = NewFSM(
- "start",
- Events{
- {Name: "run", Src: []string{"start"}, Dst: "end"},
- },
- Callbacks{
- "run": func(e *Event) {
- fsm.Current() // Should not result in a panic / deadlock
- },
- },
- )
- fsm.Event("run")
- }
- func TestThreadSafetyRaceCondition(t *testing.T) {
- fsm := NewFSM(
- "start",
- Events{
- {Name: "run", Src: []string{"start"}, Dst: "end"},
- },
- Callbacks{
- "run": func(e *Event) {
- },
- },
- )
- var wg sync.WaitGroup
- wg.Add(1)
- go func() {
- defer wg.Done()
- _ = fsm.Current()
- }()
- fsm.Event("run")
- wg.Wait()
- }
- func TestDoubleTransition(t *testing.T) {
- var fsm *FSM
- var wg sync.WaitGroup
- wg.Add(2)
- fsm = NewFSM(
- "start",
- Events{
- {Name: "run", Src: []string{"start"}, Dst: "end"},
- },
- Callbacks{
- "before_run": func(e *Event) {
- wg.Done()
- // Imagine a concurrent event coming in of the same type while
- // the data access mutex is unlocked because the current transition
- // is running its event callbacks, getting around the "active"
- // transition checks
- if len(e.Args) == 0 {
- // Must be concurrent so the test may pass when we add a mutex that synchronizes
- // calls to Event(...). It will then fail as an inappropriate transition as we
- // have changed state.
- go func() {
- if err := fsm.Event("run", "second run"); err != nil {
- fmt.Println(err)
- wg.Done() // It should fail, and then we unfreeze the test.
- }
- }()
- time.Sleep(20 * time.Millisecond)
- } else {
- panic("Was able to reissue an event mid-transition")
- }
- },
- },
- )
- if err := fsm.Event("run"); err != nil {
- fmt.Println(err)
- }
- wg.Wait()
- }
- func TestNoTransition(t *testing.T) {
- fsm := NewFSM(
- "start",
- Events{
- {Name: "run", Src: []string{"start"}, Dst: "start"},
- },
- Callbacks{},
- )
- err := fsm.Event("run")
- if _, ok := err.(NoTransitionError); !ok {
- t.Error("expected 'NoTransitionError'")
- }
- }
- func ExampleNewFSM() {
- fsm := NewFSM(
- "green",
- Events{
- {Name: "warn", Src: []string{"green"}, Dst: "yellow"},
- {Name: "panic", Src: []string{"yellow"}, Dst: "red"},
- {Name: "panic", Src: []string{"green"}, Dst: "red"},
- {Name: "calm", Src: []string{"red"}, Dst: "yellow"},
- {Name: "clear", Src: []string{"yellow"}, Dst: "green"},
- },
- Callbacks{
- "before_warn": func(e *Event) {
- fmt.Println("before_warn")
- },
- "before_event": func(e *Event) {
- fmt.Println("before_event")
- },
- "leave_green": func(e *Event) {
- fmt.Println("leave_green")
- },
- "leave_state": func(e *Event) {
- fmt.Println("leave_state")
- },
- "enter_yellow": func(e *Event) {
- fmt.Println("enter_yellow")
- },
- "enter_state": func(e *Event) {
- fmt.Println("enter_state")
- },
- "after_warn": func(e *Event) {
- fmt.Println("after_warn")
- },
- "after_event": func(e *Event) {
- fmt.Println("after_event")
- },
- },
- )
- fmt.Println(fsm.Current())
- err := fsm.Event("warn")
- if err != nil {
- fmt.Println(err)
- }
- fmt.Println(fsm.Current())
- // Output:
- // green
- // before_warn
- // before_event
- // leave_green
- // leave_state
- // enter_yellow
- // enter_state
- // after_warn
- // after_event
- // yellow
- }
- func ExampleFSM_Current() {
- fsm := NewFSM(
- "closed",
- Events{
- {Name: "open", Src: []string{"closed"}, Dst: "open"},
- {Name: "close", Src: []string{"open"}, Dst: "closed"},
- },
- Callbacks{},
- )
- fmt.Println(fsm.Current())
- // Output: closed
- }
- func ExampleFSM_Is() {
- fsm := NewFSM(
- "closed",
- Events{
- {Name: "open", Src: []string{"closed"}, Dst: "open"},
- {Name: "close", Src: []string{"open"}, Dst: "closed"},
- },
- Callbacks{},
- )
- fmt.Println(fsm.Is("closed"))
- fmt.Println(fsm.Is("open"))
- // Output:
- // true
- // false
- }
- func ExampleFSM_Can() {
- fsm := NewFSM(
- "closed",
- Events{
- {Name: "open", Src: []string{"closed"}, Dst: "open"},
- {Name: "close", Src: []string{"open"}, Dst: "closed"},
- },
- Callbacks{},
- )
- fmt.Println(fsm.Can("open"))
- fmt.Println(fsm.Can("close"))
- // Output:
- // true
- // false
- }
- func ExampleFSM_AvailableTransitions() {
- fsm := NewFSM(
- "closed",
- Events{
- {Name: "open", Src: []string{"closed"}, Dst: "open"},
- {Name: "close", Src: []string{"open"}, Dst: "closed"},
- {Name: "kick", Src: []string{"closed"}, Dst: "broken"},
- },
- Callbacks{},
- )
- // sort the results ordering is consistent for the output checker
- transitions := fsm.AvailableTransitions()
- sort.Strings(transitions)
- fmt.Println(transitions)
- // Output:
- // [kick open]
- }
- func ExampleFSM_Cannot() {
- fsm := NewFSM(
- "closed",
- Events{
- {Name: "open", Src: []string{"closed"}, Dst: "open"},
- {Name: "close", Src: []string{"open"}, Dst: "closed"},
- },
- Callbacks{},
- )
- fmt.Println(fsm.Cannot("open"))
- fmt.Println(fsm.Cannot("close"))
- // Output:
- // false
- // true
- }
- func ExampleFSM_Event() {
- fsm := NewFSM(
- "closed",
- Events{
- {Name: "open", Src: []string{"closed"}, Dst: "open"},
- {Name: "close", Src: []string{"open"}, Dst: "closed"},
- },
- Callbacks{},
- )
- fmt.Println(fsm.Current())
- err := fsm.Event("open")
- if err != nil {
- fmt.Println(err)
- }
- fmt.Println(fsm.Current())
- err = fsm.Event("close")
- if err != nil {
- fmt.Println(err)
- }
- fmt.Println(fsm.Current())
- // Output:
- // closed
- // open
- // closed
- }
- func ExampleFSM_Transition() {
- fsm := NewFSM(
- "closed",
- Events{
- {Name: "open", Src: []string{"closed"}, Dst: "open"},
- {Name: "close", Src: []string{"open"}, Dst: "closed"},
- },
- Callbacks{
- "leave_closed": func(e *Event) {
- e.Async()
- },
- },
- )
- err := fsm.Event("open")
- if e, ok := err.(AsyncError); !ok && e.Err != nil {
- fmt.Println(err)
- }
- fmt.Println(fsm.Current())
- err = fsm.Transition()
- if err != nil {
- fmt.Println(err)
- }
- fmt.Println(fsm.Current())
- // Output:
- // closed
- // open
- }
|