Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[core] Transition function #4954

Merged
merged 98 commits into from
Nov 12, 2024
Merged
Show file tree
Hide file tree
Changes from 12 commits
Commits
Show all changes
98 commits
Select commit Hold shift + click to select a range
9f6f015
Use defaultActionExecutor (temp)
davidkpiano Jun 4, 2024
abfe358
Convert entry/exit events
davidkpiano Jun 4, 2024
6e181ea
UnknownAction -> UnknownActionObject
davidkpiano Jun 4, 2024
f1100ab
Use defaultActionExecutor
davidkpiano Jun 5, 2024
8e25fa1
Cleanup
davidkpiano Jun 5, 2024
b2b974a
WIP
davidkpiano Jun 5, 2024
6dd475c
Merge branch 'main' into davidkpiano/transition
davidkpiano Jun 12, 2024
6533540
Merge branch 'main' into davidkpiano/transition
davidkpiano Jun 18, 2024
7641ec5
Fixing tests
davidkpiano Jun 19, 2024
9942382
Add toJSON to built-in actions
davidkpiano Jun 19, 2024
f566149
Merge branch 'main' into davidkpiano/transition
davidkpiano Jun 23, 2024
efaac40
Add docs
davidkpiano Jun 25, 2024
8b936ab
Update packages/core/src/actions/spawnChild.ts
davidkpiano Jul 3, 2024
ae70cc6
Update packages/core/src/stateUtils.ts
davidkpiano Jul 3, 2024
f3c360f
Unify convertAction
davidkpiano Jul 4, 2024
32b278c
Update packages/core/src/stateUtils.ts
davidkpiano Jul 4, 2024
b1d346e
Remove TODO
davidkpiano Jul 4, 2024
4f61ac7
Introduce `executeAction`, remove `action.execute()`
davidkpiano Jul 4, 2024
bc431d1
Enqueue actions test
davidkpiano Jul 4, 2024
96ba826
Fix type error
davidkpiano Jul 4, 2024
abf3a2c
Expose `executeAction(…)`
davidkpiano Jul 8, 2024
d97cb8f
Provide actor to action
davidkpiano Jul 8, 2024
5f6a00b
Fix type issue
davidkpiano Jul 8, 2024
c0597ee
Merge branch 'main' into davidkpiano/transition
davidkpiano Jul 11, 2024
59daf7c
Merge branch 'main' into davidkpiano/transition
davidkpiano Jul 11, 2024
1f13746
Merge branch 'main' into davidkpiano/transition
davidkpiano Jul 13, 2024
a73a0db
Changeset
davidkpiano Jul 13, 2024
4c8d491
Deprecate getNextSnapshot and getInitialSnapshot
davidkpiano Jul 14, 2024
cbb0a7f
Update packages/core/src/stateUtils.ts
davidkpiano Jul 18, 2024
18f0015
Restore getNextSnapshot.test.ts file
davidkpiano Jul 18, 2024
e2e821c
function -> exec
davidkpiano Jul 18, 2024
ad422a8
Merge branch 'main' into davidkpiano/transition
davidkpiano Jul 24, 2024
737f67f
Merge branch 'main' into davidkpiano/transition
davidkpiano Jul 30, 2024
f7991ad
Merge branch 'main' into davidkpiano/transition
davidkpiano Aug 3, 2024
97713c5
Merge branch 'main' into davidkpiano/transition
davidkpiano Aug 7, 2024
0c29c2c
Delayed raise action test
davidkpiano Aug 7, 2024
6e18296
Getting close
davidkpiano Aug 8, 2024
7146bdc
Update scheduler to handle delayed sendTo actions without an initiall…
davidkpiano Aug 9, 2024
e776f2c
Merge branch 'main' into davidkpiano/transition
davidkpiano Aug 10, 2024
3d76b58
Fix types
davidkpiano Aug 10, 2024
7e05a9a
Merge branch 'main' into davidkpiano/transition
davidkpiano Aug 17, 2024
857c96c
WIP
davidkpiano Aug 17, 2024
59eedc2
Remove test code
davidkpiano Aug 28, 2024
574c4c9
Merge branch 'main' into davidkpiano/transition
davidkpiano Sep 1, 2024
4cdd545
Default actor
davidkpiano Sep 1, 2024
d2e583f
Serialization in test
davidkpiano Sep 1, 2024
fea607b
Revert invoke.test.ts
davidkpiano Sep 5, 2024
7a33175
Cancel action execution
davidkpiano Sep 5, 2024
0c80c66
WIP
davidkpiano Sep 5, 2024
f596786
Merge branch 'main' into davidkpiano/transition
davidkpiano Sep 14, 2024
062676c
Update launch.json and jest.config.js
davidkpiano Sep 14, 2024
2d480e1
Proof of concept for invoked actions
davidkpiano Sep 14, 2024
2e450c7
Merge branch 'main' into davidkpiano/transition
davidkpiano Sep 23, 2024
a75ceb9
Include resolved input & systemId in spawnChild action
davidkpiano Sep 23, 2024
200fc12
Refactor action types to use ExecutableActionObject and add startedAt…
davidkpiano Sep 24, 2024
edb0615
Merge branch 'main' into davidkpiano/transition
davidkpiano Sep 26, 2024
96f4ac4
Add ExecutableActionsFrom
davidkpiano Sep 26, 2024
9c077d6
Clean up types
davidkpiano Sep 28, 2024
5bd5e09
Merge branch 'main' into davidkpiano/transition
davidkpiano Oct 9, 2024
cf9d549
Lint
davidkpiano Oct 9, 2024
68654be
Lint for real
davidkpiano Oct 9, 2024
7622a79
Lint lint
davidkpiano Oct 9, 2024
4f61f39
Back to any
davidkpiano Oct 9, 2024
a365418
Update packages/core/test/transition.test.ts
davidkpiano Oct 9, 2024
f6de768
use sleep
Andarist Oct 11, 2024
1977a9d
remove outdated comment
Andarist Oct 11, 2024
d202b93
add `ExecutableSendToAction` to `SpecialExecutableAction`
Andarist Oct 11, 2024
0c344da
remove `startedAt`
Andarist Oct 11, 2024
ece423d
add failing raise test case
Andarist Oct 12, 2024
9576616
tweak test title
Andarist Oct 12, 2024
85b1f67
add extra cancel tests
Andarist Oct 12, 2024
9d3f2a0
add failing test for invalid event delivery
Andarist Oct 12, 2024
4516833
Revert test (happens in main)
davidkpiano Oct 13, 2024
ab4bad0
Remove invalid test: The <cancel> element is used to cancel a delayed…
davidkpiano Oct 20, 2024
73c6b22
Fixed tests
Andarist Oct 21, 2024
07707e7
remove unused import
Andarist Oct 21, 2024
898fa99
add warn assertions
Andarist Oct 21, 2024
71324ad
Revert "Remove invalid test: The <cancel> element is used to cancel a…
Andarist Oct 23, 2024
ca2977f
make it green, make it green
Andarist Oct 25, 2024
c50d849
add action resolution capabilities to machine.executeAction
Andarist Oct 26, 2024
a7e83ce
share `resolvedInfo` between branches
Andarist Oct 26, 2024
3474c5c
bring back `ExecutableActionObject['exec']`
Andarist Oct 26, 2024
94d44f0
add a failing boilerplate for `cancel` execution
Andarist Oct 30, 2024
6242d5d
Fix cancel action
davidkpiano Nov 2, 2024
2a573fb
Add SpecialActionResolution type
davidkpiano Nov 2, 2024
91128d8
Add test for sendTo action
davidkpiano Nov 3, 2024
002670f
actorId -> targetId
davidkpiano Nov 4, 2024
a2d8fb0
Rename
davidkpiano Nov 4, 2024
b47f77f
Add tests for emit and log
davidkpiano Nov 4, 2024
c04c12c
Remove switch statement in executeAction
davidkpiano Nov 12, 2024
9772108
Merge branch 'main' into davidkpiano/transition
davidkpiano Nov 12, 2024
e87d847
Undo
davidkpiano Nov 12, 2024
a58a3f2
Remove executeAction for now
davidkpiano Nov 12, 2024
b0a7992
bring back one `toSerializableAction` call
Andarist Nov 12, 2024
c894ad4
fix one type issue
Andarist Nov 12, 2024
03dac75
tweak test titles
Andarist Nov 12, 2024
fa47906
remove redundant test
Andarist Nov 12, 2024
9ee7564
dont export `getAction`
Andarist Nov 12, 2024
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
13 changes: 8 additions & 5 deletions packages/core/src/StateMachine.ts
Original file line number Diff line number Diff line change
Expand Up @@ -46,7 +46,8 @@ import type {
SnapshotFrom,
StateMachineDefinition,
StateValue,
TransitionDefinition
TransitionDefinition,
UnknownActionObject
} from './types.ts';
import { resolveReferencedActor, toStatePath } from './utils.ts';

