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

support for system as actor #4677

Open
wants to merge 32 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
32 commits
Select commit Hold shift + click to select a range
fcd1bca
feat(core): add support for subscribing to system registration events
cevr Jan 13, 2024
ce1cdc7
refactor: put systemId in event, use single set for all subscriptions…
cevr Jan 14, 2024
ace2404
make event type consistent with others
cevr Jan 14, 2024
5c94a82
make implementation consistent with createActor
cevr Jan 15, 2024
d9bd505
update changeset
cevr Jan 15, 2024
69fe023
reuse subscribable type
cevr Jan 15, 2024
afb5807
make system subscribable, and use snapshot
cevr Jan 25, 2024
4315d52
initialize as empty actors first
cevr Jan 25, 2024
5894a57
Merge remote-tracking branch 'upstream/main' into cevr/system-subscribe
cevr Jan 25, 2024
2a68b5b
update changeset
cevr Jan 25, 2024
8d289c3
copy snapshot
cevr Jan 25, 2024
0cbb0c0
ensure new references
cevr Jan 25, 2024
3326fb3
allow useSelector to subscribe to actor system
cevr Jan 25, 2024
dd87510
remove log
cevr Jan 25, 2024
81cf130
update system snapshot on _set, ensure subscribers are only called on…
cevr Jan 25, 2024
61f0b77
add changeset for @xstate/react
cevr Jan 25, 2024
6840952
remove unused imports
cevr Jan 26, 2024
ae2bfee
ensure snapshot is updated for scheduled events as well
cevr Jan 26, 2024
2dff805
collapse test
cevr Jan 26, 2024
6f1dd44
add nested child state to useSelector test
cevr Jan 26, 2024
2f416a1
Merge remote-tracking branch 'upstream/main' into cevr/system-subscribe
cevr Feb 20, 2024
2f0fbd9
Merge remote-tracking branch 'upstream/main' into cevr/system-subscribe
cevr Mar 5, 2024
3fbe432
revert unintended changes
cevr Mar 5, 2024
2c12777
revert unintended changes
cevr Mar 5, 2024
98ac5ee
Merge remote-tracking branch 'upstream/main' into cevr/system-subscribe
cevr Apr 7, 2024
e7d105b
Merge remote-tracking branch 'upstream/main' into cevr/system-subscribe
cevr Apr 23, 2024
299d1da
revert unintended changes
cevr Apr 23, 2024
d98a89f
Merge remote-tracking branch 'upstream/main' into cevr/system-subscribe
cevr May 24, 2024
60d3db7
revert change
cevr May 24, 2024
8eec984
Merge remote-tracking branch 'upstream/main' into cevr/system-subscribe
cevr Jul 29, 2024
fe762fc
Merge remote-tracking branch 'upstream/main' into cevr/system-subscribe
cevr Aug 16, 2024
841d26c
Merge branch 'main' into cevr/system-subscribe
cevr Aug 26, 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
12 changes: 12 additions & 0 deletions .changeset/tasty-baboons-dance.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
---
'xstate': minor
---

added support for `actor.system.subscribe` to subscribe to registration and unregistration events within an actor's system.

you can use `actor.system.subscribe` in two ways:

- `actor.system.subscribe(event => ...)` subscribes to all registration/unregistration events occurring within the system
- `actor.system.subscribe(systemId, event => ...)` subscribes to the registration/unregistration events of that given systemId.

