You signed in with another tab or window. Reload to refresh your session.You signed out in another tab or window. Reload to refresh your session.You switched accounts on another tab or window. Reload to refresh your session.Dismiss alert
Copy file name to clipboardexpand all lines: README.md
+58-14
Original file line number
Diff line number
Diff line change
@@ -17,7 +17,7 @@ Please note that this spec and reference implementation are still in alpha and t
17
17
18
18
- Discrete: if your actions only deal with the state object, then every state transition is 100% predictable.
19
19
- 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.
20
-
- Recombinant: the pattern is based on [gene expression](https://en.wikipedia.org/wiki/Gene_expression), and since state machines are composed of events (`condition -> behaviour` 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.
20
+
- 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.
21
21
22
22
#### Possible Proposals
23
23
@@ -41,6 +41,16 @@ A `MachineState` consists of user-defined global variables (and is passed to eve
41
41
42
42
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.
43
43
44
+
##### Conditions and Actions
45
+
46
+
All when programs consist of `condition` and `action` pairs. The condition is a and expression that must evaluate to a boolean value.
47
+
48
+
When a `condition` evaluates to `true`, the associated `action` is then executed.
49
+
50
+
`actions` can modify the variables in the global state, but any modifications they make during a `tick` will be applied to the `state` only on the next `tick`.
51
+
52
+
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`.
53
+
44
54
##### Main loop
45
55
46
56
The goal of the main loop is to move execution forward by mutating the current `state`.
@@ -51,7 +61,11 @@ Note that any new mutations caused by actions will only appear during the next `
51
61
52
62
If multiple actions try to modify the same variable during the same `tick`, the last `action` to execute takes precedence.
53
63
54
-
The main loop will abort by default if no conditions evaluate to `true` during a single `tick`. This prevents the program from running forever.
64
+
##### Finite State Machines
65
+
66
+
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`.
67
+
68
+
This prevents the program from running forever by default, and can be disabled as needed.
55
69
56
70
#### State Manager
57
71
@@ -83,6 +97,10 @@ The main loop will abort by default if no conditions evaluate to `true` during a
83
97
84
98
-`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.
85
99
100
+
#### External inputs
101
+
102
+
`when` supports external inputs via the `@input` decorator. External inputs are readonly variables that are recorded as part of the state, but never manually
103
+
86
104
#### Note on Recombination
87
105
88
106
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.
@@ -104,7 +122,14 @@ Here are some abstract syntax examples for a full pseudo-language based on this
104
122
105
123
You can read about the original idea (slightly outdated) [in this proposal](https://gist.github.com/voodooattack/ccb1d18112720a8de5be660dbb80541c).
106
124
107
-
This is mostly pseudo-javascript with two extra `when` and `exit` keywords.
125
+
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:
126
+
127
+
-`@forever()` Must be defined a the start 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.
128
+
-`@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.
129
+
-`@unless(expression)` Prevents this action from triggering if `expression` evaluates to true. Can be used multiple times with the same action.
130
+
-`@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.
131
+
132
+
The above decorators may only precede a `when` block, and will only apply to the next encountered `when` block.
108
133
109
134
##### Examples
110
135
@@ -116,17 +141,23 @@ let current = 3; // start looking at 3
116
141
let primes = []; // array to store saved primes
117
142
118
143
// increment the counter with every tick till we hit the potential prime
144
+
@name('increment')
145
+
@unless(primes.length>=10)
119
146
when(counter<current) {
120
147
counter++;
121
148
}
122
149
123
150
// not a prime number, reset and increment current search
151
+
@name('resetNotAPrime')
152
+
@unless(primes.length>=10)
124
153
when(counter<current&¤t%counter===0) {
125
154
counter =2;
126
155
current++;
127
156
}
128
157
129
158
// if this is ever triggered, then we're dealing with a prime.
159
+
@name('capturePrime')
160
+
@unless(primes.length>=10)
130
161
when(counter>=current) {
131
162
// save the prime
132
163
primes.push(current);
@@ -136,14 +167,21 @@ when(counter >= current) {
136
167
counter =2;
137
168
current++;
138
169
}
170
+
```
171
+
172
+
To make this same machine with an explicit exit clause, simply remove all `@unless` decorators and add `@forever` at the beginning.
139
173
174
+
To make this machine exit, you must add the following anywhere in the file:
175
+
```js
140
176
// exit when we've found enough primes
177
+
@name('exitOnceDone')
141
178
when(primes.length>=10) {
142
179
exit();
143
180
}
144
181
```
145
182
146
-
Predicted exit state after `exit`:
183
+
With either option, the predicted exit state after the machine exits should be:
184
+
147
185
```json
148
186
{
149
187
"counter": 2,
@@ -177,6 +215,8 @@ See the [API documentation](https://voodooattack.github.io/when-ts/) for more in
177
215
178
216
### Usage
179
217
218
+
Some examples are located in in [examples/](examples).
219
+
180
220
- Simple example:
181
221
182
222
```typescript
@@ -191,17 +231,17 @@ class TestMachine extends EventMachine<State> {
191
231
super({ value: 0 }); // pass the initial state to the event machine
192
232
}
193
233
194
-
@when(true) // define a condition for this block to execute, in this case always
234
+
@when<State>(true) // define a condition for this block to execute, in this case always
195
235
reportOncePerTick(s:State, m:TestMachine) {
196
236
console.log(`beginning tick #${m.history.tick} with state`, s);
197
237
}
198
238
199
-
@when(state=>state.value<5) // this only executes when `value` is less than 5
200
-
incrementOncePerTick(s:State) { // increment `value` once per tick
239
+
@when<State>(state=>state.value<5) currentValue
240
+
incrementOncePerTick(s:State) { currentValue
201
241
return { value: s.value+1 };
202
242
}
203
243
204
-
@when(state=>state.value>=5) // this will only execute when `value` is >= 5
244
+
@when<State>(state=>state.value>=5) currentValue
205
245
exitWhenDone(s:State, m:TestMachine) {
206
246
console.log(`finished on tick #${m.history.tick}, exiting`, s);
0 commit comments