Expand Down Expand Up @@ -294,7 +295,8 @@ export class StateMachine<
TOutput,
TMeta
> {
return macrostep(snapshot, event, actorScope).snapshot as typeof snapshot;
return macrostep(snapshot, event, actorScope, [])
.snapshot as typeof snapshot;
}

/**
Expand Down Expand Up @@ -327,7 +329,7 @@ export class StateMachine<
TMeta
>
> {
return macrostep(snapshot, event, actorScope).microstates;
return macrostep(snapshot, event, actorScope, []).microstates;
}

public getTransitionData(
Expand Down Expand Up @@ -382,8 +384,9 @@ export class StateMachine<
preInitial,
initEvent,
actorScope,
[assign(assignment)],
internalQueue
[assign(assignment) as unknown as UnknownActionObject],
internalQueue,
undefined
) as SnapshotFrom<this>;
}

Expand Down
62 changes: 34 additions & 28 deletions packages/core/src/StateNode.ts
Original file line number Diff line number Diff line change
Expand Up @@ -30,7 +30,8 @@ import type {
AnyStateNodeConfig,
ProvidedActor,
NonReducibleUnknown,
EventDescriptor
EventDescriptor,
UnknownActionObject
} from './types.ts';
import {
createInvokeId,
Expand All @@ -41,21 +42,6 @@ import {

const EMPTY_OBJECT = {};

const toSerializableAction = (action: UnknownAction) => {
if (typeof action === 'string') {
return { type: action };
}
if (typeof action === 'function') {
if ('resolve' in action) {
return { type: (action as any).type };
}
return {
type: action.name
};
}
return action;
};

interface StateNodeOptions<
TContext extends MachineContext,
TEvent extends EventObject
Expand Down Expand Up @@ -105,11 +91,11 @@ export class StateNode<
/**
* The action(s) to be executed upon entering the state node.
*/
public entry: UnknownAction[];
public entry: UnknownActionObject[];
/**
* The action(s) to be executed upon exiting the state node.
*/
public exit: UnknownAction[];
public exit: UnknownActionObject[];
/**
* The parent state node.
*/
Expand Down Expand Up @@ -220,8 +206,28 @@ export class StateNode<
this.history =
this.config.history === true ? 'shallow' : this.config.history || false;