`actor.system.subscribe` returns a `Subscription` object you can `.unsubscribe` to at any time.
102 changes: 102 additions & 0 deletions packages/core/src/system.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ import {
AnyActorRef,
Observer,
Snapshot,
Subscription,
HomomorphicOmit,
EventObject
} from './types.ts';
Expand Down Expand Up @@ -61,6 +62,20 @@ export interface ActorSystem<T extends ActorSystemInfo> {
*/
_set: <K extends keyof T['actors']>(key: K, actorRef: T['actors'][K]) => void;
get: <K extends keyof T['actors']>(key: K) => T['actors'][K] | undefined;
subscribe: {
cevr marked this conversation as resolved.
Show resolved Hide resolved
(
listener: (
event: RegistrationEvent<
T['actors'][keyof T['actors']],
keyof T['actors'] & string
>
) => void
): Subscription;
<K extends keyof T['actors']>(
key: K,
listener: (event: RegistrationEvent<T['actors'][K], K & string>) => void
): Subscription;
};
inspect: (observer: Observer<InspectionEvent>) => void;
/**
* @internal
Expand Down Expand Up @@ -91,6 +106,8 @@ export interface ActorSystem<T extends ActorSystemInfo> {

export type AnyActorSystem = ActorSystem<any>;

const rootSubscriptionKey = '@xstate.system.root' as const;
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

added a root subscription key that seemed unique enough while following some conventions I found.

This would be used in the case actor.system.subscribe(event => ...) was used with no systemId.


let idCounter = 0;
export function createSystem<T extends ActorSystemInfo>(
rootActor: AnyActorRef,
Expand All @@ -103,6 +120,10 @@ export function createSystem<T extends ActorSystemInfo>(
const keyedActors = new Map<keyof T['actors'], AnyActorRef | undefined>();
const reverseKeyedActors = new WeakMap<AnyActorRef, keyof T['actors']>();
const observers = new Set<Observer<InspectionEvent>>();
const systemSubscriptions = new Map<
keyof T['actors'] | typeof rootSubscriptionKey,
Set<(event: RegistrationEvent<T['actors'][keyof T['actors']]>) => void>
>();
const timerMap: { [id: ScheduledEventId]: number } = {};
const clock = options.clock;

Expand Down Expand Up @@ -164,6 +185,19 @@ export function createSystem<T extends ActorSystemInfo>(
_bookId: () => `x:${idCounter++}`,
_register: (sessionId, actorRef) => {
children.set(sessionId, actorRef);
const systemId = reverseKeyedActors.get(actorRef);
if (systemId !== undefined) {
const event = {
type: `@xstate.system.actor.register.${systemId as string}`,
actorRef: actorRef as T['actors'][keyof T['actors']]
} as const;
systemSubscriptions.get(systemId)?.forEach((listener) => {
listener(event);
});
systemSubscriptions.get(rootSubscriptionKey)?.forEach((listener) => {
listener(event);
});
}
return sessionId;
},
_unregister: (actorRef) => {
Expand All @@ -173,11 +207,56 @@ export function createSystem<T extends ActorSystemInfo>(
if (systemId !== undefined) {
keyedActors.delete(systemId);
reverseKeyedActors.delete(actorRef);
const event = {
type: `@xstate.system.actor.unregister.${systemId as string}`,
actorRef: actorRef as T['actors'][keyof T['actors']]
} as const;
systemSubscriptions.get(systemId)?.forEach((listener) => {
listener(event);
});
systemSubscriptions.get(rootSubscriptionKey)?.forEach((listener) => {
listener(event);
});
}
},
get: (systemId) => {
return keyedActors.get(systemId) as T['actors'][any];
},
subscribe: <K extends keyof T['actors']>(
maybeSystemIdOrListener:
| K
| ((event: RegistrationEvent<T['actors'][keyof T['actors']]>) => void),
maybeListener?:
| ((event: RegistrationEvent<T['actors'][keyof T['actors']]>) => void)
| undefined
) => {
let listener =
typeof maybeSystemIdOrListener === 'function'
? maybeSystemIdOrListener
: maybeListener!;
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

not sure if this is okay, typescript would prevent this but there's no runtime check

let systemId =
typeof maybeSystemIdOrListener === 'function'
? rootSubscriptionKey
: maybeSystemIdOrListener;

let subscriptions = systemSubscriptions.get(systemId) || new Set();
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think we could also design this differently by using a single set, and overriding the listeners if a systemId key is present. This would also mean passing the systemId directly in the event so its easier to parse

something like:

let listener = 
  typeof maybeSystemIdOrListener === 'function' ? 
  maybeSystemIdOrListener 
  : 
    (event) => { 
      if (event.systemId === maybeSystemIdOrListener) maybeListener(event)
     }

what do you think is preferable here?

subscriptions.add(listener);
systemSubscriptions.set(systemId, subscriptions);

return {
unsubscribe: () => {
const subscriptions = systemSubscriptions.get(systemId);
if (subscriptions) {
subscriptions.delete(listener);
if (subscriptions.size === 0) {
systemSubscriptions.delete(systemId);
} else {
systemSubscriptions.set(systemId, subscriptions);
}
}
}
};
},
_set: (systemId, actorRef) => {
const existing = keyedActors.get(systemId);
if (existing && existing !== actorRef) {
Expand Down Expand Up @@ -262,3 +341,26 @@ export type InspectionEvent =
| InspectedSnapshotEvent
| InspectedEventEvent
| InspectedActorEvent;

export interface ActorRegisteredEvent<
TActorRef extends AnyActorRef,
TSystemId extends string = string
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This made sense to me if there were any plans to allow for extracting systemId types from a machine.

Let me if it doesn't!

> {
type: `@xstate.system.actor.register.${TSystemId}`;
actorRef: TActorRef;
}

export interface ActorUnregisteredEvent<
TActorRef extends AnyActorRef,
TSystemId extends string = string
> {
type: `@xstate.system.actor.unregister.${TSystemId}`;
actorRef: TActorRef;
}

export type RegistrationEvent<
TActorRef extends AnyActorRef = AnyActorRef,
TSystemId extends string = string
> =
| ActorRegisteredEvent<TActorRef, TSystemId>
| ActorUnregisteredEvent<TActorRef, TSystemId>;
212 changes: 212 additions & 0 deletions packages/core/test/system.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -535,4 +535,216 @@ describe('system', () => {

expect(spy).toHaveBeenCalledTimes(1);
});

it('should allow subscribing to system registration events', () => {
const aSystemId = 'a_child';
const bSystemId = 'b_child';
const machine = createMachine({
initial: 'a',
states: {
a: {
invoke: {
src: createMachine({}),
systemId: aSystemId
},
on: {
to_b: 'b'
}
},
b: {
invoke: {
src: createMachine({}),
systemId: bSystemId
},
on: {
to_a: 'a'
}
}
}
});

const actorRef = createActor(machine).start();

const events: string[] = [];

actorRef.system.subscribe((event) => {
events.push(event.type);
});

actorRef.send({ type: 'to_b' });
expect(events).toEqual([
`@xstate.system.actor.unregister.${aSystemId}`,
`@xstate.system.actor.register.${bSystemId}`
]);
actorRef.send({ type: 'to_a' });
expect(events).toEqual([
`@xstate.system.actor.unregister.${aSystemId}`,
`@xstate.system.actor.register.${bSystemId}`,
`@xstate.system.actor.unregister.${bSystemId}`,
`@xstate.system.actor.register.${aSystemId}`
]);
});

it('should allow unsubscribing from a system registration subscription', () => {
const aSystemId = 'a_child';
const bSystemId = 'b_child';
const machine = createMachine({
initial: 'a',
states: {
a: {
invoke: {
src: createMachine({}),
systemId: aSystemId
},
on: {
to_b: 'b'
}
},
b: {
invoke: {
src: createMachine({}),
systemId: bSystemId
},
on: {
to_a: 'a'
}
}
}
});

const actorRef = createActor(machine).start();

const events: string[] = [];
const unsubscribedEvents: string[] = [];

const subscription = actorRef.system.subscribe((event) => {
events.push(event.type);
});

actorRef.system.subscribe((event) => {
unsubscribedEvents.push(event.type);
});

actorRef.send({ type: 'to_b' });
expect(events).toEqual([
`@xstate.system.actor.unregister.${aSystemId}`,
`@xstate.system.actor.register.${bSystemId}`
]);
expect(unsubscribedEvents).toEqual([
`@xstate.system.actor.unregister.${aSystemId}`,
`@xstate.system.actor.register.${bSystemId}`
]);

subscription.unsubscribe();
actorRef.send({ type: 'to_a' });

expect(events).toEqual([
`@xstate.system.actor.unregister.${aSystemId}`,
`@xstate.system.actor.register.${bSystemId}`
]);
expect(unsubscribedEvents).toEqual([
`@xstate.system.actor.unregister.${aSystemId}`,
`@xstate.system.actor.register.${bSystemId}`,
`@xstate.system.actor.unregister.${bSystemId}`,
`@xstate.system.actor.register.${aSystemId}`
]);
});

it('should allow subscribing to system registration events of a specific sysyemId', () => {
const aSystemId = 'a_child';
const bSystemId = 'b_child';
const machine = createMachine({
initial: 'a',
states: {
a: {
invoke: {
systemId: aSystemId,
src: createMachine({})
},
on: {
to_b: 'b'
}
},
b: {
invoke: {
src: createMachine({}),
systemId: bSystemId
},
on: {
to_a: 'a'
}
}
}
});

const actorRef = createActor(machine).start();

const aEvents: string[] = [];
const bEvents: string[] = [];

actorRef.system.subscribe(aSystemId, (event) => {
aEvents.push(event.type);
});

actorRef.system.subscribe(bSystemId, (event) => {
bEvents.push(event.type);
});

actorRef.send({ type: 'to_b' });
expect(aEvents).toEqual([`@xstate.system.actor.unregister.${aSystemId}`]);
expect(bEvents).toEqual([`@xstate.system.actor.register.${bSystemId}`]);
actorRef.send({ type: 'to_a' });
expect(aEvents).toEqual([
`@xstate.system.actor.unregister.${aSystemId}`,
`@xstate.system.actor.register.${aSystemId}`
]);
expect(bEvents).toEqual([
`@xstate.system.actor.register.${bSystemId}`,
`@xstate.system.actor.unregister.${bSystemId}`
]);
});

it('should allow unsubscribing from a system registration subscription of a specific systemId', () => {
const aSystemId = 'a_child';
const machine = createMachine({
initial: 'a',
states: {
a: {
invoke: {
systemId: aSystemId,
src: createMachine({})
},
on: {
to_b: 'b'
}
},
b: {
invoke: {
src: createMachine({}),
systemId: 'b_child'
},
on: {
to_a: 'a'
}
}
}
});

const actorRef = createActor(machine).start();

const events: string[] = [];
const unsubscribedEvents: string[] = [];

const subscription = actorRef.system.subscribe(aSystemId, (event) => {
events.push(event.type);
});

actorRef.send({ type: 'to_b' });
expect(events).toEqual([`@xstate.system.actor.unregister.${aSystemId}`]);

subscription.unsubscribe();

actorRef.send({ type: 'to_a' });
expect(events).toEqual([`@xstate.system.actor.unregister.${aSystemId}`]);
});
});