diff --git a/README.md b/README.md index b96e4fd..c692dee 100644 --- a/README.md +++ b/README.md @@ -1,4 +1,4 @@ -# When: TypeScript Reference Implementation +# When: TypeScript Implementation ##### A software design pattern for building event-based recombinant state machines [![npm](https://img.shields.io/npm/v/when-ts.svg)](https://www.npmjs.com/package/when-ts) @@ -9,208 +9,13 @@ ### Introduction -**The latest version of this spec can be found in the [`devel` branch](https://github.com/voodooattack/when-ts/blob/devel/README.md), please read the spec there if that's what you're after.** +**The latest version of this README can be found in the [`devel` branch](https://github.com/voodooattack/when-ts/blob/devel/README.md), please read the spec there if that's what you're after.** -This is a reference implementation for a new software design pattern that allows for composable event-based state machines with complete (including temporal) control over their state. +The spec for the abstract syntax and the design pattern itself can be found in [the spec/ subdirectory](spec/when.md). Please read the specs before delving into the implementation itself to get a good understanding of how things work``. -Please note that this spec and reference implementation are still in alpha and the specs are not yet final. +This is a reference implementation for a new software design pattern that allows for composable event-based state machines with complete (including temporal) control over the state. -#### Features: - -- Discrete: if your actions only deal with the state object, then every state transition is 100% predictable. -- Temporal: time can be rewound at any given moment (tick) by default, and the state machine will transition to a previously known state in time, along with any future information in the form of an optional state mutation to apply. -- Recombinant: the pattern is based on [gene expression](https://en.wikipedia.org/wiki/Gene_expression), and since state machines are composed of events (`condition -> action` pairs) that are quite similar to how real genes are theorised to work (`activation region -> coding region`), this means that genetic recombination can be applied to `when` state machines by transferring new events from one machine to another. Mutating the machine (DNA) by transferring events (genes) from one machine to the other will introduce new behaviour. - -#### Possible Proposals - -Here are some possible expansions on the idea. These require further discussion before they're mature enough to include: - -- Sexual reproduction of state machines: possible use of a similar mechanic to the one used in organic cells to combine two different programs (DNA) by randomly selecting an equal half of each. -- Mutation: Possible, but difficult since we can't swap code like basepairs. The simplest possible mutation would be a random swap of conditions between two randomly selected actions. - -This would all lead to more emergent behaviour in agents produced by recombination. - -#### Pattern - -*The following is a description of the pattern itself, and not any specific implementation.* - -This pattern itself is completely generic and can be implemented in any programming language available today with varying degrees of ease, depending on the features of the target language. - -##### Program state - -A `MachineState` consists of user-defined global variables (and is passed to every condition and action as the first argument in the reference implementation). - -An external tick counter (`history.tick`) exists and can be considered part of the state (but is not included inside the state object). It is a special variable that is automatically incremented with every new tick. Can be used to reference discrete points in time. - -##### Conditions and Actions - -All when programs consist of `condition` and `action` pairs. The condition is a and expression that must evaluate to a boolean value. - -When a `condition` evaluates to `true`, the associated `action` is then executed. - -`actions` can modify the variables in the current state, but any modifications they make during a `tick` will be applied to the `state` only on the next `tick`. - -If a conflict between two or more `actions` trying to modify the same variable during a `tick` happens, the last `action` to be invoked will override the previous value set by any earlier `actions` during the current `tick`. - -##### Main loop - -The goal of the main loop is to move execution forward by mutating the current `state`. - -To do this, `when` implements a loop that constantly evaluates a set of rules (`program`). Every iteration of this loop is called a `tick`, and whenever a condition evaluates to `true`, the `action` associated with the condition is evaluated. `actions` can modify non-constant global variables with values for the next `state`. - -Note that any new mutations caused by actions will only appear during the next `tick`. This is to prevent interactions between different `actions` during the same `tick`. - -If multiple actions try to modify the same variable during the same `tick`, the last `action` to execute takes precedence. - -##### Finite State Machines - -By default, the state machines built with `when` will be finite, this means that the main loop will halt by default if it exhausts all possible conditions and none evaluate to `true` and trigger an action during the same `tick`. - -This prevents the program from running forever by default, and can be disabled as needed. - -#### State Manager - -- A State Manager (`history`) is accessible from events. It is responsible for managing an array of previous states (`history.records`), in which states are recorded as the program advances. - -- A state machine can exit by calling `exit()` from any event, the returned value is the last recorded state. A single argument can be passed to `exit()` to override the returned state. - -- Events can use `history.tick` to access the current tick counter. - -- Events can access the last recorded states from `history.currentState`. - -- Events can access the next state being actively mutated by the current tick through the read-only property `history.nextState`. - -- The state can be rewound to a previously recorded state using the `history.rewind(t)` method. `history.rewind(2)` will cause the program to rewind to the discrete tick `t` (the tick counter will be decremented as needed). If this occurs inside an event handler, further events will not be processed. - -- `history.rewind` accepts a second parameter with optional variable to pass after rewinding to the past state, `history.rewind(2, { backToTheFuture: true })` will rewind to tick `2` and mutate the past state by setting the variable `backToTheFuture` to `true`. - -- State history can be erased at any time using `history.clear();`. - -- State recording can be configured or disabled at any time by manipulating `history.limit`. - -- Setting a finite `limit` during startup is strongly advised. `history.limit` defaults to `Infinity`. - -**Examples of `limit`:** - -- `history.limit = Infinity;` Record an infinite amount of state. (This is the default, which may cause memory issues if your state objects are very big and/or your program stays running for a long time) - -- `history.limit = 4;` Only record the most recent 4 states. Discards any stored older states. - -- `history.limit = 0;` No further state recording allowed, and acts the same as `history.limit = 1`. Discards any older history, and `history.record` will only show the previous state. - -#### External inputs - -`when` supports external inputs via the `@input` decorator. External inputs are readonly variables that are recorded as part of the state, but never manually updated. - -#### Note on Recombination - -This is not part of the current spec, but is currently offered by the TypeScript reference implementation. You can combine any two machines by calling `machine1.recombine(machine2)`, see the [TypeScript API documentation](https://voodooattack.github.io/when-ts/) for more details. - -##### How it can be useful for emergent behaviour: - -For emergent behaviour to be meaningful, the machines in questions must attribute the same 'meaning' to the same variable names. - -A `health` variable for an NPC will usually have the same meaning for two different state machines when it comes to behaviour, and for the sake of argument, let us assume two different behaviours in two different machines: - -1. A machine has a `when` clause that causes the NPC to flee on low health (by controlling movement). -2. Another machine attacks on low health (controlling a bow and arrow) - -When both traits are present in a single machine, the NPC will potentially exhibit both behaviour simultaneously and run away while shooting, once they have low health. - -#### Abstract Syntax - -Here are some abstract syntax examples for a full pseudo-language based on this pattern. In this theoretical language, the program itself is a state machine, variables of the `MachineState` are global variables, and all of the primitives described above are part of the language itself. - -This is mostly pseudo-javascript with two extra `when` and `exit` keywords, and using a hypothetical decorator syntax to specify action metadata. - -The decorators are completely optional, and the currently proposed ones are: - -##### Action decorators: - -Action decorators may only precede a `when` block, and will only apply to that block. - -- `@name(action_name)` Associate a name with an action to be make it possible for inhibitors to reference it elsewhere. Can only be used once per action. - -- `@unless(expression)` Prevents this action from triggering if `expression` evaluates to true. Can be used multiple times with the same action. - -- `@inhibitedBy(action_name)` Prevents this action from triggering if another by `action_name` will execute during this tick. Can be used multiple times with the same action and different inhibitors. - -- `@priority(expression)` Sets a priority for the action. (Default is 0) This will influence the order of evaluation inside the main loop. Actions with lower priority values are evaluated last, while actions with higher priority values are evaluated first, meaning that they will take precedence if there's a conflict from multiple actions trying to update the same variable during the same tick. Can be a literal numeric value or an expression that returns a signed numeric value. - -##### Control decorators: - -- `@forever()` Must be defined at the very beginning of the program, and tells the state machine not to halt due to inactivity. In this case, the machine must explicitly end its execution via a call to `exit()`. Accepts no arguments. - -- `@input(policy?: 'once'|'always'|function, input_name?)` Implementation dependent. Defaults to `once`. Must precede a constant/readonly variable declaration. Tells `when` to poll an external value and record its value as part of the state. The interpretation of what an input is depends on the implementation. It can be a command-line argument, a memory address, or an hardware interrupt. The `policy` argument specifies how frequently the polling is done: `once` is exactly once at startup, `always` is once per `tick`. `function` is user-defined function that implements custom logic and returns a boolean. - -##### Examples - -- A prime number generator: - -```typescript -// maximum number of primes to brute-force before exiting, -// note that this variable is a readonly external input, -// and is read only once on startup. -@input('once') -const maxPrimes: number = 10; - -let counter = 2; // starting counting up from 2 -let current = 3; // start looking at 3 -let primes = []; // array to store saved primes - -// increment the counter with every tick till we hit the potential prime -@name('increment') -@unless(primes.length >= maxPrimes) -when(counter < current) { - counter++; -} - -// not a prime number, reset and increment current search -@name('resetNotAPrime') -@unless(primes.length >= maxPrimes) -when(counter < current && current % counter === 0) { - counter = 2; - current++; -} - -// if this is ever triggered, then we're dealing with a prime. -@name('capturePrime') -@unless(primes.length >= maxPrimes) -when(counter >= current) { - // save the prime - primes.push(current); - // print it to the console - console.log(current); - // reset the variables and look for the next one - counter = 2; - current++; -} -``` - -To make this same machine with an explicit exit clause, simply remove all `@unless` decorators and add `@forever` at the beginning. - -To make this machine exit, you must add the following anywhere in the file: -```js -// exit when we've found enough primes -@name('exitOnceDone') -when(primes.length >= 10) { - exit(); -} -``` - -With either option, the predicted exit state after the machine exits should be: - -```json -{ - "counter": 2, - "current": 30, - "primes": [ 2, 3, 5, 7, 11, 13, 17, 19, 23, 29 ] -} -``` - -Note: more complex examples are coming soon. - -## TypeScript Reference Implementation +## When: TypeScript Implementation ### Installation diff --git a/package.json b/package.json index dfbfcfa..54f571c 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "when-ts", - "version": "1.0.0-alpha.5", + "version": "1.0.0", "description": "When: A software design pattern for building event-based recombinant state machines.", "main": "dist/lib/src/index.js", "types": "dist/types/src/index.d.ts", @@ -13,7 +13,7 @@ "prepublish": "npm run build && npm run typedoc && npm run test", "test": "jest --coverage", "test:coveralls": "cat ./coverage/lcov.info | ./node_modules/coveralls/bin/coveralls.js", - "commit": "git-cz" + "commit": "npm run lint && git-cz" }, "engines": { "node": ">=8.12.0" diff --git a/spec/when.md b/spec/when.md new file mode 100644 index 0000000..e52ea24 --- /dev/null +++ b/spec/when.md @@ -0,0 +1,196 @@ +# When: recombinant event-based state machines. + +## Features: + +- Discrete: if your actions only deal with the state object, then every state transition is 100% predictable. +- Temporal: time can be rewound at any given moment (tick) by default, and the state machine will transition to a previously known state in time, along with any future information in the form of an optional state mutation to apply. +- Recombinant: the pattern is based on [gene expression](https://en.wikipedia.org/wiki/Gene_expression), and since state machines are composed of events (`condition -> action` pairs) that are quite similar to how real genes are theorised to work (`activation region -> coding region`), this means that genetic recombination can be applied to `when` state machines by transferring new events from one machine to another. Mutating the machine (DNA) by transferring events (genes) from one machine to the other will introduce new behaviour. + +### Possible Proposals + +Here are some possible expansions on the idea. These require further discussion before they're mature enough to include: + +- Sexual reproduction of state machines: possible use of a similar mechanic to the one used in organic cells to combine two different programs (DNA) by randomly selecting an equal half of each. +- Mutation: Possible, but difficult since we can't swap code like basepairs. The simplest possible mutation would be a random swap of conditions between two randomly selected actions. + +This would all lead to more emergent behaviour in agents produced by recombination. + +### Pattern + +*The following is a description of the pattern itself, and not any specific implementation.* + +This pattern itself is completely generic and can be implemented in any programming language available today with varying degrees of ease, depending on the features of the target language. + +#### Program state + +A `MachineState` consists of user-defined global variables (and is passed to every condition and action as the first argument in the reference implementation). + +An external tick counter (`history.tick`) exists and can be considered part of the state (but is not included inside the state object). It is a special variable that is automatically incremented with every new tick. Can be used to reference discrete points in time. + +#### Conditions and Actions + +All when programs consist of `condition` and `action` pairs. The condition is a and expression that must evaluate to a boolean value. + +When a `condition` evaluates to `true`, the associated `action` is then executed. + +`actions` can modify the variables in the current state, but any modifications they make during a `tick` will be applied to the `state` only on the next `tick`. + +If a conflict between two or more `actions` trying to modify the same variable during a `tick` happens, the last `action` to be invoked will override the previous value set by any earlier `actions` during the current `tick`. + +#### Main loop + +The goal of the main loop is to move execution forward by mutating the current `state`. + +To do this, `when` implements a loop that constantly evaluates a set of rules (`program`). Every iteration of this loop is called a `tick`, and whenever a condition evaluates to `true`, the `action` associated with the condition is evaluated. `actions` can modify non-constant global variables with values for the next `state`. + +Note that any new mutations caused by actions will only appear during the next `tick`. This is to prevent interactions between different `actions` during the same `tick`. + +If multiple actions try to modify the same variable during the same `tick`, the last `action` to execute takes precedence. + +#### Finite State Machines + +By default, the state machines built with `when` will be finite, this means that the main loop will halt by default if it exhausts all possible conditions and none evaluate to `true` and trigger an action during the same `tick`. + +This prevents the program from running forever by default, and can be disabled as needed. + +### State Manager + +- A State Manager (`history`) is accessible from events. It is responsible for managing an array of previous states (`history.records`), in which states are recorded as the program advances. + +- A state machine can exit by calling `exit()` from any event, the returned value is the last recorded state. A single argument can be passed to `exit()` to override the returned state. + +- Events can use `history.tick` to access the current tick counter. + +- Events can access the last recorded states from `history.currentState`. + +- Events can access the next state being actively mutated by the current tick through the read-only property `history.nextState`. + +- The state can be rewound to a previously recorded state using the `history.rewind(t)` method. `history.rewind(2)` will cause the program to rewind to the discrete tick `t` (the tick counter will be decremented as needed). If this occurs inside an event handler, further events will not be processed. + +- `history.rewind` accepts a second parameter with optional variable to pass after rewinding to the past state, `history.rewind(2, { backToTheFuture: true })` will rewind to tick `2` and mutate the past state by setting the variable `backToTheFuture` to `true`. + +- State history can be erased at any time using `history.clear();`. + +- State recording can be configured or disabled at any time by manipulating `history.limit`. + +- Setting a finite `limit` during startup is strongly advised. `history.limit` defaults to `Infinity`. + +**Examples of `limit`:** + +- `history.limit = Infinity;` Record an infinite amount of state. (This is the default, which may cause memory issues if your state objects are very big and/or your program stays running for a long time) + +- `history.limit = 4;` Only record the most recent 4 states. Discards any stored older states. + +- `history.limit = 0;` No further state recording allowed, and acts the same as `history.limit = 1`. Discards any older history, and `history.record` will only show the previous state. + +### External inputs + +`when` supports external inputs via the `@input` decorator. External inputs are readonly variables that are recorded as part of the state, but never manually updated. + +### Note on Recombination + +This is not part of the current spec, but is currently offered by the TypeScript reference implementation. You can combine any two machines by calling `machine1.recombine(machine2)`, see the [TypeScript API documentation](https://voodooattack.github.io/when-ts/) for more details. + +#### How it can be useful for emergent behaviour: + +For emergent behaviour to be meaningful, the machines in questions must attribute the same 'meaning' to the same variable names. + +A `health` variable for an NPC will usually have the same meaning for two different state machines when it comes to behaviour, and for the sake of argument, let us assume two different behaviours in two different machines: + +1. A machine has a `when` clause that causes the NPC to flee on low health (by controlling movement). +2. Another machine attacks on low health (controlling a bow and arrow) + +When both traits are present in a single machine, the NPC will potentially exhibit both behaviour simultaneously and run away while shooting, once they have low health. + +### Abstract Syntax + +Here are some abstract syntax examples for a full pseudo-language based on this pattern. In this theoretical language, the program itself is a state machine, variables of the `MachineState` are global variables, and all of the primitives described above are part of the language itself. + +This is mostly pseudo-javascript with two extra `when` and `exit` keywords, and using a hypothetical decorator syntax to specify action metadata. + +The decorators are completely optional, and the currently proposed ones are: + +#### Action decorators: + +Action decorators may only precede a `when` block, and will only apply to that block. + +- `@name(action_name)` Associate a name with an action to be make it possible for inhibitors to reference it elsewhere. Can only be used once per action. + +- `@unless(expression)` Prevents this action from triggering if `expression` evaluates to true. Can be used multiple times with the same action. + +- `@inhibitedBy(action_name)` Prevents this action from triggering if another by `action_name` will execute during this tick. Can be used multiple times with the same action and different inhibitors. + +- `@priority(expression)` Sets a priority for the action. (Default is 0) This will influence the order of evaluation inside the main loop. Actions with lower priority values are evaluated last, while actions with higher priority values are evaluated first, meaning that they will take precedence if there's a conflict from multiple actions trying to update the same variable during the same tick. Can be a literal numeric value or an expression that returns a signed numeric value. + +#### Control decorators: + +- `@forever()` Must be defined at the very beginning of the program, and tells the state machine not to halt due to inactivity. In this case, the machine must explicitly end its execution via a call to `exit()`. Accepts no arguments. + +- `@input(policy?: 'once'|'always'|function, input_name?)` Implementation dependent. Defaults to `once`. Must precede a constant/readonly variable declaration. Tells `when` to poll an external value and record its value as part of the state. The interpretation of what an input is depends on the implementation. It can be a command-line argument, a memory address, or an hardware interrupt. The `policy` argument specifies how frequently the polling is done: `once` is exactly once at startup, `always` is once per `tick`. `function` is user-defined function that implements custom logic and returns a boolean. + +### Examples + +- A prime number generator: + +```typescript +// maximum number of primes to brute-force before exiting, +// note that this variable is a readonly external input, +// and is read only once on startup. +@input('once') +const maxPrimes: number = 10; + +let counter = 2; // starting counting up from 2 +let current = 3; // start looking at 3 +let primes = []; // array to store saved primes + +// increment the counter with every tick till we hit the potential prime +@name('increment') +@unless(primes.length >= maxPrimes) +when(counter < current) { + counter++; +} + +// not a prime number, reset and increment current search +@name('resetNotAPrime') +@unless(primes.length >= maxPrimes) +when(counter < current && current % counter === 0) { + counter = 2; + current++; +} + +// if this is ever triggered, then we're dealing with a prime. +@name('capturePrime') +@unless(primes.length >= maxPrimes) +when(counter >= current) { + // save the prime + primes.push(current); + // print it to the console + console.log(current); + // reset the variables and look for the next one + counter = 2; + current++; +} +``` + +To make this same machine with an explicit exit clause, simply remove all `@unless` decorators and add `@forever` at the beginning. + +To make this machine exit, you must add the following anywhere in the file: +```js +// exit when we've found enough primes +@name('exitOnceDone') +when(primes.length >= 10) { + exit(); +} +``` + +With either option, the predicted exit state after the machine exits should be: + +```json +{ + "counter": 2, + "current": 30, + "primes": [ 2, 3, 5, 7, 11, 13, 17, 19, 23, 29 ] +} +``` + +Note: more complex examples are coming soon. \ No newline at end of file diff --git a/src/historyManager.ts b/src/historyManager.ts index 88e7718..f155e72 100644 --- a/src/historyManager.ts +++ b/src/historyManager.ts @@ -19,7 +19,7 @@ export class HistoryManager = {}; private _maxHistory: number = Infinity; private _tick: number = 0; - private _nextState: Partial>; + private _nextState: Readonly & I>; private _records: (S & Readonly)[] = []; /** @@ -164,7 +164,7 @@ export class HistoryManager; + let nextState = this.nextState; if (this.tick === 0) nextState = Object.assign(nextState, this._records.pop() || {}); // discard old tick 0 this._records.push(nextState as any); diff --git a/src/interfaces.ts b/src/interfaces.ts index c57362f..0a30033 100644 --- a/src/interfaces.ts +++ b/src/interfaces.ts @@ -49,8 +49,6 @@ export type ActivationAction = (state: Readonly, machine: StateMachine) => number; - - /** * The HistoryManager interface allows for state manipulation and the rewinding of a program. */ @@ -59,7 +57,7 @@ export interface IHistory>; readonly currentState: Readonly; readonly initialState: Readonly; - readonly nextState: Readonly> & Readonly; + readonly nextState: Readonly & I>; /** * Limit the maximum number of past history states kept on record. diff --git a/src/stateMachine.ts b/src/stateMachine.ts index 549d2e9..7fc8f46 100644 --- a/src/stateMachine.ts +++ b/src/stateMachine.ts @@ -1,7 +1,7 @@ -import { actionMetadataKey, inputMetadataKey, priorityMetadataKey } from './metadataKeys'; import { HistoryManager } from './historyManager'; -import { ActivationAction, ActivationCond, MachineInputSource, MachineState } from './index'; +import { ActivationAction, ActivationCond, MachineInputSource, MachineState, PriorityExpression } from './index'; import { IHistory } from './interfaces'; +import { actionMetadataKey, inputMetadataKey, priorityMetadataKey } from './metadataKeys'; import { getAllMethods, InputOf, StateOf } from './util'; export type StateCombiner, @@ -19,6 +19,12 @@ export type StateCombiner, ): (S1 & S2) }; +export type ProgramEntry = + { + action: ActivationAction; + priority: number | PriorityExpression; + } + /** * Your state machine should inherit the `StateMachine` class. */ @@ -28,8 +34,7 @@ export class StateMachine, - ActivationAction> = new Map(); + private _program: Map, ProgramEntry> = new Map(); private readonly _history: HistoryManager; private _exitState?: Readonly; @@ -39,18 +44,14 @@ export class StateMachine, any, number][] = []; - for (let m of properties) { - if (Reflect.hasMetadata(actionMetadataKey, m)) { - const cond = Reflect.getMetadata(actionMetadataKey, m); - const priority = Reflect.getMetadata(priorityMetadataKey, m); - program.push([cond, m, typeof priority === 'number' ? priority : 0]); + const properties: ActivationAction[] = getAllMethods(this) as any; + for (let action of properties) { + if (Reflect.hasMetadata(actionMetadataKey, action)) { + const cond = Reflect.getMetadata(actionMetadataKey, action); + const priority = Reflect.getMetadata(priorityMetadataKey, action); + this._program.set(cond, { priority, action }); } } - this._program = new Map( - program.sort(([,,a], [,,b]) => a - b) as any - ); this._history = new HistoryManager(this, initialState, inputSource, inputSource ? Reflect.getMetadata(inputMetadataKey, inputSource) : [] ); @@ -64,7 +65,7 @@ export class StateMachine { + get history(): IHistory { return this._history; } @@ -74,19 +75,41 @@ export class StateMachine, ProgramEntry][] = []; + // for(let [key, entry] of this._program) { + // const priority = (typeof entry.priority === 'number' ? + // entry.priority : entry.priority(currentState, this)) || 0; + // actions.splice(priority, 0, [key, entry]); + // } + let actions = + Array.from(this._program.entries()) + .map(([cond, entry]) => + ({ + cond, + entry: { + action: entry.action, + priority: (typeof entry.priority === 'function' ? + entry.priority(currentState, this) : entry.priority) || 0 + } + }) + ) + .sort(({ entry: { priority: p1 } }, { entry: { priority: p2 } }) => p1 - p2); + for (let { cond, entry: { action } } of actions) { if (this.history.tick !== currentTick && !this.exitState) { // abort current tick on rewind. // always report at least 1 action fired in this case. return Math.max(1, fired); } - if (this.exitState) + if (this.exitState) { break; + } if (cond.call(this, this.history.currentState, this)) { - const newState = body.call(this, this.history.currentState, this); + const newState = action.call(this, this.history.currentState, this); if (newState) { this._history._mutateTick(newState); } @@ -94,8 +117,9 @@ export class StateMachine = { andWhen(cond: ActivationCond | true): WhenDecoratorWithChain; unless(condition: ActivationCond): WhenDecoratorWithChain; inhibitedBy(inhibitor: keyof M): WhenDecoratorWithChain; - priority(p: number): WhenDecoratorWithChain; + priority(p: number|PriorityExpression): WhenDecoratorWithChain; } export type WhenDecoratorWithChain = MethodDecorator & WhenDecoratorChainResult; diff --git a/tests/stateMachine.test.ts b/tests/stateMachine.test.ts index 5b433ed..b6388c1 100644 --- a/tests/stateMachine.test.ts +++ b/tests/stateMachine.test.ts @@ -608,17 +608,23 @@ describe('StateMachine', () => { return { value: s.value - 1 }; } - @when(state => state.value < 5) + @when(state => state.value < 5).priority(-10) incrementOncePerTick(s: State) { return { value: s.value + 1 }; } + @when((_, m) => m.history.tick > 4) + .priority(state => state.value % 2 === 0 ? 1000 : 0) + mul(s: State) { + return { value: s.value * 2 }; + } + } const test = new TestMachine(); const result = test.run(); - expect(result).toEqual({ value: -5 }); + expect(result).toEqual({ value: -8 }); });