this.entry = toArray(this.config.entry).slice();
this.exit = toArray(this.config.exit).slice();
const convertAction = (
davidkpiano marked this conversation as resolved.
Show resolved Hide resolved
action: UnknownAction,
kind: 'entry' | 'exit',
i: number
): UnknownActionObject => {
if (typeof action === 'string') {
return { type: action };
}
if (typeof action === 'function' && !('resolve' in action)) {
const type = `${this.id}|${kind}:${i}`;
this.machine.implementations.actions[type] = action as any;
davidkpiano marked this conversation as resolved.
Show resolved Hide resolved
return { type };
}
return action as any;
};

this.entry = toArray(this.config.entry as UnknownAction).map((a, i) =>
convertAction(a, 'entry', i)
);
this.exit = toArray(this.config.exit as UnknownAction).map((a, i) =>
convertAction(a, 'exit', i)
);

this.meta = this.config.meta;
this.output =
Expand All @@ -233,8 +239,8 @@ export class StateNode<
public _initialize() {
this.transitions = formatTransitions(this);
if (this.config.always) {
this.always = toTransitionConfigArray(this.config.always).map((t) =>
formatTransition(this, NULL_EVENT, t)
this.always = toTransitionConfigArray(this.config.always).map((t, i) =>
formatTransition(this, NULL_EVENT, t, i)
);
}

Expand All @@ -256,13 +262,13 @@ export class StateNode<
? {
target: this.initial.target,
source: this,
actions: this.initial.actions.map(toSerializableAction),
actions: this.initial.actions,
eventType: null as any,
reenter: false,
toJSON: () => ({
target: this.initial!.target!.map((t) => `#${t.id}`),
source: `#${this.id}`,
actions: this.initial!.actions.map(toSerializableAction),
actions: this.initial!.actions,
eventType: null as any
})
}
Expand All @@ -274,10 +280,10 @@ export class StateNode<
on: this.on,
transitions: [...this.transitions.values()].flat().map((t) => ({
...t,
actions: t.actions.map(toSerializableAction)
actions: t.actions
})),
entry: this.entry.map(toSerializableAction),
exit: this.exit.map(toSerializableAction),
entry: this.entry,
exit: this.exit,
meta: this.meta,
order: this.order || -1,
output: this.output,
Expand Down Expand Up @@ -391,7 +397,7 @@ export class StateNode<
event: TEvent
): TransitionDefinition<TContext, TEvent>[] | undefined {
const eventType = event.type;
const actions: UnknownAction[] = [];
const actions: UnknownActionObject[] = [];

let selectedTransition: TransitionDefinition<TContext, TEvent> | undefined;

Expand Down
6 changes: 5 additions & 1 deletion packages/core/src/actions/assign.ts
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
import isDevelopment from '#is-development';
import { cloneMachineSnapshot } from '../State.ts';
import { executingCustomAction } from '../createActor.ts';
import { Spawner, createSpawner } from '../spawn.ts';
import { executingCustomAction } from '../stateUtils.ts';
import type {
ActionArgs,
AnyActorScope,
Expand Down Expand Up @@ -177,5 +177,9 @@ export function assign<

assign.resolve = resolveAssign;

assign.toJSON = () => ({
...assign
});
davidkpiano marked this conversation as resolved.
Show resolved Hide resolved

return assign;
}
4 changes: 4 additions & 0 deletions packages/core/src/actions/cancel.ts
Original file line number Diff line number Diff line change
Expand Up @@ -97,5 +97,9 @@ export function cancel<
cancel.resolve = resolveCancel;
cancel.execute = executeCancel;

cancel.toJSON = () => ({
...cancel
});

return cancel;
}
6 changes: 5 additions & 1 deletion packages/core/src/actions/emit.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
import isDevelopment from '#is-development';
import { executingCustomAction } from '../stateUtils.ts';
import { executingCustomAction } from '../createActor.ts';
import {
ActionArgs,
AnyActorScope,
Expand Down Expand Up @@ -143,5 +143,9 @@ export function emit<
emit.resolve = resolveEmit;
emit.execute = executeEmit;

emit.toJSON = () => ({
...emit
});

return emit;
}
4 changes: 4 additions & 0 deletions packages/core/src/actions/enqueueActions.ts
davidkpiano marked this conversation as resolved.
Show resolved Hide resolved
Original file line number Diff line number Diff line change
Expand Up @@ -304,5 +304,9 @@ export function enqueueActions<
enqueueActions.collect = collect;
enqueueActions.resolve = resolveEnqueueActions;

enqueueActions.toJSON = () => ({
...enqueueActions
});

return enqueueActions;
}
4 changes: 4 additions & 0 deletions packages/core/src/actions/log.ts
Original file line number Diff line number Diff line change
Expand Up @@ -95,5 +95,9 @@ export function log<
log.resolve = resolveLog;
log.execute = executeLog;

