Skip to content

Commit baf52f7

Browse files
authored
feat(machine): allow for partial auto states via handlers (#208)
1 parent 4e9974f commit baf52f7

15 files changed

+224
-138
lines changed

docs/cookbook.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -82,7 +82,7 @@ func (h *Handlers) AnyFoo(e *am.Event) {}
8282

8383
```go
8484
// always called as a transition from Any state to Any state
85-
func (h *Handlers) AnyAny(e *am.Event) bool {}
85+
func (h *Handlers) AnyEnter(e *am.Event) bool {}
8686
```
8787

8888
## Global final handler

docs/manual.md

+14-16
Original file line numberDiff line numberDiff line change
@@ -584,7 +584,7 @@ goroutine, with a timeout of [`Machine.HandlerTimeout`](https://pkg.go.dev/githu
584584
```go
585585
// can Foo activate?
586586
func (h *Handlers) FooEnter(e *am.Event) bool {}
587-
// when Foo active, can Bar activate?
587+
// with Foo active, can Bar activate?
588588
func (h *Handlers) FooBar(e *am.Event) {}
589589
// Foo activates
590590
func (h *Handlers) FooState(e *am.Event) {}
@@ -597,8 +597,6 @@ func (h *Handlers) FooEnd(e *am.Event) {}
597597
List of handlers during a transition from `Foo` to `Bar`, in the order of execution:
598598

599599
- `FooExit` - [negotiation handler](#negotiation-handlers)
600-
- `FooAny` - [negotiation handler](#negotiation-handlers)
601-
- `AnyBar` - [negotiation handler](#negotiation-handlers)
602600
- `BarEnter` - [negotiation handler](#negotiation-handlers)
603601
- `FooBar` - [negotiation handler](#negotiation-handlers)
604602
- `FooEnd` - [final handler](#final-handlers)
@@ -611,14 +609,13 @@ which can be set via [`am.Opts`](https://pkg.go.dev/github.com/pancsta/asyncmach
611609

612610
### Self Handlers
613611

614-
Self handler is a final handler for states which were active **before and after** a transition (all no-change active
615-
states). The name is a doubled name of the state (eg `FooFoo`).
612+
Self handler is a negotiation handler for states which were active **before and after** a transition (all no-change
613+
active states). The name is a doubled name of the state (eg `FooFoo`).
616614

617615
List of handlers during a transition from `Foo` to `Foo Bar`, in the order of execution:
618616

619-
- `AnyBar` - [negotiation handler](#negotiation-handlers)
620617
- `BarEnter` - [negotiation handler](#negotiation-handlers)
621-
- `FooFoo` - [final handler](#final-handlers) and **self handler**
618+
- `FooFoo` - [negotiation handler](#negotiation-handlers) and **self handler**
622619
- `BarState` - [final handler](#final-handlers)
623620

624621
Self handlers provide a simple alternative to [`Multi` states](#multi-states), while fully maintaining [state clocks](#clock-and-context).
@@ -754,13 +751,18 @@ end
754751
### Negotiation Handlers
755752

756753
```go
754+
// can Foo activate?
757755
func (h *Handlers) FooEnter(e *am.Event) bool {}
756+
// can Foo de-activate?
758757
func (h *Handlers) FooExit(e *am.Event) bool {}
758+
// with Bar active, can Foo activate?
759+
func (h *Handlers) BarFoo(e *am.Event) bool {}
759760
```
760761

761-
**Negotiation handlers** `Enter` and `Exit` are called for every state which is going to be activated or de-activated. They
762-
are allowed to cancel a transition by optionally returning `false`. **Negotiation handlers** are limited to read-only
763-
operations, or at least to side effects free ones. Their purpose is to make sure that
762+
**Negotiation handlers** `Enter` and `Exit` are called for every state which is going to be activated or de-activated.
763+
State-state handler (eg `FooBar`) are `Foo` is active and `Bar` wants to activate. They are allowed to cancel a
764+
transition by optionally returning `false`. **Negotiation handlers** are limited to read-only operations, or at least to
765+
side effects free ones. Their purpose is to make sure that
764766
[final transition handlers](#final-handlers) are good to go.
765767

766768
```go
@@ -812,10 +814,6 @@ mach.Add1("Foo", nil) // ->am.Canceled
812814
```go
813815
func (h *Handlers) FooState(e *am.Event) {}
814816
func (h *Handlers) FooEnd(e *am.Event) {}
815-
func (h *Handlers) FooBar(e *am.Event) {}
816-
func (h *Handlers) BarFoo(e *am.Event) {}
817-
func (h *Handlers) AnyFoo(e *am.Event) {}
818-
func (h *Handlers) FooAny(e *am.Event) {}
819817
```
820818

821819
Final handlers `State` and `End` are where the main handler logic resides. After the transition gets accepted by
@@ -858,10 +856,10 @@ func (h *Handlers) ProcessingFileState(e *am.Event) {
858856
859857
### Global Handlers
860858
861-
`AnyAny` is the first negotiation handler and always gets executed.
859+
`AnyEnter` is the first negotiation handler and always gets executed.
862860
863861
```go
864-
func (d *Debugger) AnyAny(e *am.Event) bool {
862+
func (d *Debugger) AnyEnter(e *am.Event) bool {
865863
tx := e.Transition()
866864
867865
// ...

pkg/machine/machine.go

+11-6
Original file line numberDiff line numberDiff line change
@@ -569,7 +569,7 @@ func (m *Machine) WhenArgs(
569569
ch := make(chan struct{})
570570

571571
m.MustParseStates(S{state})
572-
name := state + "State"
572+
name := state + SuffixState
573573

574574
// locks
575575
m.indexWhenArgsLock.Lock()
@@ -1173,7 +1173,7 @@ func (m *Machine) queueMutation(
11731173
//
11741174
// Note: usage of Eval is discouraged. But if you have to, use AM_DETECT_EVAL in
11751175
// tests for deadlock detection. Most usages of eval can be replaced with
1176-
// atomics.
1176+
// atomics or returning from mutation via channels.
11771177
func (m *Machine) Eval(source string, fn func(), ctx context.Context) bool {
11781178
if m.disposed.Load() {
11791179
return false
@@ -1349,7 +1349,12 @@ func (m *Machine) BindHandlers(handlers any) error {
13491349
h := m.newHandler(handlers, name, &v, methodNames)
13501350
old := m.getHandlers(false)
13511351
m.setHandlers(false, append(old, h))
1352-
m.log(LogOps, "[handlers] bind %s", name)
1352+
if name != "" {
1353+
m.log(LogOps, "[handlers] bind %s", name)
1354+
} else {
1355+
// index for anon handlers
1356+
m.log(LogOps, "[handlers] bind %d", len(old))
1357+
}
13531358

13541359
return nil
13551360
}
@@ -1959,7 +1964,7 @@ func (m *Machine) processWhenArgs(e *Event) {
19591964

19601965
argNames := jw(slices.AppendSeq(S{}, maps.Keys(binding.args)), ",")
19611966
// FooState -> Foo
1962-
name := e.Name[0 : len(e.Name)-len("State")]
1967+
name := e.Name[0 : len(e.Name)-len(SuffixState)]
19631968
m.log(LogOps, "[whenArgs:match] %s (%s)", name, argNames)
19641969
// args match - doDispose and close outside the mutex
19651970
chToClose = append(chToClose, binding.ch)
@@ -2327,8 +2332,8 @@ func (m *Machine) processHandlers(e *Event) (Result, bool) {
23272332
case timeout:
23282333

23292334
return Canceled, handlerCalled
2330-
case strings.HasSuffix(e.Name, "State"):
2331-
case strings.HasSuffix(e.Name, "End"):
2335+
case strings.HasSuffix(e.Name, SuffixState):
2336+
case strings.HasSuffix(e.Name, SuffixEnd):
23322337
// returns from State and End handlers are ignored
23332338
default:
23342339
if !ret {

0 commit comments

Comments
 (0)