log.toJSON = () => ({
...log
});

return log;
}
6 changes: 5 additions & 1 deletion packages/core/src/actions/raise.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
import isDevelopment from '#is-development';
import { executingCustomAction } from '../stateUtils.ts';
import { executingCustomAction } from '../createActor.ts';
import {
ActionArgs,
ActionFunction,
Expand Down Expand Up @@ -165,5 +165,9 @@ export function raise<
raise.resolve = resolveRaise;
raise.execute = executeRaise;

raise.toJSON = () => ({
...raise
});

return raise;
}
6 changes: 5 additions & 1 deletion packages/core/src/actions/send.ts
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
import isDevelopment from '#is-development';
import { XSTATE_ERROR } from '../constants.ts';
import { createErrorActorEvent } from '../eventUtils.ts';
import { executingCustomAction } from '../stateUtils.ts';
import { executingCustomAction } from '../createActor.ts';
import {
ActionArgs,
ActionFunction,
Expand Down Expand Up @@ -258,6 +258,10 @@ export function sendTo<
sendTo.retryResolve = retryResolveSendTo;
sendTo.execute = executeSendTo;

sendTo.toJSON = () => ({
...sendTo
});

return sendTo;
}

Expand Down
6 changes: 5 additions & 1 deletion packages/core/src/actions/spawnChild.ts
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
import isDevelopment from '#is-development';
import { cloneMachineSnapshot } from '../State.ts';
import { ProcessingStatus, createActor } from '../createActor.ts';
import { executingCustomAction } from '../stateUtils.ts';
// import { executingCustomAction } from '../createActor.ts';
davidkpiano marked this conversation as resolved.
Show resolved Hide resolved
import {
ActionArgs,
ActionFunction,
Expand Down Expand Up @@ -225,5 +225,9 @@ export function spawnChild<
spawnChild.resolve = resolveSpawn;
spawnChild.execute = executeSpawn;

spawnChild.toJSON = () => ({
...spawnChild
});

return spawnChild;
}
5 changes: 4 additions & 1 deletion packages/core/src/actions/stopChild.ts
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,6 @@ import { cloneMachineSnapshot } from '../State.ts';
import { ProcessingStatus } from '../createActor.ts';
import {
ActionArgs,
ActorRef,
AnyActorRef,
AnyActorScope,
AnyMachineSnapshot,
Expand Down Expand Up @@ -116,6 +115,10 @@ export function stopChild<
stop.resolve = resolveStop;
stop.execute = executeStop;

stop.toJSON = () => ({
...stop
});

return stop;
}

Expand Down
25 changes: 25 additions & 0 deletions packages/core/src/createActor.ts
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,8 @@ import { reportUnhandledError } from './reportUnhandledError.ts';
import { symbolObservable } from './symbolObservable.ts';
import { AnyActorSystem, Clock, createSystem } from './system.ts';

export let executingCustomAction: ((...args: any[]) => void) | false = false;

import type {
ActorScope,
AnyActorLogic,
Expand Down Expand Up @@ -205,6 +207,29 @@ export class Actor<TLogic extends AnyActorLogic>
for (const handler of Array.from(allListeners)) {
handler(emittedEvent);
}
},
actionExecutor: (action) => {
const exec = () => {
this._actorScope.system._sendInspectionEvent({
type: '@xstate.action',
actorRef: this,
action: {
type: action.type,
params: action.params as any // TODO: fix types
}
});
try {
executingCustomAction = action.execute;
action.execute();
} finally {
executingCustomAction = false;
}
};
if (this._processingStatus === ProcessingStatus.Running) {
exec();
} else {
this._deferred.push(exec);
}
}
};

Expand Down
3 changes: 2 additions & 1 deletion packages/core/src/getNextSnapshot.ts
Original file line number Diff line number Diff line change
Expand Up @@ -27,7 +27,8 @@ export function createInertActorScope<T extends AnyActorLogic>(
sessionId: '',
stopChild: () => {},
system: self.system,
emit: () => {}
emit: () => {},
actionExecutor: () => {}
};

return inertActorScope;
Expand Down
1 change: 1 addition & 0 deletions packages/core/src/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -38,6 +38,7 @@ export {
type Interpreter
};
export { assertEvent } from './assert.ts';
export { transition } from './transition.ts';

declare global {
interface SymbolConstructor {
Expand Down
Loading
Loading