atom.io 0.46.15 → 0.46.17

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,7 +1,8 @@
1
1
  import { Fn, RootStore, Store, Transceiver } from "atom.io/internal";
2
2
  import * as AtomIO from "atom.io";
3
- import { AtomToken, ReadonlyPureSelectorToken, RegularAtomToken, WritableToken } from "atom.io";
3
+ import { AtomToken, ReadonlyPureSelectorToken, RegularAtomToken, ViewOf, WritableToken } from "atom.io";
4
4
  import { Canonical, Json } from "atom.io/json";
5
+ import { UList } from "atom.io/transceivers/u-list";
5
6
  import { ContinuityToken, RoomKey, Socket, SocketKey, UserKey } from "atom.io/realtime";
6
7
 
7
8
  //#region src/realtime-client/continuity/register-and-attempt-confirmed-update.d.ts
@@ -38,6 +39,8 @@ declare function pushState<J extends Json.Serializable>(store: Store, socket: So
38
39
  declare const mySocketKeyAtom: RegularAtomToken<SocketKey | undefined>;
39
40
  declare const myUserKeyAtom: RegularAtomToken<UserKey | null>;
40
41
  declare const myRoomKeySelector: ReadonlyPureSelectorToken<RoomKey | null>;
42
+ declare const usersHereSelector: ReadonlyPureSelectorToken<ViewOf<UList<UserKey> | null>>;
43
+ declare const roomOwnerSelector: ReadonlyPureSelectorToken<UserKey | null>;
41
44
  //#endregion
42
45
  //#region src/realtime-client/realtime-client-stores/client-sync-store.d.ts
43
46
  declare const optimisticUpdateQueueAtom: AtomIO.RegularAtomToken<AtomIO.TransactionOutcomeEvent<any>[]>;
@@ -46,5 +49,5 @@ declare const confirmedUpdateQueueAtom: AtomIO.RegularAtomToken<AtomIO.Transacti
46
49
  //#region src/realtime-client/sync-continuity.d.ts
47
50
  declare function syncContinuity(store: RootStore, socket: Socket, continuity: ContinuityToken): () => void;
48
51
  //#endregion
49
- export { confirmedUpdateQueueAtom, myRoomKeySelector, mySocketKeyAtom, myUserKeyAtom, optimisticUpdateQueueAtom, pullAtom, pullAtomFamilyMember, pullMutableAtom, pullMutableAtomFamilyMember, pullSelector, pullSelectorFamilyMember, pushState, syncContinuity, useConcealState, useRegisterAndAttemptConfirmedUpdate, useRevealState };
52
+ export { confirmedUpdateQueueAtom, myRoomKeySelector, mySocketKeyAtom, myUserKeyAtom, optimisticUpdateQueueAtom, pullAtom, pullAtomFamilyMember, pullMutableAtom, pullMutableAtomFamilyMember, pullSelector, pullSelectorFamilyMember, pushState, roomOwnerSelector, syncContinuity, useConcealState, useRegisterAndAttemptConfirmedUpdate, useRevealState, usersHereSelector };
50
53
  //# sourceMappingURL=index.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.ts","names":["mySocketKeyAtom: RegularAtomToken<SocketKey | undefined>","myUserKeyAtom: RegularAtomToken<UserKey | null>","myRoomKeySelector: ReadonlyPureSelectorToken<RoomKey | null>","optimisticUpdateQueueAtom: AtomIO.RegularAtomToken<\n\tAtomIO.TransactionOutcomeEvent<any>[]\n>","confirmedUpdateQueueAtom: AtomIO.RegularAtomToken<\n\tAtomIO.TransactionOutcomeEvent<any>[]\n>"],"sources":["../../src/realtime-client/continuity/register-and-attempt-confirmed-update.ts","../../src/realtime-client/continuity/use-conceal-state.ts","../../src/realtime-client/continuity/use-reveal-state.ts","../../src/realtime-client/pull-atom.ts","../../src/realtime-client/pull-atom-family-member.ts","../../src/realtime-client/pull-mutable-atom.ts","../../src/realtime-client/pull-mutable-atom-family-member.ts","../../src/realtime-client/pull-selector.ts","../../src/realtime-client/pull-selector-family-member.ts","../../src/realtime-client/push-state.ts","../../src/realtime-client/realtime-client-stores/client-main-store.ts","../../src/realtime-client/realtime-client-stores/client-sync-store.ts","../../src/realtime-client/sync-continuity.ts"],"sourcesContent":[],"mappings":";;;;;;;cAiBa,8CAEJ,0CAEC,oCACoB,MAAA,CAAO,wBAClC,MAAA,CAAO,iBAAiB,mCAEE,MAAA,CAAO,wBACjC,MAAA,CAAO,iBAAiB,uBAId,MAAA,CAAO,wBAAwB,MAAA,CAAO,iBAAiB,OACjE,IAAA,CAAK;;;iBC1BQ,eAAA,QAAuB,oBACnB,UAAU,IAAA,CAAK;;;iBCHnB,cAAA,QAAsB,mBACnB,IAAA,CAAK;;;iBCGR,mBAAmB,IAAA,CAAK,qBAChC,eACC,eACD,MAAA,CAAO,iBAAiB;;;iBCHhB,+BACL,IAAA,CAAK,wBACL,UJQX,CAAA,KAAa,EINL,KJMK,EAAA,MAAA,EILJ,MJKI,EAAA,MAAA,EIJJ,MAAA,CAAO,eJkBR,CIlBwB,CJkBxB,EIlB2B,CJkB3B,EAAA,GAAA,CAAA,EAAA,GAAA,EIjBF,OJiBE,CIjBM,CJiBN,CAAA,CAAA,EAAA,GAAA,GAAA,IAAA;;;iBKxBQ,0BAA0B,mCAClC,eACC,eACD,MAAA,CAAO,iBAAiB;;;iBCGhB,sCACL,sCACA,UNEX,CAAA,KAAa,EMAL,KNAK,EAAA,MAAA,EMCJ,MNDI,EAAA,MAAA,EMEJ,MAAA,CAAO,sBNYR,CMZ+B,CNY/B,EMZkC,CNYlC,CAAA,EAAA,GAAA,EMXF,ONWE,CMXM,CNWN,CAAA,CAAA,EAAA,GAAA,GAAA,IAAA;;;iBOzBQ,uBACR,eACC,eACD,MAAA,CAAO,cAAc;;;iBCDb,sCAAsC,URStD,CAAA,KAAa,EQRL,KRQK,EAAA,MAAA,EQPJ,MROI,EAAA,WAcL,EQpBM,MAAA,CAAO,mBRoBb,CQpBiC,CRoBjC,EQpBoC,CRoBpC,CAAA,EAAA,GAAA,EQnBF,ORmBE,CQnBM,CRmBN,CAAA,CAAA,EAAA,GAAA,GAAA,IAAA;;;iBStBQ,oBAAoB,IAAA,CAAK,qBACjC,eACC,eACD,cAAc;;;cCPTA,iBAAiB,iBAAiB;cAKlCC,eAAe,iBAAiB;cAchCC,mBAAmB,0BAA0B;;;cCtB7CC,2BAA2B,MAAA,CAAO,iBAC9C,MAAA,CAAO;cAMKC,0BAA0B,MAAA,CAAO,iBAC7C,MAAA,CAAO;;;iBCUQ,cAAA,QACR,mBACC,oBACI"}
1
+ {"version":3,"file":"index.d.ts","names":["mySocketKeyAtom: RegularAtomToken<SocketKey | undefined>","myUserKeyAtom: RegularAtomToken<UserKey | null>","myRoomKeySelector: ReadonlyPureSelectorToken<RoomKey | null>","usersHereSelector: ReadonlyPureSelectorToken<\n\tViewOf<UList<UserKey> | null>\n>","roomOwnerSelector: ReadonlyPureSelectorToken<UserKey | null>","optimisticUpdateQueueAtom: AtomIO.RegularAtomToken<\n\tAtomIO.TransactionOutcomeEvent<any>[]\n>","confirmedUpdateQueueAtom: AtomIO.RegularAtomToken<\n\tAtomIO.TransactionOutcomeEvent<any>[]\n>"],"sources":["../../src/realtime-client/continuity/register-and-attempt-confirmed-update.ts","../../src/realtime-client/continuity/use-conceal-state.ts","../../src/realtime-client/continuity/use-reveal-state.ts","../../src/realtime-client/pull-atom.ts","../../src/realtime-client/pull-atom-family-member.ts","../../src/realtime-client/pull-mutable-atom.ts","../../src/realtime-client/pull-mutable-atom-family-member.ts","../../src/realtime-client/pull-selector.ts","../../src/realtime-client/pull-selector-family-member.ts","../../src/realtime-client/push-state.ts","../../src/realtime-client/realtime-client-stores/client-main-store.ts","../../src/realtime-client/realtime-client-stores/client-sync-store.ts","../../src/realtime-client/sync-continuity.ts"],"sourcesContent":[],"mappings":";;;;;;;;cAiBa,8CAEJ,0CAEC,oCACoB,MAAA,CAAO,wBAClC,MAAA,CAAO,iBAAiB,mCAEE,MAAA,CAAO,wBACjC,MAAA,CAAO,iBAAiB,uBAId,MAAA,CAAO,wBAAwB,MAAA,CAAO,iBAAiB,OACjE,IAAA,CAAK;;;iBC1BQ,eAAA,QAAuB,oBACnB,UAAU,IAAA,CAAK;;;iBCHnB,cAAA,QAAsB,mBACnB,IAAA,CAAK;;;iBCGR,mBAAmB,IAAA,CAAK,qBAChC,eACC,eACD,MAAA,CAAO,iBAAiB;;;iBCHhB,+BACL,IAAA,CAAK,wBACL,kBAEH,eACC,gBACA,MAAA,CAAO,gBAAgB,GAAG,cAC7B,QAAQ;;;iBCPE,0BAA0B,mCAClC,eACC,eACD,MAAA,CAAO,iBAAiB;;;iBCGhB,sCACL,sCACA,kBAEH,eACC,gBACA,MAAA,CAAO,uBAAuB,GAAG,SACpC,QAAQ;;;iBCdE,uBACR,eACC,eACD,MAAA,CAAO,cAAc;;;iBCDb,sCAAsC,kBAC9C,eACC,qBACK,MAAA,CAAO,oBAAoB,GAAG,SACtC,QAAQ;;;iBCHE,oBAAoB,IAAA,CAAK,qBACjC,eACC,eACD,cAAc;;;cCFTA,iBAAiB,iBAAiB;cAKlCC,eAAe,iBAAiB;cAchCC,mBAAmB,0BAA0B;cAqB7CC,mBAAmB,0BAC/B,OAAO,MAAM;cAYDC,mBAAmB,0BAA0B;;;cC7D7CC,2BAA2B,MAAA,CAAO,iBAC9C,MAAA,CAAO;cAMKC,0BAA0B,MAAA,CAAO,iBAC7C,MAAA,CAAO;;;iBCUQ,cAAA,QACR,mBACC,oBACI"}
@@ -2,7 +2,7 @@ import { Future, actUponStore, assignTransactionToContinuity, disposeAtom, findI
2
2
  import * as AtomIO from "atom.io";
3
3
  import { atom, getInternalRelations, selector } from "atom.io";
4
4
  import { parseJson } from "atom.io/json";
5
- import { employSocket, mutexAtoms, usersInRooms } from "atom.io/realtime";
5
+ import { employSocket, mutexAtoms, ownersOfRooms, usersInRooms } from "atom.io/realtime";
6
6
 
7
7
  //#region src/realtime-client/realtime-client-stores/client-main-store.ts
8
8
  const mySocketKeyAtom = atom({
@@ -30,6 +30,26 @@ const myRoomKeySelector = selector({
30
30
  return null;
31
31
  }
32
32
  });
33
+ const usersHereSelector = selector({
34
+ key: `usersHere`,
35
+ get: ({ get, find }) => {
36
+ const myRoomKey = get(myRoomKeySelector);
37
+ if (!myRoomKey) return null;
38
+ const [usersInRoomsAtoms] = getInternalRelations(usersInRooms, `split`);
39
+ return get(find(usersInRoomsAtoms, myRoomKey));
40
+ }
41
+ });
42
+ const roomOwnerSelector = selector({
43
+ key: `roomOwner`,
44
+ get: ({ get }) => {
45
+ const myRoomKey = get(myRoomKeySelector);
46
+ if (!myRoomKey) return null;
47
+ const [, ownerOfRoomsAtoms] = getInternalRelations(ownersOfRooms, `split`);
48
+ const owner = get(ownerOfRoomsAtoms, myRoomKey);
49
+ for (const userKey of owner) return userKey;
50
+ return null;
51
+ }
52
+ });
33
53
 
34
54
  //#endregion
35
55
  //#region src/realtime-client/realtime-client-stores/client-sync-store.ts
@@ -420,5 +440,5 @@ function syncContinuity(store, socket, continuity) {
420
440
  }
421
441
 
422
442
  //#endregion
423
- export { confirmedUpdateQueueAtom, myRoomKeySelector, mySocketKeyAtom, myUserKeyAtom, optimisticUpdateQueueAtom, pullAtom, pullAtomFamilyMember, pullMutableAtom, pullMutableAtomFamilyMember, pullSelector, pullSelectorFamilyMember, pushState, syncContinuity, useConcealState, useRegisterAndAttemptConfirmedUpdate, useRevealState };
443
+ export { confirmedUpdateQueueAtom, myRoomKeySelector, mySocketKeyAtom, myUserKeyAtom, optimisticUpdateQueueAtom, pullAtom, pullAtomFamilyMember, pullMutableAtom, pullMutableAtomFamilyMember, pullSelector, pullSelectorFamilyMember, pushState, roomOwnerSelector, syncContinuity, useConcealState, useRegisterAndAttemptConfirmedUpdate, useRevealState, usersHereSelector };
424
444
  //# sourceMappingURL=index.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"index.js","names":["mySocketKeyAtom: RegularAtomToken<SocketKey | undefined>","myUserKeyAtom: RegularAtomToken<UserKey | null>","myRoomKeySelector: ReadonlyPureSelectorToken<RoomKey | null>","optimisticUpdateQueueAtom: AtomIO.RegularAtomToken<\n\tAtomIO.TransactionOutcomeEvent<any>[]\n>","confirmedUpdateQueueAtom: AtomIO.RegularAtomToken<\n\tAtomIO.TransactionOutcomeEvent<any>[]\n>","continuityEpoch: number | undefined","k: any","v: any","subscriptions: WeakMap<Socket, Map<string, SubData>>","socketIds: WeakMap<Socket, string | undefined>","atom","k: any","v: any"],"sources":["../../src/realtime-client/realtime-client-stores/client-main-store.ts","../../src/realtime-client/realtime-client-stores/client-sync-store.ts","../../src/realtime-client/continuity/register-and-attempt-confirmed-update.ts","../../src/realtime-client/continuity/use-conceal-state.ts","../../src/realtime-client/continuity/use-reveal-state.ts","../../src/realtime-client/create-subscriber.ts","../../src/realtime-client/pull-atom.ts","../../src/realtime-client/pull-atom-family-member.ts","../../src/realtime-client/pull-mutable-atom.ts","../../src/realtime-client/pull-mutable-atom-family-member.ts","../../src/realtime-client/pull-selector-roots.ts","../../src/realtime-client/pull-selector.ts","../../src/realtime-client/pull-selector-family-member.ts","../../src/realtime-client/push-state.ts","../../src/realtime-client/sync-continuity.ts"],"sourcesContent":["import type { ReadonlyPureSelectorToken, RegularAtomToken } from \"atom.io\"\nimport { atom, getInternalRelations, selector } from \"atom.io\"\nimport type { RoomKey, SocketKey, UserKey } from \"atom.io/realtime\"\nimport { usersInRooms } from \"atom.io/realtime\"\n\nexport const mySocketKeyAtom: RegularAtomToken<SocketKey | undefined> = atom({\n\tkey: `mySocketKey`,\n\tdefault: undefined,\n})\n\nexport const myUserKeyAtom: RegularAtomToken<UserKey | null> = atom({\n\tkey: `myUserKey`,\n\tdefault: null,\n\teffects: [\n\t\t(userKey) => {\n\t\t\tif (typeof window !== `undefined`) {\n\t\t\t\tvoid import(`atom.io/web`).then(({ storageSync }) => {\n\t\t\t\t\tstorageSync(globalThis.localStorage, JSON, `myUserKey`)(userKey)\n\t\t\t\t})\n\t\t\t}\n\t\t},\n\t],\n})\n\nexport const myRoomKeySelector: ReadonlyPureSelectorToken<RoomKey | null> =\n\tselector({\n\t\tkey: `myRoomKey`,\n\t\tget: ({ get }) => {\n\t\t\tif (\n\t\t\t\t`process` in globalThis &&\n\t\t\t\t`env` in process &&\n\t\t\t\t`REALTIME_ROOM_KEY` in process.env\n\t\t\t) {\n\t\t\t\t// if a room running server-side wants its own key, this is where it lives\n\t\t\t\treturn process.env[`REALTIME_ROOM_KEY`] as RoomKey\n\t\t\t}\n\t\t\tconst myUserKey = get(myUserKeyAtom)\n\t\t\tif (!myUserKey) return null\n\t\t\tconst [, usersInRoomsAtoms] = getInternalRelations(usersInRooms, `split`)\n\t\t\tconst roomKeys = get(usersInRoomsAtoms, myUserKey)\n\t\t\tfor (const roomKey of roomKeys) return roomKey\n\t\t\treturn null\n\t\t},\n\t})\n","import * as AtomIO from \"atom.io\"\n\nexport const optimisticUpdateQueueAtom: AtomIO.RegularAtomToken<\n\tAtomIO.TransactionOutcomeEvent<any>[]\n> = AtomIO.atom<AtomIO.TransactionOutcomeEvent<any>[]>({\n\tkey: `optimisticUpdateQueue`,\n\tdefault: () => [],\n})\n\nexport const confirmedUpdateQueueAtom: AtomIO.RegularAtomToken<\n\tAtomIO.TransactionOutcomeEvent<any>[]\n> = AtomIO.atom<AtomIO.TransactionOutcomeEvent<any>[]>({\n\tkey: `confirmedUpdateQueue`,\n\tdefault: () => [],\n})\n","import type * as AtomIO from \"atom.io\"\nimport type { Fn, RootStore } from \"atom.io/internal\"\nimport {\n\tactUponStore,\n\tgetEpochNumberOfContinuity,\n\tingestTransactionOutcomeEvent,\n\tsetEpochNumberOfContinuity,\n\tsetIntoStore,\n} from \"atom.io/internal\"\nimport type { Json } from \"atom.io/json\"\nimport type { Socket } from \"atom.io/realtime\"\n\nimport {\n\tconfirmedUpdateQueueAtom,\n\toptimisticUpdateQueueAtom,\n} from \"../realtime-client-stores\"\n\nexport const useRegisterAndAttemptConfirmedUpdate =\n\t(\n\t\tstore: RootStore,\n\t\tcontinuityKey: string,\n\t\tsocket: Socket,\n\t\toptimisticUpdates: readonly AtomIO.TransactionOutcomeEvent<\n\t\t\tAtomIO.TransactionToken<Fn>\n\t\t>[],\n\t\tconfirmedUpdates: readonly AtomIO.TransactionOutcomeEvent<\n\t\t\tAtomIO.TransactionToken<Fn>\n\t\t>[],\n\t) =>\n\t(\n\t\tconfirmed: AtomIO.TransactionOutcomeEvent<AtomIO.TransactionToken<Fn>> &\n\t\t\tJson.Serializable,\n\t): void => {\n\t\tfunction reconcileEpoch(\n\t\t\toptimisticUpdate: AtomIO.TransactionOutcomeEvent<\n\t\t\t\tAtomIO.TransactionToken<Fn>\n\t\t\t>,\n\t\t\tconfirmedUpdate: AtomIO.TransactionOutcomeEvent<\n\t\t\t\tAtomIO.TransactionToken<Fn>\n\t\t\t>,\n\t\t): void {\n\t\t\tstore.logger.info(\n\t\t\t\t`🧑‍⚖️`,\n\t\t\t\t`continuity`,\n\t\t\t\tcontinuityKey,\n\t\t\t\t`reconciling updates`,\n\t\t\t)\n\t\t\tsetIntoStore(store, optimisticUpdateQueueAtom, (queue) => {\n\t\t\t\tqueue.shift()\n\t\t\t\treturn queue\n\t\t\t})\n\t\t\tif (optimisticUpdate.id === confirmedUpdate.id) {\n\t\t\t\tconst clientResult = JSON.stringify(optimisticUpdate.subEvents)\n\t\t\t\tconst serverResult = JSON.stringify(confirmedUpdate.subEvents)\n\t\t\t\tif (clientResult === serverResult) {\n\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t`✅`,\n\t\t\t\t\t\t`continuity`,\n\t\t\t\t\t\tcontinuityKey,\n\t\t\t\t\t\t`results for ${optimisticUpdate.id} match between client and server`,\n\t\t\t\t\t)\n\t\t\t\t\tsocket.emit(`ack:${continuityKey}`, confirmedUpdate.epoch)\n\t\t\t\t\treturn\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\t// id mismatch\n\t\t\t\tstore.logger.info(\n\t\t\t\t\t`❌`,\n\t\t\t\t\t`continuity`,\n\t\t\t\t\tcontinuityKey,\n\t\t\t\t\t`thought update #${confirmedUpdate.epoch} was ${optimisticUpdate.token.key}:${optimisticUpdate.id}, but it was actually ${confirmedUpdate.token.key}:${confirmedUpdate.id}`,\n\t\t\t\t)\n\t\t\t}\n\t\t\tstore.logger.info(\n\t\t\t\t`🧑‍⚖️`,\n\t\t\t\t`continuity`,\n\t\t\t\tcontinuityKey,\n\t\t\t\t`updates do not match`,\n\t\t\t\toptimisticUpdate,\n\t\t\t\tconfirmedUpdate,\n\t\t\t)\n\t\t\tconst reversedOptimisticUpdates = optimisticUpdates.toReversed()\n\t\t\tfor (const subsequentOptimistic of reversedOptimisticUpdates) {\n\t\t\t\tingestTransactionOutcomeEvent(store, subsequentOptimistic, `oldValue`)\n\t\t\t}\n\t\t\tstore.logger.info(\n\t\t\t\t`⏪`,\n\t\t\t\t`continuity`,\n\t\t\t\tcontinuityKey,\n\t\t\t\t`undid optimistic updates:`,\n\t\t\t\treversedOptimisticUpdates,\n\t\t\t)\n\t\t\tingestTransactionOutcomeEvent(store, optimisticUpdate, `oldValue`)\n\t\t\tstore.logger.info(\n\t\t\t\t`⏪`,\n\t\t\t\t`continuity`,\n\t\t\t\tcontinuityKey,\n\t\t\t\t`undid zeroth optimistic update`,\n\t\t\t\toptimisticUpdate,\n\t\t\t)\n\t\t\tingestTransactionOutcomeEvent(store, confirmedUpdate, `newValue`)\n\t\t\tstore.logger.info(\n\t\t\t\t`⏩`,\n\t\t\t\t`continuity`,\n\t\t\t\tcontinuityKey,\n\t\t\t\t`applied confirmed update`,\n\t\t\t\tconfirmedUpdate,\n\t\t\t)\n\t\t\tsocket.emit(`ack:${continuityKey}`, confirmedUpdate.epoch)\n\n\t\t\tfor (const subsequentOptimistic of optimisticUpdates) {\n\t\t\t\tconst token = {\n\t\t\t\t\ttype: `transaction`,\n\t\t\t\t\tkey: subsequentOptimistic.token.key,\n\t\t\t\t} as const\n\t\t\t\tconst { id, params } = subsequentOptimistic\n\t\t\t\tactUponStore(store, token, id)(...params)\n\t\t\t}\n\t\t\tstore.logger.info(\n\t\t\t\t`⏩`,\n\t\t\t\t`continuity`,\n\t\t\t\tcontinuityKey,\n\t\t\t\t`reapplied subsequent optimistic updates:`,\n\t\t\t\toptimisticUpdates,\n\t\t\t)\n\t\t}\n\n\t\tstore.logger.info(\n\t\t\t`🧑‍⚖️`,\n\t\t\t`continuity`,\n\t\t\tcontinuityKey,\n\t\t\t`integrating confirmed update`,\n\t\t\t{ confirmedUpdate: confirmed, confirmedUpdates, optimisticUpdates },\n\t\t)\n\t\tconst zerothOptimisticUpdate = optimisticUpdates[0]\n\t\tif (zerothOptimisticUpdate) {\n\t\t\tstore.logger.info(\n\t\t\t\t`🧑‍⚖️`,\n\t\t\t\t`continuity`,\n\t\t\t\tcontinuityKey,\n\t\t\t\t`has optimistic updates to reconcile`,\n\t\t\t)\n\t\t\tif (confirmed.epoch === zerothOptimisticUpdate.epoch) {\n\t\t\t\tstore.logger.info(\n\t\t\t\t\t`🧑‍⚖️`,\n\t\t\t\t\t`continuity`,\n\t\t\t\t\tcontinuityKey,\n\t\t\t\t\t`epoch of confirmed update #${confirmed.epoch} matches zeroth optimistic update`,\n\t\t\t\t)\n\t\t\t\treconcileEpoch(zerothOptimisticUpdate, confirmed)\n\t\t\t\tfor (const nextConfirmed of confirmedUpdates) {\n\t\t\t\t\tconst nextOptimistic = optimisticUpdates[0]\n\t\t\t\t\tif (nextConfirmed.epoch === nextOptimistic?.epoch) {\n\t\t\t\t\t\treconcileEpoch(nextOptimistic, nextConfirmed)\n\t\t\t\t\t} else {\n\t\t\t\t\t\tbreak\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\t// epoch mismatch\n\t\t\t\tstore.logger.info(\n\t\t\t\t\t`🧑‍⚖️`,\n\t\t\t\t\t`continuity`,\n\t\t\t\t\tcontinuityKey,\n\t\t\t\t\t`epoch of confirmed update #${confirmed.epoch} does not match zeroth optimistic update #${zerothOptimisticUpdate.epoch}`,\n\t\t\t\t)\n\t\t\t\tconst confirmedUpdateIsAlreadyEnqueued = confirmedUpdates.some(\n\t\t\t\t\t(update) => update.epoch === confirmed.epoch,\n\t\t\t\t)\n\t\t\t\tif (!confirmedUpdateIsAlreadyEnqueued) {\n\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t`👈`,\n\t\t\t\t\t\t`continuity`,\n\t\t\t\t\t\tcontinuityKey,\n\t\t\t\t\t\t`pushing confirmed update to queue`,\n\t\t\t\t\t\tconfirmed,\n\t\t\t\t\t)\n\t\t\t\t\tsetIntoStore(store, confirmedUpdateQueueAtom, (queue) => {\n\t\t\t\t\t\tqueue.push(confirmed)\n\t\t\t\t\t\tqueue.sort((a, b) => a.epoch - b.epoch)\n\t\t\t\t\t\treturn queue\n\t\t\t\t\t})\n\t\t\t\t}\n\t\t\t}\n\t\t} else {\n\t\t\tstore.logger.info(\n\t\t\t\t`🧑‍⚖️`,\n\t\t\t\t`continuity`,\n\t\t\t\tcontinuityKey,\n\t\t\t\t`has no optimistic updates to deal with`,\n\t\t\t)\n\t\t\tlet continuityEpoch: number | undefined\n\t\t\tcontinuityEpoch = getEpochNumberOfContinuity(store, continuityKey)\n\n\t\t\tif (continuityEpoch === confirmed.epoch - 1) {\n\t\t\t\tstore.logger.info(\n\t\t\t\t\t`✅`,\n\t\t\t\t\t`continuity`,\n\t\t\t\t\tcontinuityKey,\n\t\t\t\t\t`integrating update #${confirmed.epoch} (${confirmed.token.key} ${confirmed.id})`,\n\t\t\t\t)\n\t\t\t\tingestTransactionOutcomeEvent(store, confirmed, `newValue`)\n\t\t\t\tsocket.emit(`ack:${continuityKey}`, confirmed.epoch)\n\t\t\t\tsetEpochNumberOfContinuity(store, continuityKey, confirmed.epoch)\n\t\t\t} else if (continuityEpoch !== undefined) {\n\t\t\t\tstore.logger.info(\n\t\t\t\t\t`🧑‍⚖️`,\n\t\t\t\t\t`continuity`,\n\t\t\t\t\tcontinuityKey,\n\t\t\t\t\t`received update #${confirmed.epoch} but still waiting for update #${\n\t\t\t\t\t\tcontinuityEpoch + 1\n\t\t\t\t\t}`,\n\t\t\t\t\t{\n\t\t\t\t\t\tclientEpoch: continuityEpoch,\n\t\t\t\t\t\tserverEpoch: confirmed.epoch,\n\t\t\t\t\t},\n\t\t\t\t)\n\t\t\t\tconst confirmedUpdateIsAlreadyEnqueued = confirmedUpdates.some(\n\t\t\t\t\t(update) => update.epoch === confirmed.epoch,\n\t\t\t\t)\n\t\t\t\tif (confirmedUpdateIsAlreadyEnqueued) {\n\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t`👍`,\n\t\t\t\t\t\t`continuity`,\n\t\t\t\t\t\tcontinuityKey,\n\t\t\t\t\t\t`confirmed update #${confirmed.epoch} is already enqueued`,\n\t\t\t\t\t)\n\t\t\t\t} else {\n\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t`👈`,\n\t\t\t\t\t\t`continuity`,\n\t\t\t\t\t\tcontinuityKey,\n\t\t\t\t\t\t`pushing confirmed update #${confirmed.epoch} to queue`,\n\t\t\t\t\t)\n\t\t\t\t\tsetIntoStore(store, confirmedUpdateQueueAtom, (queue) => {\n\t\t\t\t\t\tqueue.push(confirmed)\n\t\t\t\t\t\tqueue.sort((a, b) => a.epoch - b.epoch)\n\t\t\t\t\t\treturn queue\n\t\t\t\t\t})\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t}\n","import type { AtomToken } from \"atom.io\"\nimport type { Store } from \"atom.io/internal\"\nimport { disposeAtom } from \"atom.io/internal\"\nimport type { Json } from \"atom.io/json\"\n\nexport function useConcealState(store: Store) {\n\treturn (concealed: AtomToken<Json.Serializable>[]): void => {\n\t\tfor (const token of concealed) {\n\t\t\tdisposeAtom(store, token)\n\t\t}\n\t}\n}\n","import { setIntoStore, type Store } from \"atom.io/internal\"\nimport type { Json } from \"atom.io/json\"\n\nexport function useRevealState(store: Store) {\n\treturn (revealed: Json.Array): void => {\n\t\tlet i = 0\n\t\tlet k: any\n\t\tlet v: any\n\t\tfor (const x of revealed) {\n\t\t\tif (i % 2 === 0) {\n\t\t\t\tk = x\n\t\t\t} else {\n\t\t\t\tv = x\n\t\t\t\tsetIntoStore(store, k, v)\n\t\t\t}\n\t\t\ti++\n\t\t}\n\t}\n}\n","import { Future } from \"atom.io/internal\"\nimport type { Socket } from \"atom.io/realtime\"\n\ntype SubData = { refcount: number; timer: Future<void> }\n\nconst subscriptions: WeakMap<Socket, Map<string, SubData>> = new WeakMap()\nconst socketIds: WeakMap<Socket, string | undefined> = new WeakMap()\n\nfunction getSubMap(socket: Socket): Map<string, SubData> {\n\tlet subMap = subscriptions.get(socket)\n\tif (subMap === undefined) {\n\t\tsubMap = new Map()\n\t\tsubscriptions.set(socket, subMap)\n\t}\n\treturn subMap\n}\n\nexport function createSubscriber<K extends string>(\n\tsocket: Socket,\n\tkey: K,\n\topen: (key: K) => () => void,\n): () => void {\n\tconst knownSocketId = socketIds.get(socket)\n\tif (knownSocketId !== socket.id) {\n\t\tsocketIds.set(socket, socket.id)\n\t\tsubscriptions.delete(socket)\n\t}\n\tconst subMap = getSubMap(socket)\n\tlet sub = subMap.get(key)\n\n\tif (sub) {\n\t\tsub.timer.use(new Promise<void>(() => {}))\n\t\tsub.refcount++\n\t} else {\n\t\tsub = { refcount: 1, timer: new Future<void>(() => {}) }\n\t\tsubMap.set(key, sub)\n\t\tconst close = open(key)\n\t\tvoid sub.timer.then(() => {\n\t\t\tclose()\n\t\t\tsubMap.delete(key)\n\t\t})\n\t}\n\treturn () => {\n\t\tsub.refcount--\n\n\t\tif (sub.refcount === 0) {\n\t\t\tconst timeout = new Promise<void>((resolve) => {\n\t\t\t\tsetTimeout(resolve, 50)\n\t\t\t})\n\t\t\tsub.timer.use(timeout)\n\t\t}\n\t}\n}\n","import type * as AtomIO from \"atom.io\"\nimport { setIntoStore, type Store } from \"atom.io/internal\"\nimport type { Json } from \"atom.io/json\"\nimport { employSocket, type Socket } from \"atom.io/realtime\"\n\nimport { createSubscriber } from \"./create-subscriber\"\n\nexport function pullAtom<J extends Json.Serializable>(\n\tstore: Store,\n\tsocket: Socket,\n\ttoken: AtomIO.RegularAtomToken<J, any, any>,\n): () => void {\n\treturn createSubscriber(socket, token.key, (key) => {\n\t\tconst stopWatching = employSocket(socket, `serve:${key}`, (data: J) => {\n\t\t\tsetIntoStore(store, token, data)\n\t\t})\n\t\tsocket.emit(`sub:${token.key}`)\n\t\treturn () => {\n\t\t\tsocket.emit(`unsub:${key}`)\n\t\t\tstopWatching()\n\t\t}\n\t})\n}\n","import type * as AtomIO from \"atom.io\"\nimport { findInStore, setIntoStore, type Store } from \"atom.io/internal\"\nimport type { Canonical, Json } from \"atom.io/json\"\nimport { employSocket, type Socket } from \"atom.io/realtime\"\n\nimport { createSubscriber } from \"./create-subscriber\"\n\nexport function pullAtomFamilyMember<\n\tJ extends Json.Serializable,\n\tK extends Canonical,\n>(\n\tstore: Store,\n\tsocket: Socket,\n\tfamily: AtomIO.AtomFamilyToken<J, K, any>,\n\tkey: NoInfer<K>,\n): () => void {\n\tconst token = findInStore(store, family, key)\n\treturn createSubscriber(socket, token.key, () => {\n\t\tconst stopWatching = employSocket(\n\t\t\tsocket,\n\t\t\t`serve:${token.key}`,\n\t\t\t(data: J) => {\n\t\t\t\tsetIntoStore(store, token, data)\n\t\t\t},\n\t\t)\n\t\tsocket.emit(`sub:${family.key}`, key)\n\t\treturn () => {\n\t\t\tsocket.emit(`unsub:${token.key}`)\n\t\t\tstopWatching()\n\t\t}\n\t})\n}\n","import type * as AtomIO from \"atom.io\"\nimport type { AsJSON, SignalFrom, Store, Transceiver } from \"atom.io/internal\"\nimport { getJsonToken, getUpdateToken, setIntoStore } from \"atom.io/internal\"\nimport { employSocket, type Socket } from \"atom.io/realtime\"\n\nimport { createSubscriber } from \"./create-subscriber\"\n\nexport function pullMutableAtom<T extends Transceiver<any, any, any>>(\n\tstore: Store,\n\tsocket: Socket,\n\ttoken: AtomIO.MutableAtomToken<T>,\n): () => void {\n\tconst jsonToken = getJsonToken(store, token)\n\tconst updateToken = getUpdateToken(token)\n\treturn createSubscriber(socket, token.key, () => {\n\t\tconst stopWatchingForInit = employSocket(\n\t\t\tsocket,\n\t\t\t`init:${token.key}`,\n\t\t\t(data: AsJSON<T>) => {\n\t\t\t\tsetIntoStore(store, jsonToken, data)\n\t\t\t},\n\t\t)\n\t\tconst stopWatchingForUpdate = employSocket(\n\t\t\tsocket,\n\t\t\t`next:${token.key}`,\n\t\t\t(data: SignalFrom<T>) => {\n\t\t\t\tsetIntoStore(store, updateToken, data)\n\t\t\t},\n\t\t)\n\t\tsocket.emit(`sub:${token.key}`)\n\t\treturn () => {\n\t\t\tsocket.emit(`unsub:${token.key}`)\n\t\t\tstopWatchingForInit()\n\t\t\tstopWatchingForUpdate()\n\t\t}\n\t})\n}\n","import type * as AtomIO from \"atom.io\"\nimport type { AsJSON, SignalFrom, Store, Transceiver } from \"atom.io/internal\"\nimport {\n\tfindInStore,\n\tgetJsonToken,\n\tgetUpdateToken,\n\tsetIntoStore,\n} from \"atom.io/internal\"\nimport type { Canonical } from \"atom.io/json\"\nimport { employSocket, type Socket } from \"atom.io/realtime\"\n\nimport { createSubscriber } from \"./create-subscriber\"\n\nexport function pullMutableAtomFamilyMember<\n\tT extends Transceiver<any, any, any>,\n\tK extends Canonical,\n>(\n\tstore: Store,\n\tsocket: Socket,\n\tfamily: AtomIO.MutableAtomFamilyToken<T, K>,\n\tkey: NoInfer<K>,\n): () => void {\n\tconst token = findInStore(store, family, key)\n\tconst jsonToken = getJsonToken(store, token)\n\tconst trackerToken = getUpdateToken(token)\n\treturn createSubscriber(socket, token.key, () => {\n\t\tconst stopWatchingForInit = employSocket(\n\t\t\tsocket,\n\t\t\t`init:${token.key}`,\n\t\t\t(data: AsJSON<T>) => {\n\t\t\t\tsetIntoStore(store, jsonToken, data)\n\t\t\t},\n\t\t)\n\t\tconst stopWatchingForUpdate = employSocket(\n\t\t\tsocket,\n\t\t\t`next:${token.key}`,\n\t\t\t(data: SignalFrom<T>) => {\n\t\t\t\tsetIntoStore(store, trackerToken, data)\n\t\t\t},\n\t\t)\n\t\tsocket.emit(`sub:${family.key}`, key)\n\t\treturn () => {\n\t\t\tsocket.emit(`unsub:${token.key}`)\n\t\t\tstopWatchingForInit()\n\t\t\tstopWatchingForUpdate()\n\t\t}\n\t})\n}\n","import type { AtomToken, SelectorToken } from \"atom.io\"\nimport type { Store } from \"atom.io/internal\"\nimport { getFamilyOfToken, subscribeToState } from \"atom.io/internal\"\nimport { parseJson } from \"atom.io/json\"\nimport type { Socket } from \"atom.io/realtime\"\n\nimport { pullAtom } from \"./pull-atom\"\nimport { pullAtomFamilyMember } from \"./pull-atom-family-member\"\nimport { pullMutableAtom } from \"./pull-mutable-atom\"\nimport { pullMutableAtomFamilyMember } from \"./pull-mutable-atom-family-member\"\n\nexport function pullSelectorRoots(\n\tstore: Store,\n\tsocket: Socket,\n\tselectorToken: SelectorToken<any>,\n): () => void {\n\tconst atomSubscriptions = new Map<string, () => void>()\n\n\tconst start = () => {\n\t\tconst atomKeys = store.selectorAtoms.getRelatedKeys(selectorToken.key)\n\t\tif (atomKeys) {\n\t\t\tfor (const [atomKey, unsub] of atomSubscriptions) {\n\t\t\t\tif (!atomKeys.has(atomKey)) {\n\t\t\t\t\tunsub()\n\t\t\t\t\tatomSubscriptions.delete(atomKey)\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tfor (const atomKey of atomKeys) {\n\t\t\t\tif (atomSubscriptions.has(atomKey)) {\n\t\t\t\t\tcontinue\n\t\t\t\t}\n\t\t\t\tconst atom = store.atoms.get(atomKey) as AtomToken<any, any>\n\t\t\t\tswitch (atom.type) {\n\t\t\t\t\tcase `atom`: {\n\t\t\t\t\t\tif (atom.family) {\n\t\t\t\t\t\t\tconst { subKey: serializedSubKey } = atom.family\n\t\t\t\t\t\t\tconst subKey = parseJson(serializedSubKey)\n\t\t\t\t\t\t\tconst family = getFamilyOfToken(store, atom)\n\t\t\t\t\t\t\tatomSubscriptions.set(\n\t\t\t\t\t\t\t\tatomKey,\n\t\t\t\t\t\t\t\tpullAtomFamilyMember(store, socket, family, subKey),\n\t\t\t\t\t\t\t)\n\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\tatomSubscriptions.set(atomKey, pullAtom(store, socket, atom))\n\t\t\t\t\t\t}\n\t\t\t\t\t\tbreak\n\t\t\t\t\t}\n\t\t\t\t\tcase `mutable_atom`: {\n\t\t\t\t\t\tif (atom.family) {\n\t\t\t\t\t\t\tconst { subKey: serializedSubKey } = atom.family\n\t\t\t\t\t\t\tconst subKey = parseJson(serializedSubKey)\n\t\t\t\t\t\t\tconst family = getFamilyOfToken(store, atom)\n\t\t\t\t\t\t\tatomSubscriptions.set(\n\t\t\t\t\t\t\t\tatomKey,\n\t\t\t\t\t\t\t\tpullMutableAtomFamilyMember(store, socket, family, subKey),\n\t\t\t\t\t\t\t)\n\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\tatomSubscriptions.set(\n\t\t\t\t\t\t\t\tatomKey,\n\t\t\t\t\t\t\t\tpullMutableAtom(store, socket, atom),\n\t\t\t\t\t\t\t)\n\t\t\t\t\t\t}\n\t\t\t\t\t\tbreak\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t}\n\n\tconst unsubFromSelector = subscribeToState(\n\t\tstore,\n\t\tselectorToken,\n\t\t`pull-watches-dependencies`,\n\t\t() => {\n\t\t\tstart()\n\t\t},\n\t)\n\n\tstart()\n\n\treturn () => {\n\t\tfor (const [, unsub] of atomSubscriptions) unsub()\n\t\tatomSubscriptions.clear()\n\t\tunsubFromSelector()\n\t}\n}\n","import type * as AtomIO from \"atom.io\"\nimport type { Store } from \"atom.io/internal\"\nimport type { Socket } from \"atom.io/realtime\"\n\nimport { pullSelectorRoots } from \"./pull-selector-roots\"\n\nexport function pullSelector<T>(\n\tstore: Store,\n\tsocket: Socket,\n\ttoken: AtomIO.SelectorToken<T>,\n): () => void {\n\treturn pullSelectorRoots(store, socket, token)\n}\n","import type * as AtomIO from \"atom.io\"\nimport type { Store } from \"atom.io/internal\"\nimport { findInStore } from \"atom.io/internal\"\nimport type { Canonical } from \"atom.io/json\"\nimport type { Socket } from \"atom.io/realtime\"\n\nimport { pullSelectorRoots } from \"./pull-selector-roots\"\n\nexport function pullSelectorFamilyMember<T, K extends Canonical>(\n\tstore: Store,\n\tsocket: Socket,\n\tfamilyToken: AtomIO.SelectorFamilyToken<T, K>,\n\tkey: NoInfer<K>,\n): () => void {\n\tconst token = findInStore(store, familyToken, key)\n\treturn pullSelectorRoots(store, socket, token)\n}\n","import type { WritableToken } from \"atom.io\"\nimport type { Store } from \"atom.io/internal\"\nimport { setIntoStore, subscribeToState } from \"atom.io/internal\"\nimport type { Json } from \"atom.io/json\"\nimport type { Socket } from \"atom.io/realtime\"\nimport { employSocket, mutexAtoms } from \"atom.io/realtime\"\n\nimport { createSubscriber } from \"./create-subscriber\"\n\nexport function pushState<J extends Json.Serializable>(\n\tstore: Store,\n\tsocket: Socket,\n\ttoken: WritableToken<J>,\n): () => void {\n\treturn createSubscriber(socket, `push:${token.key}`, () => {\n\t\tlet stopWatching = employSocket(\n\t\t\tsocket,\n\t\t\t`claim-result:${token.key}`,\n\t\t\t(success: boolean) => {\n\t\t\t\tif (!success) return\n\t\t\t\tstopWatching()\n\t\t\t\tsetIntoStore(store, mutexAtoms, token.key, true)\n\t\t\t\tstopWatching = subscribeToState(store, token, `push`, ({ newValue }) => {\n\t\t\t\t\tsocket.emit(`pub:${token.key}`, newValue)\n\t\t\t\t})\n\t\t\t},\n\t\t)\n\n\t\tsocket.emit(`claim:${token.key}`)\n\n\t\treturn () => {\n\t\t\tsocket.emit(`unclaim:${token.key}`)\n\t\t\tstopWatching()\n\t\t}\n\t})\n}\n","import type { RootStore } from \"atom.io/internal\"\nimport {\n\tassignTransactionToContinuity,\n\tgetFromStore,\n\tgetJsonToken,\n\tsetEpochNumberOfContinuity,\n\tsetIntoStore,\n\tsubscribeToTransaction,\n} from \"atom.io/internal\"\nimport type { Json } from \"atom.io/json\"\nimport type { ContinuityToken, Socket } from \"atom.io/realtime\"\n\nimport { useRegisterAndAttemptConfirmedUpdate } from \"./continuity/register-and-attempt-confirmed-update\"\nimport { useConcealState } from \"./continuity/use-conceal-state\"\nimport { useRevealState } from \"./continuity/use-reveal-state\"\nimport {\n\tconfirmedUpdateQueueAtom,\n\toptimisticUpdateQueueAtom,\n} from \"./realtime-client-stores\"\n\nexport function syncContinuity(\n\tstore: RootStore,\n\tsocket: Socket,\n\tcontinuity: ContinuityToken,\n): () => void {\n\tconst continuityKey = continuity.key\n\tconst optimisticUpdates = getFromStore(store, optimisticUpdateQueueAtom)\n\tconst confirmedUpdates = getFromStore(store, confirmedUpdateQueueAtom)\n\n\tconst initializeContinuity = (epoch: number, payload: Json.Array) => {\n\t\tsocket.off(`continuity-init:${continuityKey}`, initializeContinuity)\n\t\tlet i = 0\n\t\tlet k: any\n\t\tlet v: any\n\t\tfor (const x of payload) {\n\t\t\tif (i % 2 === 0) {\n\t\t\t\tk = x\n\t\t\t} else {\n\t\t\t\tv = x\n\t\t\t\tif (`type` in k && k.type === `mutable_atom`) {\n\t\t\t\t\tk = getJsonToken(store, k)\n\t\t\t\t}\n\t\t\t\tsetIntoStore(store, k, v)\n\t\t\t}\n\t\t\ti++\n\t\t}\n\t\tsetEpochNumberOfContinuity(store, continuityKey, epoch)\n\t}\n\tsocket.off(`continuity-init:${continuityKey}`)\n\tsocket.on(`continuity-init:${continuityKey}`, initializeContinuity)\n\n\tconst registerAndAttemptConfirmedUpdate = useRegisterAndAttemptConfirmedUpdate(\n\t\tstore,\n\t\tcontinuityKey,\n\t\tsocket,\n\t\toptimisticUpdates,\n\t\tconfirmedUpdates,\n\t)\n\tsocket.off(`tx-new:${continuityKey}`)\n\tsocket.on(`tx-new:${continuityKey}`, registerAndAttemptConfirmedUpdate)\n\n\tconst unsubscribeFunctions = continuity.actions.map((transaction) => {\n\t\tassignTransactionToContinuity(store, continuityKey, transaction.key)\n\t\tconst unsubscribeFromTransactionUpdates = subscribeToTransaction(\n\t\t\tstore,\n\t\t\ttransaction,\n\t\t\t`tx-run:${continuityKey}`,\n\t\t\t(clientUpdate) => {\n\t\t\t\tstore.logger.info(\n\t\t\t\t\t`🤞`,\n\t\t\t\t\t`continuity`,\n\t\t\t\t\tcontinuityKey,\n\t\t\t\t\t`enqueuing optimistic update`,\n\t\t\t\t)\n\t\t\t\tconst optimisticUpdateIndex = optimisticUpdates.findIndex(\n\t\t\t\t\t(update) => update.id === clientUpdate.id,\n\t\t\t\t)\n\t\t\t\tif (optimisticUpdateIndex === -1) {\n\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t`🤞`,\n\t\t\t\t\t\t`continuity`,\n\t\t\t\t\t\tcontinuityKey,\n\t\t\t\t\t\t`enqueuing new optimistic update`,\n\t\t\t\t\t)\n\t\t\t\t\tsetIntoStore(store, optimisticUpdateQueueAtom, (queue) => {\n\t\t\t\t\t\tqueue.push(clientUpdate)\n\t\t\t\t\t\tqueue.sort((a, b) => a.epoch - b.epoch)\n\t\t\t\t\t\treturn queue\n\t\t\t\t\t})\n\t\t\t\t} else {\n\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t`🤞`,\n\t\t\t\t\t\t`continuity`,\n\t\t\t\t\t\tcontinuityKey,\n\t\t\t\t\t\t`replacing existing optimistic update at index ${optimisticUpdateIndex}`,\n\t\t\t\t\t)\n\t\t\t\t\tsetIntoStore(store, optimisticUpdateQueueAtom, (queue) => {\n\t\t\t\t\t\tqueue[optimisticUpdateIndex] = clientUpdate\n\t\t\t\t\t\treturn queue\n\t\t\t\t\t})\n\t\t\t\t}\n\t\t\t\tsocket.emit(`tx-run:${continuityKey}`, {\n\t\t\t\t\tid: clientUpdate.id,\n\t\t\t\t\ttoken: transaction,\n\t\t\t\t\tparams: clientUpdate.params,\n\t\t\t\t})\n\t\t\t},\n\t\t)\n\t\treturn unsubscribeFromTransactionUpdates\n\t})\n\n\tconst revealState = useRevealState(store)\n\tconst concealState = useConcealState(store)\n\tsocket.on(`reveal:${continuityKey}`, revealState)\n\tsocket.on(`conceal:${continuityKey}`, concealState)\n\n\tsocket.emit(`get:${continuityKey}`)\n\treturn () => {\n\t\tsocket.off(`continuity-init:${continuityKey}`)\n\t\tsocket.off(`tx-new:${continuityKey}`)\n\t\tfor (const unsubscribe of unsubscribeFunctions) unsubscribe()\n\t\t// socket.emit(`unsub:${continuityKey}`)\n\t}\n}\n"],"mappings":"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"}
1
+ {"version":3,"file":"index.js","names":["mySocketKeyAtom: RegularAtomToken<SocketKey | undefined>","myUserKeyAtom: RegularAtomToken<UserKey | null>","myRoomKeySelector: ReadonlyPureSelectorToken<RoomKey | null>","usersHereSelector: ReadonlyPureSelectorToken<\n\tViewOf<UList<UserKey> | null>\n>","roomOwnerSelector: ReadonlyPureSelectorToken<UserKey | null>","optimisticUpdateQueueAtom: AtomIO.RegularAtomToken<\n\tAtomIO.TransactionOutcomeEvent<any>[]\n>","confirmedUpdateQueueAtom: AtomIO.RegularAtomToken<\n\tAtomIO.TransactionOutcomeEvent<any>[]\n>","continuityEpoch: number | undefined","k: any","v: any","subscriptions: WeakMap<Socket, Map<string, SubData>>","socketIds: WeakMap<Socket, string | undefined>","atom","k: any","v: any"],"sources":["../../src/realtime-client/realtime-client-stores/client-main-store.ts","../../src/realtime-client/realtime-client-stores/client-sync-store.ts","../../src/realtime-client/continuity/register-and-attempt-confirmed-update.ts","../../src/realtime-client/continuity/use-conceal-state.ts","../../src/realtime-client/continuity/use-reveal-state.ts","../../src/realtime-client/create-subscriber.ts","../../src/realtime-client/pull-atom.ts","../../src/realtime-client/pull-atom-family-member.ts","../../src/realtime-client/pull-mutable-atom.ts","../../src/realtime-client/pull-mutable-atom-family-member.ts","../../src/realtime-client/pull-selector-roots.ts","../../src/realtime-client/pull-selector.ts","../../src/realtime-client/pull-selector-family-member.ts","../../src/realtime-client/push-state.ts","../../src/realtime-client/sync-continuity.ts"],"sourcesContent":["import type {\n\tReadonlyPureSelectorToken,\n\tRegularAtomToken,\n\tViewOf,\n} from \"atom.io\"\nimport { atom, getInternalRelations, selector } from \"atom.io\"\nimport type { RoomKey, SocketKey, UserKey } from \"atom.io/realtime\"\nimport { ownersOfRooms, usersInRooms } from \"atom.io/realtime\"\nimport type { UList } from \"atom.io/transceivers/u-list\"\n\nexport const mySocketKeyAtom: RegularAtomToken<SocketKey | undefined> = atom({\n\tkey: `mySocketKey`,\n\tdefault: undefined,\n})\n\nexport const myUserKeyAtom: RegularAtomToken<UserKey | null> = atom({\n\tkey: `myUserKey`,\n\tdefault: null,\n\teffects: [\n\t\t(userKey) => {\n\t\t\tif (typeof window !== `undefined`) {\n\t\t\t\tvoid import(`atom.io/web`).then(({ storageSync }) => {\n\t\t\t\t\tstorageSync(globalThis.localStorage, JSON, `myUserKey`)(userKey)\n\t\t\t\t})\n\t\t\t}\n\t\t},\n\t],\n})\n\nexport const myRoomKeySelector: ReadonlyPureSelectorToken<RoomKey | null> =\n\tselector({\n\t\tkey: `myRoomKey`,\n\t\tget: ({ get }) => {\n\t\t\tif (\n\t\t\t\t`process` in globalThis &&\n\t\t\t\t`env` in process &&\n\t\t\t\t`REALTIME_ROOM_KEY` in process.env\n\t\t\t) {\n\t\t\t\t// if a room running server-side wants its own key, this is where it lives\n\t\t\t\treturn process.env[`REALTIME_ROOM_KEY`] as RoomKey\n\t\t\t}\n\t\t\tconst myUserKey = get(myUserKeyAtom)\n\t\t\tif (!myUserKey) return null\n\t\t\tconst [, usersInRoomsAtoms] = getInternalRelations(usersInRooms, `split`)\n\t\t\tconst roomKeys = get(usersInRoomsAtoms, myUserKey)\n\t\t\tfor (const roomKey of roomKeys) return roomKey\n\t\t\treturn null\n\t\t},\n\t})\n\nexport const usersHereSelector: ReadonlyPureSelectorToken<\n\tViewOf<UList<UserKey> | null>\n> = selector({\n\tkey: `usersHere`,\n\tget: ({ get, find }) => {\n\t\tconst myRoomKey = get(myRoomKeySelector)\n\t\tif (!myRoomKey) return null\n\t\tconst [usersInRoomsAtoms] = getInternalRelations(usersInRooms, `split`)\n\t\tconst users = get(find(usersInRoomsAtoms, myRoomKey))\n\t\treturn users\n\t},\n})\n\nexport const roomOwnerSelector: ReadonlyPureSelectorToken<UserKey | null> =\n\tselector({\n\t\tkey: `roomOwner`,\n\t\tget: ({ get }) => {\n\t\t\tconst myRoomKey = get(myRoomKeySelector)\n\t\t\tif (!myRoomKey) return null\n\t\t\tconst [, ownerOfRoomsAtoms] = getInternalRelations(ownersOfRooms, `split`)\n\t\t\tconst owner = get(ownerOfRoomsAtoms, myRoomKey)\n\t\t\tfor (const userKey of owner) return userKey\n\t\t\treturn null\n\t\t},\n\t})\n","import * as AtomIO from \"atom.io\"\n\nexport const optimisticUpdateQueueAtom: AtomIO.RegularAtomToken<\n\tAtomIO.TransactionOutcomeEvent<any>[]\n> = AtomIO.atom<AtomIO.TransactionOutcomeEvent<any>[]>({\n\tkey: `optimisticUpdateQueue`,\n\tdefault: () => [],\n})\n\nexport const confirmedUpdateQueueAtom: AtomIO.RegularAtomToken<\n\tAtomIO.TransactionOutcomeEvent<any>[]\n> = AtomIO.atom<AtomIO.TransactionOutcomeEvent<any>[]>({\n\tkey: `confirmedUpdateQueue`,\n\tdefault: () => [],\n})\n","import type * as AtomIO from \"atom.io\"\nimport type { Fn, RootStore } from \"atom.io/internal\"\nimport {\n\tactUponStore,\n\tgetEpochNumberOfContinuity,\n\tingestTransactionOutcomeEvent,\n\tsetEpochNumberOfContinuity,\n\tsetIntoStore,\n} from \"atom.io/internal\"\nimport type { Json } from \"atom.io/json\"\nimport type { Socket } from \"atom.io/realtime\"\n\nimport {\n\tconfirmedUpdateQueueAtom,\n\toptimisticUpdateQueueAtom,\n} from \"../realtime-client-stores\"\n\nexport const useRegisterAndAttemptConfirmedUpdate =\n\t(\n\t\tstore: RootStore,\n\t\tcontinuityKey: string,\n\t\tsocket: Socket,\n\t\toptimisticUpdates: readonly AtomIO.TransactionOutcomeEvent<\n\t\t\tAtomIO.TransactionToken<Fn>\n\t\t>[],\n\t\tconfirmedUpdates: readonly AtomIO.TransactionOutcomeEvent<\n\t\t\tAtomIO.TransactionToken<Fn>\n\t\t>[],\n\t) =>\n\t(\n\t\tconfirmed: AtomIO.TransactionOutcomeEvent<AtomIO.TransactionToken<Fn>> &\n\t\t\tJson.Serializable,\n\t): void => {\n\t\tfunction reconcileEpoch(\n\t\t\toptimisticUpdate: AtomIO.TransactionOutcomeEvent<\n\t\t\t\tAtomIO.TransactionToken<Fn>\n\t\t\t>,\n\t\t\tconfirmedUpdate: AtomIO.TransactionOutcomeEvent<\n\t\t\t\tAtomIO.TransactionToken<Fn>\n\t\t\t>,\n\t\t): void {\n\t\t\tstore.logger.info(\n\t\t\t\t`🧑‍⚖️`,\n\t\t\t\t`continuity`,\n\t\t\t\tcontinuityKey,\n\t\t\t\t`reconciling updates`,\n\t\t\t)\n\t\t\tsetIntoStore(store, optimisticUpdateQueueAtom, (queue) => {\n\t\t\t\tqueue.shift()\n\t\t\t\treturn queue\n\t\t\t})\n\t\t\tif (optimisticUpdate.id === confirmedUpdate.id) {\n\t\t\t\tconst clientResult = JSON.stringify(optimisticUpdate.subEvents)\n\t\t\t\tconst serverResult = JSON.stringify(confirmedUpdate.subEvents)\n\t\t\t\tif (clientResult === serverResult) {\n\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t`✅`,\n\t\t\t\t\t\t`continuity`,\n\t\t\t\t\t\tcontinuityKey,\n\t\t\t\t\t\t`results for ${optimisticUpdate.id} match between client and server`,\n\t\t\t\t\t)\n\t\t\t\t\tsocket.emit(`ack:${continuityKey}`, confirmedUpdate.epoch)\n\t\t\t\t\treturn\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\t// id mismatch\n\t\t\t\tstore.logger.info(\n\t\t\t\t\t`❌`,\n\t\t\t\t\t`continuity`,\n\t\t\t\t\tcontinuityKey,\n\t\t\t\t\t`thought update #${confirmedUpdate.epoch} was ${optimisticUpdate.token.key}:${optimisticUpdate.id}, but it was actually ${confirmedUpdate.token.key}:${confirmedUpdate.id}`,\n\t\t\t\t)\n\t\t\t}\n\t\t\tstore.logger.info(\n\t\t\t\t`🧑‍⚖️`,\n\t\t\t\t`continuity`,\n\t\t\t\tcontinuityKey,\n\t\t\t\t`updates do not match`,\n\t\t\t\toptimisticUpdate,\n\t\t\t\tconfirmedUpdate,\n\t\t\t)\n\t\t\tconst reversedOptimisticUpdates = optimisticUpdates.toReversed()\n\t\t\tfor (const subsequentOptimistic of reversedOptimisticUpdates) {\n\t\t\t\tingestTransactionOutcomeEvent(store, subsequentOptimistic, `oldValue`)\n\t\t\t}\n\t\t\tstore.logger.info(\n\t\t\t\t`⏪`,\n\t\t\t\t`continuity`,\n\t\t\t\tcontinuityKey,\n\t\t\t\t`undid optimistic updates:`,\n\t\t\t\treversedOptimisticUpdates,\n\t\t\t)\n\t\t\tingestTransactionOutcomeEvent(store, optimisticUpdate, `oldValue`)\n\t\t\tstore.logger.info(\n\t\t\t\t`⏪`,\n\t\t\t\t`continuity`,\n\t\t\t\tcontinuityKey,\n\t\t\t\t`undid zeroth optimistic update`,\n\t\t\t\toptimisticUpdate,\n\t\t\t)\n\t\t\tingestTransactionOutcomeEvent(store, confirmedUpdate, `newValue`)\n\t\t\tstore.logger.info(\n\t\t\t\t`⏩`,\n\t\t\t\t`continuity`,\n\t\t\t\tcontinuityKey,\n\t\t\t\t`applied confirmed update`,\n\t\t\t\tconfirmedUpdate,\n\t\t\t)\n\t\t\tsocket.emit(`ack:${continuityKey}`, confirmedUpdate.epoch)\n\n\t\t\tfor (const subsequentOptimistic of optimisticUpdates) {\n\t\t\t\tconst token = {\n\t\t\t\t\ttype: `transaction`,\n\t\t\t\t\tkey: subsequentOptimistic.token.key,\n\t\t\t\t} as const\n\t\t\t\tconst { id, params } = subsequentOptimistic\n\t\t\t\tactUponStore(store, token, id)(...params)\n\t\t\t}\n\t\t\tstore.logger.info(\n\t\t\t\t`⏩`,\n\t\t\t\t`continuity`,\n\t\t\t\tcontinuityKey,\n\t\t\t\t`reapplied subsequent optimistic updates:`,\n\t\t\t\toptimisticUpdates,\n\t\t\t)\n\t\t}\n\n\t\tstore.logger.info(\n\t\t\t`🧑‍⚖️`,\n\t\t\t`continuity`,\n\t\t\tcontinuityKey,\n\t\t\t`integrating confirmed update`,\n\t\t\t{ confirmedUpdate: confirmed, confirmedUpdates, optimisticUpdates },\n\t\t)\n\t\tconst zerothOptimisticUpdate = optimisticUpdates[0]\n\t\tif (zerothOptimisticUpdate) {\n\t\t\tstore.logger.info(\n\t\t\t\t`🧑‍⚖️`,\n\t\t\t\t`continuity`,\n\t\t\t\tcontinuityKey,\n\t\t\t\t`has optimistic updates to reconcile`,\n\t\t\t)\n\t\t\tif (confirmed.epoch === zerothOptimisticUpdate.epoch) {\n\t\t\t\tstore.logger.info(\n\t\t\t\t\t`🧑‍⚖️`,\n\t\t\t\t\t`continuity`,\n\t\t\t\t\tcontinuityKey,\n\t\t\t\t\t`epoch of confirmed update #${confirmed.epoch} matches zeroth optimistic update`,\n\t\t\t\t)\n\t\t\t\treconcileEpoch(zerothOptimisticUpdate, confirmed)\n\t\t\t\tfor (const nextConfirmed of confirmedUpdates) {\n\t\t\t\t\tconst nextOptimistic = optimisticUpdates[0]\n\t\t\t\t\tif (nextConfirmed.epoch === nextOptimistic?.epoch) {\n\t\t\t\t\t\treconcileEpoch(nextOptimistic, nextConfirmed)\n\t\t\t\t\t} else {\n\t\t\t\t\t\tbreak\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\t// epoch mismatch\n\t\t\t\tstore.logger.info(\n\t\t\t\t\t`🧑‍⚖️`,\n\t\t\t\t\t`continuity`,\n\t\t\t\t\tcontinuityKey,\n\t\t\t\t\t`epoch of confirmed update #${confirmed.epoch} does not match zeroth optimistic update #${zerothOptimisticUpdate.epoch}`,\n\t\t\t\t)\n\t\t\t\tconst confirmedUpdateIsAlreadyEnqueued = confirmedUpdates.some(\n\t\t\t\t\t(update) => update.epoch === confirmed.epoch,\n\t\t\t\t)\n\t\t\t\tif (!confirmedUpdateIsAlreadyEnqueued) {\n\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t`👈`,\n\t\t\t\t\t\t`continuity`,\n\t\t\t\t\t\tcontinuityKey,\n\t\t\t\t\t\t`pushing confirmed update to queue`,\n\t\t\t\t\t\tconfirmed,\n\t\t\t\t\t)\n\t\t\t\t\tsetIntoStore(store, confirmedUpdateQueueAtom, (queue) => {\n\t\t\t\t\t\tqueue.push(confirmed)\n\t\t\t\t\t\tqueue.sort((a, b) => a.epoch - b.epoch)\n\t\t\t\t\t\treturn queue\n\t\t\t\t\t})\n\t\t\t\t}\n\t\t\t}\n\t\t} else {\n\t\t\tstore.logger.info(\n\t\t\t\t`🧑‍⚖️`,\n\t\t\t\t`continuity`,\n\t\t\t\tcontinuityKey,\n\t\t\t\t`has no optimistic updates to deal with`,\n\t\t\t)\n\t\t\tlet continuityEpoch: number | undefined\n\t\t\tcontinuityEpoch = getEpochNumberOfContinuity(store, continuityKey)\n\n\t\t\tif (continuityEpoch === confirmed.epoch - 1) {\n\t\t\t\tstore.logger.info(\n\t\t\t\t\t`✅`,\n\t\t\t\t\t`continuity`,\n\t\t\t\t\tcontinuityKey,\n\t\t\t\t\t`integrating update #${confirmed.epoch} (${confirmed.token.key} ${confirmed.id})`,\n\t\t\t\t)\n\t\t\t\tingestTransactionOutcomeEvent(store, confirmed, `newValue`)\n\t\t\t\tsocket.emit(`ack:${continuityKey}`, confirmed.epoch)\n\t\t\t\tsetEpochNumberOfContinuity(store, continuityKey, confirmed.epoch)\n\t\t\t} else if (continuityEpoch !== undefined) {\n\t\t\t\tstore.logger.info(\n\t\t\t\t\t`🧑‍⚖️`,\n\t\t\t\t\t`continuity`,\n\t\t\t\t\tcontinuityKey,\n\t\t\t\t\t`received update #${confirmed.epoch} but still waiting for update #${\n\t\t\t\t\t\tcontinuityEpoch + 1\n\t\t\t\t\t}`,\n\t\t\t\t\t{\n\t\t\t\t\t\tclientEpoch: continuityEpoch,\n\t\t\t\t\t\tserverEpoch: confirmed.epoch,\n\t\t\t\t\t},\n\t\t\t\t)\n\t\t\t\tconst confirmedUpdateIsAlreadyEnqueued = confirmedUpdates.some(\n\t\t\t\t\t(update) => update.epoch === confirmed.epoch,\n\t\t\t\t)\n\t\t\t\tif (confirmedUpdateIsAlreadyEnqueued) {\n\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t`👍`,\n\t\t\t\t\t\t`continuity`,\n\t\t\t\t\t\tcontinuityKey,\n\t\t\t\t\t\t`confirmed update #${confirmed.epoch} is already enqueued`,\n\t\t\t\t\t)\n\t\t\t\t} else {\n\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t`👈`,\n\t\t\t\t\t\t`continuity`,\n\t\t\t\t\t\tcontinuityKey,\n\t\t\t\t\t\t`pushing confirmed update #${confirmed.epoch} to queue`,\n\t\t\t\t\t)\n\t\t\t\t\tsetIntoStore(store, confirmedUpdateQueueAtom, (queue) => {\n\t\t\t\t\t\tqueue.push(confirmed)\n\t\t\t\t\t\tqueue.sort((a, b) => a.epoch - b.epoch)\n\t\t\t\t\t\treturn queue\n\t\t\t\t\t})\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t}\n","import type { AtomToken } from \"atom.io\"\nimport type { Store } from \"atom.io/internal\"\nimport { disposeAtom } from \"atom.io/internal\"\nimport type { Json } from \"atom.io/json\"\n\nexport function useConcealState(store: Store) {\n\treturn (concealed: AtomToken<Json.Serializable>[]): void => {\n\t\tfor (const token of concealed) {\n\t\t\tdisposeAtom(store, token)\n\t\t}\n\t}\n}\n","import { setIntoStore, type Store } from \"atom.io/internal\"\nimport type { Json } from \"atom.io/json\"\n\nexport function useRevealState(store: Store) {\n\treturn (revealed: Json.Array): void => {\n\t\tlet i = 0\n\t\tlet k: any\n\t\tlet v: any\n\t\tfor (const x of revealed) {\n\t\t\tif (i % 2 === 0) {\n\t\t\t\tk = x\n\t\t\t} else {\n\t\t\t\tv = x\n\t\t\t\tsetIntoStore(store, k, v)\n\t\t\t}\n\t\t\ti++\n\t\t}\n\t}\n}\n","import { Future } from \"atom.io/internal\"\nimport type { Socket } from \"atom.io/realtime\"\n\ntype SubData = { refcount: number; timer: Future<void> }\n\nconst subscriptions: WeakMap<Socket, Map<string, SubData>> = new WeakMap()\nconst socketIds: WeakMap<Socket, string | undefined> = new WeakMap()\n\nfunction getSubMap(socket: Socket): Map<string, SubData> {\n\tlet subMap = subscriptions.get(socket)\n\tif (subMap === undefined) {\n\t\tsubMap = new Map()\n\t\tsubscriptions.set(socket, subMap)\n\t}\n\treturn subMap\n}\n\nexport function createSubscriber<K extends string>(\n\tsocket: Socket,\n\tkey: K,\n\topen: (key: K) => () => void,\n): () => void {\n\tconst knownSocketId = socketIds.get(socket)\n\tif (knownSocketId !== socket.id) {\n\t\tsocketIds.set(socket, socket.id)\n\t\tsubscriptions.delete(socket)\n\t}\n\tconst subMap = getSubMap(socket)\n\tlet sub = subMap.get(key)\n\n\tif (sub) {\n\t\tsub.timer.use(new Promise<void>(() => {}))\n\t\tsub.refcount++\n\t} else {\n\t\tsub = { refcount: 1, timer: new Future<void>(() => {}) }\n\t\tsubMap.set(key, sub)\n\t\tconst close = open(key)\n\t\tvoid sub.timer.then(() => {\n\t\t\tclose()\n\t\t\tsubMap.delete(key)\n\t\t})\n\t}\n\treturn () => {\n\t\tsub.refcount--\n\n\t\tif (sub.refcount === 0) {\n\t\t\tconst timeout = new Promise<void>((resolve) => {\n\t\t\t\tsetTimeout(resolve, 50)\n\t\t\t})\n\t\t\tsub.timer.use(timeout)\n\t\t}\n\t}\n}\n","import type * as AtomIO from \"atom.io\"\nimport { setIntoStore, type Store } from \"atom.io/internal\"\nimport type { Json } from \"atom.io/json\"\nimport { employSocket, type Socket } from \"atom.io/realtime\"\n\nimport { createSubscriber } from \"./create-subscriber\"\n\nexport function pullAtom<J extends Json.Serializable>(\n\tstore: Store,\n\tsocket: Socket,\n\ttoken: AtomIO.RegularAtomToken<J, any, any>,\n): () => void {\n\treturn createSubscriber(socket, token.key, (key) => {\n\t\tconst stopWatching = employSocket(socket, `serve:${key}`, (data: J) => {\n\t\t\tsetIntoStore(store, token, data)\n\t\t})\n\t\tsocket.emit(`sub:${token.key}`)\n\t\treturn () => {\n\t\t\tsocket.emit(`unsub:${key}`)\n\t\t\tstopWatching()\n\t\t}\n\t})\n}\n","import type * as AtomIO from \"atom.io\"\nimport { findInStore, setIntoStore, type Store } from \"atom.io/internal\"\nimport type { Canonical, Json } from \"atom.io/json\"\nimport { employSocket, type Socket } from \"atom.io/realtime\"\n\nimport { createSubscriber } from \"./create-subscriber\"\n\nexport function pullAtomFamilyMember<\n\tJ extends Json.Serializable,\n\tK extends Canonical,\n>(\n\tstore: Store,\n\tsocket: Socket,\n\tfamily: AtomIO.AtomFamilyToken<J, K, any>,\n\tkey: NoInfer<K>,\n): () => void {\n\tconst token = findInStore(store, family, key)\n\treturn createSubscriber(socket, token.key, () => {\n\t\tconst stopWatching = employSocket(\n\t\t\tsocket,\n\t\t\t`serve:${token.key}`,\n\t\t\t(data: J) => {\n\t\t\t\tsetIntoStore(store, token, data)\n\t\t\t},\n\t\t)\n\t\tsocket.emit(`sub:${family.key}`, key)\n\t\treturn () => {\n\t\t\tsocket.emit(`unsub:${token.key}`)\n\t\t\tstopWatching()\n\t\t}\n\t})\n}\n","import type * as AtomIO from \"atom.io\"\nimport type { AsJSON, SignalFrom, Store, Transceiver } from \"atom.io/internal\"\nimport { getJsonToken, getUpdateToken, setIntoStore } from \"atom.io/internal\"\nimport { employSocket, type Socket } from \"atom.io/realtime\"\n\nimport { createSubscriber } from \"./create-subscriber\"\n\nexport function pullMutableAtom<T extends Transceiver<any, any, any>>(\n\tstore: Store,\n\tsocket: Socket,\n\ttoken: AtomIO.MutableAtomToken<T>,\n): () => void {\n\tconst jsonToken = getJsonToken(store, token)\n\tconst updateToken = getUpdateToken(token)\n\treturn createSubscriber(socket, token.key, () => {\n\t\tconst stopWatchingForInit = employSocket(\n\t\t\tsocket,\n\t\t\t`init:${token.key}`,\n\t\t\t(data: AsJSON<T>) => {\n\t\t\t\tsetIntoStore(store, jsonToken, data)\n\t\t\t},\n\t\t)\n\t\tconst stopWatchingForUpdate = employSocket(\n\t\t\tsocket,\n\t\t\t`next:${token.key}`,\n\t\t\t(data: SignalFrom<T>) => {\n\t\t\t\tsetIntoStore(store, updateToken, data)\n\t\t\t},\n\t\t)\n\t\tsocket.emit(`sub:${token.key}`)\n\t\treturn () => {\n\t\t\tsocket.emit(`unsub:${token.key}`)\n\t\t\tstopWatchingForInit()\n\t\t\tstopWatchingForUpdate()\n\t\t}\n\t})\n}\n","import type * as AtomIO from \"atom.io\"\nimport type { AsJSON, SignalFrom, Store, Transceiver } from \"atom.io/internal\"\nimport {\n\tfindInStore,\n\tgetJsonToken,\n\tgetUpdateToken,\n\tsetIntoStore,\n} from \"atom.io/internal\"\nimport type { Canonical } from \"atom.io/json\"\nimport { employSocket, type Socket } from \"atom.io/realtime\"\n\nimport { createSubscriber } from \"./create-subscriber\"\n\nexport function pullMutableAtomFamilyMember<\n\tT extends Transceiver<any, any, any>,\n\tK extends Canonical,\n>(\n\tstore: Store,\n\tsocket: Socket,\n\tfamily: AtomIO.MutableAtomFamilyToken<T, K>,\n\tkey: NoInfer<K>,\n): () => void {\n\tconst token = findInStore(store, family, key)\n\tconst jsonToken = getJsonToken(store, token)\n\tconst trackerToken = getUpdateToken(token)\n\treturn createSubscriber(socket, token.key, () => {\n\t\tconst stopWatchingForInit = employSocket(\n\t\t\tsocket,\n\t\t\t`init:${token.key}`,\n\t\t\t(data: AsJSON<T>) => {\n\t\t\t\tsetIntoStore(store, jsonToken, data)\n\t\t\t},\n\t\t)\n\t\tconst stopWatchingForUpdate = employSocket(\n\t\t\tsocket,\n\t\t\t`next:${token.key}`,\n\t\t\t(data: SignalFrom<T>) => {\n\t\t\t\tsetIntoStore(store, trackerToken, data)\n\t\t\t},\n\t\t)\n\t\tsocket.emit(`sub:${family.key}`, key)\n\t\treturn () => {\n\t\t\tsocket.emit(`unsub:${token.key}`)\n\t\t\tstopWatchingForInit()\n\t\t\tstopWatchingForUpdate()\n\t\t}\n\t})\n}\n","import type { AtomToken, SelectorToken } from \"atom.io\"\nimport type { Store } from \"atom.io/internal\"\nimport { getFamilyOfToken, subscribeToState } from \"atom.io/internal\"\nimport { parseJson } from \"atom.io/json\"\nimport type { Socket } from \"atom.io/realtime\"\n\nimport { pullAtom } from \"./pull-atom\"\nimport { pullAtomFamilyMember } from \"./pull-atom-family-member\"\nimport { pullMutableAtom } from \"./pull-mutable-atom\"\nimport { pullMutableAtomFamilyMember } from \"./pull-mutable-atom-family-member\"\n\nexport function pullSelectorRoots(\n\tstore: Store,\n\tsocket: Socket,\n\tselectorToken: SelectorToken<any>,\n): () => void {\n\tconst atomSubscriptions = new Map<string, () => void>()\n\n\tconst start = () => {\n\t\tconst atomKeys = store.selectorAtoms.getRelatedKeys(selectorToken.key)\n\t\tif (atomKeys) {\n\t\t\tfor (const [atomKey, unsub] of atomSubscriptions) {\n\t\t\t\tif (!atomKeys.has(atomKey)) {\n\t\t\t\t\tunsub()\n\t\t\t\t\tatomSubscriptions.delete(atomKey)\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tfor (const atomKey of atomKeys) {\n\t\t\t\tif (atomSubscriptions.has(atomKey)) {\n\t\t\t\t\tcontinue\n\t\t\t\t}\n\t\t\t\tconst atom = store.atoms.get(atomKey) as AtomToken<any, any>\n\t\t\t\tswitch (atom.type) {\n\t\t\t\t\tcase `atom`: {\n\t\t\t\t\t\tif (atom.family) {\n\t\t\t\t\t\t\tconst { subKey: serializedSubKey } = atom.family\n\t\t\t\t\t\t\tconst subKey = parseJson(serializedSubKey)\n\t\t\t\t\t\t\tconst family = getFamilyOfToken(store, atom)\n\t\t\t\t\t\t\tatomSubscriptions.set(\n\t\t\t\t\t\t\t\tatomKey,\n\t\t\t\t\t\t\t\tpullAtomFamilyMember(store, socket, family, subKey),\n\t\t\t\t\t\t\t)\n\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\tatomSubscriptions.set(atomKey, pullAtom(store, socket, atom))\n\t\t\t\t\t\t}\n\t\t\t\t\t\tbreak\n\t\t\t\t\t}\n\t\t\t\t\tcase `mutable_atom`: {\n\t\t\t\t\t\tif (atom.family) {\n\t\t\t\t\t\t\tconst { subKey: serializedSubKey } = atom.family\n\t\t\t\t\t\t\tconst subKey = parseJson(serializedSubKey)\n\t\t\t\t\t\t\tconst family = getFamilyOfToken(store, atom)\n\t\t\t\t\t\t\tatomSubscriptions.set(\n\t\t\t\t\t\t\t\tatomKey,\n\t\t\t\t\t\t\t\tpullMutableAtomFamilyMember(store, socket, family, subKey),\n\t\t\t\t\t\t\t)\n\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\tatomSubscriptions.set(\n\t\t\t\t\t\t\t\tatomKey,\n\t\t\t\t\t\t\t\tpullMutableAtom(store, socket, atom),\n\t\t\t\t\t\t\t)\n\t\t\t\t\t\t}\n\t\t\t\t\t\tbreak\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t}\n\n\tconst unsubFromSelector = subscribeToState(\n\t\tstore,\n\t\tselectorToken,\n\t\t`pull-watches-dependencies`,\n\t\t() => {\n\t\t\tstart()\n\t\t},\n\t)\n\n\tstart()\n\n\treturn () => {\n\t\tfor (const [, unsub] of atomSubscriptions) unsub()\n\t\tatomSubscriptions.clear()\n\t\tunsubFromSelector()\n\t}\n}\n","import type * as AtomIO from \"atom.io\"\nimport type { Store } from \"atom.io/internal\"\nimport type { Socket } from \"atom.io/realtime\"\n\nimport { pullSelectorRoots } from \"./pull-selector-roots\"\n\nexport function pullSelector<T>(\n\tstore: Store,\n\tsocket: Socket,\n\ttoken: AtomIO.SelectorToken<T>,\n): () => void {\n\treturn pullSelectorRoots(store, socket, token)\n}\n","import type * as AtomIO from \"atom.io\"\nimport type { Store } from \"atom.io/internal\"\nimport { findInStore } from \"atom.io/internal\"\nimport type { Canonical } from \"atom.io/json\"\nimport type { Socket } from \"atom.io/realtime\"\n\nimport { pullSelectorRoots } from \"./pull-selector-roots\"\n\nexport function pullSelectorFamilyMember<T, K extends Canonical>(\n\tstore: Store,\n\tsocket: Socket,\n\tfamilyToken: AtomIO.SelectorFamilyToken<T, K>,\n\tkey: NoInfer<K>,\n): () => void {\n\tconst token = findInStore(store, familyToken, key)\n\treturn pullSelectorRoots(store, socket, token)\n}\n","import type { WritableToken } from \"atom.io\"\nimport type { Store } from \"atom.io/internal\"\nimport { setIntoStore, subscribeToState } from \"atom.io/internal\"\nimport type { Json } from \"atom.io/json\"\nimport type { Socket } from \"atom.io/realtime\"\nimport { employSocket, mutexAtoms } from \"atom.io/realtime\"\n\nimport { createSubscriber } from \"./create-subscriber\"\n\nexport function pushState<J extends Json.Serializable>(\n\tstore: Store,\n\tsocket: Socket,\n\ttoken: WritableToken<J>,\n): () => void {\n\treturn createSubscriber(socket, `push:${token.key}`, () => {\n\t\tlet stopWatching = employSocket(\n\t\t\tsocket,\n\t\t\t`claim-result:${token.key}`,\n\t\t\t(success: boolean) => {\n\t\t\t\tif (!success) return\n\t\t\t\tstopWatching()\n\t\t\t\tsetIntoStore(store, mutexAtoms, token.key, true)\n\t\t\t\tstopWatching = subscribeToState(store, token, `push`, ({ newValue }) => {\n\t\t\t\t\tsocket.emit(`pub:${token.key}`, newValue)\n\t\t\t\t})\n\t\t\t},\n\t\t)\n\n\t\tsocket.emit(`claim:${token.key}`)\n\n\t\treturn () => {\n\t\t\tsocket.emit(`unclaim:${token.key}`)\n\t\t\tstopWatching()\n\t\t}\n\t})\n}\n","import type { RootStore } from \"atom.io/internal\"\nimport {\n\tassignTransactionToContinuity,\n\tgetFromStore,\n\tgetJsonToken,\n\tsetEpochNumberOfContinuity,\n\tsetIntoStore,\n\tsubscribeToTransaction,\n} from \"atom.io/internal\"\nimport type { Json } from \"atom.io/json\"\nimport type { ContinuityToken, Socket } from \"atom.io/realtime\"\n\nimport { useRegisterAndAttemptConfirmedUpdate } from \"./continuity/register-and-attempt-confirmed-update\"\nimport { useConcealState } from \"./continuity/use-conceal-state\"\nimport { useRevealState } from \"./continuity/use-reveal-state\"\nimport {\n\tconfirmedUpdateQueueAtom,\n\toptimisticUpdateQueueAtom,\n} from \"./realtime-client-stores\"\n\nexport function syncContinuity(\n\tstore: RootStore,\n\tsocket: Socket,\n\tcontinuity: ContinuityToken,\n): () => void {\n\tconst continuityKey = continuity.key\n\tconst optimisticUpdates = getFromStore(store, optimisticUpdateQueueAtom)\n\tconst confirmedUpdates = getFromStore(store, confirmedUpdateQueueAtom)\n\n\tconst initializeContinuity = (epoch: number, payload: Json.Array) => {\n\t\tsocket.off(`continuity-init:${continuityKey}`, initializeContinuity)\n\t\tlet i = 0\n\t\tlet k: any\n\t\tlet v: any\n\t\tfor (const x of payload) {\n\t\t\tif (i % 2 === 0) {\n\t\t\t\tk = x\n\t\t\t} else {\n\t\t\t\tv = x\n\t\t\t\tif (`type` in k && k.type === `mutable_atom`) {\n\t\t\t\t\tk = getJsonToken(store, k)\n\t\t\t\t}\n\t\t\t\tsetIntoStore(store, k, v)\n\t\t\t}\n\t\t\ti++\n\t\t}\n\t\tsetEpochNumberOfContinuity(store, continuityKey, epoch)\n\t}\n\tsocket.off(`continuity-init:${continuityKey}`)\n\tsocket.on(`continuity-init:${continuityKey}`, initializeContinuity)\n\n\tconst registerAndAttemptConfirmedUpdate = useRegisterAndAttemptConfirmedUpdate(\n\t\tstore,\n\t\tcontinuityKey,\n\t\tsocket,\n\t\toptimisticUpdates,\n\t\tconfirmedUpdates,\n\t)\n\tsocket.off(`tx-new:${continuityKey}`)\n\tsocket.on(`tx-new:${continuityKey}`, registerAndAttemptConfirmedUpdate)\n\n\tconst unsubscribeFunctions = continuity.actions.map((transaction) => {\n\t\tassignTransactionToContinuity(store, continuityKey, transaction.key)\n\t\tconst unsubscribeFromTransactionUpdates = subscribeToTransaction(\n\t\t\tstore,\n\t\t\ttransaction,\n\t\t\t`tx-run:${continuityKey}`,\n\t\t\t(clientUpdate) => {\n\t\t\t\tstore.logger.info(\n\t\t\t\t\t`🤞`,\n\t\t\t\t\t`continuity`,\n\t\t\t\t\tcontinuityKey,\n\t\t\t\t\t`enqueuing optimistic update`,\n\t\t\t\t)\n\t\t\t\tconst optimisticUpdateIndex = optimisticUpdates.findIndex(\n\t\t\t\t\t(update) => update.id === clientUpdate.id,\n\t\t\t\t)\n\t\t\t\tif (optimisticUpdateIndex === -1) {\n\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t`🤞`,\n\t\t\t\t\t\t`continuity`,\n\t\t\t\t\t\tcontinuityKey,\n\t\t\t\t\t\t`enqueuing new optimistic update`,\n\t\t\t\t\t)\n\t\t\t\t\tsetIntoStore(store, optimisticUpdateQueueAtom, (queue) => {\n\t\t\t\t\t\tqueue.push(clientUpdate)\n\t\t\t\t\t\tqueue.sort((a, b) => a.epoch - b.epoch)\n\t\t\t\t\t\treturn queue\n\t\t\t\t\t})\n\t\t\t\t} else {\n\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t`🤞`,\n\t\t\t\t\t\t`continuity`,\n\t\t\t\t\t\tcontinuityKey,\n\t\t\t\t\t\t`replacing existing optimistic update at index ${optimisticUpdateIndex}`,\n\t\t\t\t\t)\n\t\t\t\t\tsetIntoStore(store, optimisticUpdateQueueAtom, (queue) => {\n\t\t\t\t\t\tqueue[optimisticUpdateIndex] = clientUpdate\n\t\t\t\t\t\treturn queue\n\t\t\t\t\t})\n\t\t\t\t}\n\t\t\t\tsocket.emit(`tx-run:${continuityKey}`, {\n\t\t\t\t\tid: clientUpdate.id,\n\t\t\t\t\ttoken: transaction,\n\t\t\t\t\tparams: clientUpdate.params,\n\t\t\t\t})\n\t\t\t},\n\t\t)\n\t\treturn unsubscribeFromTransactionUpdates\n\t})\n\n\tconst revealState = useRevealState(store)\n\tconst concealState = useConcealState(store)\n\tsocket.on(`reveal:${continuityKey}`, revealState)\n\tsocket.on(`conceal:${continuityKey}`, concealState)\n\n\tsocket.emit(`get:${continuityKey}`)\n\treturn () => {\n\t\tsocket.off(`continuity-init:${continuityKey}`)\n\t\tsocket.off(`tx-new:${continuityKey}`)\n\t\tfor (const unsubscribe of unsubscribeFunctions) unsubscribe()\n\t\t// socket.emit(`unsub:${continuityKey}`)\n\t}\n}\n"],"mappings":"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"}
@@ -1,6 +1,6 @@
1
1
  import { RootStore, Subject, Transceiver } from "atom.io/internal";
2
2
  import * as AtomIO from "atom.io";
3
- import { Hierarchy, JoinToken, Loadable, MutableAtomToken, RegularAtomFamilyToken, WritableToken } from "atom.io";
3
+ import { Hierarchy, JoinToken, Loadable, MutableAtomToken, ReadableFamilyToken, RegularAtomFamilyToken, WritableToken } from "atom.io";
4
4
  import { Canonical, Json, stringified } from "atom.io/json";
5
5
  import { UList } from "atom.io/transceivers/u-list";
6
6
  import { ContinuityToken, GuardedSocket, RoomKey, RoomSocketInterface, Socket, SocketKey, StandardSchemaV1, UserKey } from "atom.io/realtime";
@@ -143,6 +143,7 @@ declare function destroyRoom({
143
143
  }: DestroyRoomConfig): (roomKey: RoomKey) => void;
144
144
  type ProvideRoomsConfig<RoomNames extends string> = {
145
145
  resolveRoomScript: (path: RoomNames) => [string, string[]];
146
+ roomAdminsToken: ReadableFamilyToken<boolean, UserKey>;
146
147
  roomNames: RoomNames[];
147
148
  roomTimeLimit?: number;
148
149
  userKey: UserKey;
@@ -150,10 +151,11 @@ type ProvideRoomsConfig<RoomNames extends string> = {
150
151
  socket: Socket;
151
152
  };
152
153
  declare function provideRooms<RoomNames extends string>({
153
- store,
154
- socket,
155
154
  resolveRoomScript,
155
+ roomAdminsToken,
156
156
  roomNames,
157
+ socket,
158
+ store,
157
159
  userKey
158
160
  }: ProvideRoomsConfig<RoomNames>): () => void;
159
161
  //#endregion
@@ -163,7 +165,7 @@ declare function realtimeAtomFamilyProvider({
163
165
  socket,
164
166
  consumer,
165
167
  store
166
- }: ServerConfig): <J extends Json.Serializable, K extends Canonical>(family: AtomIO.RegularAtomFamilyToken<J, K>, index: AtomIO.ReadableToken<Iterable<NoInfer<K>>> | Iterable<NoInfer<K>>) => () => void;
168
+ }: ServerConfig): <J extends Json.Serializable, K extends Canonical>(family: AtomIO.RegularAtomFamilyToken<J, K>, index: AtomIO.ReadableToken<Iterable<NoInfer<K>> | null> | Iterable<NoInfer<K>>) => () => void;
167
169
  //#endregion
168
170
  //#region src/realtime-server/realtime-mutable-family-provider.d.ts
169
171
  type MutableFamilyProvider = ReturnType<typeof realtimeMutableFamilyProvider>;
@@ -171,7 +173,7 @@ declare function realtimeMutableFamilyProvider({
171
173
  socket,
172
174
  consumer,
173
175
  store
174
- }: ServerConfig): <T extends Transceiver<any, any, any>, K extends Canonical>(family: AtomIO.MutableAtomFamilyToken<T, K>, index: AtomIO.ReadableToken<Iterable<NoInfer<K>>> | Iterable<NoInfer<K>>) => () => void;
176
+ }: ServerConfig): <T extends Transceiver<any, any, any>, K extends Canonical>(family: AtomIO.MutableAtomFamilyToken<T, K>, index: AtomIO.ReadableToken<Iterable<NoInfer<K>> | null> | Iterable<NoInfer<K>>) => () => void;
175
177
  //#endregion
176
178
  //#region src/realtime-server/realtime-mutable-provider.d.ts
177
179
  type MutableProvider = ReturnType<typeof realtimeMutableProvider>;
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.ts","names":["ATOM_IO_REALTIME_SERVER_ROOMS: RoomMap","ROOMS: RoomMap","roomMeta: { count: number }","socketAtoms: RegularAtomFamilyToken<Socket | null, SocketKey>","socketKeysAtom: MutableAtomToken<UList<SocketKey>>","onlineUsersAtom: MutableAtomToken<UList<UserKey>>","usersOfSockets: JoinToken<\n\t`user`,\n\tUserKey,\n\t`socket`,\n\tSocketKey,\n\t`1:1`\n>"],"sources":["../../src/realtime-server/continuity/provide-continuity.ts","../../src/realtime-server/ipc-sockets/custom-socket.ts","../../src/realtime-server/ipc-sockets/child-socket.ts","../../src/realtime-server/ipc-sockets/parent-socket.ts","../../src/realtime-server/provide-rooms.ts","../../src/realtime-server/realtime-family-provider.ts","../../src/realtime-server/realtime-mutable-family-provider.ts","../../src/realtime-server/realtime-mutable-provider.ts","../../src/realtime-server/realtime-state-provider.ts","../../src/realtime-server/realtime-state-receiver.ts","../../src/realtime-server/server-config.ts","../../src/realtime-server/server-socket-state.ts"],"sourcesContent":[],"mappings":"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"}
1
+ {"version":3,"file":"index.d.ts","names":["ATOM_IO_REALTIME_SERVER_ROOMS: RoomMap","ROOMS: RoomMap","roomMeta: { count: number }","socketAtoms: RegularAtomFamilyToken<Socket | null, SocketKey>","socketKeysAtom: MutableAtomToken<UList<SocketKey>>","onlineUsersAtom: MutableAtomToken<UList<UserKey>>","usersOfSockets: JoinToken<\n\t`user`,\n\tUserKey,\n\t`socket`,\n\tSocketKey,\n\t`1:1`\n>"],"sources":["../../src/realtime-server/continuity/provide-continuity.ts","../../src/realtime-server/ipc-sockets/custom-socket.ts","../../src/realtime-server/ipc-sockets/child-socket.ts","../../src/realtime-server/ipc-sockets/parent-socket.ts","../../src/realtime-server/provide-rooms.ts","../../src/realtime-server/realtime-family-provider.ts","../../src/realtime-server/realtime-mutable-family-provider.ts","../../src/realtime-server/realtime-mutable-provider.ts","../../src/realtime-server/realtime-state-provider.ts","../../src/realtime-server/realtime-state-receiver.ts","../../src/realtime-server/server-config.ts","../../src/realtime-server/server-socket-state.ts"],"sourcesContent":[],"mappings":"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"}
@@ -603,7 +603,7 @@ function realtimeMutableFamilyProvider({ socket, consumer, store = IMPLICIT.STOR
603
603
  let exposedSubKeys;
604
604
  if (dynamicIndex) exposedSubKeys = getFromStore(store, dynamicIndex);
605
605
  else exposedSubKeys = staticIndex;
606
- if (isAvailable(exposedSubKeys, subKey)) {
606
+ if (exposedSubKeys && isAvailable(exposedSubKeys, subKey)) {
607
607
  store.logger.info(`👀`, `user`, consumer, `is approved for a subscription to`, subKey, `in family "${family.key}"`);
608
608
  exposeFamilyMembers(subKey);
609
609
  } else {
@@ -614,6 +614,7 @@ function realtimeMutableFamilyProvider({ socket, consumer, store = IMPLICIT.STOR
614
614
  }));
615
615
  if (dynamicIndex) coreSubscriptions.add(subscribeToState(store, dynamicIndex, `expose-family:${family.key}:${socket.id}`, ({ newValue: newExposedSubKeys }) => {
616
616
  store.logger.info(`👀`, `user`, consumer, `has the following keys available for family "${family.key}"`, newExposedSubKeys);
617
+ if (newExposedSubKeys === null) return;
617
618
  for (const subKey of newExposedSubKeys) if (familyMemberSubscriptionsWanted.has(stringifyJson(subKey))) {
618
619
  store.logger.info(`👀`, `user`, consumer, `is retroactively approved for a subscription to`, subKey, `in family "${family.key}"`);
619
620
  exposeFamilyMembers(subKey);
@@ -787,7 +788,8 @@ function destroyRoom({ store, socket, userKey }) {
787
788
  store.logger.info(`📡`, `socket`, socket.id ?? `[ID MISSING?!]`, `👤 ${userKey} failed to delete ${roomKey}; its owner is ${owner}`);
788
789
  };
789
790
  }
790
- function provideRooms({ store = IMPLICIT.STORE, socket, resolveRoomScript, roomNames, userKey }) {
791
+ function provideRooms({ resolveRoomScript, roomAdminsToken, roomNames, socket, store = IMPLICIT.STORE, userKey }) {
792
+ const isAdmin = getFromStore(store, roomAdminsToken, userKey);
791
793
  const roomSocket = guardSocket(socket, createRoomSocketGuard(roomNames));
792
794
  const unsubFromRoomKeys = realtimeMutableProvider({
793
795
  socket,
@@ -816,17 +818,19 @@ function provideRooms({ store = IMPLICIT.STORE, socket, resolveRoomScript, roomN
816
818
  enterRoom(userRoomKey);
817
819
  break;
818
820
  }
819
- roomSocket.on(`createRoom`, spawnRoom({
820
- store,
821
- socket,
822
- userKey,
823
- resolveRoomScript
824
- }));
825
- roomSocket.on(`deleteRoom`, destroyRoom({
826
- store,
827
- socket,
828
- userKey
829
- }));
821
+ if (isAdmin) {
822
+ roomSocket.on(`createRoom`, spawnRoom({
823
+ store,
824
+ socket,
825
+ userKey,
826
+ resolveRoomScript
827
+ }));
828
+ roomSocket.on(`deleteRoom`, destroyRoom({
829
+ store,
830
+ socket,
831
+ userKey
832
+ }));
833
+ }
830
834
  return () => {
831
835
  unsubFromRoomKeys();
832
836
  unsubFromUsersInRooms();
@@ -917,7 +921,7 @@ function realtimeAtomFamilyProvider({ socket, consumer, store = IMPLICIT.STORE }
917
921
  let exposedSubKeys;
918
922
  if (dynamicIndex) exposedSubKeys = getFromStore(store, dynamicIndex);
919
923
  else exposedSubKeys = staticIndex;
920
- if (isAvailable$1(exposedSubKeys, subKey)) {
924
+ if (exposedSubKeys && isAvailable$1(exposedSubKeys, subKey)) {
921
925
  store.logger.info(`👀`, `user`, consumer, `is approved for a subscription to`, subKey, `in family "${family.key}"`);
922
926
  exposeFamilyMembers(subKey);
923
927
  } else {
@@ -928,6 +932,7 @@ function realtimeAtomFamilyProvider({ socket, consumer, store = IMPLICIT.STORE }
928
932
  }));
929
933
  if (dynamicIndex) coreSubscriptions.add(subscribeToState(store, dynamicIndex, `expose-family:${family.key}:${socket.id}`, ({ newValue: newExposedSubKeys }) => {
930
934
  store.logger.info(`👀`, `user`, consumer, `has the following keys available for family "${family.key}"`, newExposedSubKeys);
935
+ if (newExposedSubKeys === null) return;
931
936
  for (const subKey of newExposedSubKeys) if (familyMemberSubscriptionsWanted.has(stringifyJson(subKey))) {
932
937
  store.logger.info(`👀`, `user`, consumer, `is retroactively approved for a subscription to`, subKey, `in family "${family.key}"`);
933
938
  exposeFamilyMembers(subKey);
@@ -1 +1 @@
1
- {"version":3,"file":"index.js","names":["redactorAtoms: RegularAtomFamilyToken<\n\t{\n\t\tocclude: (updates: TransactionSubEvent[]) => TransactionSubEvent[]\n\t},\n\tUserKey\n>","unacknowledgedUpdatesAtoms: RegularAtomFamilyToken<\n\tContinuitySyncTransactionUpdate[],\n\tUserKey\n>","atom","initialPayload: Json.Serializable[]","atom","cleanupRelay","exposedSubKeys: Iterable<K>","ROOMS: RoomMap","roomMeta: { count: number }","room","roomQueue: [string, ...Json.Array][]","forward: AllEventsListener<EventsMap>","roomKeySchema: StandardSchemaV1<Json.Array, [RoomKey]>","isAvailable","exposedSubKeys: Iterable<K>","socketAtoms: RegularAtomFamilyToken<Socket | null, SocketKey>","socketKeysAtom: MutableAtomToken<UList<SocketKey>>","onlineUsersAtom: MutableAtomToken<UList<UserKey>>","usersOfSockets: JoinToken<\n\t`user`,\n\tUserKey,\n\t`socket`,\n\tSocketKey,\n\t`1:1`\n>","serverConfig: UserServerConfig"],"sources":["../../src/realtime-server/continuity/continuity-store.ts","../../src/realtime-server/continuity/provide-outcomes.ts","../../src/realtime-server/continuity/provide-perspectives.ts","../../src/realtime-server/continuity/provide-startup-payloads.ts","../../src/realtime-server/continuity/receive-action-requests.ts","../../src/realtime-server/continuity/track-acknowledgements.ts","../../src/realtime-server/continuity/provide-continuity.ts","../../src/realtime-server/ipc-sockets/custom-socket.ts","../../src/realtime-server/ipc-sockets/parent-socket.ts","../../src/realtime-server/ipc-sockets/child-socket.ts","../../src/realtime-server/realtime-mutable-family-provider.ts","../../src/realtime-server/realtime-mutable-provider.ts","../../src/realtime-server/provide-rooms.ts","../../src/realtime-server/realtime-family-provider.ts","../../src/realtime-server/realtime-state-provider.ts","../../src/realtime-server/realtime-state-receiver.ts","../../src/realtime-server/server-socket-state.ts","../../src/realtime-server/server-config.ts"],"sourcesContent":["import type {\n\tRegularAtomFamilyToken,\n\tTransactionOutcomeEvent,\n\tTransactionSubEvent,\n} from \"atom.io\"\nimport { atomFamily } from \"atom.io\"\nimport type { UserKey } from \"atom.io/realtime\"\n\nexport function redactTransactionUpdateContent(\n\tvisibleStateKeys: string[],\n\tupdates: TransactionSubEvent[],\n): TransactionSubEvent[] {\n\treturn updates\n\t\t.map((update): TransactionSubEvent => {\n\t\t\tswitch (update.type) {\n\t\t\t\tcase `transaction_outcome`: {\n\t\t\t\t\tconst redacted = redactTransactionUpdateContent(\n\t\t\t\t\t\tvisibleStateKeys,\n\t\t\t\t\t\tupdate.subEvents,\n\t\t\t\t\t)\n\t\t\t\t\treturn { ...update, subEvents: redacted }\n\t\t\t\t}\n\t\t\t\tcase `atom_update`:\n\t\t\t\tcase `molecule_creation`:\n\t\t\t\tcase `molecule_disposal`:\n\t\t\t\tcase `molecule_transfer`:\n\t\t\t\tcase `state_creation`:\n\t\t\t\tcase `state_disposal`:\n\t\t\t\t\treturn update\n\t\t\t}\n\t\t})\n\t\t.filter((update) => {\n\t\t\tswitch (update.type) {\n\t\t\t\tcase `atom_update`:\n\t\t\t\tcase `state_creation`:\n\t\t\t\tcase `state_disposal`:\n\t\t\t\t\treturn visibleStateKeys.includes(update.token.key)\n\t\t\t\tcase `molecule_creation`:\n\t\t\t\tcase `transaction_outcome`:\n\t\t\t\tcase `molecule_disposal`:\n\t\t\t\tcase `molecule_transfer`:\n\t\t\t\t\treturn true\n\t\t\t}\n\t\t})\n}\n\nexport const redactorAtoms: RegularAtomFamilyToken<\n\t{\n\t\tocclude: (updates: TransactionSubEvent[]) => TransactionSubEvent[]\n\t},\n\tUserKey\n> = atomFamily({\n\tkey: `redactor`,\n\tdefault: { occlude: (updates) => updates },\n})\n\nexport type ContinuitySyncTransactionUpdate = Pick<\n\tTransactionOutcomeEvent<any>,\n\t`epoch` | `id` | `output` | `subEvents` | `token`\n>\nexport const unacknowledgedUpdatesAtoms: RegularAtomFamilyToken<\n\tContinuitySyncTransactionUpdate[],\n\tUserKey\n> = atomFamily({\n\tkey: `unacknowledgedUpdates`,\n\tdefault: () => [],\n})\n","import type { Store } from \"atom.io/internal\"\nimport {\n\tfindInStore,\n\tgetFromStore,\n\tgetUpdateToken,\n\tsetIntoStore,\n\tsubscribeToTransaction,\n} from \"atom.io/internal\"\nimport type { Json } from \"atom.io/json\"\nimport type { ContinuityToken, Socket, UserKey } from \"atom.io/realtime\"\n\nimport {\n\tredactTransactionUpdateContent,\n\tunacknowledgedUpdatesAtoms,\n} from \"./continuity-store\"\n\nexport function provideOutcomes(\n\tstore: Store,\n\tsocket: Socket,\n\tcontinuity: ContinuityToken,\n\tuserKey: UserKey,\n): () => void {\n\tconst continuityKey = continuity.key\n\tconst unsubscribeFunctions = new Set<() => void>()\n\n\tfor (const transaction of continuity.actions) {\n\t\tconst unsubscribeFromTransaction = subscribeToTransaction(\n\t\t\tstore,\n\t\t\ttransaction,\n\t\t\t`sync-continuity:${continuityKey}:${userKey}`,\n\t\t\t(outcomes) => {\n\t\t\t\ttry {\n\t\t\t\t\tconst visibleKeys = continuity.globals\n\t\t\t\t\t\t.map((atom) => {\n\t\t\t\t\t\t\tif (atom.type === `atom`) {\n\t\t\t\t\t\t\t\treturn atom.key\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\treturn getUpdateToken(atom).key\n\t\t\t\t\t\t})\n\t\t\t\t\t\t.concat(\n\t\t\t\t\t\t\tcontinuity.perspectives.flatMap((perspective) => {\n\t\t\t\t\t\t\t\tconst { viewAtoms } = perspective\n\t\t\t\t\t\t\t\tconst userPerspectiveTokenState = findInStore(\n\t\t\t\t\t\t\t\t\tstore,\n\t\t\t\t\t\t\t\t\tviewAtoms,\n\t\t\t\t\t\t\t\t\tuserKey,\n\t\t\t\t\t\t\t\t)\n\t\t\t\t\t\t\t\tconst visibleTokens = getFromStore(\n\t\t\t\t\t\t\t\t\tstore,\n\t\t\t\t\t\t\t\t\tuserPerspectiveTokenState,\n\t\t\t\t\t\t\t\t)\n\t\t\t\t\t\t\t\treturn visibleTokens.map((token) => {\n\t\t\t\t\t\t\t\t\tconst key =\n\t\t\t\t\t\t\t\t\t\ttoken.type === `mutable_atom` ? `*` + token.key : token.key\n\t\t\t\t\t\t\t\t\treturn key\n\t\t\t\t\t\t\t\t})\n\t\t\t\t\t\t\t}),\n\t\t\t\t\t\t)\n\t\t\t\t\tconst redactedUpdates = redactTransactionUpdateContent(\n\t\t\t\t\t\tvisibleKeys,\n\t\t\t\t\t\toutcomes.subEvents,\n\t\t\t\t\t)\n\t\t\t\t\tconst redactedUpdate = {\n\t\t\t\t\t\t...outcomes,\n\t\t\t\t\t\tupdates: redactedUpdates,\n\t\t\t\t\t}\n\t\t\t\t\tsetIntoStore(store, unacknowledgedUpdatesAtoms, userKey, (updates) => {\n\t\t\t\t\t\tif (redactedUpdate) {\n\t\t\t\t\t\t\tupdates.push(redactedUpdate)\n\t\t\t\t\t\t\tupdates.sort((a, b) => a.epoch - b.epoch)\n\t\t\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t\t\t`👍`,\n\t\t\t\t\t\t\t\t`continuity`,\n\t\t\t\t\t\t\t\tcontinuityKey,\n\t\t\t\t\t\t\t\t`${userKey} unacknowledged update queue now has`,\n\t\t\t\t\t\t\t\tupdates.length,\n\t\t\t\t\t\t\t\t`items`,\n\t\t\t\t\t\t\t)\n\t\t\t\t\t\t}\n\t\t\t\t\t\treturn updates\n\t\t\t\t\t})\n\n\t\t\t\t\tsocket.emit(\n\t\t\t\t\t\t`tx-new:${continuityKey}`,\n\t\t\t\t\t\tredactedUpdate as unknown as Json.Serializable,\n\t\t\t\t\t)\n\t\t\t\t} catch (thrown) {\n\t\t\t\t\tif (thrown instanceof Error) {\n\t\t\t\t\t\tstore.logger.error(\n\t\t\t\t\t\t\t`❌`,\n\t\t\t\t\t\t\t`continuity`,\n\t\t\t\t\t\t\tcontinuityKey,\n\t\t\t\t\t\t\t`${userKey} failed to send update from transaction ${transaction.key} to ${userKey}`,\n\t\t\t\t\t\t\tthrown.message,\n\t\t\t\t\t\t)\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t},\n\t\t)\n\t\tunsubscribeFunctions.add(unsubscribeFromTransaction)\n\t}\n\treturn () => {\n\t\tfor (const unsubscribe of unsubscribeFunctions) unsubscribe()\n\t}\n}\n","import type { Store } from \"atom.io/internal\"\nimport {\n\tfindInStore,\n\tgetFromStore,\n\tgetJsonToken,\n\tsubscribeToState,\n} from \"atom.io/internal\"\nimport type { ContinuityToken, Socket, UserKey } from \"atom.io/realtime\"\n\nexport function providePerspectives(\n\tstore: Store,\n\tsocket: Socket,\n\tcontinuity: ContinuityToken,\n\tuserKey: UserKey,\n): () => void {\n\tconst continuityKey = continuity.key\n\tconst unsubFns = new Set<() => void>()\n\tfor (const perspective of continuity.perspectives) {\n\t\tconst { viewAtoms } = perspective\n\t\tconst userViewState = findInStore(store, viewAtoms, userKey)\n\t\tconst unsubscribeFromUserView = subscribeToState(\n\t\t\tstore,\n\t\t\tuserViewState,\n\t\t\t`sync-continuity:${continuityKey}:${userKey}:perspective:${perspective.resourceAtoms.key}`,\n\t\t\t({ oldValue, newValue }) => {\n\t\t\t\tconst oldKeys = oldValue?.map((token) => token.key)\n\t\t\t\tconst newKeys = newValue.map((token) => token.key)\n\t\t\t\tconst concealed = oldValue?.filter(\n\t\t\t\t\t(token) => !newKeys.includes(token.key),\n\t\t\t\t)\n\t\t\t\tconst revealed = newValue\n\t\t\t\t\t.filter((token) => !oldKeys?.includes(token.key))\n\t\t\t\t\t.flatMap((token) => {\n\t\t\t\t\t\tconst resourceToken =\n\t\t\t\t\t\t\ttoken.type === `mutable_atom` ? getJsonToken(store, token) : token\n\t\t\t\t\t\tconst resource = getFromStore(store, resourceToken)\n\t\t\t\t\t\treturn [resourceToken, resource]\n\t\t\t\t\t})\n\t\t\t\tstore.logger.info(\n\t\t\t\t\t`👁`,\n\t\t\t\t\t`atom`,\n\t\t\t\t\tperspective.resourceAtoms.key,\n\t\t\t\t\t`${userKey} has a new perspective`,\n\t\t\t\t\t{ oldKeys, newKeys, revealed, concealed },\n\t\t\t\t)\n\t\t\t\tif (revealed.length > 0) {\n\t\t\t\t\tsocket.emit(`reveal:${continuityKey}`, revealed)\n\t\t\t\t}\n\t\t\t\tif (concealed && concealed.length > 0) {\n\t\t\t\t\tsocket.emit(`conceal:${continuityKey}`, concealed)\n\t\t\t\t}\n\t\t\t},\n\t\t)\n\t\tunsubFns.add(unsubscribeFromUserView)\n\t}\n\treturn () => {\n\t\tfor (const unsubscribe of unsubFns) unsubscribe()\n\t}\n}\n","import type { Store } from \"atom.io/internal\"\nimport {\n\tfindInStore,\n\tgetFromStore,\n\tgetJsonToken,\n\tisRootStore,\n} from \"atom.io/internal\"\nimport type { Json } from \"atom.io/json\"\nimport type { ContinuityToken, Socket, UserKey } from \"atom.io/realtime\"\nimport { employSocket } from \"atom.io/realtime\"\n\nexport function provideStartupPayloads(\n\tstore: Store,\n\tsocket: Socket,\n\tcontinuity: ContinuityToken,\n\tuserKey: UserKey,\n): () => void {\n\tconst continuityKey = continuity.key\n\tfunction sendInitialPayload(): void {\n\t\tconst initialPayload: Json.Serializable[] = []\n\t\tfor (const atom of continuity.globals) {\n\t\t\tconst resourceToken =\n\t\t\t\tatom.type === `mutable_atom` ? getJsonToken(store, atom) : atom\n\t\t\tconst resource = getFromStore(store, resourceToken)\n\t\t\tinitialPayload.push(resourceToken, resource)\n\t\t}\n\t\tfor (const perspective of continuity.perspectives) {\n\t\t\tconst { viewAtoms, resourceAtoms } = perspective\n\t\t\tconst userViewState = findInStore(store, viewAtoms, userKey)\n\t\t\tconst userView = getFromStore(store, userViewState)\n\t\t\tstore.logger.info(`👁`, `atom`, resourceAtoms.key, `${userKey} can see`, {\n\t\t\t\tviewAtoms,\n\t\t\t\tresourceAtoms,\n\t\t\t\tuserView,\n\t\t\t})\n\t\t\tfor (const visibleToken of userView) {\n\t\t\t\tconst resourceToken =\n\t\t\t\t\tvisibleToken.type === `mutable_atom`\n\t\t\t\t\t\t? getJsonToken(store, visibleToken)\n\t\t\t\t\t\t: visibleToken\n\t\t\t\tconst resource = getFromStore(store, resourceToken)\n\n\t\t\t\tinitialPayload.push(resourceToken, resource)\n\t\t\t}\n\t\t}\n\n\t\tconst epoch = isRootStore(store)\n\t\t\t? (store.transactionMeta.epoch.get(continuityKey) ?? null)\n\t\t\t: null\n\n\t\tsocket.emit(`continuity-init:${continuityKey}`, epoch, initialPayload)\n\t}\n\treturn employSocket(socket, `get:${continuityKey}`, sendInitialPayload)\n}\n","import type { TransactionOutcomeEvent, TransactionToken } from \"atom.io\"\nimport type { Store } from \"atom.io/internal\"\nimport { actUponStore } from \"atom.io/internal\"\nimport type { Json, JsonIO } from \"atom.io/json\"\nimport type { ContinuityToken, Socket } from \"atom.io/realtime\"\nimport { employSocket } from \"atom.io/realtime\"\n\nexport function receiveActionRequests(\n\tstore: Store,\n\tsocket: Socket,\n\tcontinuity: ContinuityToken,\n\tuserKey: string,\n): () => void {\n\tconst continuityKey = continuity.key\n\n\treturn employSocket(\n\t\tsocket,\n\t\t`tx-run:${continuityKey}`,\n\t\tfunction serveTransactionRequest(\n\t\t\ttxOutcome: Json.Serializable &\n\t\t\t\tPick<\n\t\t\t\t\tTransactionOutcomeEvent<TransactionToken<JsonIO>>,\n\t\t\t\t\t`id` | `params` | `token`\n\t\t\t\t>,\n\t\t) {\n\t\t\tstore.logger.info(`🛎️`, `continuity`, continuityKey, `received`, txOutcome)\n\t\t\tconst transactionKey = txOutcome.token.key\n\t\t\tconst updateId = txOutcome.id\n\t\t\tconst performanceKey = `tx-run:${transactionKey}:${updateId}`\n\t\t\tconst performanceKeyStart = `${performanceKey}:start`\n\t\t\tconst performanceKeyEnd = `${performanceKey}:end`\n\t\t\tperformance.mark(performanceKeyStart)\n\t\t\ttry {\n\t\t\t\tactUponStore(\n\t\t\t\t\tstore,\n\t\t\t\t\t{ type: `transaction`, key: transactionKey },\n\t\t\t\t\tupdateId,\n\t\t\t\t)(...txOutcome.params)\n\t\t\t} catch (thrown) {\n\t\t\t\tif (thrown instanceof Error) {\n\t\t\t\t\tstore.logger.error(\n\t\t\t\t\t\t`❌`,\n\t\t\t\t\t\t`continuity`,\n\t\t\t\t\t\tcontinuityKey,\n\t\t\t\t\t\t`failed to run transaction ${transactionKey} from ${userKey} with update ${updateId}`,\n\t\t\t\t\t\tthrown.message,\n\t\t\t\t\t)\n\t\t\t\t}\n\t\t\t}\n\t\t\tperformance.mark(performanceKeyEnd)\n\t\t\tconst metric = performance.measure(\n\t\t\t\tperformanceKey,\n\t\t\t\tperformanceKeyStart,\n\t\t\t\tperformanceKeyEnd,\n\t\t\t)\n\t\t\tstore.logger.info(\n\t\t\t\t`🚀`,\n\t\t\t\t`transaction`,\n\t\t\t\ttransactionKey,\n\t\t\t\tupdateId,\n\t\t\t\tuserKey,\n\t\t\t\tmetric.duration,\n\t\t\t)\n\t\t},\n\t)\n}\n","import type { Store } from \"atom.io/internal\"\nimport { getFromStore, setIntoStore } from \"atom.io/internal\"\nimport type { ContinuityToken, Socket, UserKey } from \"atom.io/realtime\"\nimport { employSocket } from \"atom.io/realtime\"\n\nimport { unacknowledgedUpdatesAtoms } from \"./continuity-store\"\n\nexport function trackAcknowledgements(\n\tstore: Store,\n\tsocket: Socket,\n\tcontinuity: ContinuityToken,\n\tuserKey: UserKey,\n): () => void {\n\tconst continuityKey = continuity.key\n\tconst userUnacknowledgedUpdates = getFromStore(\n\t\tstore,\n\t\tunacknowledgedUpdatesAtoms,\n\t\tuserKey,\n\t)\n\tfunction trackClientAcknowledgement(epoch: number): void {\n\t\tstore.logger.info(\n\t\t\t`👍`,\n\t\t\t`continuity`,\n\t\t\tcontinuityKey,\n\t\t\t`${userKey} acknowledged epoch ${epoch}`,\n\t\t)\n\t\tconst isUnacknowledged = userUnacknowledgedUpdates[0]?.epoch === epoch\n\t\tif (isUnacknowledged) {\n\t\t\tsetIntoStore(store, unacknowledgedUpdatesAtoms, userKey, (updates) => {\n\t\t\t\tupdates.shift()\n\t\t\t\tstore.logger.info(\n\t\t\t\t\t`👍`,\n\t\t\t\t\t`continuity`,\n\t\t\t\t\tcontinuityKey,\n\t\t\t\t\t`${userKey} unacknowledged update queue now has`,\n\t\t\t\t\tupdates.length,\n\t\t\t\t\t`items`,\n\t\t\t\t)\n\t\t\t\treturn updates\n\t\t\t})\n\t\t}\n\t}\n\treturn employSocket(socket, `ack:${continuityKey}`, trackClientAcknowledgement)\n}\n","import { getFromStore, IMPLICIT } from \"atom.io/internal\"\nimport type { Json } from \"atom.io/json\"\nimport type { ContinuityToken, UserKey } from \"atom.io/realtime\"\n\nimport type { ServerConfig } from \"..\"\nimport { unacknowledgedUpdatesAtoms } from \"./continuity-store\"\nimport { provideOutcomes } from \"./provide-outcomes\"\nimport { providePerspectives } from \"./provide-perspectives\"\nimport { provideStartupPayloads } from \"./provide-startup-payloads\"\nimport { receiveActionRequests } from \"./receive-action-requests\"\nimport { trackAcknowledgements } from \"./track-acknowledgements\"\n\nexport type ProvideContinuity = (\n\tcontinuity: ContinuityToken,\n\tuserKey: UserKey,\n) => () => void\nexport function prepareToProvideContinuity({\n\tsocket,\n\tstore = IMPLICIT.STORE,\n}: ServerConfig): ProvideContinuity {\n\treturn function syncRealtimeContinuity(continuity, userKey) {\n\t\tconst continuityKey = continuity.key\n\n\t\tconst unacknowledgedUpdates = getFromStore(\n\t\t\tstore,\n\t\t\tunacknowledgedUpdatesAtoms,\n\t\t\tuserKey,\n\t\t)\n\t\tfor (const unacknowledgedUpdate of unacknowledgedUpdates) {\n\t\t\tsocket.emit(\n\t\t\t\t`tx-new:${continuityKey}`,\n\t\t\t\tunacknowledgedUpdate as Json.Serializable,\n\t\t\t)\n\t\t}\n\n\t\tconst subscriptions = new Set<() => void>()\n\t\tconst clearSubscriptions = () => {\n\t\t\tfor (const unsubscribe of subscriptions) unsubscribe()\n\t\t\tsubscriptions.clear()\n\t\t}\n\n\t\tsubscriptions.add(providePerspectives(store, socket, continuity, userKey))\n\t\tsubscriptions.add(provideOutcomes(store, socket, continuity, userKey))\n\t\tsubscriptions.add(provideStartupPayloads(store, socket, continuity, userKey))\n\t\tsubscriptions.add(receiveActionRequests(store, socket, continuity, userKey))\n\t\tsubscriptions.add(trackAcknowledgements(store, socket, continuity, userKey))\n\n\t\treturn clearSubscriptions\n\t}\n}\n","import type { Json, stringified } from \"atom.io/json\"\nimport type { Socket } from \"atom.io/realtime\"\n\nexport type Events = Json.Object<string, Json.Serializable[]>\n\nexport type EventPayload<\n\treceiveRelay extends Events,\n\tK extends string & keyof receiveRelay = string & keyof receiveRelay,\n> = [string, ...receiveRelay[K]]\n\nexport interface EventBuffer<\n\tE extends Events,\n\tK extends string & keyof E = string & keyof E,\n> extends Buffer {\n\ttoString(): stringified<EventPayload<E, K>>\n}\n\nexport abstract class CustomSocket<I extends Events, O extends Events>\n\timplements Socket\n{\n\tprotected listeners: Map<keyof O, Set<(...args: Json.Array) => void>>\n\tprotected globalListeners: Set<(event: string, ...args: Json.Array) => void>\n\tprotected globalListenersOutgoing: Set<\n\t\t(event: string, ...args: Json.Array) => void\n\t>\n\tprotected handleEvent<K extends string & keyof I>(\n\t\t...args: EventPayload<I, K>\n\t): void {\n\t\tconst [event, ...rest] = args\n\t\tfor (const listener of this.globalListeners) {\n\t\t\tlistener(event, ...rest)\n\t\t}\n\t\tconst listeners = this.listeners.get(event)\n\t\tif (listeners) {\n\t\t\tfor (const listener of listeners) {\n\t\t\t\tlistener(...rest)\n\t\t\t}\n\t\t}\n\t}\n\n\tpublic id = `no_id_retrieved`\n\tpublic emit: <Event extends string & keyof I>(\n\t\tevent: Event,\n\t\t...args: I[Event]\n\t) => CustomSocket<I, O>\n\n\tpublic constructor(\n\t\temit: <Event extends keyof I>(\n\t\t\tevent: Event,\n\t\t\t...args: I[Event]\n\t\t) => CustomSocket<I, O>,\n\t) {\n\t\tthis.emit = (...args) => {\n\t\t\tfor (const listener of this.globalListenersOutgoing) {\n\t\t\t\tlistener(...args)\n\t\t\t}\n\t\t\treturn emit(...args)\n\t\t}\n\t\tthis.listeners = new Map()\n\t\tthis.globalListeners = new Set()\n\t\tthis.globalListenersOutgoing = new Set()\n\t}\n\n\tpublic on<Event extends keyof O>(\n\t\tevent: Event,\n\t\tlistener: (...args: O[Event]) => void,\n\t): this {\n\t\tconst listeners = this.listeners.get(event)\n\t\tif (listeners) {\n\t\t\tlisteners.add(listener)\n\t\t} else {\n\t\t\tthis.listeners.set(event, new Set([listener]))\n\t\t}\n\t\treturn this\n\t}\n\n\tpublic onAny(listener: (event: string, ...args: Json.Array) => void): this {\n\t\tthis.globalListeners.add(listener)\n\t\treturn this\n\t}\n\n\tpublic onAnyOutgoing(\n\t\tlistener: (event: string, ...args: Json.Array) => void,\n\t): this {\n\t\tthis.globalListenersOutgoing.add(listener)\n\t\treturn this\n\t}\n\n\tpublic off<Event extends keyof O>(\n\t\tevent: Event,\n\t\tlistener?: (...args: O[Event]) => void,\n\t): this {\n\t\tconst listeners = this.listeners.get(event)\n\t\tif (listeners) {\n\t\t\tif (listener) {\n\t\t\t\tlisteners.delete(listener)\n\t\t\t} else {\n\t\t\t\tthis.listeners.delete(event)\n\t\t\t}\n\t\t}\n\t\treturn this\n\t}\n\n\tpublic offAny(listener: (event: string, ...args: Json.Array) => void): this {\n\t\tthis.globalListeners.delete(listener)\n\t\treturn this\n\t}\n}\n","import type { Readable, Writable } from \"node:stream\"\n\nimport { Subject } from \"atom.io/internal\"\nimport type { Json } from \"atom.io/json\"\nimport { parseJson, stringifyJson } from \"atom.io/json\"\nimport type { UserKey } from \"atom.io/realtime\"\nimport { UList } from \"atom.io/transceivers/u-list\"\n\nimport type { StderrLog } from \"./child-socket\"\nimport type { EventBuffer, EventPayload, Events } from \"./custom-socket\"\nimport { CustomSocket } from \"./custom-socket\"\n\nexport const PROOF_OF_LIFE_SIGNAL = `ALIVE`\n\nexport class SubjectSocket<\n\tI extends Events,\n\tO extends Events,\n> extends CustomSocket<I, O> {\n\tpublic in: Subject<EventPayload<I>>\n\tpublic out: Subject<EventPayload<O>>\n\tpublic id = `no_id_retrieved`\n\tpublic disposalEffects: (() => void)[] = []\n\n\tpublic constructor(id: string) {\n\t\tsuper((...args) => {\n\t\t\tthis.out.next(args as any)\n\t\t\treturn this\n\t\t})\n\t\tthis.id = id\n\t\tthis.in = new Subject()\n\t\tthis.out = new Subject()\n\t\tthis.in.subscribe(`socket`, (event) => {\n\t\t\tthis.handleEvent(...event)\n\t\t})\n\t}\n\n\tpublic dispose(): void {\n\t\tfor (const dispose of this.disposalEffects) {\n\t\t\tdispose()\n\t\t}\n\t}\n}\n\nexport type ParentProcess = {\n\tpid?: number | undefined\n\tstdin: Readable\n\tstdout: Writable\n\tstderr: Writable\n\texit: (code?: number) => void\n}\n\nexport class ParentSocket<\n\tI extends Events & {\n\t\t[user: UserKey]: [string, ...Json.Array[]]\n\t},\n\tO extends Events & {\n\t\t[id in string as `user::${id}`]: [string, ...Json.Array[]]\n\t} & {\n\t\t/* eslint-disable quotes */\n\t\t\"user-joins\": [key: UserKey]\n\t\t\"user-leaves\": [key: UserKey]\n\t\t/* eslint-enable quotes */\n\t},\n\tP extends ParentProcess = ParentProcess,\n> extends CustomSocket<I, O> {\n\tprotected incompleteData = ``\n\tprotected unprocessedEvents: string[] = []\n\tprotected relays: Map<UserKey, SubjectSocket<any, any>>\n\tprotected initRelay: (\n\t\tsocket: SubjectSocket<any, any>,\n\t\tuserKey: UserKey,\n\t) => (() => void) | void\n\tpublic proc: P\n\n\tpublic id = `#####`\n\n\tprotected log(...args: StderrLog): void {\n\t\tthis.proc.stderr.write(\n\t\t\tstringifyJson(\n\t\t\t\targs.map((arg) =>\n\t\t\t\t\targ instanceof UList ? `{ ${arg.toJSON().join(` | `)} }` : arg,\n\t\t\t\t),\n\t\t\t) + `\\x03`,\n\t\t)\n\t}\n\tpublic logger = {\n\t\tinfo: (...args: Json.Array): void => {\n\t\t\tthis.log(`i`, ...args)\n\t\t},\n\t\twarn: (...args: Json.Array): void => {\n\t\t\tthis.log(`w`, ...args)\n\t\t},\n\t\terror: (...args: Json.Array): void => {\n\t\t\tthis.log(`e`, ...args)\n\t\t},\n\t}\n\n\tpublic constructor(proc: P) {\n\t\tsuper((event, ...args) => {\n\t\t\tconst stringifiedEvent = JSON.stringify([event, ...args])\n\t\t\tthis.proc.stdout.write(stringifiedEvent + `\\x03`)\n\t\t\treturn this\n\t\t})\n\t\tthis.proc = proc\n\t\tthis.proc.stdin.resume()\n\t\tthis.relays = new Map()\n\t\tthis.initRelay = () => {\n\t\t\tthis.logger.info(`🔗`, `nothing to relay`)\n\t\t}\n\n\t\tthis.proc.stdin.on(\n\t\t\t`data`,\n\t\t\t<K extends string & keyof I>(buffer: EventBuffer<I, K>) => {\n\t\t\t\tconst chunk = buffer.toString()\n\t\t\t\tconst pieces = chunk.split(`\\x03`)\n\t\t\t\tconst initialMaybeWellFormed = pieces[0]\n\t\t\t\tpieces[0] = this.incompleteData + initialMaybeWellFormed\n\t\t\t\tlet idx = 0\n\t\t\t\tfor (const piece of pieces) {\n\t\t\t\t\tif (piece === ``) {\n\t\t\t\t\t\tcontinue\n\t\t\t\t\t}\n\t\t\t\t\ttry {\n\t\t\t\t\t\tconst jsonPiece = parseJson(piece)\n\t\t\t\t\t\tthis.logger.info(`🎰`, `received`, jsonPiece)\n\t\t\t\t\t\tthis.handleEvent(...(jsonPiece as EventPayload<I>))\n\t\t\t\t\t\tthis.incompleteData = ``\n\t\t\t\t\t} catch (thrown0) {\n\t\t\t\t\t\tif (thrown0 instanceof Error) {\n\t\t\t\t\t\t\tthis.logger.error(\n\t\t\t\t\t\t\t\t[\n\t\t\t\t\t\t\t\t\t`received malformed data from parent process:`,\n\t\t\t\t\t\t\t\t\t``,\n\t\t\t\t\t\t\t\t\tpiece,\n\t\t\t\t\t\t\t\t\t``,\n\t\t\t\t\t\t\t\t\tthrown0.message,\n\t\t\t\t\t\t\t\t].join(`\\n❌\\t`),\n\t\t\t\t\t\t\t)\n\t\t\t\t\t\t}\n\t\t\t\t\t\ttry {\n\t\t\t\t\t\t\tif (idx === 0) {\n\t\t\t\t\t\t\t\tthis.incompleteData = piece\n\t\t\t\t\t\t\t\tconst maybeActualJsonPiece = parseJson(initialMaybeWellFormed)\n\t\t\t\t\t\t\t\tthis.logger.info(`🎰`, `received`, maybeActualJsonPiece)\n\t\t\t\t\t\t\t\tthis.handleEvent(...(maybeActualJsonPiece as EventPayload<I>))\n\t\t\t\t\t\t\t\tthis.incompleteData = ``\n\t\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\t\tthis.incompleteData += piece\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t} catch (thrown1) {\n\t\t\t\t\t\t\tif (thrown1 instanceof Error) {\n\t\t\t\t\t\t\t\tthis.logger.error(\n\t\t\t\t\t\t\t\t\t[\n\t\t\t\t\t\t\t\t\t\t`received malformed data from parent process:`,\n\t\t\t\t\t\t\t\t\t\t``,\n\t\t\t\t\t\t\t\t\t\tinitialMaybeWellFormed,\n\t\t\t\t\t\t\t\t\t\t``,\n\t\t\t\t\t\t\t\t\t\tthrown1.message,\n\t\t\t\t\t\t\t\t\t].join(`\\n❌\\t`),\n\t\t\t\t\t\t\t\t)\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t\t++idx\n\t\t\t\t}\n\t\t\t},\n\t\t)\n\n\t\tthis.on(`exit`, () => {\n\t\t\tthis.logger.info(`🔥`, this.id, `received \"exit\"`)\n\t\t\tthis.proc.exit(0)\n\t\t})\n\n\t\tif (this.proc.pid) {\n\t\t\tthis.id = this.proc.pid?.toString()\n\t\t}\n\n\t\tthis.on(`user-joins`, (userKey: UserKey) => {\n\t\t\tthis.logger.info(`👤`, userKey, `joined`)\n\t\t\tconst existingRelay = this.relays.get(userKey)\n\t\t\tif (existingRelay) {\n\t\t\t\tthis.logger.info(`🔗`, `reattaching relay services for`, userKey)\n\t\t\t\tconst cleanupRelay = this.initRelay(existingRelay, userKey)\n\t\t\t\tif (cleanupRelay) {\n\t\t\t\t\texistingRelay.disposalEffects.push(cleanupRelay)\n\t\t\t\t}\n\t\t\t\tthis.on(userKey, (...data) => {\n\t\t\t\t\trelay.in.next(data)\n\t\t\t\t})\n\t\t\t\texistingRelay.disposalEffects.push(\n\t\t\t\t\texistingRelay.out.subscribe(`socket`, (data) => {\n\t\t\t\t\t\tthis.emit(userKey, ...(data as any))\n\t\t\t\t\t}),\n\t\t\t\t)\n\t\t\t\treturn\n\t\t\t}\n\t\t\tconst relay = new SubjectSocket(userKey)\n\t\t\tthis.relays.set(userKey, relay)\n\t\t\tthis.logger.info(`🔗`, `attaching relay services for`, userKey)\n\t\t\tconst cleanupRelay = this.initRelay(relay, userKey)\n\t\t\tif (cleanupRelay) {\n\t\t\t\trelay.disposalEffects.push(cleanupRelay)\n\t\t\t}\n\t\t\tthis.on(userKey, (...data) => {\n\t\t\t\trelay.in.next(data)\n\t\t\t})\n\t\t\trelay.disposalEffects.push(\n\t\t\t\trelay.out.subscribe(`socket`, (data) => {\n\t\t\t\t\tthis.emit(userKey, ...(data as any))\n\t\t\t\t}),\n\t\t\t)\n\t\t})\n\n\t\tthis.on(`user-leaves`, (userKey: UserKey) => {\n\t\t\tconst relay = this.relays.get(userKey)\n\t\t\tthis.off(userKey)\n\t\t\tif (relay) {\n\t\t\t\trelay.dispose()\n\t\t\t\tthis.relays.delete(userKey)\n\t\t\t}\n\t\t})\n\n\t\tthis.proc.stdout.write(PROOF_OF_LIFE_SIGNAL)\n\t}\n\n\tpublic receiveRelay(\n\t\tattachServices: (\n\t\t\tsocket: SubjectSocket<any, any>,\n\t\t\tuserKey: UserKey,\n\t\t) => (() => void) | void,\n\t): void {\n\t\tthis.initRelay = attachServices\n\t\tthis.logger.info(`🔗`, `ready to relay`)\n\t}\n}\n","import type { Readable, Writable } from \"node:stream\"\n\nimport type { Json, stringified } from \"atom.io/json\"\nimport { parseJson } from \"atom.io/json\"\n\nimport type { EventBuffer, EventPayload, Events } from \"./custom-socket\"\nimport { CustomSocket } from \"./custom-socket\"\nimport { PROOF_OF_LIFE_SIGNAL } from \"./parent-socket\"\n\n/* eslint-disable no-console */\n\nexport type ChildProcess = {\n\tpid?: number | undefined\n\tstdin: Writable\n\tstdout: Readable\n\tstderr: Readable\n}\n\nexport type StderrLog = [`e` | `i` | `w`, ...Json.Array]\n\nexport class ChildSocket<\n\tI extends Events,\n\tO extends Events,\n\tP extends ChildProcess = ChildProcess,\n> extends CustomSocket<I, O> {\n\tprotected incompleteData = ``\n\tprotected unprocessedEvents: string[] = []\n\tprotected incompleteLog = ``\n\tprotected unprocessedLogs: string[] = []\n\n\tpublic id = `#####`\n\n\tpublic proc: P\n\tpublic key: string\n\tpublic logger: Pick<Console, `error` | `info` | `warn`>\n\n\tprotected handleLog(log: StderrLog): void {\n\t\tif (Array.isArray(log)) {\n\t\t\tconst [level, ...rest] = log\n\t\t\tswitch (level) {\n\t\t\t\tcase `i`:\n\t\t\t\t\tthis.logger.info(...rest)\n\t\t\t\t\tbreak\n\t\t\t\tcase `w`:\n\t\t\t\t\tthis.logger.warn(...rest)\n\t\t\t\t\tbreak\n\t\t\t\tcase `e`:\n\t\t\t\t\tthis.logger.error(...rest)\n\t\t\t\t\tbreak\n\t\t\t}\n\t\t}\n\t}\n\n\tpublic constructor(\n\t\tproc: P,\n\t\tkey: string,\n\t\tlogger?: Pick<Console, `error` | `info` | `warn`>,\n\t) {\n\t\tsuper((event, ...args) => {\n\t\t\tconst stringifiedEvent = JSON.stringify([event, ...args]) + `\\x03`\n\t\t\tthis.proc.stdin.write(stringifiedEvent)\n\t\t\treturn this\n\t\t})\n\n\t\tthis.proc = proc\n\t\tthis.key = key\n\t\tthis.logger = logger ?? {\n\t\t\tinfo: (...args: unknown[]) => {\n\t\t\t\tconsole.info(this.id, this.key, ...args)\n\t\t\t},\n\t\t\twarn: (...args: unknown[]) => {\n\t\t\t\tconsole.warn(this.id, this.key, ...args)\n\t\t\t},\n\t\t\terror: (...args: unknown[]) => {\n\t\t\t\tconsole.error(this.id, this.key, ...args)\n\t\t\t},\n\t\t}\n\t\tthis.proc.stdout.on(\n\t\t\t`data`,\n\t\t\t<K extends string & keyof I>(buffer: EventBuffer<I, K>) => {\n\t\t\t\tif (buffer[0] === 27 && buffer[1] === 91 && buffer[2] === 50) {\n\t\t\t\t\tthis.logger.info(`STDOUT TERMINAL CLEAR`, buffer)\n\t\t\t\t\treturn\n\t\t\t\t}\n\t\t\t\tconst chunk = buffer.toString()\n\n\t\t\t\tif (chunk === PROOF_OF_LIFE_SIGNAL) {\n\t\t\t\t\treturn\n\t\t\t\t}\n\n\t\t\t\tconst pieces = chunk.split(`\\x03`)\n\t\t\t\tconst initialMaybeWellFormed = pieces[0]\n\t\t\t\tpieces[0] = this.incompleteData + initialMaybeWellFormed\n\t\t\t\tlet idx = 0\n\t\t\t\tfor (const piece of pieces) {\n\t\t\t\t\tif (piece === ``) {\n\t\t\t\t\t\tcontinue\n\t\t\t\t\t}\n\t\t\t\t\ttry {\n\t\t\t\t\t\tconst jsonPiece = parseJson(piece as stringified<EventPayload<I, K>>)\n\t\t\t\t\t\tthis.logger.info(`💸`, `emitted`, jsonPiece)\n\t\t\t\t\t\tthis.handleEvent(...jsonPiece)\n\t\t\t\t\t\tthis.incompleteData = ``\n\t\t\t\t\t} catch (thrown0) {\n\t\t\t\t\t\tif (thrown0 instanceof Error) {\n\t\t\t\t\t\t\tconsole.error(\n\t\t\t\t\t\t\t\t[\n\t\t\t\t\t\t\t\t\t`❌ Malformed data received from child process:`,\n\t\t\t\t\t\t\t\t\t``,\n\t\t\t\t\t\t\t\t\t...piece.split(`\\n`),\n\t\t\t\t\t\t\t\t\t``,\n\t\t\t\t\t\t\t\t\tthrown0.message,\n\t\t\t\t\t\t\t\t].join(`\\n❌\\t`) + `\\n`,\n\t\t\t\t\t\t\t)\n\t\t\t\t\t\t}\n\t\t\t\t\t\ttry {\n\t\t\t\t\t\t\tif (idx === 0) {\n\t\t\t\t\t\t\t\tthis.incompleteData = piece\n\t\t\t\t\t\t\t\tconst maybeActualJsonPiece = parseJson(\n\t\t\t\t\t\t\t\t\tinitialMaybeWellFormed as stringified<EventPayload<I, K>>,\n\t\t\t\t\t\t\t\t)\n\t\t\t\t\t\t\t\tthis.logger.info(`💸`, `emitted`, maybeActualJsonPiece)\n\t\t\t\t\t\t\t\tthis.handleEvent(...maybeActualJsonPiece)\n\t\t\t\t\t\t\t\tthis.incompleteData = ``\n\t\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\t\tthis.incompleteData += piece\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t} catch (thrown1) {\n\t\t\t\t\t\t\tif (thrown1 instanceof Error) {\n\t\t\t\t\t\t\t\tconsole.error(\n\t\t\t\t\t\t\t\t\t[\n\t\t\t\t\t\t\t\t\t\t`❌ Malformed data received from child process:`,\n\t\t\t\t\t\t\t\t\t\t``,\n\t\t\t\t\t\t\t\t\t\t...initialMaybeWellFormed.split(`\\n`),\n\t\t\t\t\t\t\t\t\t\t``,\n\t\t\t\t\t\t\t\t\t\tthrown1.message,\n\t\t\t\t\t\t\t\t\t].join(`\\n❌\\t`) + `\\n`,\n\t\t\t\t\t\t\t\t)\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t\t++idx\n\t\t\t\t}\n\t\t\t},\n\t\t)\n\t\tthis.proc.stderr.on(`data`, (buffer: Buffer) => {\n\t\t\tif (buffer[0] === 27 && buffer[1] === 91 && buffer[2] === 50) {\n\t\t\t\tthis.logger.info(`STDERR TERMINAL CLEAR`, buffer)\n\t\t\t\treturn\n\t\t\t}\n\t\t\tconst chunk = buffer.toString()\n\n\t\t\tconst pieces = chunk.split(`\\x03`)\n\t\t\tconst initialMaybeWellFormed = pieces[0]\n\t\t\tpieces[0] = this.incompleteData + initialMaybeWellFormed\n\t\t\tlet idx = 0\n\t\t\tfor (const piece of pieces) {\n\t\t\t\tif (piece === ``) {\n\t\t\t\t\tcontinue\n\t\t\t\t}\n\t\t\t\ttry {\n\t\t\t\t\tconst jsonPiece = parseJson(piece as stringified<StderrLog>)\n\t\t\t\t\tthis.handleLog(jsonPiece)\n\t\t\t\t\tthis.incompleteData = ``\n\t\t\t\t} catch (thrown0) {\n\t\t\t\t\tif (thrown0 instanceof Error) {\n\t\t\t\t\t\tthis.logger.error(\n\t\t\t\t\t\t\t[\n\t\t\t\t\t\t\t\t`❌ Malformed log received from child process:`,\n\t\t\t\t\t\t\t\t``,\n\t\t\t\t\t\t\t\t...piece.split(`\\n`),\n\t\t\t\t\t\t\t\t``,\n\t\t\t\t\t\t\t\tthrown0.message,\n\t\t\t\t\t\t\t].join(`\\n❌\\t`) + `\\n`,\n\t\t\t\t\t\t)\n\t\t\t\t\t}\n\t\t\t\t\ttry {\n\t\t\t\t\t\tif (idx === 0) {\n\t\t\t\t\t\t\tthis.incompleteData = piece\n\t\t\t\t\t\t\tconst maybeActualJsonPiece = parseJson(\n\t\t\t\t\t\t\t\tinitialMaybeWellFormed as stringified<StderrLog>,\n\t\t\t\t\t\t\t)\n\t\t\t\t\t\t\tthis.handleLog(maybeActualJsonPiece)\n\t\t\t\t\t\t\tthis.incompleteData = ``\n\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\tthis.incompleteData += piece\n\t\t\t\t\t\t}\n\t\t\t\t\t} catch (thrown1) {\n\t\t\t\t\t\tif (thrown1 instanceof Error) {\n\t\t\t\t\t\t\tthis.logger.error(\n\t\t\t\t\t\t\t\t[\n\t\t\t\t\t\t\t\t\t`❌ Malformed log received from child process:`,\n\t\t\t\t\t\t\t\t\t``,\n\t\t\t\t\t\t\t\t\t...initialMaybeWellFormed.split(`\\n`),\n\t\t\t\t\t\t\t\t\t``,\n\t\t\t\t\t\t\t\t\tthrown1.message,\n\t\t\t\t\t\t\t\t].join(`\\n❌\\t`) + `\\n`,\n\t\t\t\t\t\t\t)\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t\t++idx\n\t\t\t}\n\t\t})\n\t\tthis.proc.stdin.once(`error`, (err: { code: string }) => {\n\t\t\tif (err.code === `EPIPE`) {\n\t\t\t\tconsole.error(`EPIPE error during write`, this.proc.stdin)\n\t\t\t}\n\t\t})\n\t\tif (proc.pid) {\n\t\t\tthis.id = proc.pid.toString()\n\t\t}\n\t}\n}\n","import type * as AtomIO from \"atom.io\"\nimport type { Transceiver } from \"atom.io/internal\"\nimport {\n\tfindInStore,\n\tgetFromStore,\n\tgetJsonToken,\n\tgetUpdateToken,\n\tIMPLICIT,\n\tsubscribeToState,\n} from \"atom.io/internal\"\nimport type { Canonical, stringified } from \"atom.io/json\"\nimport { stringifyJson } from \"atom.io/json\"\nimport { employSocket } from \"atom.io/realtime\"\n\nimport type { ServerConfig } from \".\"\n\nconst isAvailable = <K extends Canonical>(\n\texposedSubKeys: Iterable<K>,\n\tsubKey: K,\n): boolean => {\n\tfor (const exposedSubKey of exposedSubKeys) {\n\t\tif (stringifyJson(exposedSubKey) === stringifyJson(subKey)) {\n\t\t\treturn true\n\t\t}\n\t}\n\treturn false\n}\n\nexport type MutableFamilyProvider = ReturnType<\n\ttypeof realtimeMutableFamilyProvider\n>\nexport function realtimeMutableFamilyProvider({\n\tsocket,\n\tconsumer,\n\tstore = IMPLICIT.STORE,\n}: ServerConfig) {\n\treturn function mutableFamilyProvider<\n\t\tT extends Transceiver<any, any, any>,\n\t\tK extends Canonical,\n\t>(\n\t\tfamily: AtomIO.MutableAtomFamilyToken<T, K>,\n\t\tindex: AtomIO.ReadableToken<Iterable<NoInfer<K>>> | Iterable<NoInfer<K>>,\n\t): () => void {\n\t\tconst [dynamicIndex, staticIndex]:\n\t\t\t| [AtomIO.ReadableToken<Iterable<NoInfer<K>>>, undefined]\n\t\t\t| [undefined, Iterable<NoInfer<K>>] = (() => {\n\t\t\tif (typeof index === `object` && `key` in index && `type` in index) {\n\t\t\t\treturn [index, undefined] as const\n\t\t\t}\n\t\t\treturn [undefined, index] as const\n\t\t})()\n\n\t\tconst coreSubscriptions = new Set<() => void>()\n\t\tconst clearCoreSubscriptions = () => {\n\t\t\tfor (const unsub of coreSubscriptions) unsub()\n\t\t\tcoreSubscriptions.clear()\n\t\t}\n\t\tconst familyMemberSubscriptionsWanted = new Set<stringified<K>>()\n\t\tconst familyMemberSubscriptions = new Map<string, () => void>()\n\t\tconst clearFamilySubscriptions = () => {\n\t\t\tfor (const unsub of familyMemberSubscriptions.values()) unsub()\n\t\t\tfamilyMemberSubscriptions.clear()\n\t\t}\n\n\t\tconst fillUnsubRequest = (key: string) => {\n\t\t\tconst unsubUnsub = familyMemberSubscriptions.get(`${key}:unsub`)\n\t\t\tif (unsubUnsub) {\n\t\t\t\tunsubUnsub()\n\t\t\t\tfamilyMemberSubscriptions.delete(`${key}:unsub`)\n\t\t\t}\n\t\t\tconst unsub = familyMemberSubscriptions.get(key)\n\t\t\tif (unsub) {\n\t\t\t\tunsub()\n\t\t\t\tfamilyMemberSubscriptions.delete(key)\n\t\t\t}\n\t\t}\n\n\t\tconst exposeFamilyMembers = (subKey: K) => {\n\t\t\tconst token = findInStore(store, family, subKey)\n\t\t\tgetFromStore(store, token)\n\t\t\tconst jsonToken = getJsonToken(store, token)\n\t\t\tconst updateToken = getUpdateToken(token)\n\t\t\tsocket.emit(`init:${token.key}`, getFromStore(store, jsonToken))\n\t\t\tfamilyMemberSubscriptions.set(\n\t\t\t\ttoken.key,\n\t\t\t\tsubscribeToState(\n\t\t\t\t\tstore,\n\t\t\t\t\tupdateToken,\n\t\t\t\t\t`expose-family:${family.key}:${socket.id}`,\n\t\t\t\t\t({ newValue }) => {\n\t\t\t\t\t\tsocket.emit(`next:${token.key}`, newValue)\n\t\t\t\t\t},\n\t\t\t\t),\n\t\t\t)\n\t\t\tfamilyMemberSubscriptions.set(\n\t\t\t\t`${token.key}:unsub`,\n\t\t\t\temploySocket(socket, `unsub:${token.key}`, () => {\n\t\t\t\t\tfillUnsubRequest(token.key)\n\t\t\t\t}),\n\t\t\t)\n\t\t}\n\n\t\tconst start = () => {\n\t\t\tstore.logger.info(\n\t\t\t\t`👀`,\n\t\t\t\t`user`,\n\t\t\t\tconsumer,\n\t\t\t\t`can subscribe to family \"${family.key}\"`,\n\t\t\t)\n\t\t\tcoreSubscriptions.add(\n\t\t\t\temploySocket(socket, `sub:${family.key}`, (subKey: K) => {\n\t\t\t\t\tlet exposedSubKeys: Iterable<K>\n\t\t\t\t\tif (dynamicIndex) {\n\t\t\t\t\t\texposedSubKeys = getFromStore(store, dynamicIndex)\n\t\t\t\t\t} else {\n\t\t\t\t\t\texposedSubKeys = staticIndex\n\t\t\t\t\t}\n\t\t\t\t\tconst shouldExpose = isAvailable(exposedSubKeys, subKey)\n\t\t\t\t\tif (shouldExpose) {\n\t\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t\t`👀`,\n\t\t\t\t\t\t\t`user`,\n\t\t\t\t\t\t\tconsumer,\n\t\t\t\t\t\t\t`is approved for a subscription to`,\n\t\t\t\t\t\t\tsubKey,\n\t\t\t\t\t\t\t`in family \"${family.key}\"`,\n\t\t\t\t\t\t)\n\t\t\t\t\t\texposeFamilyMembers(subKey)\n\t\t\t\t\t} else {\n\t\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t\t`❌`,\n\t\t\t\t\t\t\t`user`,\n\t\t\t\t\t\t\tconsumer,\n\t\t\t\t\t\t\t`is denied for a subscription to`,\n\t\t\t\t\t\t\tsubKey,\n\t\t\t\t\t\t\t`in family \"${family.key}\"`,\n\t\t\t\t\t\t)\n\t\t\t\t\t\tfamilyMemberSubscriptionsWanted.add(stringifyJson(subKey))\n\t\t\t\t\t\tsocket.emit(`unavailable:${family.key}`, subKey)\n\t\t\t\t\t}\n\t\t\t\t}),\n\t\t\t)\n\t\t\tif (dynamicIndex) {\n\t\t\t\tcoreSubscriptions.add(\n\t\t\t\t\tsubscribeToState(\n\t\t\t\t\t\tstore,\n\t\t\t\t\t\tdynamicIndex,\n\t\t\t\t\t\t`expose-family:${family.key}:${socket.id}`,\n\t\t\t\t\t\t({ newValue: newExposedSubKeys }) => {\n\t\t\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t\t\t`👀`,\n\t\t\t\t\t\t\t\t`user`,\n\t\t\t\t\t\t\t\tconsumer,\n\t\t\t\t\t\t\t\t`has the following keys available for family \"${family.key}\"`,\n\t\t\t\t\t\t\t\tnewExposedSubKeys,\n\t\t\t\t\t\t\t)\n\t\t\t\t\t\t\tfor (const subKey of newExposedSubKeys) {\n\t\t\t\t\t\t\t\tif (familyMemberSubscriptionsWanted.has(stringifyJson(subKey))) {\n\t\t\t\t\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t\t\t\t\t`👀`,\n\t\t\t\t\t\t\t\t\t\t`user`,\n\t\t\t\t\t\t\t\t\t\tconsumer,\n\t\t\t\t\t\t\t\t\t\t`is retroactively approved for a subscription to`,\n\t\t\t\t\t\t\t\t\t\tsubKey,\n\t\t\t\t\t\t\t\t\t\t`in family \"${family.key}\"`,\n\t\t\t\t\t\t\t\t\t)\n\t\t\t\t\t\t\t\t\texposeFamilyMembers(subKey)\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t},\n\t\t\t\t\t),\n\t\t\t\t)\n\t\t\t}\n\t\t}\n\n\t\tstart()\n\n\t\treturn () => {\n\t\t\tclearCoreSubscriptions()\n\t\t\tclearFamilySubscriptions()\n\t\t}\n\t}\n}\n","import type * as AtomIO from \"atom.io\"\nimport type { Transceiver } from \"atom.io/internal\"\nimport {\n\tgetFromStore,\n\tgetJsonToken,\n\tgetUpdateToken,\n\tIMPLICIT,\n\tsubscribeToState,\n} from \"atom.io/internal\"\nimport type { Json } from \"atom.io/json\"\nimport { employSocket } from \"atom.io/realtime\"\n\nimport type { ServerConfig } from \".\"\n\nexport type MutableProvider = ReturnType<typeof realtimeMutableProvider>\nexport function realtimeMutableProvider({\n\tsocket,\n\tconsumer,\n\tstore = IMPLICIT.STORE,\n}: ServerConfig) {\n\treturn function mutableProvider<\n\t\tCore extends Transceiver<any, Json.Serializable, Json.Serializable>,\n\t>(token: AtomIO.MutableAtomToken<Core>): () => void {\n\t\tconst subscriptions = new Set<() => void>()\n\t\tconst clearSubscriptions = () => {\n\t\t\tfor (const unsub of subscriptions) unsub()\n\t\t\tsubscriptions.clear()\n\t\t}\n\n\t\tconst jsonToken = getJsonToken(store, token)\n\t\tconst trackerToken = getUpdateToken(token)\n\n\t\tconst start = () => {\n\t\t\tstore.logger.info(\n\t\t\t\t`👀`,\n\t\t\t\t`user`,\n\t\t\t\tconsumer,\n\t\t\t\t`can subscribe to state \"${token.key}\"`,\n\t\t\t)\n\t\t\tsubscriptions.add(\n\t\t\t\temploySocket(socket, `sub:${token.key}`, () => {\n\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t`👀`,\n\t\t\t\t\t\t`user`,\n\t\t\t\t\t\tconsumer,\n\t\t\t\t\t\t`subscribes to state \"${token.key}\"`,\n\t\t\t\t\t)\n\t\t\t\t\tclearSubscriptions()\n\t\t\t\t\tsocket.emit(`init:${token.key}`, getFromStore(store, jsonToken))\n\t\t\t\t\tsubscriptions.add(\n\t\t\t\t\t\tsubscribeToState(\n\t\t\t\t\t\t\tstore,\n\t\t\t\t\t\t\ttrackerToken,\n\t\t\t\t\t\t\t`expose-single:${socket.id}`,\n\t\t\t\t\t\t\t({ newValue }) => {\n\t\t\t\t\t\t\t\tsocket.emit(`next:${token.key}`, newValue)\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t),\n\t\t\t\t\t)\n\t\t\t\t\tsubscriptions.add(\n\t\t\t\t\t\temploySocket(socket, `unsub:${token.key}`, () => {\n\t\t\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t\t\t`🙈`,\n\t\t\t\t\t\t\t\t`user`,\n\t\t\t\t\t\t\t\tconsumer,\n\t\t\t\t\t\t\t\t`unsubscribes from state \"${token.key}\"`,\n\t\t\t\t\t\t\t)\n\t\t\t\t\t\t\tclearSubscriptions()\n\t\t\t\t\t\t\tstart()\n\t\t\t\t\t\t}),\n\t\t\t\t\t)\n\t\t\t\t}),\n\t\t\t)\n\t\t}\n\n\t\tstart()\n\n\t\treturn clearSubscriptions\n\t}\n}\n","import type { ChildProcessWithoutNullStreams } from \"node:child_process\"\nimport { spawn } from \"node:child_process\"\n\nimport type { RootStore } from \"atom.io/internal\"\nimport {\n\teditRelationsInStore,\n\tfindInStore,\n\tfindRelationsInStore,\n\tgetFromStore,\n\tgetInternalRelationsFromStore,\n\tIMPLICIT,\n\tsetIntoStore,\n} from \"atom.io/internal\"\nimport type { Json } from \"atom.io/json\"\nimport type {\n\tAllEventsListener,\n\tEventsMap,\n\tGuardedSocket,\n\tRoomKey,\n\tRoomSocketInterface,\n\tSocket,\n\tSocketGuard,\n\tStandardSchemaV1,\n\tUserKey,\n} from \"atom.io/realtime\"\nimport {\n\tguardSocket,\n\tisRoomKey,\n\townersOfRooms,\n\troomKeysAtom,\n\tusersInRooms,\n\tvisibilityFromRoomSelector,\n\tvisibleUsersInRoomsSelector,\n} from \"atom.io/realtime\"\n\nimport { ChildSocket, PROOF_OF_LIFE_SIGNAL } from \"./ipc-sockets\"\nimport { realtimeMutableFamilyProvider } from \"./realtime-mutable-family-provider\"\nimport { realtimeMutableProvider } from \"./realtime-mutable-provider\"\n\nexport type RoomMap = Map<\n\tstring,\n\tChildSocket<any, any, ChildProcessWithoutNullStreams>\n>\n\ndeclare global {\n\tvar ATOM_IO_REALTIME_SERVER_ROOMS: RoomMap\n}\nexport const ROOMS: RoomMap =\n\tglobalThis.ATOM_IO_REALTIME_SERVER_ROOMS ??\n\t(globalThis.ATOM_IO_REALTIME_SERVER_ROOMS = new Map())\n\nexport const roomMeta: { count: number } = { count: 0 }\n\nexport type SpawnRoomConfig<RoomNames extends string> = {\n\tstore: RootStore\n\tsocket: Socket\n\tuserKey: UserKey\n\tresolveRoomScript: (roomName: RoomNames) => [string, string[]]\n}\nexport function spawnRoom<RoomNames extends string>({\n\tstore,\n\tsocket,\n\tuserKey,\n\tresolveRoomScript,\n}: SpawnRoomConfig<RoomNames>): (\n\troomName: RoomNames,\n) => Promise<ChildSocket<any, any>> {\n\treturn async (roomName) => {\n\t\tstore.logger.info(\n\t\t\t`📡`,\n\t\t\t`socket`,\n\t\t\tsocket.id ?? `[ID MISSING?!]`,\n\t\t\t`👤 ${userKey} spawns room ${roomName}`,\n\t\t)\n\t\tconst roomKey = `room::${roomMeta.count++}-${roomName}` satisfies RoomKey\n\t\tconst [command, args] = resolveRoomScript(roomName)\n\t\tconst child = await new Promise<ChildProcessWithoutNullStreams>(\n\t\t\t(resolve) => {\n\t\t\t\tconst room = spawn(command, args, {\n\t\t\t\t\tenv: { ...process.env, REALTIME_ROOM_KEY: roomKey },\n\t\t\t\t})\n\t\t\t\tconst resolver = (data: Buffer) => {\n\t\t\t\t\tconst chunk = data.toString()\n\t\t\t\t\tif (chunk === PROOF_OF_LIFE_SIGNAL) {\n\t\t\t\t\t\troom.stdout.off(`data`, resolver)\n\t\t\t\t\t\tresolve(room)\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t\troom.stdout.on(`data`, resolver)\n\t\t\t},\n\t\t)\n\n\t\tconst room = new ChildSocket(child, roomKey)\n\t\tROOMS.set(roomKey, room)\n\t\tsetIntoStore(store, roomKeysAtom, (index) => (index.add(roomKey), index))\n\t\teditRelationsInStore(store, ownersOfRooms, (relations) => {\n\t\t\trelations.set({ room: roomKey, user: userKey })\n\t\t})\n\n\t\tconst provideMutableFamily = realtimeMutableFamilyProvider({\n\t\t\tsocket: room,\n\t\t\tconsumer: roomKey,\n\t\t\tstore,\n\t\t})\n\n\t\tconst ownersOfRoomsAtoms = getInternalRelationsFromStore(\n\t\t\tstore,\n\t\t\townersOfRooms,\n\t\t)\n\t\tconst unsubFromOwnerKeys = provideMutableFamily(ownersOfRoomsAtoms, [\n\t\t\troomKey,\n\t\t])\n\t\tconst usersInRoomsAtoms = getInternalRelationsFromStore(store, usersInRooms)\n\t\tconst unsubFromUsersInRooms = provideMutableFamily(\n\t\t\tusersInRoomsAtoms,\n\t\t\tfindInStore(store, visibilityFromRoomSelector, roomKey),\n\t\t)\n\n\t\troom.on(`close`, () => {\n\t\t\tunsubFromOwnerKeys()\n\t\t\tunsubFromUsersInRooms()\n\t\t\tdestroyRoom({ store, socket, userKey })(roomKey)\n\t\t})\n\n\t\treturn room\n\t}\n}\n\nexport type ProvideEnterAndExitConfig = {\n\tstore: RootStore\n\tsocket: Socket\n\troomSocket: GuardedSocket<RoomSocketInterface<string>>\n\tuserKey: UserKey\n}\nexport function provideEnterAndExit({\n\tstore,\n\tsocket,\n\troomSocket,\n\tuserKey,\n}: ProvideEnterAndExitConfig): (roomKey: RoomKey) => void {\n\tconst enterRoom = (roomKey: RoomKey): void => {\n\t\tstore.logger.info(\n\t\t\t`📡`,\n\t\t\t`socket`,\n\t\t\tsocket.id ?? `[ID MISSING?!]`,\n\t\t\t`👤 ${userKey} enters ${roomKey}`,\n\t\t)\n\t\tconst childSocket = ROOMS.get(roomKey)\n\t\tif (!childSocket) {\n\t\t\tstore.logger.error(`❌`, `unknown`, roomKey, `no room found with this id`)\n\t\t\treturn\n\t\t}\n\t\tconst toUser = socket.emit.bind(socket)\n\t\tchildSocket.on(userKey, toUser)\n\n\t\tconst roomQueue: [string, ...Json.Array][] = []\n\t\tconst pushToRoomQueue = (payload: [string, ...Json.Array]): void => {\n\t\t\troomQueue.push(payload)\n\t\t}\n\t\tlet toRoom = pushToRoomQueue\n\t\tconst forward: AllEventsListener<EventsMap> = (...payload) => {\n\t\t\ttoRoom([userKey, ...payload])\n\t\t}\n\t\tsocket.onAny(forward)\n\n\t\tconst dcUserFromRoom = () => {\n\t\t\tstore.logger.info(\n\t\t\t\t`📡`,\n\t\t\t\t`socket`,\n\t\t\t\tsocket.id ?? `[ID MISSING?!]`,\n\t\t\t\t`👤 ${userKey} is has lost connection to ${roomKey}`,\n\t\t\t)\n\t\t\tsocket.offAny(forward)\n\t\t\tchildSocket.off(userKey, toUser)\n\t\t\ttoRoom([`user-leaves`, userKey])\n\t\t}\n\n\t\tconst exitRoom = () => {\n\t\t\tstore.logger.info(\n\t\t\t\t`📡`,\n\t\t\t\t`socket`,\n\t\t\t\tsocket.id ?? `[ID MISSING?!]`,\n\t\t\t\t`👤 ${userKey} leaves ${roomKey}`,\n\t\t\t)\n\t\t\tdcUserFromRoom()\n\t\t\teditRelationsInStore(store, usersInRooms, (relations) => {\n\t\t\t\trelations.delete({ room: roomKey, user: userKey })\n\t\t\t})\n\n\t\t\troomSocket.off(`leaveRoom`, exitRoom)\n\t\t\troomSocket.on(`joinRoom`, enterRoom)\n\t\t}\n\n\t\tconst userIsAlreadyInRoom = getFromStore(\n\t\t\tstore,\n\t\t\tgetInternalRelationsFromStore(store, usersInRooms),\n\t\t\troomKey,\n\t\t).has(userKey as any)\n\n\t\tif (!userIsAlreadyInRoom) {\n\t\t\teditRelationsInStore(store, usersInRooms, (relations) => {\n\t\t\t\trelations.set({ room: roomKey, user: userKey })\n\t\t\t})\n\t\t}\n\t\tchildSocket.emit(`user-joins`, userKey)\n\n\t\ttoRoom = (payload) => {\n\t\t\tchildSocket.emit(...payload)\n\t\t}\n\t\twhile (roomQueue.length > 0) {\n\t\t\tconst payload = roomQueue.shift()\n\t\t\tif (payload) toRoom(payload)\n\t\t}\n\n\t\tsocket.on(`disconnect`, dcUserFromRoom)\n\t\troomSocket.on(`leaveRoom`, exitRoom)\n\t\troomSocket.off(`joinRoom`, enterRoom)\n\t}\n\troomSocket.on(`joinRoom`, enterRoom)\n\treturn enterRoom\n}\n\nexport type DestroyRoomConfig = {\n\tstore: RootStore\n\tsocket: Socket\n\tuserKey: UserKey\n}\nexport function destroyRoom({\n\tstore,\n\tsocket,\n\tuserKey,\n}: DestroyRoomConfig): (roomKey: RoomKey) => void {\n\treturn (roomKey: RoomKey) => {\n\t\tstore.logger.info(\n\t\t\t`📡`,\n\t\t\t`socket`,\n\t\t\tsocket.id ?? `[ID MISSING?!]`,\n\t\t\t`👤 ${userKey} attempts to delete ${roomKey}`,\n\t\t)\n\t\tconst owner = getFromStore(\n\t\t\tstore,\n\t\t\tfindRelationsInStore(store, ownersOfRooms, roomKey).userKeyOfRoom,\n\t\t)\n\t\tif (owner === userKey) {\n\t\t\tstore.logger.info(\n\t\t\t\t`📡`,\n\t\t\t\t`socket`,\n\t\t\t\tsocket.id ?? `[ID MISSING?!]`,\n\t\t\t\t`👤 ${userKey} deletes ${roomKey}`,\n\t\t\t)\n\t\t\tsetIntoStore(store, roomKeysAtom, (s) => (s.delete(roomKey), s))\n\t\t\teditRelationsInStore(store, usersInRooms, (relations) => {\n\t\t\t\trelations.delete({ room: roomKey })\n\t\t\t})\n\t\t\tconst room = ROOMS.get(roomKey)\n\t\t\tif (room) {\n\t\t\t\troom.emit(`exit`)\n\t\t\t\tROOMS.delete(roomKey)\n\t\t\t}\n\t\t\treturn\n\t\t}\n\t\tstore.logger.info(\n\t\t\t`📡`,\n\t\t\t`socket`,\n\t\t\tsocket.id ?? `[ID MISSING?!]`,\n\t\t\t`👤 ${userKey} failed to delete ${roomKey}; its owner is ${owner}`,\n\t\t)\n\t}\n}\n\nexport type ProvideRoomsConfig<RoomNames extends string> = {\n\tresolveRoomScript: (path: RoomNames) => [string, string[]]\n\troomNames: RoomNames[]\n\troomTimeLimit?: number\n\tuserKey: UserKey\n\tstore: RootStore\n\tsocket: Socket\n}\nexport function provideRooms<RoomNames extends string>({\n\tstore = IMPLICIT.STORE,\n\tsocket,\n\tresolveRoomScript,\n\troomNames,\n\tuserKey,\n}: ProvideRoomsConfig<RoomNames>): () => void {\n\tconst roomSocket = guardSocket<RoomSocketInterface<RoomNames>>(\n\t\tsocket,\n\t\tcreateRoomSocketGuard(roomNames),\n\t)\n\tconst exposeMutable = realtimeMutableProvider({\n\t\tsocket,\n\t\tstore,\n\t\tconsumer: userKey,\n\t})\n\tconst unsubFromRoomKeys = exposeMutable(roomKeysAtom)\n\tconst usersInRoomsAtoms = getInternalRelationsFromStore(store, usersInRooms)\n\tconst [, usersInRoomsAtomsUsersOnly] = getInternalRelationsFromStore(\n\t\tstore,\n\t\tusersInRooms,\n\t\t`split`,\n\t)\n\tconst usersWhoseRoomsCanBeSeenSelector = findInStore(\n\t\tstore,\n\t\tvisibleUsersInRoomsSelector,\n\t\tuserKey,\n\t)\n\tconst ownersOfRoomsAtoms = getInternalRelationsFromStore(store, ownersOfRooms)\n\tconst exposeMutableFamily = realtimeMutableFamilyProvider({\n\t\tsocket,\n\t\tstore,\n\t\tconsumer: userKey,\n\t})\n\tconst unsubFromUsersInRooms = exposeMutableFamily(\n\t\tusersInRoomsAtoms,\n\t\tusersWhoseRoomsCanBeSeenSelector,\n\t)\n\tconst unsubFromOwnersOfRooms = exposeMutableFamily(\n\t\townersOfRoomsAtoms,\n\t\tusersWhoseRoomsCanBeSeenSelector,\n\t)\n\tconst enterRoom = provideEnterAndExit({\n\t\tstore,\n\t\tsocket,\n\t\troomSocket,\n\t\tuserKey,\n\t})\n\n\tconst userRoomSet = getFromStore(store, usersInRoomsAtomsUsersOnly, userKey)\n\tfor (const userRoomKey of userRoomSet) {\n\t\tenterRoom(userRoomKey)\n\t\tbreak\n\t}\n\troomSocket.on(\n\t\t`createRoom`,\n\t\tspawnRoom({ store, socket, userKey, resolveRoomScript }),\n\t)\n\troomSocket.on(`deleteRoom`, destroyRoom({ store, socket, userKey }))\n\treturn () => {\n\t\tunsubFromRoomKeys()\n\t\tunsubFromUsersInRooms()\n\t\tunsubFromOwnersOfRooms()\n\t}\n}\n\nconst roomKeySchema: StandardSchemaV1<Json.Array, [RoomKey]> = {\n\t\"~standard\": {\n\t\tversion: 1,\n\t\tvendor: `atom.io`,\n\t\tvalidate: ([maybeRoomKey]: Json.Array) => {\n\t\t\tif (typeof maybeRoomKey === `string`) {\n\t\t\t\tif (isRoomKey(maybeRoomKey)) {\n\t\t\t\t\treturn { value: [maybeRoomKey] }\n\t\t\t\t}\n\t\t\t\treturn {\n\t\t\t\t\tissues: [\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tmessage: `Room key must start with \"room::\"`,\n\t\t\t\t\t\t},\n\t\t\t\t\t],\n\t\t\t\t}\n\t\t\t}\n\t\t\treturn {\n\t\t\t\tissues: [\n\t\t\t\t\t{\n\t\t\t\t\t\tmessage: `Room key must be a string`,\n\t\t\t\t\t},\n\t\t\t\t],\n\t\t\t}\n\t\t},\n\t},\n}\n\nfunction createRoomSocketGuard<RoomNames extends string>(\n\troomNames: RoomNames[],\n): SocketGuard<RoomSocketInterface<RoomNames>> {\n\treturn {\n\t\tcreateRoom: {\n\t\t\t\"~standard\": {\n\t\t\t\tversion: 1,\n\t\t\t\tvendor: `atom.io`,\n\t\t\t\tvalidate: ([maybeRoomName]) => {\n\t\t\t\t\tif (roomNames.includes(maybeRoomName as RoomNames)) {\n\t\t\t\t\t\treturn { value: [maybeRoomName as RoomNames] }\n\t\t\t\t\t}\n\t\t\t\t\treturn {\n\t\t\t\t\t\tissues: [\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tmessage:\n\t\t\t\t\t\t\t\t\t`Room name must be one of the following:\\n - ` +\n\t\t\t\t\t\t\t\t\troomNames.join(`\\n - `),\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t],\n\t\t\t\t\t}\n\t\t\t\t},\n\t\t\t},\n\t\t},\n\t\tjoinRoom: roomKeySchema,\n\t\tdeleteRoom: roomKeySchema,\n\t\tleaveRoom: {\n\t\t\t\"~standard\": {\n\t\t\t\tversion: 1,\n\t\t\t\tvendor: `atom.io`,\n\t\t\t\tvalidate: () => ({ value: [] }),\n\t\t\t},\n\t\t},\n\t}\n}\n","import type * as AtomIO from \"atom.io\"\nimport {\n\tfindInStore,\n\tgetFromStore,\n\tIMPLICIT,\n\tsubscribeToState,\n} from \"atom.io/internal\"\nimport type { Canonical, Json, stringified } from \"atom.io/json\"\nimport { stringifyJson } from \"atom.io/json\"\nimport { employSocket } from \"atom.io/realtime\"\n\nimport type { ServerConfig } from \".\"\n\nexport type FamilyProvider = ReturnType<typeof realtimeAtomFamilyProvider>\nexport function realtimeAtomFamilyProvider({\n\tsocket,\n\tconsumer,\n\tstore = IMPLICIT.STORE,\n}: ServerConfig) {\n\treturn function familyProvider<\n\t\tJ extends Json.Serializable,\n\t\tK extends Canonical,\n\t>(\n\t\tfamily: AtomIO.RegularAtomFamilyToken<J, K>,\n\t\tindex: AtomIO.ReadableToken<Iterable<NoInfer<K>>> | Iterable<NoInfer<K>>,\n\t): () => void {\n\t\tconst [dynamicIndex, staticIndex]:\n\t\t\t| [AtomIO.ReadableToken<Iterable<NoInfer<K>>>, undefined]\n\t\t\t| [undefined, Iterable<NoInfer<K>>] = (() => {\n\t\t\tif (typeof index === `object` && `key` in index && `type` in index) {\n\t\t\t\treturn [index, undefined] as const\n\t\t\t}\n\t\t\treturn [undefined, index] as const\n\t\t})()\n\n\t\tconst coreSubscriptions = new Set<() => void>()\n\t\tconst clearCoreSubscriptions = () => {\n\t\t\tfor (const unsub of coreSubscriptions) unsub()\n\t\t\tcoreSubscriptions.clear()\n\t\t}\n\t\tconst familyMemberSubscriptionsWanted = new Set<stringified<K>>()\n\t\tconst familyMemberSubscriptions = new Map<string, () => void>()\n\t\tconst clearFamilySubscriptions = () => {\n\t\t\tfor (const unsub of familyMemberSubscriptions.values()) unsub()\n\t\t\tfamilyMemberSubscriptions.clear()\n\t\t}\n\n\t\tconst fillUnsubRequest = (key: string) => {\n\t\t\tconst unsubUnsub = familyMemberSubscriptions.get(`${key}:unsub`)\n\t\t\tif (unsubUnsub) {\n\t\t\t\tunsubUnsub()\n\t\t\t\tfamilyMemberSubscriptions.delete(`${key}:unsub`)\n\t\t\t}\n\t\t\tconst unsub = familyMemberSubscriptions.get(key)\n\t\t\tif (unsub) {\n\t\t\t\tunsub()\n\t\t\t\tfamilyMemberSubscriptions.delete(key)\n\t\t\t}\n\t\t}\n\n\t\tconst exposeFamilyMembers = (subKey: K) => {\n\t\t\tconst token = findInStore(store, family, subKey)\n\t\t\tgetFromStore(store, token)\n\t\t\tsocket.emit(`serve:${token.key}`, getFromStore(store, token))\n\t\t\tfamilyMemberSubscriptions.set(\n\t\t\t\ttoken.key,\n\t\t\t\tsubscribeToState(\n\t\t\t\t\tstore,\n\t\t\t\t\ttoken,\n\t\t\t\t\t`expose-family:${family.key}:${socket.id}`,\n\t\t\t\t\t({ newValue }) => {\n\t\t\t\t\t\tsocket.emit(`serve:${token.key}`, newValue)\n\t\t\t\t\t},\n\t\t\t\t),\n\t\t\t)\n\t\t\tfamilyMemberSubscriptions.set(\n\t\t\t\t`${token.key}:unsub`,\n\t\t\t\temploySocket(socket, `unsub:${token.key}`, () => {\n\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t`🙈`,\n\t\t\t\t\t\t`user`,\n\t\t\t\t\t\tconsumer,\n\t\t\t\t\t\t`unsubscribed from state \"${token.key}\"`,\n\t\t\t\t\t)\n\t\t\t\t\tfillUnsubRequest(token.key)\n\t\t\t\t}),\n\t\t\t)\n\t\t}\n\n\t\tconst isAvailable = (exposedSubKeys: Iterable<K>, subKey: K): boolean => {\n\t\t\tfor (const exposedSubKey of exposedSubKeys) {\n\t\t\t\tif (stringifyJson(exposedSubKey) === stringifyJson(subKey)) {\n\t\t\t\t\treturn true\n\t\t\t\t}\n\t\t\t}\n\t\t\treturn false\n\t\t}\n\n\t\tconst start = () => {\n\t\t\tstore.logger.info(\n\t\t\t\t`👀`,\n\t\t\t\t`user`,\n\t\t\t\tconsumer,\n\t\t\t\t`can subscribe to family \"${family.key}\"`,\n\t\t\t)\n\t\t\tcoreSubscriptions.add(\n\t\t\t\temploySocket(socket, `sub:${family.key}`, (subKey: K) => {\n\t\t\t\t\tlet exposedSubKeys: Iterable<K>\n\t\t\t\t\tif (dynamicIndex) {\n\t\t\t\t\t\texposedSubKeys = getFromStore(store, dynamicIndex)\n\t\t\t\t\t} else {\n\t\t\t\t\t\texposedSubKeys = staticIndex\n\t\t\t\t\t}\n\t\t\t\t\tconst shouldExpose = isAvailable(exposedSubKeys, subKey)\n\t\t\t\t\tif (shouldExpose) {\n\t\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t\t`👀`,\n\t\t\t\t\t\t\t`user`,\n\t\t\t\t\t\t\tconsumer,\n\t\t\t\t\t\t\t`is approved for a subscription to`,\n\t\t\t\t\t\t\tsubKey,\n\t\t\t\t\t\t\t`in family \"${family.key}\"`,\n\t\t\t\t\t\t)\n\t\t\t\t\t\texposeFamilyMembers(subKey)\n\t\t\t\t\t} else {\n\t\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t\t`❌`,\n\t\t\t\t\t\t\t`user`,\n\t\t\t\t\t\t\tconsumer,\n\t\t\t\t\t\t\t`is denied for a subscription to`,\n\t\t\t\t\t\t\tsubKey,\n\t\t\t\t\t\t\t`in family \"${family.key}\"`,\n\t\t\t\t\t\t)\n\t\t\t\t\t\tfamilyMemberSubscriptionsWanted.add(stringifyJson(subKey))\n\t\t\t\t\t\tsocket.emit(`unavailable:${family.key}`, subKey)\n\t\t\t\t\t}\n\t\t\t\t}),\n\t\t\t)\n\t\t\tif (dynamicIndex) {\n\t\t\t\tcoreSubscriptions.add(\n\t\t\t\t\tsubscribeToState(\n\t\t\t\t\t\tstore,\n\t\t\t\t\t\tdynamicIndex,\n\t\t\t\t\t\t`expose-family:${family.key}:${socket.id}`,\n\t\t\t\t\t\t({ newValue: newExposedSubKeys }) => {\n\t\t\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t\t\t`👀`,\n\t\t\t\t\t\t\t\t`user`,\n\t\t\t\t\t\t\t\tconsumer,\n\t\t\t\t\t\t\t\t`has the following keys available for family \"${family.key}\"`,\n\t\t\t\t\t\t\t\tnewExposedSubKeys,\n\t\t\t\t\t\t\t)\n\t\t\t\t\t\t\tfor (const subKey of newExposedSubKeys) {\n\t\t\t\t\t\t\t\tif (familyMemberSubscriptionsWanted.has(stringifyJson(subKey))) {\n\t\t\t\t\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t\t\t\t\t`👀`,\n\t\t\t\t\t\t\t\t\t\t`user`,\n\t\t\t\t\t\t\t\t\t\tconsumer,\n\t\t\t\t\t\t\t\t\t\t`is retroactively approved for a subscription to`,\n\t\t\t\t\t\t\t\t\t\tsubKey,\n\t\t\t\t\t\t\t\t\t\t`in family \"${family.key}\"`,\n\t\t\t\t\t\t\t\t\t)\n\t\t\t\t\t\t\t\t\texposeFamilyMembers(subKey)\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t},\n\t\t\t\t\t),\n\t\t\t\t)\n\t\t\t}\n\t\t}\n\n\t\tstart()\n\n\t\treturn () => {\n\t\t\tclearCoreSubscriptions()\n\t\t\tclearFamilySubscriptions()\n\t\t}\n\t}\n}\n","import type * as AtomIO from \"atom.io\"\nimport { getFromStore, IMPLICIT, subscribeToState } from \"atom.io/internal\"\nimport type { Json } from \"atom.io/json\"\nimport { employSocket } from \"atom.io/realtime\"\n\nimport type { ServerConfig } from \".\"\n\nfunction isReadableToken(input: unknown): input is AtomIO.ReadableToken<any> {\n\treturn (\n\t\ttypeof input === `object` &&\n\t\tinput !== null &&\n\t\t`key` in input &&\n\t\t`type` in input\n\t)\n}\n\nexport type StateProvider = ReturnType<typeof realtimeStateProvider>\nexport function realtimeStateProvider({\n\tsocket,\n\tconsumer,\n\tstore = IMPLICIT.STORE,\n}: ServerConfig) {\n\tstore.logger.info(`🔌`, `user`, consumer, `initialized state provider`)\n\treturn function stateProvider<C extends Json.Serializable, S extends C>(\n\t\tclientToken: AtomIO.WritableToken<C>,\n\t\tserverData:\n\t\t\t| AtomIO.ReadableToken<S>\n\t\t\t| S = clientToken as AtomIO.ReadableToken<S>,\n\t): () => void {\n\t\tconst isStatic = !isReadableToken(serverData)\n\n\t\tconst subscriptions = new Set<() => void>()\n\t\tconst clearSubscriptions = () => {\n\t\t\tfor (const unsub of subscriptions) unsub()\n\t\t\tsubscriptions.clear()\n\t\t}\n\n\t\tconst start = () => {\n\t\t\tif (isStatic) {\n\t\t\t\tstore.logger.info(\n\t\t\t\t\t`👀`,\n\t\t\t\t\t`user`,\n\t\t\t\t\tconsumer,\n\t\t\t\t\t`will be served`,\n\t\t\t\t\tserverData,\n\t\t\t\t\t`as \"${clientToken.key}\"`,\n\t\t\t\t)\n\t\t\t} else {\n\t\t\t\tstore.logger.info(\n\t\t\t\t\t`👀`,\n\t\t\t\t\t`user`,\n\t\t\t\t\tconsumer,\n\t\t\t\t\t`can subscribe to state \"${serverData.key}\" as \"${clientToken.key}\"`,\n\t\t\t\t)\n\t\t\t}\n\t\t\tsubscriptions.add(\n\t\t\t\temploySocket(socket, `sub:${clientToken.key}`, () => {\n\t\t\t\t\tif (isStatic) {\n\t\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t\t`👀`,\n\t\t\t\t\t\t\t`user`,\n\t\t\t\t\t\t\tconsumer,\n\t\t\t\t\t\t\t`requests`,\n\t\t\t\t\t\t\t`\"${clientToken.key}\"`,\n\t\t\t\t\t\t)\n\t\t\t\t\t\tsocket.emit(`serve:${clientToken.key}`, serverData)\n\t\t\t\t\t} else {\n\t\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t\t`👀`,\n\t\t\t\t\t\t\t`user`,\n\t\t\t\t\t\t\tconsumer,\n\t\t\t\t\t\t\t`subscribes to state \"${serverData.key}\"`,\n\t\t\t\t\t\t\tclientToken === serverData\n\t\t\t\t\t\t\t\t? `directly`\n\t\t\t\t\t\t\t\t: `as \"${clientToken.key}\"`,\n\t\t\t\t\t\t)\n\t\t\t\t\t\tclearSubscriptions()\n\t\t\t\t\t\tsocket.emit(\n\t\t\t\t\t\t\t`serve:${clientToken.key}`,\n\t\t\t\t\t\t\tgetFromStore(store, serverData),\n\t\t\t\t\t\t)\n\t\t\t\t\t\tsubscriptions.add(\n\t\t\t\t\t\t\tsubscribeToState(\n\t\t\t\t\t\t\t\tstore,\n\t\t\t\t\t\t\t\tserverData,\n\t\t\t\t\t\t\t\t`expose-single:${socket.id}`,\n\t\t\t\t\t\t\t\t({ newValue }) => {\n\t\t\t\t\t\t\t\t\tsocket.emit(`serve:${clientToken.key}`, newValue)\n\t\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t),\n\t\t\t\t\t\t)\n\t\t\t\t\t\tsubscriptions.add(\n\t\t\t\t\t\t\temploySocket(socket, `unsub:${serverData.key}`, () => {\n\t\t\t\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t\t\t\t`🙈`,\n\t\t\t\t\t\t\t\t\t`user`,\n\t\t\t\t\t\t\t\t\tconsumer,\n\t\t\t\t\t\t\t\t\t`unsubscribes from state \"${serverData.key}\", served`,\n\t\t\t\t\t\t\t\t\tclientToken === serverData\n\t\t\t\t\t\t\t\t\t\t? `directly`\n\t\t\t\t\t\t\t\t\t\t: `as \"${clientToken.key}\"`,\n\t\t\t\t\t\t\t\t)\n\t\t\t\t\t\t\t\tclearSubscriptions()\n\t\t\t\t\t\t\t\tstart()\n\t\t\t\t\t\t\t}),\n\t\t\t\t\t\t)\n\t\t\t\t\t}\n\t\t\t\t}),\n\t\t\t)\n\t\t}\n\n\t\tstart()\n\n\t\treturn clearSubscriptions\n\t}\n}\n","import type { WritableToken } from \"atom.io\"\nimport {\n\tfindInStore,\n\tgetFromStore,\n\tIMPLICIT,\n\toperateOnStore,\n\tOWN_OP,\n\tsetIntoStore,\n\tsubscribeToState,\n} from \"atom.io/internal\"\nimport type { Json } from \"atom.io/json\"\nimport type { SocketKey, StandardSchemaV1 } from \"atom.io/realtime\"\nimport { employSocket, mutexAtoms } from \"atom.io/realtime\"\n\nimport type { ServerConfig } from \".\"\n\nexport type StateReceiver = ReturnType<typeof realtimeStateReceiver>\nexport function realtimeStateReceiver({\n\tsocket,\n\tconsumer,\n\tstore = IMPLICIT.STORE,\n}: ServerConfig) {\n\treturn function stateReceiver<S extends Json.Serializable, C extends S>(\n\t\tschema: StandardSchemaV1<unknown, C>,\n\t\tclientToken: WritableToken<C>,\n\t\tserverToken: WritableToken<S> = clientToken,\n\t): () => void {\n\t\tconst socketKey = `socket::${socket.id}` satisfies SocketKey\n\t\tconst mutexAtom = findInStore(store, mutexAtoms, serverToken.key)\n\n\t\tconst subscriptions = new Set<() => void>()\n\t\tconst clearSubscriptions = () => {\n\t\t\tfor (const unsub of subscriptions) unsub()\n\t\t\tsubscriptions.clear()\n\t\t}\n\n\t\tconst permitPublish = () => {\n\t\t\tclearSubscriptions()\n\t\t\tsubscriptions.add(\n\t\t\t\temploySocket(socket, `pub:${clientToken.key}`, async (newValue) => {\n\t\t\t\t\tconst parsed = await schema[`~standard`].validate(newValue)\n\t\t\t\t\tif (parsed.issues) {\n\t\t\t\t\t\tstore.logger.error(\n\t\t\t\t\t\t\t`❌`,\n\t\t\t\t\t\t\t`user`,\n\t\t\t\t\t\t\tconsumer,\n\t\t\t\t\t\t\t`attempted to publish invalid value`,\n\t\t\t\t\t\t\tnewValue,\n\t\t\t\t\t\t\t`to state \"${serverToken.key}\"`,\n\t\t\t\t\t\t)\n\t\t\t\t\t\treturn\n\t\t\t\t\t}\n\t\t\t\t\tsetIntoStore(store, serverToken, parsed.value)\n\t\t\t\t}),\n\t\t\t)\n\t\t\tsubscriptions.add(\n\t\t\t\temploySocket(socket, `unclaim:${clientToken.key}`, () => {\n\t\t\t\t\tsetIntoStore(store, mutexAtom, false)\n\t\t\t\t\tclearSubscriptions()\n\t\t\t\t\tstart()\n\t\t\t\t}),\n\t\t\t)\n\t\t}\n\n\t\tconst start = () => {\n\t\t\tsubscriptions.add(\n\t\t\t\temploySocket(socket, `claim:${clientToken.key}`, () => {\n\t\t\t\t\tif (getFromStore(store, mutexAtom)) {\n\t\t\t\t\t\tclearSubscriptions()\n\t\t\t\t\t\tsubscriptions.add(\n\t\t\t\t\t\t\tsubscribeToState(store, mutexAtom, socketKey, () => {\n\t\t\t\t\t\t\t\tconst currentValue = getFromStore(store, mutexAtom)\n\t\t\t\t\t\t\t\tif (currentValue === false) {\n\t\t\t\t\t\t\t\t\toperateOnStore(OWN_OP, store, mutexAtom, true)\n\t\t\t\t\t\t\t\t\tpermitPublish()\n\t\t\t\t\t\t\t\t\tsocket.emit(`claim-result:${clientToken.key}`, true)\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t}),\n\t\t\t\t\t\t)\n\t\t\t\t\t\tsocket.emit(`claim-result:${clientToken.key}`, false)\n\t\t\t\t\t\treturn\n\t\t\t\t\t}\n\t\t\t\t\tsetIntoStore(store, mutexAtom, true)\n\t\t\t\t\tpermitPublish()\n\t\t\t\t\tsocket.emit(`claim-result:${clientToken.key}`, true)\n\t\t\t\t}),\n\t\t\t)\n\t\t}\n\n\t\tstart()\n\n\t\treturn clearSubscriptions\n\t}\n}\n","import type {\n\tHierarchy,\n\tJoinToken,\n\tMutableAtomToken,\n\tRegularAtomFamilyToken,\n} from \"atom.io\"\nimport { atomFamily, join, mutableAtom } from \"atom.io\"\nimport type { RoomKey, Socket, SocketKey, UserKey } from \"atom.io/realtime\"\nimport { isSocketKey, isUserKey } from \"atom.io/realtime\"\nimport { UList } from \"atom.io/transceivers/u-list\"\n\nexport type SocketSystemHierarchy = Hierarchy<\n\t[\n\t\t{\n\t\t\tabove: `root`\n\t\t\tbelow: [UserKey, SocketKey, RoomKey]\n\t\t},\n\t]\n>\n\nexport const socketAtoms: RegularAtomFamilyToken<Socket | null, SocketKey> =\n\tatomFamily({\n\t\tkey: `sockets`,\n\t\tdefault: null,\n\t})\n\nexport const socketKeysAtom: MutableAtomToken<UList<SocketKey>> = mutableAtom({\n\tkey: `socketKeys`,\n\tclass: UList,\n})\nexport const onlineUsersAtom: MutableAtomToken<UList<UserKey>> = mutableAtom({\n\tkey: `onlineUsers`,\n\tclass: UList,\n})\nexport const usersOfSockets: JoinToken<\n\t`user`,\n\tUserKey,\n\t`socket`,\n\tSocketKey,\n\t`1:1`\n> = join({\n\tkey: `usersOfSockets`,\n\tbetween: [`user`, `socket`],\n\tcardinality: `1:1`,\n\tisAType: isUserKey,\n\tisBType: isSocketKey,\n})\n","import type { IncomingHttpHeaders } from \"node:http\"\nimport type { ParsedUrlQuery } from \"node:querystring\"\n\nimport type { Loadable } from \"atom.io\"\nimport { Realm } from \"atom.io\"\nimport type { RootStore } from \"atom.io/internal\"\nimport {\n\teditRelationsInStore,\n\tfindInStore,\n\tfindRelationsInStore,\n\tgetFromStore,\n\tIMPLICIT,\n\tsetIntoStore,\n} from \"atom.io/internal\"\nimport type { RoomKey, Socket, SocketKey, UserKey } from \"atom.io/realtime\"\nimport { myUserKeyAtom } from \"atom.io/realtime-client\"\nimport type { Server } from \"socket.io\"\n\nimport { realtimeStateProvider } from \"./realtime-state-provider\"\nimport type { SocketSystemHierarchy } from \"./server-socket-state\"\nimport {\n\tonlineUsersAtom,\n\tsocketAtoms,\n\tsocketKeysAtom,\n\tusersOfSockets,\n} from \"./server-socket-state\"\n\nexport type ServerConfig = {\n\tsocket: Socket\n\tconsumer: RoomKey | UserKey\n\tstore?: RootStore\n}\nexport type UserServerConfig = {\n\tsocket: Socket\n\tconsumer: UserKey\n\tstore?: RootStore\n}\n\n/** Socket Handshake details--taken from socket.io */\nexport type Handshake = {\n\t/** The headers sent as part of the handshake */\n\theaders: IncomingHttpHeaders\n\t/** The date of creation (as string) */\n\ttime: string\n\t/** The ip of the client */\n\taddress: string\n\t/** Whether the connection is cross-domain */\n\txdomain: boolean\n\t/** Whether the connection is secure */\n\tsecure: boolean\n\t/** The date of creation (as unix timestamp) */\n\tissued: number\n\t/** The request URL string */\n\turl: string\n\t/** The query object */\n\tquery: ParsedUrlQuery\n\t/** The auth object */\n\tauth: {\n\t\t[key: string]: any\n\t}\n}\n\nexport function realtime(\n\tserver: Server,\n\tauth: (handshake: Handshake) => Loadable<Error | UserKey>,\n\tonConnect: (config: UserServerConfig) => Loadable<() => Loadable<void>>,\n\tstore: RootStore = IMPLICIT.STORE,\n): () => Promise<void> {\n\tconst socketRealm = new Realm<SocketSystemHierarchy>(store)\n\n\tserver\n\t\t.use(async (socket, next) => {\n\t\t\tconst result = await auth(socket.handshake)\n\t\t\tif (result instanceof Error) {\n\t\t\t\tstore.logger.error(\n\t\t\t\t\t`📡`,\n\t\t\t\t\t`socket`,\n\t\t\t\t\tsocket.id,\n\t\t\t\t\t`failed to authenticate`,\n\t\t\t\t\tresult.message,\n\t\t\t\t)\n\t\t\t\tnext(result)\n\t\t\t\treturn\n\t\t\t}\n\t\t\tconst userClaim = socketRealm.allocate(`root`, result)\n\t\t\tconst socketClaim = socketRealm.allocate(`root`, `socket::${socket.id}`)\n\t\t\tconst socketState = findInStore(store, socketAtoms, socketClaim)\n\t\t\tsetIntoStore(store, socketState, socket)\n\t\t\teditRelationsInStore(store, usersOfSockets, (relations) => {\n\t\t\t\trelations.set(userClaim, socketClaim)\n\t\t\t})\n\t\t\tsetIntoStore(store, onlineUsersAtom, (index) => index.add(userClaim))\n\t\t\tsetIntoStore(store, socketKeysAtom, (index) => index.add(socketClaim))\n\t\t\tnext()\n\t\t})\n\t\t.on(`connection`, async (socket) => {\n\t\t\tconst socketKey = `socket::${socket.id}` satisfies SocketKey\n\t\t\tconst userKeySelector = findRelationsInStore(\n\t\t\t\tstore,\n\t\t\t\tusersOfSockets,\n\t\t\t\tsocketKey,\n\t\t\t).userKeyOfSocket\n\t\t\tconst userKey = getFromStore(store, userKeySelector)!\n\t\t\tconst serverConfig: UserServerConfig = { store, socket, consumer: userKey }\n\t\t\tconst provideState = realtimeStateProvider(serverConfig)\n\t\t\tconst unsubFromMyUserKey = provideState(myUserKeyAtom, userKey)\n\n\t\t\tconst disposeServices = await onConnect(serverConfig)\n\n\t\t\tsocket.on(`disconnect`, async () => {\n\t\t\t\tstore.logger.info(`📡`, `socket`, socketKey, `👤 ${userKey} disconnects`)\n\t\t\t\tawait disposeServices()\n\t\t\t\tunsubFromMyUserKey()\n\t\t\t\teditRelationsInStore(store, usersOfSockets, (rel) =>\n\t\t\t\t\trel.delete(socketKey),\n\t\t\t\t)\n\t\t\t\tsetIntoStore(\n\t\t\t\t\tstore,\n\t\t\t\t\tonlineUsersAtom,\n\t\t\t\t\t(keys) => (keys.delete(userKey), keys),\n\t\t\t\t)\n\t\t\t\tsetIntoStore(\n\t\t\t\t\tstore,\n\t\t\t\t\tsocketKeysAtom,\n\t\t\t\t\t(keys) => (keys.delete(socketKey), keys),\n\t\t\t\t)\n\t\t\t})\n\t\t})\n\n\tconst disposeAll = async () => {\n\t\tawait server.close()\n\t}\n\treturn disposeAll\n}\n"],"mappings":"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"}
1
+ {"version":3,"file":"index.js","names":["redactorAtoms: RegularAtomFamilyToken<\n\t{\n\t\tocclude: (updates: TransactionSubEvent[]) => TransactionSubEvent[]\n\t},\n\tUserKey\n>","unacknowledgedUpdatesAtoms: RegularAtomFamilyToken<\n\tContinuitySyncTransactionUpdate[],\n\tUserKey\n>","atom","initialPayload: Json.Serializable[]","atom","cleanupRelay","exposedSubKeys: Iterable<K> | null","ROOMS: RoomMap","roomMeta: { count: number }","room","roomQueue: [string, ...Json.Array][]","forward: AllEventsListener<EventsMap>","roomKeySchema: StandardSchemaV1<Json.Array, [RoomKey]>","isAvailable","exposedSubKeys: Iterable<K> | null","socketAtoms: RegularAtomFamilyToken<Socket | null, SocketKey>","socketKeysAtom: MutableAtomToken<UList<SocketKey>>","onlineUsersAtom: MutableAtomToken<UList<UserKey>>","usersOfSockets: JoinToken<\n\t`user`,\n\tUserKey,\n\t`socket`,\n\tSocketKey,\n\t`1:1`\n>","serverConfig: UserServerConfig"],"sources":["../../src/realtime-server/continuity/continuity-store.ts","../../src/realtime-server/continuity/provide-outcomes.ts","../../src/realtime-server/continuity/provide-perspectives.ts","../../src/realtime-server/continuity/provide-startup-payloads.ts","../../src/realtime-server/continuity/receive-action-requests.ts","../../src/realtime-server/continuity/track-acknowledgements.ts","../../src/realtime-server/continuity/provide-continuity.ts","../../src/realtime-server/ipc-sockets/custom-socket.ts","../../src/realtime-server/ipc-sockets/parent-socket.ts","../../src/realtime-server/ipc-sockets/child-socket.ts","../../src/realtime-server/realtime-mutable-family-provider.ts","../../src/realtime-server/realtime-mutable-provider.ts","../../src/realtime-server/provide-rooms.ts","../../src/realtime-server/realtime-family-provider.ts","../../src/realtime-server/realtime-state-provider.ts","../../src/realtime-server/realtime-state-receiver.ts","../../src/realtime-server/server-socket-state.ts","../../src/realtime-server/server-config.ts"],"sourcesContent":["import type {\n\tRegularAtomFamilyToken,\n\tTransactionOutcomeEvent,\n\tTransactionSubEvent,\n} from \"atom.io\"\nimport { atomFamily } from \"atom.io\"\nimport type { UserKey } from \"atom.io/realtime\"\n\nexport function redactTransactionUpdateContent(\n\tvisibleStateKeys: string[],\n\tupdates: TransactionSubEvent[],\n): TransactionSubEvent[] {\n\treturn updates\n\t\t.map((update): TransactionSubEvent => {\n\t\t\tswitch (update.type) {\n\t\t\t\tcase `transaction_outcome`: {\n\t\t\t\t\tconst redacted = redactTransactionUpdateContent(\n\t\t\t\t\t\tvisibleStateKeys,\n\t\t\t\t\t\tupdate.subEvents,\n\t\t\t\t\t)\n\t\t\t\t\treturn { ...update, subEvents: redacted }\n\t\t\t\t}\n\t\t\t\tcase `atom_update`:\n\t\t\t\tcase `molecule_creation`:\n\t\t\t\tcase `molecule_disposal`:\n\t\t\t\tcase `molecule_transfer`:\n\t\t\t\tcase `state_creation`:\n\t\t\t\tcase `state_disposal`:\n\t\t\t\t\treturn update\n\t\t\t}\n\t\t})\n\t\t.filter((update) => {\n\t\t\tswitch (update.type) {\n\t\t\t\tcase `atom_update`:\n\t\t\t\tcase `state_creation`:\n\t\t\t\tcase `state_disposal`:\n\t\t\t\t\treturn visibleStateKeys.includes(update.token.key)\n\t\t\t\tcase `molecule_creation`:\n\t\t\t\tcase `transaction_outcome`:\n\t\t\t\tcase `molecule_disposal`:\n\t\t\t\tcase `molecule_transfer`:\n\t\t\t\t\treturn true\n\t\t\t}\n\t\t})\n}\n\nexport const redactorAtoms: RegularAtomFamilyToken<\n\t{\n\t\tocclude: (updates: TransactionSubEvent[]) => TransactionSubEvent[]\n\t},\n\tUserKey\n> = atomFamily({\n\tkey: `redactor`,\n\tdefault: { occlude: (updates) => updates },\n})\n\nexport type ContinuitySyncTransactionUpdate = Pick<\n\tTransactionOutcomeEvent<any>,\n\t`epoch` | `id` | `output` | `subEvents` | `token`\n>\nexport const unacknowledgedUpdatesAtoms: RegularAtomFamilyToken<\n\tContinuitySyncTransactionUpdate[],\n\tUserKey\n> = atomFamily({\n\tkey: `unacknowledgedUpdates`,\n\tdefault: () => [],\n})\n","import type { Store } from \"atom.io/internal\"\nimport {\n\tfindInStore,\n\tgetFromStore,\n\tgetUpdateToken,\n\tsetIntoStore,\n\tsubscribeToTransaction,\n} from \"atom.io/internal\"\nimport type { Json } from \"atom.io/json\"\nimport type { ContinuityToken, Socket, UserKey } from \"atom.io/realtime\"\n\nimport {\n\tredactTransactionUpdateContent,\n\tunacknowledgedUpdatesAtoms,\n} from \"./continuity-store\"\n\nexport function provideOutcomes(\n\tstore: Store,\n\tsocket: Socket,\n\tcontinuity: ContinuityToken,\n\tuserKey: UserKey,\n): () => void {\n\tconst continuityKey = continuity.key\n\tconst unsubscribeFunctions = new Set<() => void>()\n\n\tfor (const transaction of continuity.actions) {\n\t\tconst unsubscribeFromTransaction = subscribeToTransaction(\n\t\t\tstore,\n\t\t\ttransaction,\n\t\t\t`sync-continuity:${continuityKey}:${userKey}`,\n\t\t\t(outcomes) => {\n\t\t\t\ttry {\n\t\t\t\t\tconst visibleKeys = continuity.globals\n\t\t\t\t\t\t.map((atom) => {\n\t\t\t\t\t\t\tif (atom.type === `atom`) {\n\t\t\t\t\t\t\t\treturn atom.key\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\treturn getUpdateToken(atom).key\n\t\t\t\t\t\t})\n\t\t\t\t\t\t.concat(\n\t\t\t\t\t\t\tcontinuity.perspectives.flatMap((perspective) => {\n\t\t\t\t\t\t\t\tconst { viewAtoms } = perspective\n\t\t\t\t\t\t\t\tconst userPerspectiveTokenState = findInStore(\n\t\t\t\t\t\t\t\t\tstore,\n\t\t\t\t\t\t\t\t\tviewAtoms,\n\t\t\t\t\t\t\t\t\tuserKey,\n\t\t\t\t\t\t\t\t)\n\t\t\t\t\t\t\t\tconst visibleTokens = getFromStore(\n\t\t\t\t\t\t\t\t\tstore,\n\t\t\t\t\t\t\t\t\tuserPerspectiveTokenState,\n\t\t\t\t\t\t\t\t)\n\t\t\t\t\t\t\t\treturn visibleTokens.map((token) => {\n\t\t\t\t\t\t\t\t\tconst key =\n\t\t\t\t\t\t\t\t\t\ttoken.type === `mutable_atom` ? `*` + token.key : token.key\n\t\t\t\t\t\t\t\t\treturn key\n\t\t\t\t\t\t\t\t})\n\t\t\t\t\t\t\t}),\n\t\t\t\t\t\t)\n\t\t\t\t\tconst redactedUpdates = redactTransactionUpdateContent(\n\t\t\t\t\t\tvisibleKeys,\n\t\t\t\t\t\toutcomes.subEvents,\n\t\t\t\t\t)\n\t\t\t\t\tconst redactedUpdate = {\n\t\t\t\t\t\t...outcomes,\n\t\t\t\t\t\tupdates: redactedUpdates,\n\t\t\t\t\t}\n\t\t\t\t\tsetIntoStore(store, unacknowledgedUpdatesAtoms, userKey, (updates) => {\n\t\t\t\t\t\tif (redactedUpdate) {\n\t\t\t\t\t\t\tupdates.push(redactedUpdate)\n\t\t\t\t\t\t\tupdates.sort((a, b) => a.epoch - b.epoch)\n\t\t\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t\t\t`👍`,\n\t\t\t\t\t\t\t\t`continuity`,\n\t\t\t\t\t\t\t\tcontinuityKey,\n\t\t\t\t\t\t\t\t`${userKey} unacknowledged update queue now has`,\n\t\t\t\t\t\t\t\tupdates.length,\n\t\t\t\t\t\t\t\t`items`,\n\t\t\t\t\t\t\t)\n\t\t\t\t\t\t}\n\t\t\t\t\t\treturn updates\n\t\t\t\t\t})\n\n\t\t\t\t\tsocket.emit(\n\t\t\t\t\t\t`tx-new:${continuityKey}`,\n\t\t\t\t\t\tredactedUpdate as unknown as Json.Serializable,\n\t\t\t\t\t)\n\t\t\t\t} catch (thrown) {\n\t\t\t\t\tif (thrown instanceof Error) {\n\t\t\t\t\t\tstore.logger.error(\n\t\t\t\t\t\t\t`❌`,\n\t\t\t\t\t\t\t`continuity`,\n\t\t\t\t\t\t\tcontinuityKey,\n\t\t\t\t\t\t\t`${userKey} failed to send update from transaction ${transaction.key} to ${userKey}`,\n\t\t\t\t\t\t\tthrown.message,\n\t\t\t\t\t\t)\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t},\n\t\t)\n\t\tunsubscribeFunctions.add(unsubscribeFromTransaction)\n\t}\n\treturn () => {\n\t\tfor (const unsubscribe of unsubscribeFunctions) unsubscribe()\n\t}\n}\n","import type { Store } from \"atom.io/internal\"\nimport {\n\tfindInStore,\n\tgetFromStore,\n\tgetJsonToken,\n\tsubscribeToState,\n} from \"atom.io/internal\"\nimport type { ContinuityToken, Socket, UserKey } from \"atom.io/realtime\"\n\nexport function providePerspectives(\n\tstore: Store,\n\tsocket: Socket,\n\tcontinuity: ContinuityToken,\n\tuserKey: UserKey,\n): () => void {\n\tconst continuityKey = continuity.key\n\tconst unsubFns = new Set<() => void>()\n\tfor (const perspective of continuity.perspectives) {\n\t\tconst { viewAtoms } = perspective\n\t\tconst userViewState = findInStore(store, viewAtoms, userKey)\n\t\tconst unsubscribeFromUserView = subscribeToState(\n\t\t\tstore,\n\t\t\tuserViewState,\n\t\t\t`sync-continuity:${continuityKey}:${userKey}:perspective:${perspective.resourceAtoms.key}`,\n\t\t\t({ oldValue, newValue }) => {\n\t\t\t\tconst oldKeys = oldValue?.map((token) => token.key)\n\t\t\t\tconst newKeys = newValue.map((token) => token.key)\n\t\t\t\tconst concealed = oldValue?.filter(\n\t\t\t\t\t(token) => !newKeys.includes(token.key),\n\t\t\t\t)\n\t\t\t\tconst revealed = newValue\n\t\t\t\t\t.filter((token) => !oldKeys?.includes(token.key))\n\t\t\t\t\t.flatMap((token) => {\n\t\t\t\t\t\tconst resourceToken =\n\t\t\t\t\t\t\ttoken.type === `mutable_atom` ? getJsonToken(store, token) : token\n\t\t\t\t\t\tconst resource = getFromStore(store, resourceToken)\n\t\t\t\t\t\treturn [resourceToken, resource]\n\t\t\t\t\t})\n\t\t\t\tstore.logger.info(\n\t\t\t\t\t`👁`,\n\t\t\t\t\t`atom`,\n\t\t\t\t\tperspective.resourceAtoms.key,\n\t\t\t\t\t`${userKey} has a new perspective`,\n\t\t\t\t\t{ oldKeys, newKeys, revealed, concealed },\n\t\t\t\t)\n\t\t\t\tif (revealed.length > 0) {\n\t\t\t\t\tsocket.emit(`reveal:${continuityKey}`, revealed)\n\t\t\t\t}\n\t\t\t\tif (concealed && concealed.length > 0) {\n\t\t\t\t\tsocket.emit(`conceal:${continuityKey}`, concealed)\n\t\t\t\t}\n\t\t\t},\n\t\t)\n\t\tunsubFns.add(unsubscribeFromUserView)\n\t}\n\treturn () => {\n\t\tfor (const unsubscribe of unsubFns) unsubscribe()\n\t}\n}\n","import type { Store } from \"atom.io/internal\"\nimport {\n\tfindInStore,\n\tgetFromStore,\n\tgetJsonToken,\n\tisRootStore,\n} from \"atom.io/internal\"\nimport type { Json } from \"atom.io/json\"\nimport type { ContinuityToken, Socket, UserKey } from \"atom.io/realtime\"\nimport { employSocket } from \"atom.io/realtime\"\n\nexport function provideStartupPayloads(\n\tstore: Store,\n\tsocket: Socket,\n\tcontinuity: ContinuityToken,\n\tuserKey: UserKey,\n): () => void {\n\tconst continuityKey = continuity.key\n\tfunction sendInitialPayload(): void {\n\t\tconst initialPayload: Json.Serializable[] = []\n\t\tfor (const atom of continuity.globals) {\n\t\t\tconst resourceToken =\n\t\t\t\tatom.type === `mutable_atom` ? getJsonToken(store, atom) : atom\n\t\t\tconst resource = getFromStore(store, resourceToken)\n\t\t\tinitialPayload.push(resourceToken, resource)\n\t\t}\n\t\tfor (const perspective of continuity.perspectives) {\n\t\t\tconst { viewAtoms, resourceAtoms } = perspective\n\t\t\tconst userViewState = findInStore(store, viewAtoms, userKey)\n\t\t\tconst userView = getFromStore(store, userViewState)\n\t\t\tstore.logger.info(`👁`, `atom`, resourceAtoms.key, `${userKey} can see`, {\n\t\t\t\tviewAtoms,\n\t\t\t\tresourceAtoms,\n\t\t\t\tuserView,\n\t\t\t})\n\t\t\tfor (const visibleToken of userView) {\n\t\t\t\tconst resourceToken =\n\t\t\t\t\tvisibleToken.type === `mutable_atom`\n\t\t\t\t\t\t? getJsonToken(store, visibleToken)\n\t\t\t\t\t\t: visibleToken\n\t\t\t\tconst resource = getFromStore(store, resourceToken)\n\n\t\t\t\tinitialPayload.push(resourceToken, resource)\n\t\t\t}\n\t\t}\n\n\t\tconst epoch = isRootStore(store)\n\t\t\t? (store.transactionMeta.epoch.get(continuityKey) ?? null)\n\t\t\t: null\n\n\t\tsocket.emit(`continuity-init:${continuityKey}`, epoch, initialPayload)\n\t}\n\treturn employSocket(socket, `get:${continuityKey}`, sendInitialPayload)\n}\n","import type { TransactionOutcomeEvent, TransactionToken } from \"atom.io\"\nimport type { Store } from \"atom.io/internal\"\nimport { actUponStore } from \"atom.io/internal\"\nimport type { Json, JsonIO } from \"atom.io/json\"\nimport type { ContinuityToken, Socket } from \"atom.io/realtime\"\nimport { employSocket } from \"atom.io/realtime\"\n\nexport function receiveActionRequests(\n\tstore: Store,\n\tsocket: Socket,\n\tcontinuity: ContinuityToken,\n\tuserKey: string,\n): () => void {\n\tconst continuityKey = continuity.key\n\n\treturn employSocket(\n\t\tsocket,\n\t\t`tx-run:${continuityKey}`,\n\t\tfunction serveTransactionRequest(\n\t\t\ttxOutcome: Json.Serializable &\n\t\t\t\tPick<\n\t\t\t\t\tTransactionOutcomeEvent<TransactionToken<JsonIO>>,\n\t\t\t\t\t`id` | `params` | `token`\n\t\t\t\t>,\n\t\t) {\n\t\t\tstore.logger.info(`🛎️`, `continuity`, continuityKey, `received`, txOutcome)\n\t\t\tconst transactionKey = txOutcome.token.key\n\t\t\tconst updateId = txOutcome.id\n\t\t\tconst performanceKey = `tx-run:${transactionKey}:${updateId}`\n\t\t\tconst performanceKeyStart = `${performanceKey}:start`\n\t\t\tconst performanceKeyEnd = `${performanceKey}:end`\n\t\t\tperformance.mark(performanceKeyStart)\n\t\t\ttry {\n\t\t\t\tactUponStore(\n\t\t\t\t\tstore,\n\t\t\t\t\t{ type: `transaction`, key: transactionKey },\n\t\t\t\t\tupdateId,\n\t\t\t\t)(...txOutcome.params)\n\t\t\t} catch (thrown) {\n\t\t\t\tif (thrown instanceof Error) {\n\t\t\t\t\tstore.logger.error(\n\t\t\t\t\t\t`❌`,\n\t\t\t\t\t\t`continuity`,\n\t\t\t\t\t\tcontinuityKey,\n\t\t\t\t\t\t`failed to run transaction ${transactionKey} from ${userKey} with update ${updateId}`,\n\t\t\t\t\t\tthrown.message,\n\t\t\t\t\t)\n\t\t\t\t}\n\t\t\t}\n\t\t\tperformance.mark(performanceKeyEnd)\n\t\t\tconst metric = performance.measure(\n\t\t\t\tperformanceKey,\n\t\t\t\tperformanceKeyStart,\n\t\t\t\tperformanceKeyEnd,\n\t\t\t)\n\t\t\tstore.logger.info(\n\t\t\t\t`🚀`,\n\t\t\t\t`transaction`,\n\t\t\t\ttransactionKey,\n\t\t\t\tupdateId,\n\t\t\t\tuserKey,\n\t\t\t\tmetric.duration,\n\t\t\t)\n\t\t},\n\t)\n}\n","import type { Store } from \"atom.io/internal\"\nimport { getFromStore, setIntoStore } from \"atom.io/internal\"\nimport type { ContinuityToken, Socket, UserKey } from \"atom.io/realtime\"\nimport { employSocket } from \"atom.io/realtime\"\n\nimport { unacknowledgedUpdatesAtoms } from \"./continuity-store\"\n\nexport function trackAcknowledgements(\n\tstore: Store,\n\tsocket: Socket,\n\tcontinuity: ContinuityToken,\n\tuserKey: UserKey,\n): () => void {\n\tconst continuityKey = continuity.key\n\tconst userUnacknowledgedUpdates = getFromStore(\n\t\tstore,\n\t\tunacknowledgedUpdatesAtoms,\n\t\tuserKey,\n\t)\n\tfunction trackClientAcknowledgement(epoch: number): void {\n\t\tstore.logger.info(\n\t\t\t`👍`,\n\t\t\t`continuity`,\n\t\t\tcontinuityKey,\n\t\t\t`${userKey} acknowledged epoch ${epoch}`,\n\t\t)\n\t\tconst isUnacknowledged = userUnacknowledgedUpdates[0]?.epoch === epoch\n\t\tif (isUnacknowledged) {\n\t\t\tsetIntoStore(store, unacknowledgedUpdatesAtoms, userKey, (updates) => {\n\t\t\t\tupdates.shift()\n\t\t\t\tstore.logger.info(\n\t\t\t\t\t`👍`,\n\t\t\t\t\t`continuity`,\n\t\t\t\t\tcontinuityKey,\n\t\t\t\t\t`${userKey} unacknowledged update queue now has`,\n\t\t\t\t\tupdates.length,\n\t\t\t\t\t`items`,\n\t\t\t\t)\n\t\t\t\treturn updates\n\t\t\t})\n\t\t}\n\t}\n\treturn employSocket(socket, `ack:${continuityKey}`, trackClientAcknowledgement)\n}\n","import { getFromStore, IMPLICIT } from \"atom.io/internal\"\nimport type { Json } from \"atom.io/json\"\nimport type { ContinuityToken, UserKey } from \"atom.io/realtime\"\n\nimport type { ServerConfig } from \"..\"\nimport { unacknowledgedUpdatesAtoms } from \"./continuity-store\"\nimport { provideOutcomes } from \"./provide-outcomes\"\nimport { providePerspectives } from \"./provide-perspectives\"\nimport { provideStartupPayloads } from \"./provide-startup-payloads\"\nimport { receiveActionRequests } from \"./receive-action-requests\"\nimport { trackAcknowledgements } from \"./track-acknowledgements\"\n\nexport type ProvideContinuity = (\n\tcontinuity: ContinuityToken,\n\tuserKey: UserKey,\n) => () => void\nexport function prepareToProvideContinuity({\n\tsocket,\n\tstore = IMPLICIT.STORE,\n}: ServerConfig): ProvideContinuity {\n\treturn function syncRealtimeContinuity(continuity, userKey) {\n\t\tconst continuityKey = continuity.key\n\n\t\tconst unacknowledgedUpdates = getFromStore(\n\t\t\tstore,\n\t\t\tunacknowledgedUpdatesAtoms,\n\t\t\tuserKey,\n\t\t)\n\t\tfor (const unacknowledgedUpdate of unacknowledgedUpdates) {\n\t\t\tsocket.emit(\n\t\t\t\t`tx-new:${continuityKey}`,\n\t\t\t\tunacknowledgedUpdate as Json.Serializable,\n\t\t\t)\n\t\t}\n\n\t\tconst subscriptions = new Set<() => void>()\n\t\tconst clearSubscriptions = () => {\n\t\t\tfor (const unsubscribe of subscriptions) unsubscribe()\n\t\t\tsubscriptions.clear()\n\t\t}\n\n\t\tsubscriptions.add(providePerspectives(store, socket, continuity, userKey))\n\t\tsubscriptions.add(provideOutcomes(store, socket, continuity, userKey))\n\t\tsubscriptions.add(provideStartupPayloads(store, socket, continuity, userKey))\n\t\tsubscriptions.add(receiveActionRequests(store, socket, continuity, userKey))\n\t\tsubscriptions.add(trackAcknowledgements(store, socket, continuity, userKey))\n\n\t\treturn clearSubscriptions\n\t}\n}\n","import type { Json, stringified } from \"atom.io/json\"\nimport type { Socket } from \"atom.io/realtime\"\n\nexport type Events = Json.Object<string, Json.Serializable[]>\n\nexport type EventPayload<\n\treceiveRelay extends Events,\n\tK extends string & keyof receiveRelay = string & keyof receiveRelay,\n> = [string, ...receiveRelay[K]]\n\nexport interface EventBuffer<\n\tE extends Events,\n\tK extends string & keyof E = string & keyof E,\n> extends Buffer {\n\ttoString(): stringified<EventPayload<E, K>>\n}\n\nexport abstract class CustomSocket<I extends Events, O extends Events>\n\timplements Socket\n{\n\tprotected listeners: Map<keyof O, Set<(...args: Json.Array) => void>>\n\tprotected globalListeners: Set<(event: string, ...args: Json.Array) => void>\n\tprotected globalListenersOutgoing: Set<\n\t\t(event: string, ...args: Json.Array) => void\n\t>\n\tprotected handleEvent<K extends string & keyof I>(\n\t\t...args: EventPayload<I, K>\n\t): void {\n\t\tconst [event, ...rest] = args\n\t\tfor (const listener of this.globalListeners) {\n\t\t\tlistener(event, ...rest)\n\t\t}\n\t\tconst listeners = this.listeners.get(event)\n\t\tif (listeners) {\n\t\t\tfor (const listener of listeners) {\n\t\t\t\tlistener(...rest)\n\t\t\t}\n\t\t}\n\t}\n\n\tpublic id = `no_id_retrieved`\n\tpublic emit: <Event extends string & keyof I>(\n\t\tevent: Event,\n\t\t...args: I[Event]\n\t) => CustomSocket<I, O>\n\n\tpublic constructor(\n\t\temit: <Event extends keyof I>(\n\t\t\tevent: Event,\n\t\t\t...args: I[Event]\n\t\t) => CustomSocket<I, O>,\n\t) {\n\t\tthis.emit = (...args) => {\n\t\t\tfor (const listener of this.globalListenersOutgoing) {\n\t\t\t\tlistener(...args)\n\t\t\t}\n\t\t\treturn emit(...args)\n\t\t}\n\t\tthis.listeners = new Map()\n\t\tthis.globalListeners = new Set()\n\t\tthis.globalListenersOutgoing = new Set()\n\t}\n\n\tpublic on<Event extends keyof O>(\n\t\tevent: Event,\n\t\tlistener: (...args: O[Event]) => void,\n\t): this {\n\t\tconst listeners = this.listeners.get(event)\n\t\tif (listeners) {\n\t\t\tlisteners.add(listener)\n\t\t} else {\n\t\t\tthis.listeners.set(event, new Set([listener]))\n\t\t}\n\t\treturn this\n\t}\n\n\tpublic onAny(listener: (event: string, ...args: Json.Array) => void): this {\n\t\tthis.globalListeners.add(listener)\n\t\treturn this\n\t}\n\n\tpublic onAnyOutgoing(\n\t\tlistener: (event: string, ...args: Json.Array) => void,\n\t): this {\n\t\tthis.globalListenersOutgoing.add(listener)\n\t\treturn this\n\t}\n\n\tpublic off<Event extends keyof O>(\n\t\tevent: Event,\n\t\tlistener?: (...args: O[Event]) => void,\n\t): this {\n\t\tconst listeners = this.listeners.get(event)\n\t\tif (listeners) {\n\t\t\tif (listener) {\n\t\t\t\tlisteners.delete(listener)\n\t\t\t} else {\n\t\t\t\tthis.listeners.delete(event)\n\t\t\t}\n\t\t}\n\t\treturn this\n\t}\n\n\tpublic offAny(listener: (event: string, ...args: Json.Array) => void): this {\n\t\tthis.globalListeners.delete(listener)\n\t\treturn this\n\t}\n}\n","import type { Readable, Writable } from \"node:stream\"\n\nimport { Subject } from \"atom.io/internal\"\nimport type { Json } from \"atom.io/json\"\nimport { parseJson, stringifyJson } from \"atom.io/json\"\nimport type { UserKey } from \"atom.io/realtime\"\nimport { UList } from \"atom.io/transceivers/u-list\"\n\nimport type { StderrLog } from \"./child-socket\"\nimport type { EventBuffer, EventPayload, Events } from \"./custom-socket\"\nimport { CustomSocket } from \"./custom-socket\"\n\nexport const PROOF_OF_LIFE_SIGNAL = `ALIVE`\n\nexport class SubjectSocket<\n\tI extends Events,\n\tO extends Events,\n> extends CustomSocket<I, O> {\n\tpublic in: Subject<EventPayload<I>>\n\tpublic out: Subject<EventPayload<O>>\n\tpublic id = `no_id_retrieved`\n\tpublic disposalEffects: (() => void)[] = []\n\n\tpublic constructor(id: string) {\n\t\tsuper((...args) => {\n\t\t\tthis.out.next(args as any)\n\t\t\treturn this\n\t\t})\n\t\tthis.id = id\n\t\tthis.in = new Subject()\n\t\tthis.out = new Subject()\n\t\tthis.in.subscribe(`socket`, (event) => {\n\t\t\tthis.handleEvent(...event)\n\t\t})\n\t}\n\n\tpublic dispose(): void {\n\t\tfor (const dispose of this.disposalEffects) {\n\t\t\tdispose()\n\t\t}\n\t}\n}\n\nexport type ParentProcess = {\n\tpid?: number | undefined\n\tstdin: Readable\n\tstdout: Writable\n\tstderr: Writable\n\texit: (code?: number) => void\n}\n\nexport class ParentSocket<\n\tI extends Events & {\n\t\t[user: UserKey]: [string, ...Json.Array[]]\n\t},\n\tO extends Events & {\n\t\t[id in string as `user::${id}`]: [string, ...Json.Array[]]\n\t} & {\n\t\t/* eslint-disable quotes */\n\t\t\"user-joins\": [key: UserKey]\n\t\t\"user-leaves\": [key: UserKey]\n\t\t/* eslint-enable quotes */\n\t},\n\tP extends ParentProcess = ParentProcess,\n> extends CustomSocket<I, O> {\n\tprotected incompleteData = ``\n\tprotected unprocessedEvents: string[] = []\n\tprotected relays: Map<UserKey, SubjectSocket<any, any>>\n\tprotected initRelay: (\n\t\tsocket: SubjectSocket<any, any>,\n\t\tuserKey: UserKey,\n\t) => (() => void) | void\n\tpublic proc: P\n\n\tpublic id = `#####`\n\n\tprotected log(...args: StderrLog): void {\n\t\tthis.proc.stderr.write(\n\t\t\tstringifyJson(\n\t\t\t\targs.map((arg) =>\n\t\t\t\t\targ instanceof UList ? `{ ${arg.toJSON().join(` | `)} }` : arg,\n\t\t\t\t),\n\t\t\t) + `\\x03`,\n\t\t)\n\t}\n\tpublic logger = {\n\t\tinfo: (...args: Json.Array): void => {\n\t\t\tthis.log(`i`, ...args)\n\t\t},\n\t\twarn: (...args: Json.Array): void => {\n\t\t\tthis.log(`w`, ...args)\n\t\t},\n\t\terror: (...args: Json.Array): void => {\n\t\t\tthis.log(`e`, ...args)\n\t\t},\n\t}\n\n\tpublic constructor(proc: P) {\n\t\tsuper((event, ...args) => {\n\t\t\tconst stringifiedEvent = JSON.stringify([event, ...args])\n\t\t\tthis.proc.stdout.write(stringifiedEvent + `\\x03`)\n\t\t\treturn this\n\t\t})\n\t\tthis.proc = proc\n\t\tthis.proc.stdin.resume()\n\t\tthis.relays = new Map()\n\t\tthis.initRelay = () => {\n\t\t\tthis.logger.info(`🔗`, `nothing to relay`)\n\t\t}\n\n\t\tthis.proc.stdin.on(\n\t\t\t`data`,\n\t\t\t<K extends string & keyof I>(buffer: EventBuffer<I, K>) => {\n\t\t\t\tconst chunk = buffer.toString()\n\t\t\t\tconst pieces = chunk.split(`\\x03`)\n\t\t\t\tconst initialMaybeWellFormed = pieces[0]\n\t\t\t\tpieces[0] = this.incompleteData + initialMaybeWellFormed\n\t\t\t\tlet idx = 0\n\t\t\t\tfor (const piece of pieces) {\n\t\t\t\t\tif (piece === ``) {\n\t\t\t\t\t\tcontinue\n\t\t\t\t\t}\n\t\t\t\t\ttry {\n\t\t\t\t\t\tconst jsonPiece = parseJson(piece)\n\t\t\t\t\t\tthis.logger.info(`🎰`, `received`, jsonPiece)\n\t\t\t\t\t\tthis.handleEvent(...(jsonPiece as EventPayload<I>))\n\t\t\t\t\t\tthis.incompleteData = ``\n\t\t\t\t\t} catch (thrown0) {\n\t\t\t\t\t\tif (thrown0 instanceof Error) {\n\t\t\t\t\t\t\tthis.logger.error(\n\t\t\t\t\t\t\t\t[\n\t\t\t\t\t\t\t\t\t`received malformed data from parent process:`,\n\t\t\t\t\t\t\t\t\t``,\n\t\t\t\t\t\t\t\t\tpiece,\n\t\t\t\t\t\t\t\t\t``,\n\t\t\t\t\t\t\t\t\tthrown0.message,\n\t\t\t\t\t\t\t\t].join(`\\n❌\\t`),\n\t\t\t\t\t\t\t)\n\t\t\t\t\t\t}\n\t\t\t\t\t\ttry {\n\t\t\t\t\t\t\tif (idx === 0) {\n\t\t\t\t\t\t\t\tthis.incompleteData = piece\n\t\t\t\t\t\t\t\tconst maybeActualJsonPiece = parseJson(initialMaybeWellFormed)\n\t\t\t\t\t\t\t\tthis.logger.info(`🎰`, `received`, maybeActualJsonPiece)\n\t\t\t\t\t\t\t\tthis.handleEvent(...(maybeActualJsonPiece as EventPayload<I>))\n\t\t\t\t\t\t\t\tthis.incompleteData = ``\n\t\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\t\tthis.incompleteData += piece\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t} catch (thrown1) {\n\t\t\t\t\t\t\tif (thrown1 instanceof Error) {\n\t\t\t\t\t\t\t\tthis.logger.error(\n\t\t\t\t\t\t\t\t\t[\n\t\t\t\t\t\t\t\t\t\t`received malformed data from parent process:`,\n\t\t\t\t\t\t\t\t\t\t``,\n\t\t\t\t\t\t\t\t\t\tinitialMaybeWellFormed,\n\t\t\t\t\t\t\t\t\t\t``,\n\t\t\t\t\t\t\t\t\t\tthrown1.message,\n\t\t\t\t\t\t\t\t\t].join(`\\n❌\\t`),\n\t\t\t\t\t\t\t\t)\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t\t++idx\n\t\t\t\t}\n\t\t\t},\n\t\t)\n\n\t\tthis.on(`exit`, () => {\n\t\t\tthis.logger.info(`🔥`, this.id, `received \"exit\"`)\n\t\t\tthis.proc.exit(0)\n\t\t})\n\n\t\tif (this.proc.pid) {\n\t\t\tthis.id = this.proc.pid?.toString()\n\t\t}\n\n\t\tthis.on(`user-joins`, (userKey: UserKey) => {\n\t\t\tthis.logger.info(`👤`, userKey, `joined`)\n\t\t\tconst existingRelay = this.relays.get(userKey)\n\t\t\tif (existingRelay) {\n\t\t\t\tthis.logger.info(`🔗`, `reattaching relay services for`, userKey)\n\t\t\t\tconst cleanupRelay = this.initRelay(existingRelay, userKey)\n\t\t\t\tif (cleanupRelay) {\n\t\t\t\t\texistingRelay.disposalEffects.push(cleanupRelay)\n\t\t\t\t}\n\t\t\t\tthis.on(userKey, (...data) => {\n\t\t\t\t\trelay.in.next(data)\n\t\t\t\t})\n\t\t\t\texistingRelay.disposalEffects.push(\n\t\t\t\t\texistingRelay.out.subscribe(`socket`, (data) => {\n\t\t\t\t\t\tthis.emit(userKey, ...(data as any))\n\t\t\t\t\t}),\n\t\t\t\t)\n\t\t\t\treturn\n\t\t\t}\n\t\t\tconst relay = new SubjectSocket(userKey)\n\t\t\tthis.relays.set(userKey, relay)\n\t\t\tthis.logger.info(`🔗`, `attaching relay services for`, userKey)\n\t\t\tconst cleanupRelay = this.initRelay(relay, userKey)\n\t\t\tif (cleanupRelay) {\n\t\t\t\trelay.disposalEffects.push(cleanupRelay)\n\t\t\t}\n\t\t\tthis.on(userKey, (...data) => {\n\t\t\t\trelay.in.next(data)\n\t\t\t})\n\t\t\trelay.disposalEffects.push(\n\t\t\t\trelay.out.subscribe(`socket`, (data) => {\n\t\t\t\t\tthis.emit(userKey, ...(data as any))\n\t\t\t\t}),\n\t\t\t)\n\t\t})\n\n\t\tthis.on(`user-leaves`, (userKey: UserKey) => {\n\t\t\tconst relay = this.relays.get(userKey)\n\t\t\tthis.off(userKey)\n\t\t\tif (relay) {\n\t\t\t\trelay.dispose()\n\t\t\t\tthis.relays.delete(userKey)\n\t\t\t}\n\t\t})\n\n\t\tthis.proc.stdout.write(PROOF_OF_LIFE_SIGNAL)\n\t}\n\n\tpublic receiveRelay(\n\t\tattachServices: (\n\t\t\tsocket: SubjectSocket<any, any>,\n\t\t\tuserKey: UserKey,\n\t\t) => (() => void) | void,\n\t): void {\n\t\tthis.initRelay = attachServices\n\t\tthis.logger.info(`🔗`, `ready to relay`)\n\t}\n}\n","import type { Readable, Writable } from \"node:stream\"\n\nimport type { Json, stringified } from \"atom.io/json\"\nimport { parseJson } from \"atom.io/json\"\n\nimport type { EventBuffer, EventPayload, Events } from \"./custom-socket\"\nimport { CustomSocket } from \"./custom-socket\"\nimport { PROOF_OF_LIFE_SIGNAL } from \"./parent-socket\"\n\n/* eslint-disable no-console */\n\nexport type ChildProcess = {\n\tpid?: number | undefined\n\tstdin: Writable\n\tstdout: Readable\n\tstderr: Readable\n}\n\nexport type StderrLog = [`e` | `i` | `w`, ...Json.Array]\n\nexport class ChildSocket<\n\tI extends Events,\n\tO extends Events,\n\tP extends ChildProcess = ChildProcess,\n> extends CustomSocket<I, O> {\n\tprotected incompleteData = ``\n\tprotected unprocessedEvents: string[] = []\n\tprotected incompleteLog = ``\n\tprotected unprocessedLogs: string[] = []\n\n\tpublic id = `#####`\n\n\tpublic proc: P\n\tpublic key: string\n\tpublic logger: Pick<Console, `error` | `info` | `warn`>\n\n\tprotected handleLog(log: StderrLog): void {\n\t\tif (Array.isArray(log)) {\n\t\t\tconst [level, ...rest] = log\n\t\t\tswitch (level) {\n\t\t\t\tcase `i`:\n\t\t\t\t\tthis.logger.info(...rest)\n\t\t\t\t\tbreak\n\t\t\t\tcase `w`:\n\t\t\t\t\tthis.logger.warn(...rest)\n\t\t\t\t\tbreak\n\t\t\t\tcase `e`:\n\t\t\t\t\tthis.logger.error(...rest)\n\t\t\t\t\tbreak\n\t\t\t}\n\t\t}\n\t}\n\n\tpublic constructor(\n\t\tproc: P,\n\t\tkey: string,\n\t\tlogger?: Pick<Console, `error` | `info` | `warn`>,\n\t) {\n\t\tsuper((event, ...args) => {\n\t\t\tconst stringifiedEvent = JSON.stringify([event, ...args]) + `\\x03`\n\t\t\tthis.proc.stdin.write(stringifiedEvent)\n\t\t\treturn this\n\t\t})\n\n\t\tthis.proc = proc\n\t\tthis.key = key\n\t\tthis.logger = logger ?? {\n\t\t\tinfo: (...args: unknown[]) => {\n\t\t\t\tconsole.info(this.id, this.key, ...args)\n\t\t\t},\n\t\t\twarn: (...args: unknown[]) => {\n\t\t\t\tconsole.warn(this.id, this.key, ...args)\n\t\t\t},\n\t\t\terror: (...args: unknown[]) => {\n\t\t\t\tconsole.error(this.id, this.key, ...args)\n\t\t\t},\n\t\t}\n\t\tthis.proc.stdout.on(\n\t\t\t`data`,\n\t\t\t<K extends string & keyof I>(buffer: EventBuffer<I, K>) => {\n\t\t\t\tif (buffer[0] === 27 && buffer[1] === 91 && buffer[2] === 50) {\n\t\t\t\t\tthis.logger.info(`STDOUT TERMINAL CLEAR`, buffer)\n\t\t\t\t\treturn\n\t\t\t\t}\n\t\t\t\tconst chunk = buffer.toString()\n\n\t\t\t\tif (chunk === PROOF_OF_LIFE_SIGNAL) {\n\t\t\t\t\treturn\n\t\t\t\t}\n\n\t\t\t\tconst pieces = chunk.split(`\\x03`)\n\t\t\t\tconst initialMaybeWellFormed = pieces[0]\n\t\t\t\tpieces[0] = this.incompleteData + initialMaybeWellFormed\n\t\t\t\tlet idx = 0\n\t\t\t\tfor (const piece of pieces) {\n\t\t\t\t\tif (piece === ``) {\n\t\t\t\t\t\tcontinue\n\t\t\t\t\t}\n\t\t\t\t\ttry {\n\t\t\t\t\t\tconst jsonPiece = parseJson(piece as stringified<EventPayload<I, K>>)\n\t\t\t\t\t\tthis.logger.info(`💸`, `emitted`, jsonPiece)\n\t\t\t\t\t\tthis.handleEvent(...jsonPiece)\n\t\t\t\t\t\tthis.incompleteData = ``\n\t\t\t\t\t} catch (thrown0) {\n\t\t\t\t\t\tif (thrown0 instanceof Error) {\n\t\t\t\t\t\t\tconsole.error(\n\t\t\t\t\t\t\t\t[\n\t\t\t\t\t\t\t\t\t`❌ Malformed data received from child process:`,\n\t\t\t\t\t\t\t\t\t``,\n\t\t\t\t\t\t\t\t\t...piece.split(`\\n`),\n\t\t\t\t\t\t\t\t\t``,\n\t\t\t\t\t\t\t\t\tthrown0.message,\n\t\t\t\t\t\t\t\t].join(`\\n❌\\t`) + `\\n`,\n\t\t\t\t\t\t\t)\n\t\t\t\t\t\t}\n\t\t\t\t\t\ttry {\n\t\t\t\t\t\t\tif (idx === 0) {\n\t\t\t\t\t\t\t\tthis.incompleteData = piece\n\t\t\t\t\t\t\t\tconst maybeActualJsonPiece = parseJson(\n\t\t\t\t\t\t\t\t\tinitialMaybeWellFormed as stringified<EventPayload<I, K>>,\n\t\t\t\t\t\t\t\t)\n\t\t\t\t\t\t\t\tthis.logger.info(`💸`, `emitted`, maybeActualJsonPiece)\n\t\t\t\t\t\t\t\tthis.handleEvent(...maybeActualJsonPiece)\n\t\t\t\t\t\t\t\tthis.incompleteData = ``\n\t\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\t\tthis.incompleteData += piece\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t} catch (thrown1) {\n\t\t\t\t\t\t\tif (thrown1 instanceof Error) {\n\t\t\t\t\t\t\t\tconsole.error(\n\t\t\t\t\t\t\t\t\t[\n\t\t\t\t\t\t\t\t\t\t`❌ Malformed data received from child process:`,\n\t\t\t\t\t\t\t\t\t\t``,\n\t\t\t\t\t\t\t\t\t\t...initialMaybeWellFormed.split(`\\n`),\n\t\t\t\t\t\t\t\t\t\t``,\n\t\t\t\t\t\t\t\t\t\tthrown1.message,\n\t\t\t\t\t\t\t\t\t].join(`\\n❌\\t`) + `\\n`,\n\t\t\t\t\t\t\t\t)\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t\t++idx\n\t\t\t\t}\n\t\t\t},\n\t\t)\n\t\tthis.proc.stderr.on(`data`, (buffer: Buffer) => {\n\t\t\tif (buffer[0] === 27 && buffer[1] === 91 && buffer[2] === 50) {\n\t\t\t\tthis.logger.info(`STDERR TERMINAL CLEAR`, buffer)\n\t\t\t\treturn\n\t\t\t}\n\t\t\tconst chunk = buffer.toString()\n\n\t\t\tconst pieces = chunk.split(`\\x03`)\n\t\t\tconst initialMaybeWellFormed = pieces[0]\n\t\t\tpieces[0] = this.incompleteData + initialMaybeWellFormed\n\t\t\tlet idx = 0\n\t\t\tfor (const piece of pieces) {\n\t\t\t\tif (piece === ``) {\n\t\t\t\t\tcontinue\n\t\t\t\t}\n\t\t\t\ttry {\n\t\t\t\t\tconst jsonPiece = parseJson(piece as stringified<StderrLog>)\n\t\t\t\t\tthis.handleLog(jsonPiece)\n\t\t\t\t\tthis.incompleteData = ``\n\t\t\t\t} catch (thrown0) {\n\t\t\t\t\tif (thrown0 instanceof Error) {\n\t\t\t\t\t\tthis.logger.error(\n\t\t\t\t\t\t\t[\n\t\t\t\t\t\t\t\t`❌ Malformed log received from child process:`,\n\t\t\t\t\t\t\t\t``,\n\t\t\t\t\t\t\t\t...piece.split(`\\n`),\n\t\t\t\t\t\t\t\t``,\n\t\t\t\t\t\t\t\tthrown0.message,\n\t\t\t\t\t\t\t].join(`\\n❌\\t`) + `\\n`,\n\t\t\t\t\t\t)\n\t\t\t\t\t}\n\t\t\t\t\ttry {\n\t\t\t\t\t\tif (idx === 0) {\n\t\t\t\t\t\t\tthis.incompleteData = piece\n\t\t\t\t\t\t\tconst maybeActualJsonPiece = parseJson(\n\t\t\t\t\t\t\t\tinitialMaybeWellFormed as stringified<StderrLog>,\n\t\t\t\t\t\t\t)\n\t\t\t\t\t\t\tthis.handleLog(maybeActualJsonPiece)\n\t\t\t\t\t\t\tthis.incompleteData = ``\n\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\tthis.incompleteData += piece\n\t\t\t\t\t\t}\n\t\t\t\t\t} catch (thrown1) {\n\t\t\t\t\t\tif (thrown1 instanceof Error) {\n\t\t\t\t\t\t\tthis.logger.error(\n\t\t\t\t\t\t\t\t[\n\t\t\t\t\t\t\t\t\t`❌ Malformed log received from child process:`,\n\t\t\t\t\t\t\t\t\t``,\n\t\t\t\t\t\t\t\t\t...initialMaybeWellFormed.split(`\\n`),\n\t\t\t\t\t\t\t\t\t``,\n\t\t\t\t\t\t\t\t\tthrown1.message,\n\t\t\t\t\t\t\t\t].join(`\\n❌\\t`) + `\\n`,\n\t\t\t\t\t\t\t)\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t\t++idx\n\t\t\t}\n\t\t})\n\t\tthis.proc.stdin.once(`error`, (err: { code: string }) => {\n\t\t\tif (err.code === `EPIPE`) {\n\t\t\t\tconsole.error(`EPIPE error during write`, this.proc.stdin)\n\t\t\t}\n\t\t})\n\t\tif (proc.pid) {\n\t\t\tthis.id = proc.pid.toString()\n\t\t}\n\t}\n}\n","import type * as AtomIO from \"atom.io\"\nimport type { Transceiver } from \"atom.io/internal\"\nimport {\n\tfindInStore,\n\tgetFromStore,\n\tgetJsonToken,\n\tgetUpdateToken,\n\tIMPLICIT,\n\tsubscribeToState,\n} from \"atom.io/internal\"\nimport type { Canonical, stringified } from \"atom.io/json\"\nimport { stringifyJson } from \"atom.io/json\"\nimport { employSocket } from \"atom.io/realtime\"\n\nimport type { ServerConfig } from \".\"\n\nconst isAvailable = <K extends Canonical>(\n\texposedSubKeys: Iterable<K>,\n\tsubKey: K,\n): boolean => {\n\tfor (const exposedSubKey of exposedSubKeys) {\n\t\tif (stringifyJson(exposedSubKey) === stringifyJson(subKey)) {\n\t\t\treturn true\n\t\t}\n\t}\n\treturn false\n}\n\nexport type MutableFamilyProvider = ReturnType<\n\ttypeof realtimeMutableFamilyProvider\n>\nexport function realtimeMutableFamilyProvider({\n\tsocket,\n\tconsumer,\n\tstore = IMPLICIT.STORE,\n}: ServerConfig) {\n\treturn function mutableFamilyProvider<\n\t\tT extends Transceiver<any, any, any>,\n\t\tK extends Canonical,\n\t>(\n\t\tfamily: AtomIO.MutableAtomFamilyToken<T, K>,\n\t\tindex:\n\t\t\t| AtomIO.ReadableToken<Iterable<NoInfer<K>> | null>\n\t\t\t| Iterable<NoInfer<K>>,\n\t): () => void {\n\t\tconst [dynamicIndex, staticIndex]:\n\t\t\t| [AtomIO.ReadableToken<Iterable<NoInfer<K>> | null>, undefined]\n\t\t\t| [undefined, Iterable<NoInfer<K>>] = (() => {\n\t\t\tif (typeof index === `object` && `key` in index && `type` in index) {\n\t\t\t\treturn [index, undefined] as const\n\t\t\t}\n\t\t\treturn [undefined, index] as const\n\t\t})()\n\n\t\tconst coreSubscriptions = new Set<() => void>()\n\t\tconst clearCoreSubscriptions = () => {\n\t\t\tfor (const unsub of coreSubscriptions) unsub()\n\t\t\tcoreSubscriptions.clear()\n\t\t}\n\t\tconst familyMemberSubscriptionsWanted = new Set<stringified<K>>()\n\t\tconst familyMemberSubscriptions = new Map<string, () => void>()\n\t\tconst clearFamilySubscriptions = () => {\n\t\t\tfor (const unsub of familyMemberSubscriptions.values()) unsub()\n\t\t\tfamilyMemberSubscriptions.clear()\n\t\t}\n\n\t\tconst fillUnsubRequest = (key: string) => {\n\t\t\tconst unsubUnsub = familyMemberSubscriptions.get(`${key}:unsub`)\n\t\t\tif (unsubUnsub) {\n\t\t\t\tunsubUnsub()\n\t\t\t\tfamilyMemberSubscriptions.delete(`${key}:unsub`)\n\t\t\t}\n\t\t\tconst unsub = familyMemberSubscriptions.get(key)\n\t\t\tif (unsub) {\n\t\t\t\tunsub()\n\t\t\t\tfamilyMemberSubscriptions.delete(key)\n\t\t\t}\n\t\t}\n\n\t\tconst exposeFamilyMembers = (subKey: K) => {\n\t\t\tconst token = findInStore(store, family, subKey)\n\t\t\tgetFromStore(store, token)\n\t\t\tconst jsonToken = getJsonToken(store, token)\n\t\t\tconst updateToken = getUpdateToken(token)\n\t\t\tsocket.emit(`init:${token.key}`, getFromStore(store, jsonToken))\n\t\t\tfamilyMemberSubscriptions.set(\n\t\t\t\ttoken.key,\n\t\t\t\tsubscribeToState(\n\t\t\t\t\tstore,\n\t\t\t\t\tupdateToken,\n\t\t\t\t\t`expose-family:${family.key}:${socket.id}`,\n\t\t\t\t\t({ newValue }) => {\n\t\t\t\t\t\tsocket.emit(`next:${token.key}`, newValue)\n\t\t\t\t\t},\n\t\t\t\t),\n\t\t\t)\n\t\t\tfamilyMemberSubscriptions.set(\n\t\t\t\t`${token.key}:unsub`,\n\t\t\t\temploySocket(socket, `unsub:${token.key}`, () => {\n\t\t\t\t\tfillUnsubRequest(token.key)\n\t\t\t\t}),\n\t\t\t)\n\t\t}\n\n\t\tconst start = () => {\n\t\t\tstore.logger.info(\n\t\t\t\t`👀`,\n\t\t\t\t`user`,\n\t\t\t\tconsumer,\n\t\t\t\t`can subscribe to family \"${family.key}\"`,\n\t\t\t)\n\t\t\tcoreSubscriptions.add(\n\t\t\t\temploySocket(socket, `sub:${family.key}`, (subKey: K) => {\n\t\t\t\t\tlet exposedSubKeys: Iterable<K> | null\n\t\t\t\t\tif (dynamicIndex) {\n\t\t\t\t\t\texposedSubKeys = getFromStore(store, dynamicIndex)\n\t\t\t\t\t} else {\n\t\t\t\t\t\texposedSubKeys = staticIndex\n\t\t\t\t\t}\n\t\t\t\t\tconst shouldExpose =\n\t\t\t\t\t\texposedSubKeys && isAvailable(exposedSubKeys, subKey)\n\t\t\t\t\tif (shouldExpose) {\n\t\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t\t`👀`,\n\t\t\t\t\t\t\t`user`,\n\t\t\t\t\t\t\tconsumer,\n\t\t\t\t\t\t\t`is approved for a subscription to`,\n\t\t\t\t\t\t\tsubKey,\n\t\t\t\t\t\t\t`in family \"${family.key}\"`,\n\t\t\t\t\t\t)\n\t\t\t\t\t\texposeFamilyMembers(subKey)\n\t\t\t\t\t} else {\n\t\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t\t`❌`,\n\t\t\t\t\t\t\t`user`,\n\t\t\t\t\t\t\tconsumer,\n\t\t\t\t\t\t\t`is denied for a subscription to`,\n\t\t\t\t\t\t\tsubKey,\n\t\t\t\t\t\t\t`in family \"${family.key}\"`,\n\t\t\t\t\t\t)\n\t\t\t\t\t\tfamilyMemberSubscriptionsWanted.add(stringifyJson(subKey))\n\t\t\t\t\t\tsocket.emit(`unavailable:${family.key}`, subKey)\n\t\t\t\t\t}\n\t\t\t\t}),\n\t\t\t)\n\t\t\tif (dynamicIndex) {\n\t\t\t\tcoreSubscriptions.add(\n\t\t\t\t\tsubscribeToState(\n\t\t\t\t\t\tstore,\n\t\t\t\t\t\tdynamicIndex,\n\t\t\t\t\t\t`expose-family:${family.key}:${socket.id}`,\n\t\t\t\t\t\t({ newValue: newExposedSubKeys }) => {\n\t\t\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t\t\t`👀`,\n\t\t\t\t\t\t\t\t`user`,\n\t\t\t\t\t\t\t\tconsumer,\n\t\t\t\t\t\t\t\t`has the following keys available for family \"${family.key}\"`,\n\t\t\t\t\t\t\t\tnewExposedSubKeys,\n\t\t\t\t\t\t\t)\n\t\t\t\t\t\t\tif (newExposedSubKeys === null) return\n\t\t\t\t\t\t\tfor (const subKey of newExposedSubKeys) {\n\t\t\t\t\t\t\t\tif (familyMemberSubscriptionsWanted.has(stringifyJson(subKey))) {\n\t\t\t\t\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t\t\t\t\t`👀`,\n\t\t\t\t\t\t\t\t\t\t`user`,\n\t\t\t\t\t\t\t\t\t\tconsumer,\n\t\t\t\t\t\t\t\t\t\t`is retroactively approved for a subscription to`,\n\t\t\t\t\t\t\t\t\t\tsubKey,\n\t\t\t\t\t\t\t\t\t\t`in family \"${family.key}\"`,\n\t\t\t\t\t\t\t\t\t)\n\t\t\t\t\t\t\t\t\texposeFamilyMembers(subKey)\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t},\n\t\t\t\t\t),\n\t\t\t\t)\n\t\t\t}\n\t\t}\n\n\t\tstart()\n\n\t\treturn () => {\n\t\t\tclearCoreSubscriptions()\n\t\t\tclearFamilySubscriptions()\n\t\t}\n\t}\n}\n","import type * as AtomIO from \"atom.io\"\nimport type { Transceiver } from \"atom.io/internal\"\nimport {\n\tgetFromStore,\n\tgetJsonToken,\n\tgetUpdateToken,\n\tIMPLICIT,\n\tsubscribeToState,\n} from \"atom.io/internal\"\nimport type { Json } from \"atom.io/json\"\nimport { employSocket } from \"atom.io/realtime\"\n\nimport type { ServerConfig } from \".\"\n\nexport type MutableProvider = ReturnType<typeof realtimeMutableProvider>\nexport function realtimeMutableProvider({\n\tsocket,\n\tconsumer,\n\tstore = IMPLICIT.STORE,\n}: ServerConfig) {\n\treturn function mutableProvider<\n\t\tCore extends Transceiver<any, Json.Serializable, Json.Serializable>,\n\t>(token: AtomIO.MutableAtomToken<Core>): () => void {\n\t\tconst subscriptions = new Set<() => void>()\n\t\tconst clearSubscriptions = () => {\n\t\t\tfor (const unsub of subscriptions) unsub()\n\t\t\tsubscriptions.clear()\n\t\t}\n\n\t\tconst jsonToken = getJsonToken(store, token)\n\t\tconst trackerToken = getUpdateToken(token)\n\n\t\tconst start = () => {\n\t\t\tstore.logger.info(\n\t\t\t\t`👀`,\n\t\t\t\t`user`,\n\t\t\t\tconsumer,\n\t\t\t\t`can subscribe to state \"${token.key}\"`,\n\t\t\t)\n\t\t\tsubscriptions.add(\n\t\t\t\temploySocket(socket, `sub:${token.key}`, () => {\n\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t`👀`,\n\t\t\t\t\t\t`user`,\n\t\t\t\t\t\tconsumer,\n\t\t\t\t\t\t`subscribes to state \"${token.key}\"`,\n\t\t\t\t\t)\n\t\t\t\t\tclearSubscriptions()\n\t\t\t\t\tsocket.emit(`init:${token.key}`, getFromStore(store, jsonToken))\n\t\t\t\t\tsubscriptions.add(\n\t\t\t\t\t\tsubscribeToState(\n\t\t\t\t\t\t\tstore,\n\t\t\t\t\t\t\ttrackerToken,\n\t\t\t\t\t\t\t`expose-single:${socket.id}`,\n\t\t\t\t\t\t\t({ newValue }) => {\n\t\t\t\t\t\t\t\tsocket.emit(`next:${token.key}`, newValue)\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t),\n\t\t\t\t\t)\n\t\t\t\t\tsubscriptions.add(\n\t\t\t\t\t\temploySocket(socket, `unsub:${token.key}`, () => {\n\t\t\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t\t\t`🙈`,\n\t\t\t\t\t\t\t\t`user`,\n\t\t\t\t\t\t\t\tconsumer,\n\t\t\t\t\t\t\t\t`unsubscribes from state \"${token.key}\"`,\n\t\t\t\t\t\t\t)\n\t\t\t\t\t\t\tclearSubscriptions()\n\t\t\t\t\t\t\tstart()\n\t\t\t\t\t\t}),\n\t\t\t\t\t)\n\t\t\t\t}),\n\t\t\t)\n\t\t}\n\n\t\tstart()\n\n\t\treturn clearSubscriptions\n\t}\n}\n","import type { ChildProcessWithoutNullStreams } from \"node:child_process\"\nimport { spawn } from \"node:child_process\"\n\nimport type { ReadableFamilyToken } from \"atom.io\"\nimport type { RootStore } from \"atom.io/internal\"\nimport {\n\teditRelationsInStore,\n\tfindInStore,\n\tfindRelationsInStore,\n\tgetFromStore,\n\tgetInternalRelationsFromStore,\n\tIMPLICIT,\n\tsetIntoStore,\n} from \"atom.io/internal\"\nimport type { Json } from \"atom.io/json\"\nimport type {\n\tAllEventsListener,\n\tEventsMap,\n\tGuardedSocket,\n\tRoomKey,\n\tRoomSocketInterface,\n\tSocket,\n\tSocketGuard,\n\tStandardSchemaV1,\n\tUserKey,\n} from \"atom.io/realtime\"\nimport {\n\tguardSocket,\n\tisRoomKey,\n\townersOfRooms,\n\troomKeysAtom,\n\tusersInRooms,\n\tvisibilityFromRoomSelector,\n\tvisibleUsersInRoomsSelector,\n} from \"atom.io/realtime\"\n\nimport { ChildSocket, PROOF_OF_LIFE_SIGNAL } from \"./ipc-sockets\"\nimport { realtimeMutableFamilyProvider } from \"./realtime-mutable-family-provider\"\nimport { realtimeMutableProvider } from \"./realtime-mutable-provider\"\n\nexport type RoomMap = Map<\n\tstring,\n\tChildSocket<any, any, ChildProcessWithoutNullStreams>\n>\n\ndeclare global {\n\tvar ATOM_IO_REALTIME_SERVER_ROOMS: RoomMap\n}\nexport const ROOMS: RoomMap =\n\tglobalThis.ATOM_IO_REALTIME_SERVER_ROOMS ??\n\t(globalThis.ATOM_IO_REALTIME_SERVER_ROOMS = new Map())\n\nexport const roomMeta: { count: number } = { count: 0 }\n\nexport type SpawnRoomConfig<RoomNames extends string> = {\n\tstore: RootStore\n\tsocket: Socket\n\tuserKey: UserKey\n\tresolveRoomScript: (roomName: RoomNames) => [string, string[]]\n}\nexport function spawnRoom<RoomNames extends string>({\n\tstore,\n\tsocket,\n\tuserKey,\n\tresolveRoomScript,\n}: SpawnRoomConfig<RoomNames>): (\n\troomName: RoomNames,\n) => Promise<ChildSocket<any, any>> {\n\treturn async (roomName) => {\n\t\tstore.logger.info(\n\t\t\t`📡`,\n\t\t\t`socket`,\n\t\t\tsocket.id ?? `[ID MISSING?!]`,\n\t\t\t`👤 ${userKey} spawns room ${roomName}`,\n\t\t)\n\t\tconst roomKey = `room::${roomMeta.count++}-${roomName}` satisfies RoomKey\n\t\tconst [command, args] = resolveRoomScript(roomName)\n\t\tconst child = await new Promise<ChildProcessWithoutNullStreams>(\n\t\t\t(resolve) => {\n\t\t\t\tconst room = spawn(command, args, {\n\t\t\t\t\tenv: { ...process.env, REALTIME_ROOM_KEY: roomKey },\n\t\t\t\t})\n\t\t\t\tconst resolver = (data: Buffer) => {\n\t\t\t\t\tconst chunk = data.toString()\n\t\t\t\t\tif (chunk === PROOF_OF_LIFE_SIGNAL) {\n\t\t\t\t\t\troom.stdout.off(`data`, resolver)\n\t\t\t\t\t\tresolve(room)\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t\troom.stdout.on(`data`, resolver)\n\t\t\t},\n\t\t)\n\n\t\tconst room = new ChildSocket(child, roomKey)\n\t\tROOMS.set(roomKey, room)\n\t\tsetIntoStore(store, roomKeysAtom, (index) => (index.add(roomKey), index))\n\t\teditRelationsInStore(store, ownersOfRooms, (relations) => {\n\t\t\trelations.set({ room: roomKey, user: userKey })\n\t\t})\n\n\t\tconst provideMutableFamily = realtimeMutableFamilyProvider({\n\t\t\tsocket: room,\n\t\t\tconsumer: roomKey,\n\t\t\tstore,\n\t\t})\n\n\t\tconst ownersOfRoomsAtoms = getInternalRelationsFromStore(\n\t\t\tstore,\n\t\t\townersOfRooms,\n\t\t)\n\t\tconst unsubFromOwnerKeys = provideMutableFamily(ownersOfRoomsAtoms, [\n\t\t\troomKey,\n\t\t])\n\t\tconst usersInRoomsAtoms = getInternalRelationsFromStore(store, usersInRooms)\n\t\tconst unsubFromUsersInRooms = provideMutableFamily(\n\t\t\tusersInRoomsAtoms,\n\t\t\tfindInStore(store, visibilityFromRoomSelector, roomKey),\n\t\t)\n\n\t\troom.on(`close`, () => {\n\t\t\tunsubFromOwnerKeys()\n\t\t\tunsubFromUsersInRooms()\n\t\t\tdestroyRoom({ store, socket, userKey })(roomKey)\n\t\t})\n\n\t\treturn room\n\t}\n}\n\nexport type ProvideEnterAndExitConfig = {\n\tstore: RootStore\n\tsocket: Socket\n\troomSocket: GuardedSocket<RoomSocketInterface<string>>\n\tuserKey: UserKey\n}\nexport function provideEnterAndExit({\n\tstore,\n\tsocket,\n\troomSocket,\n\tuserKey,\n}: ProvideEnterAndExitConfig): (roomKey: RoomKey) => void {\n\tconst enterRoom = (roomKey: RoomKey): void => {\n\t\tstore.logger.info(\n\t\t\t`📡`,\n\t\t\t`socket`,\n\t\t\tsocket.id ?? `[ID MISSING?!]`,\n\t\t\t`👤 ${userKey} enters ${roomKey}`,\n\t\t)\n\t\tconst childSocket = ROOMS.get(roomKey)\n\t\tif (!childSocket) {\n\t\t\tstore.logger.error(`❌`, `unknown`, roomKey, `no room found with this id`)\n\t\t\treturn\n\t\t}\n\t\tconst toUser = socket.emit.bind(socket)\n\t\tchildSocket.on(userKey, toUser)\n\n\t\tconst roomQueue: [string, ...Json.Array][] = []\n\t\tconst pushToRoomQueue = (payload: [string, ...Json.Array]): void => {\n\t\t\troomQueue.push(payload)\n\t\t}\n\t\tlet toRoom = pushToRoomQueue\n\t\tconst forward: AllEventsListener<EventsMap> = (...payload) => {\n\t\t\ttoRoom([userKey, ...payload])\n\t\t}\n\t\tsocket.onAny(forward)\n\n\t\tconst dcUserFromRoom = () => {\n\t\t\tstore.logger.info(\n\t\t\t\t`📡`,\n\t\t\t\t`socket`,\n\t\t\t\tsocket.id ?? `[ID MISSING?!]`,\n\t\t\t\t`👤 ${userKey} is has lost connection to ${roomKey}`,\n\t\t\t)\n\t\t\tsocket.offAny(forward)\n\t\t\tchildSocket.off(userKey, toUser)\n\t\t\ttoRoom([`user-leaves`, userKey])\n\t\t}\n\n\t\tconst exitRoom = () => {\n\t\t\tstore.logger.info(\n\t\t\t\t`📡`,\n\t\t\t\t`socket`,\n\t\t\t\tsocket.id ?? `[ID MISSING?!]`,\n\t\t\t\t`👤 ${userKey} leaves ${roomKey}`,\n\t\t\t)\n\t\t\tdcUserFromRoom()\n\t\t\teditRelationsInStore(store, usersInRooms, (relations) => {\n\t\t\t\trelations.delete({ room: roomKey, user: userKey })\n\t\t\t})\n\n\t\t\troomSocket.off(`leaveRoom`, exitRoom)\n\t\t\troomSocket.on(`joinRoom`, enterRoom)\n\t\t}\n\n\t\tconst userIsAlreadyInRoom = getFromStore(\n\t\t\tstore,\n\t\t\tgetInternalRelationsFromStore(store, usersInRooms),\n\t\t\troomKey,\n\t\t).has(userKey as any)\n\n\t\tif (!userIsAlreadyInRoom) {\n\t\t\teditRelationsInStore(store, usersInRooms, (relations) => {\n\t\t\t\trelations.set({ room: roomKey, user: userKey })\n\t\t\t})\n\t\t}\n\t\tchildSocket.emit(`user-joins`, userKey)\n\n\t\ttoRoom = (payload) => {\n\t\t\tchildSocket.emit(...payload)\n\t\t}\n\t\twhile (roomQueue.length > 0) {\n\t\t\tconst payload = roomQueue.shift()\n\t\t\tif (payload) toRoom(payload)\n\t\t}\n\n\t\tsocket.on(`disconnect`, dcUserFromRoom)\n\t\troomSocket.on(`leaveRoom`, exitRoom)\n\t\troomSocket.off(`joinRoom`, enterRoom)\n\t}\n\troomSocket.on(`joinRoom`, enterRoom)\n\treturn enterRoom\n}\n\nexport type DestroyRoomConfig = {\n\tstore: RootStore\n\tsocket: Socket\n\tuserKey: UserKey\n}\nexport function destroyRoom({\n\tstore,\n\tsocket,\n\tuserKey,\n}: DestroyRoomConfig): (roomKey: RoomKey) => void {\n\treturn (roomKey: RoomKey) => {\n\t\tstore.logger.info(\n\t\t\t`📡`,\n\t\t\t`socket`,\n\t\t\tsocket.id ?? `[ID MISSING?!]`,\n\t\t\t`👤 ${userKey} attempts to delete ${roomKey}`,\n\t\t)\n\t\tconst owner = getFromStore(\n\t\t\tstore,\n\t\t\tfindRelationsInStore(store, ownersOfRooms, roomKey).userKeyOfRoom,\n\t\t)\n\t\tif (owner === userKey) {\n\t\t\tstore.logger.info(\n\t\t\t\t`📡`,\n\t\t\t\t`socket`,\n\t\t\t\tsocket.id ?? `[ID MISSING?!]`,\n\t\t\t\t`👤 ${userKey} deletes ${roomKey}`,\n\t\t\t)\n\t\t\tsetIntoStore(store, roomKeysAtom, (s) => (s.delete(roomKey), s))\n\t\t\teditRelationsInStore(store, usersInRooms, (relations) => {\n\t\t\t\trelations.delete({ room: roomKey })\n\t\t\t})\n\t\t\tconst room = ROOMS.get(roomKey)\n\t\t\tif (room) {\n\t\t\t\troom.emit(`exit`)\n\t\t\t\tROOMS.delete(roomKey)\n\t\t\t}\n\t\t\treturn\n\t\t}\n\t\tstore.logger.info(\n\t\t\t`📡`,\n\t\t\t`socket`,\n\t\t\tsocket.id ?? `[ID MISSING?!]`,\n\t\t\t`👤 ${userKey} failed to delete ${roomKey}; its owner is ${owner}`,\n\t\t)\n\t}\n}\n\nexport type ProvideRoomsConfig<RoomNames extends string> = {\n\tresolveRoomScript: (path: RoomNames) => [string, string[]]\n\troomAdminsToken: ReadableFamilyToken<boolean, UserKey>\n\troomNames: RoomNames[]\n\troomTimeLimit?: number\n\tuserKey: UserKey\n\tstore: RootStore\n\tsocket: Socket\n}\nexport function provideRooms<RoomNames extends string>({\n\tresolveRoomScript,\n\troomAdminsToken,\n\troomNames,\n\tsocket,\n\tstore = IMPLICIT.STORE,\n\tuserKey,\n}: ProvideRoomsConfig<RoomNames>): () => void {\n\tconst isAdmin = getFromStore(store, roomAdminsToken, userKey)\n\tconst roomSocket = guardSocket<RoomSocketInterface<RoomNames>>(\n\t\tsocket,\n\t\tcreateRoomSocketGuard(roomNames),\n\t)\n\tconst exposeMutable = realtimeMutableProvider({\n\t\tsocket,\n\t\tstore,\n\t\tconsumer: userKey,\n\t})\n\tconst unsubFromRoomKeys = exposeMutable(roomKeysAtom)\n\tconst usersInRoomsAtoms = getInternalRelationsFromStore(store, usersInRooms)\n\tconst [, usersInRoomsAtomsUsersOnly] = getInternalRelationsFromStore(\n\t\tstore,\n\t\tusersInRooms,\n\t\t`split`,\n\t)\n\tconst usersWhoseRoomsCanBeSeenSelector = findInStore(\n\t\tstore,\n\t\tvisibleUsersInRoomsSelector,\n\t\tuserKey,\n\t)\n\tconst ownersOfRoomsAtoms = getInternalRelationsFromStore(store, ownersOfRooms)\n\tconst exposeMutableFamily = realtimeMutableFamilyProvider({\n\t\tsocket,\n\t\tstore,\n\t\tconsumer: userKey,\n\t})\n\tconst unsubFromUsersInRooms = exposeMutableFamily(\n\t\tusersInRoomsAtoms,\n\t\tusersWhoseRoomsCanBeSeenSelector,\n\t)\n\tconst unsubFromOwnersOfRooms = exposeMutableFamily(\n\t\townersOfRoomsAtoms,\n\t\tusersWhoseRoomsCanBeSeenSelector,\n\t)\n\tconst enterRoom = provideEnterAndExit({\n\t\tstore,\n\t\tsocket,\n\t\troomSocket,\n\t\tuserKey,\n\t})\n\n\tconst userRoomSet = getFromStore(store, usersInRoomsAtomsUsersOnly, userKey)\n\tfor (const userRoomKey of userRoomSet) {\n\t\tenterRoom(userRoomKey)\n\t\tbreak\n\t}\n\tif (isAdmin) {\n\t\troomSocket.on(\n\t\t\t`createRoom`,\n\t\t\tspawnRoom({ store, socket, userKey, resolveRoomScript }),\n\t\t)\n\t\troomSocket.on(`deleteRoom`, destroyRoom({ store, socket, userKey }))\n\t}\n\treturn () => {\n\t\tunsubFromRoomKeys()\n\t\tunsubFromUsersInRooms()\n\t\tunsubFromOwnersOfRooms()\n\t}\n}\n\nconst roomKeySchema: StandardSchemaV1<Json.Array, [RoomKey]> = {\n\t\"~standard\": {\n\t\tversion: 1,\n\t\tvendor: `atom.io`,\n\t\tvalidate: ([maybeRoomKey]: Json.Array) => {\n\t\t\tif (typeof maybeRoomKey === `string`) {\n\t\t\t\tif (isRoomKey(maybeRoomKey)) {\n\t\t\t\t\treturn { value: [maybeRoomKey] }\n\t\t\t\t}\n\t\t\t\treturn {\n\t\t\t\t\tissues: [\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tmessage: `Room key must start with \"room::\"`,\n\t\t\t\t\t\t},\n\t\t\t\t\t],\n\t\t\t\t}\n\t\t\t}\n\t\t\treturn {\n\t\t\t\tissues: [\n\t\t\t\t\t{\n\t\t\t\t\t\tmessage: `Room key must be a string`,\n\t\t\t\t\t},\n\t\t\t\t],\n\t\t\t}\n\t\t},\n\t},\n}\n\nfunction createRoomSocketGuard<RoomNames extends string>(\n\troomNames: RoomNames[],\n): SocketGuard<RoomSocketInterface<RoomNames>> {\n\treturn {\n\t\tcreateRoom: {\n\t\t\t\"~standard\": {\n\t\t\t\tversion: 1,\n\t\t\t\tvendor: `atom.io`,\n\t\t\t\tvalidate: ([maybeRoomName]) => {\n\t\t\t\t\tif (roomNames.includes(maybeRoomName as RoomNames)) {\n\t\t\t\t\t\treturn { value: [maybeRoomName as RoomNames] }\n\t\t\t\t\t}\n\t\t\t\t\treturn {\n\t\t\t\t\t\tissues: [\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tmessage:\n\t\t\t\t\t\t\t\t\t`Room name must be one of the following:\\n - ` +\n\t\t\t\t\t\t\t\t\troomNames.join(`\\n - `),\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t],\n\t\t\t\t\t}\n\t\t\t\t},\n\t\t\t},\n\t\t},\n\t\tjoinRoom: roomKeySchema,\n\t\tdeleteRoom: roomKeySchema,\n\t\tleaveRoom: {\n\t\t\t\"~standard\": {\n\t\t\t\tversion: 1,\n\t\t\t\tvendor: `atom.io`,\n\t\t\t\tvalidate: () => ({ value: [] }),\n\t\t\t},\n\t\t},\n\t}\n}\n","import type * as AtomIO from \"atom.io\"\nimport {\n\tfindInStore,\n\tgetFromStore,\n\tIMPLICIT,\n\tsubscribeToState,\n} from \"atom.io/internal\"\nimport type { Canonical, Json, stringified } from \"atom.io/json\"\nimport { stringifyJson } from \"atom.io/json\"\nimport { employSocket } from \"atom.io/realtime\"\n\nimport type { ServerConfig } from \".\"\n\nexport type FamilyProvider = ReturnType<typeof realtimeAtomFamilyProvider>\nexport function realtimeAtomFamilyProvider({\n\tsocket,\n\tconsumer,\n\tstore = IMPLICIT.STORE,\n}: ServerConfig) {\n\treturn function familyProvider<\n\t\tJ extends Json.Serializable,\n\t\tK extends Canonical,\n\t>(\n\t\tfamily: AtomIO.RegularAtomFamilyToken<J, K>,\n\t\tindex:\n\t\t\t| AtomIO.ReadableToken<Iterable<NoInfer<K>> | null>\n\t\t\t| Iterable<NoInfer<K>>,\n\t): () => void {\n\t\tconst [dynamicIndex, staticIndex]:\n\t\t\t| [AtomIO.ReadableToken<Iterable<NoInfer<K>> | null>, undefined]\n\t\t\t| [undefined, Iterable<NoInfer<K>>] = (() => {\n\t\t\tif (typeof index === `object` && `key` in index && `type` in index) {\n\t\t\t\treturn [index, undefined] as const\n\t\t\t}\n\t\t\treturn [undefined, index] as const\n\t\t})()\n\n\t\tconst coreSubscriptions = new Set<() => void>()\n\t\tconst clearCoreSubscriptions = () => {\n\t\t\tfor (const unsub of coreSubscriptions) unsub()\n\t\t\tcoreSubscriptions.clear()\n\t\t}\n\t\tconst familyMemberSubscriptionsWanted = new Set<stringified<K>>()\n\t\tconst familyMemberSubscriptions = new Map<string, () => void>()\n\t\tconst clearFamilySubscriptions = () => {\n\t\t\tfor (const unsub of familyMemberSubscriptions.values()) unsub()\n\t\t\tfamilyMemberSubscriptions.clear()\n\t\t}\n\n\t\tconst fillUnsubRequest = (key: string) => {\n\t\t\tconst unsubUnsub = familyMemberSubscriptions.get(`${key}:unsub`)\n\t\t\tif (unsubUnsub) {\n\t\t\t\tunsubUnsub()\n\t\t\t\tfamilyMemberSubscriptions.delete(`${key}:unsub`)\n\t\t\t}\n\t\t\tconst unsub = familyMemberSubscriptions.get(key)\n\t\t\tif (unsub) {\n\t\t\t\tunsub()\n\t\t\t\tfamilyMemberSubscriptions.delete(key)\n\t\t\t}\n\t\t}\n\n\t\tconst exposeFamilyMembers = (subKey: K) => {\n\t\t\tconst token = findInStore(store, family, subKey)\n\t\t\tgetFromStore(store, token)\n\t\t\tsocket.emit(`serve:${token.key}`, getFromStore(store, token))\n\t\t\tfamilyMemberSubscriptions.set(\n\t\t\t\ttoken.key,\n\t\t\t\tsubscribeToState(\n\t\t\t\t\tstore,\n\t\t\t\t\ttoken,\n\t\t\t\t\t`expose-family:${family.key}:${socket.id}`,\n\t\t\t\t\t({ newValue }) => {\n\t\t\t\t\t\tsocket.emit(`serve:${token.key}`, newValue)\n\t\t\t\t\t},\n\t\t\t\t),\n\t\t\t)\n\t\t\tfamilyMemberSubscriptions.set(\n\t\t\t\t`${token.key}:unsub`,\n\t\t\t\temploySocket(socket, `unsub:${token.key}`, () => {\n\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t`🙈`,\n\t\t\t\t\t\t`user`,\n\t\t\t\t\t\tconsumer,\n\t\t\t\t\t\t`unsubscribed from state \"${token.key}\"`,\n\t\t\t\t\t)\n\t\t\t\t\tfillUnsubRequest(token.key)\n\t\t\t\t}),\n\t\t\t)\n\t\t}\n\n\t\tconst isAvailable = (exposedSubKeys: Iterable<K>, subKey: K): boolean => {\n\t\t\tfor (const exposedSubKey of exposedSubKeys) {\n\t\t\t\tif (stringifyJson(exposedSubKey) === stringifyJson(subKey)) {\n\t\t\t\t\treturn true\n\t\t\t\t}\n\t\t\t}\n\t\t\treturn false\n\t\t}\n\n\t\tconst start = () => {\n\t\t\tstore.logger.info(\n\t\t\t\t`👀`,\n\t\t\t\t`user`,\n\t\t\t\tconsumer,\n\t\t\t\t`can subscribe to family \"${family.key}\"`,\n\t\t\t)\n\t\t\tcoreSubscriptions.add(\n\t\t\t\temploySocket(socket, `sub:${family.key}`, (subKey: K) => {\n\t\t\t\t\tlet exposedSubKeys: Iterable<K> | null\n\t\t\t\t\tif (dynamicIndex) {\n\t\t\t\t\t\texposedSubKeys = getFromStore(store, dynamicIndex)\n\t\t\t\t\t} else {\n\t\t\t\t\t\texposedSubKeys = staticIndex\n\t\t\t\t\t}\n\t\t\t\t\tconst shouldExpose =\n\t\t\t\t\t\texposedSubKeys && isAvailable(exposedSubKeys, subKey)\n\t\t\t\t\tif (shouldExpose) {\n\t\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t\t`👀`,\n\t\t\t\t\t\t\t`user`,\n\t\t\t\t\t\t\tconsumer,\n\t\t\t\t\t\t\t`is approved for a subscription to`,\n\t\t\t\t\t\t\tsubKey,\n\t\t\t\t\t\t\t`in family \"${family.key}\"`,\n\t\t\t\t\t\t)\n\t\t\t\t\t\texposeFamilyMembers(subKey)\n\t\t\t\t\t} else {\n\t\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t\t`❌`,\n\t\t\t\t\t\t\t`user`,\n\t\t\t\t\t\t\tconsumer,\n\t\t\t\t\t\t\t`is denied for a subscription to`,\n\t\t\t\t\t\t\tsubKey,\n\t\t\t\t\t\t\t`in family \"${family.key}\"`,\n\t\t\t\t\t\t)\n\t\t\t\t\t\tfamilyMemberSubscriptionsWanted.add(stringifyJson(subKey))\n\t\t\t\t\t\tsocket.emit(`unavailable:${family.key}`, subKey)\n\t\t\t\t\t}\n\t\t\t\t}),\n\t\t\t)\n\t\t\tif (dynamicIndex) {\n\t\t\t\tcoreSubscriptions.add(\n\t\t\t\t\tsubscribeToState(\n\t\t\t\t\t\tstore,\n\t\t\t\t\t\tdynamicIndex,\n\t\t\t\t\t\t`expose-family:${family.key}:${socket.id}`,\n\t\t\t\t\t\t({ newValue: newExposedSubKeys }) => {\n\t\t\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t\t\t`👀`,\n\t\t\t\t\t\t\t\t`user`,\n\t\t\t\t\t\t\t\tconsumer,\n\t\t\t\t\t\t\t\t`has the following keys available for family \"${family.key}\"`,\n\t\t\t\t\t\t\t\tnewExposedSubKeys,\n\t\t\t\t\t\t\t)\n\t\t\t\t\t\t\tif (newExposedSubKeys === null) return\n\t\t\t\t\t\t\tfor (const subKey of newExposedSubKeys) {\n\t\t\t\t\t\t\t\tif (familyMemberSubscriptionsWanted.has(stringifyJson(subKey))) {\n\t\t\t\t\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t\t\t\t\t`👀`,\n\t\t\t\t\t\t\t\t\t\t`user`,\n\t\t\t\t\t\t\t\t\t\tconsumer,\n\t\t\t\t\t\t\t\t\t\t`is retroactively approved for a subscription to`,\n\t\t\t\t\t\t\t\t\t\tsubKey,\n\t\t\t\t\t\t\t\t\t\t`in family \"${family.key}\"`,\n\t\t\t\t\t\t\t\t\t)\n\t\t\t\t\t\t\t\t\texposeFamilyMembers(subKey)\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t},\n\t\t\t\t\t),\n\t\t\t\t)\n\t\t\t}\n\t\t}\n\n\t\tstart()\n\n\t\treturn () => {\n\t\t\tclearCoreSubscriptions()\n\t\t\tclearFamilySubscriptions()\n\t\t}\n\t}\n}\n","import type * as AtomIO from \"atom.io\"\nimport { getFromStore, IMPLICIT, subscribeToState } from \"atom.io/internal\"\nimport type { Json } from \"atom.io/json\"\nimport { employSocket } from \"atom.io/realtime\"\n\nimport type { ServerConfig } from \".\"\n\nfunction isReadableToken(input: unknown): input is AtomIO.ReadableToken<any> {\n\treturn (\n\t\ttypeof input === `object` &&\n\t\tinput !== null &&\n\t\t`key` in input &&\n\t\t`type` in input\n\t)\n}\n\nexport type StateProvider = ReturnType<typeof realtimeStateProvider>\nexport function realtimeStateProvider({\n\tsocket,\n\tconsumer,\n\tstore = IMPLICIT.STORE,\n}: ServerConfig) {\n\tstore.logger.info(`🔌`, `user`, consumer, `initialized state provider`)\n\treturn function stateProvider<C extends Json.Serializable, S extends C>(\n\t\tclientToken: AtomIO.WritableToken<C>,\n\t\tserverData:\n\t\t\t| AtomIO.ReadableToken<S>\n\t\t\t| S = clientToken as AtomIO.ReadableToken<S>,\n\t): () => void {\n\t\tconst isStatic = !isReadableToken(serverData)\n\n\t\tconst subscriptions = new Set<() => void>()\n\t\tconst clearSubscriptions = () => {\n\t\t\tfor (const unsub of subscriptions) unsub()\n\t\t\tsubscriptions.clear()\n\t\t}\n\n\t\tconst start = () => {\n\t\t\tif (isStatic) {\n\t\t\t\tstore.logger.info(\n\t\t\t\t\t`👀`,\n\t\t\t\t\t`user`,\n\t\t\t\t\tconsumer,\n\t\t\t\t\t`will be served`,\n\t\t\t\t\tserverData,\n\t\t\t\t\t`as \"${clientToken.key}\"`,\n\t\t\t\t)\n\t\t\t} else {\n\t\t\t\tstore.logger.info(\n\t\t\t\t\t`👀`,\n\t\t\t\t\t`user`,\n\t\t\t\t\tconsumer,\n\t\t\t\t\t`can subscribe to state \"${serverData.key}\" as \"${clientToken.key}\"`,\n\t\t\t\t)\n\t\t\t}\n\t\t\tsubscriptions.add(\n\t\t\t\temploySocket(socket, `sub:${clientToken.key}`, () => {\n\t\t\t\t\tif (isStatic) {\n\t\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t\t`👀`,\n\t\t\t\t\t\t\t`user`,\n\t\t\t\t\t\t\tconsumer,\n\t\t\t\t\t\t\t`requests`,\n\t\t\t\t\t\t\t`\"${clientToken.key}\"`,\n\t\t\t\t\t\t)\n\t\t\t\t\t\tsocket.emit(`serve:${clientToken.key}`, serverData)\n\t\t\t\t\t} else {\n\t\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t\t`👀`,\n\t\t\t\t\t\t\t`user`,\n\t\t\t\t\t\t\tconsumer,\n\t\t\t\t\t\t\t`subscribes to state \"${serverData.key}\"`,\n\t\t\t\t\t\t\tclientToken === serverData\n\t\t\t\t\t\t\t\t? `directly`\n\t\t\t\t\t\t\t\t: `as \"${clientToken.key}\"`,\n\t\t\t\t\t\t)\n\t\t\t\t\t\tclearSubscriptions()\n\t\t\t\t\t\tsocket.emit(\n\t\t\t\t\t\t\t`serve:${clientToken.key}`,\n\t\t\t\t\t\t\tgetFromStore(store, serverData),\n\t\t\t\t\t\t)\n\t\t\t\t\t\tsubscriptions.add(\n\t\t\t\t\t\t\tsubscribeToState(\n\t\t\t\t\t\t\t\tstore,\n\t\t\t\t\t\t\t\tserverData,\n\t\t\t\t\t\t\t\t`expose-single:${socket.id}`,\n\t\t\t\t\t\t\t\t({ newValue }) => {\n\t\t\t\t\t\t\t\t\tsocket.emit(`serve:${clientToken.key}`, newValue)\n\t\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t),\n\t\t\t\t\t\t)\n\t\t\t\t\t\tsubscriptions.add(\n\t\t\t\t\t\t\temploySocket(socket, `unsub:${serverData.key}`, () => {\n\t\t\t\t\t\t\t\tstore.logger.info(\n\t\t\t\t\t\t\t\t\t`🙈`,\n\t\t\t\t\t\t\t\t\t`user`,\n\t\t\t\t\t\t\t\t\tconsumer,\n\t\t\t\t\t\t\t\t\t`unsubscribes from state \"${serverData.key}\", served`,\n\t\t\t\t\t\t\t\t\tclientToken === serverData\n\t\t\t\t\t\t\t\t\t\t? `directly`\n\t\t\t\t\t\t\t\t\t\t: `as \"${clientToken.key}\"`,\n\t\t\t\t\t\t\t\t)\n\t\t\t\t\t\t\t\tclearSubscriptions()\n\t\t\t\t\t\t\t\tstart()\n\t\t\t\t\t\t\t}),\n\t\t\t\t\t\t)\n\t\t\t\t\t}\n\t\t\t\t}),\n\t\t\t)\n\t\t}\n\n\t\tstart()\n\n\t\treturn clearSubscriptions\n\t}\n}\n","import type { WritableToken } from \"atom.io\"\nimport {\n\tfindInStore,\n\tgetFromStore,\n\tIMPLICIT,\n\toperateOnStore,\n\tOWN_OP,\n\tsetIntoStore,\n\tsubscribeToState,\n} from \"atom.io/internal\"\nimport type { Json } from \"atom.io/json\"\nimport type { SocketKey, StandardSchemaV1 } from \"atom.io/realtime\"\nimport { employSocket, mutexAtoms } from \"atom.io/realtime\"\n\nimport type { ServerConfig } from \".\"\n\nexport type StateReceiver = ReturnType<typeof realtimeStateReceiver>\nexport function realtimeStateReceiver({\n\tsocket,\n\tconsumer,\n\tstore = IMPLICIT.STORE,\n}: ServerConfig) {\n\treturn function stateReceiver<S extends Json.Serializable, C extends S>(\n\t\tschema: StandardSchemaV1<unknown, C>,\n\t\tclientToken: WritableToken<C>,\n\t\tserverToken: WritableToken<S> = clientToken,\n\t): () => void {\n\t\tconst socketKey = `socket::${socket.id}` satisfies SocketKey\n\t\tconst mutexAtom = findInStore(store, mutexAtoms, serverToken.key)\n\n\t\tconst subscriptions = new Set<() => void>()\n\t\tconst clearSubscriptions = () => {\n\t\t\tfor (const unsub of subscriptions) unsub()\n\t\t\tsubscriptions.clear()\n\t\t}\n\n\t\tconst permitPublish = () => {\n\t\t\tclearSubscriptions()\n\t\t\tsubscriptions.add(\n\t\t\t\temploySocket(socket, `pub:${clientToken.key}`, async (newValue) => {\n\t\t\t\t\tconst parsed = await schema[`~standard`].validate(newValue)\n\t\t\t\t\tif (parsed.issues) {\n\t\t\t\t\t\tstore.logger.error(\n\t\t\t\t\t\t\t`❌`,\n\t\t\t\t\t\t\t`user`,\n\t\t\t\t\t\t\tconsumer,\n\t\t\t\t\t\t\t`attempted to publish invalid value`,\n\t\t\t\t\t\t\tnewValue,\n\t\t\t\t\t\t\t`to state \"${serverToken.key}\"`,\n\t\t\t\t\t\t)\n\t\t\t\t\t\treturn\n\t\t\t\t\t}\n\t\t\t\t\tsetIntoStore(store, serverToken, parsed.value)\n\t\t\t\t}),\n\t\t\t)\n\t\t\tsubscriptions.add(\n\t\t\t\temploySocket(socket, `unclaim:${clientToken.key}`, () => {\n\t\t\t\t\tsetIntoStore(store, mutexAtom, false)\n\t\t\t\t\tclearSubscriptions()\n\t\t\t\t\tstart()\n\t\t\t\t}),\n\t\t\t)\n\t\t}\n\n\t\tconst start = () => {\n\t\t\tsubscriptions.add(\n\t\t\t\temploySocket(socket, `claim:${clientToken.key}`, () => {\n\t\t\t\t\tif (getFromStore(store, mutexAtom)) {\n\t\t\t\t\t\tclearSubscriptions()\n\t\t\t\t\t\tsubscriptions.add(\n\t\t\t\t\t\t\tsubscribeToState(store, mutexAtom, socketKey, () => {\n\t\t\t\t\t\t\t\tconst currentValue = getFromStore(store, mutexAtom)\n\t\t\t\t\t\t\t\tif (currentValue === false) {\n\t\t\t\t\t\t\t\t\toperateOnStore(OWN_OP, store, mutexAtom, true)\n\t\t\t\t\t\t\t\t\tpermitPublish()\n\t\t\t\t\t\t\t\t\tsocket.emit(`claim-result:${clientToken.key}`, true)\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t}),\n\t\t\t\t\t\t)\n\t\t\t\t\t\tsocket.emit(`claim-result:${clientToken.key}`, false)\n\t\t\t\t\t\treturn\n\t\t\t\t\t}\n\t\t\t\t\tsetIntoStore(store, mutexAtom, true)\n\t\t\t\t\tpermitPublish()\n\t\t\t\t\tsocket.emit(`claim-result:${clientToken.key}`, true)\n\t\t\t\t}),\n\t\t\t)\n\t\t}\n\n\t\tstart()\n\n\t\treturn clearSubscriptions\n\t}\n}\n","import type {\n\tHierarchy,\n\tJoinToken,\n\tMutableAtomToken,\n\tRegularAtomFamilyToken,\n} from \"atom.io\"\nimport { atomFamily, join, mutableAtom } from \"atom.io\"\nimport type { RoomKey, Socket, SocketKey, UserKey } from \"atom.io/realtime\"\nimport { isSocketKey, isUserKey } from \"atom.io/realtime\"\nimport { UList } from \"atom.io/transceivers/u-list\"\n\nexport type SocketSystemHierarchy = Hierarchy<\n\t[\n\t\t{\n\t\t\tabove: `root`\n\t\t\tbelow: [UserKey, SocketKey, RoomKey]\n\t\t},\n\t]\n>\n\nexport const socketAtoms: RegularAtomFamilyToken<Socket | null, SocketKey> =\n\tatomFamily({\n\t\tkey: `sockets`,\n\t\tdefault: null,\n\t})\n\nexport const socketKeysAtom: MutableAtomToken<UList<SocketKey>> = mutableAtom({\n\tkey: `socketKeys`,\n\tclass: UList,\n})\nexport const onlineUsersAtom: MutableAtomToken<UList<UserKey>> = mutableAtom({\n\tkey: `onlineUsers`,\n\tclass: UList,\n})\nexport const usersOfSockets: JoinToken<\n\t`user`,\n\tUserKey,\n\t`socket`,\n\tSocketKey,\n\t`1:1`\n> = join({\n\tkey: `usersOfSockets`,\n\tbetween: [`user`, `socket`],\n\tcardinality: `1:1`,\n\tisAType: isUserKey,\n\tisBType: isSocketKey,\n})\n","import type { IncomingHttpHeaders } from \"node:http\"\nimport type { ParsedUrlQuery } from \"node:querystring\"\n\nimport type { Loadable } from \"atom.io\"\nimport { Realm } from \"atom.io\"\nimport type { RootStore } from \"atom.io/internal\"\nimport {\n\teditRelationsInStore,\n\tfindInStore,\n\tfindRelationsInStore,\n\tgetFromStore,\n\tIMPLICIT,\n\tsetIntoStore,\n} from \"atom.io/internal\"\nimport type { RoomKey, Socket, SocketKey, UserKey } from \"atom.io/realtime\"\nimport { myUserKeyAtom } from \"atom.io/realtime-client\"\nimport type { Server } from \"socket.io\"\n\nimport { realtimeStateProvider } from \"./realtime-state-provider\"\nimport type { SocketSystemHierarchy } from \"./server-socket-state\"\nimport {\n\tonlineUsersAtom,\n\tsocketAtoms,\n\tsocketKeysAtom,\n\tusersOfSockets,\n} from \"./server-socket-state\"\n\nexport type ServerConfig = {\n\tsocket: Socket\n\tconsumer: RoomKey | UserKey\n\tstore?: RootStore\n}\nexport type UserServerConfig = {\n\tsocket: Socket\n\tconsumer: UserKey\n\tstore?: RootStore\n}\n\n/** Socket Handshake details--taken from socket.io */\nexport type Handshake = {\n\t/** The headers sent as part of the handshake */\n\theaders: IncomingHttpHeaders\n\t/** The date of creation (as string) */\n\ttime: string\n\t/** The ip of the client */\n\taddress: string\n\t/** Whether the connection is cross-domain */\n\txdomain: boolean\n\t/** Whether the connection is secure */\n\tsecure: boolean\n\t/** The date of creation (as unix timestamp) */\n\tissued: number\n\t/** The request URL string */\n\turl: string\n\t/** The query object */\n\tquery: ParsedUrlQuery\n\t/** The auth object */\n\tauth: {\n\t\t[key: string]: any\n\t}\n}\n\nexport function realtime(\n\tserver: Server,\n\tauth: (handshake: Handshake) => Loadable<Error | UserKey>,\n\tonConnect: (config: UserServerConfig) => Loadable<() => Loadable<void>>,\n\tstore: RootStore = IMPLICIT.STORE,\n): () => Promise<void> {\n\tconst socketRealm = new Realm<SocketSystemHierarchy>(store)\n\n\tserver\n\t\t.use(async (socket, next) => {\n\t\t\tconst result = await auth(socket.handshake)\n\t\t\tif (result instanceof Error) {\n\t\t\t\tstore.logger.error(\n\t\t\t\t\t`📡`,\n\t\t\t\t\t`socket`,\n\t\t\t\t\tsocket.id,\n\t\t\t\t\t`failed to authenticate`,\n\t\t\t\t\tresult.message,\n\t\t\t\t)\n\t\t\t\tnext(result)\n\t\t\t\treturn\n\t\t\t}\n\t\t\tconst userClaim = socketRealm.allocate(`root`, result)\n\t\t\tconst socketClaim = socketRealm.allocate(`root`, `socket::${socket.id}`)\n\t\t\tconst socketState = findInStore(store, socketAtoms, socketClaim)\n\t\t\tsetIntoStore(store, socketState, socket)\n\t\t\teditRelationsInStore(store, usersOfSockets, (relations) => {\n\t\t\t\trelations.set(userClaim, socketClaim)\n\t\t\t})\n\t\t\tsetIntoStore(store, onlineUsersAtom, (index) => index.add(userClaim))\n\t\t\tsetIntoStore(store, socketKeysAtom, (index) => index.add(socketClaim))\n\t\t\tnext()\n\t\t})\n\t\t.on(`connection`, async (socket) => {\n\t\t\tconst socketKey = `socket::${socket.id}` satisfies SocketKey\n\t\t\tconst userKeySelector = findRelationsInStore(\n\t\t\t\tstore,\n\t\t\t\tusersOfSockets,\n\t\t\t\tsocketKey,\n\t\t\t).userKeyOfSocket\n\t\t\tconst userKey = getFromStore(store, userKeySelector)!\n\t\t\tconst serverConfig: UserServerConfig = { store, socket, consumer: userKey }\n\t\t\tconst provideState = realtimeStateProvider(serverConfig)\n\t\t\tconst unsubFromMyUserKey = provideState(myUserKeyAtom, userKey)\n\n\t\t\tconst disposeServices = await onConnect(serverConfig)\n\n\t\t\tsocket.on(`disconnect`, async () => {\n\t\t\t\tstore.logger.info(`📡`, `socket`, socketKey, `👤 ${userKey} disconnects`)\n\t\t\t\tawait disposeServices()\n\t\t\t\tunsubFromMyUserKey()\n\t\t\t\teditRelationsInStore(store, usersOfSockets, (rel) =>\n\t\t\t\t\trel.delete(socketKey),\n\t\t\t\t)\n\t\t\t\tsetIntoStore(\n\t\t\t\t\tstore,\n\t\t\t\t\tonlineUsersAtom,\n\t\t\t\t\t(keys) => (keys.delete(userKey), keys),\n\t\t\t\t)\n\t\t\t\tsetIntoStore(\n\t\t\t\t\tstore,\n\t\t\t\t\tsocketKeysAtom,\n\t\t\t\t\t(keys) => (keys.delete(socketKey), keys),\n\t\t\t\t)\n\t\t\t})\n\t\t})\n\n\tconst disposeAll = async () => {\n\t\tawait server.close()\n\t}\n\treturn disposeAll\n}\n"],"mappings":"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"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "atom.io",
3
- "version": "0.46.15",
3
+ "version": "0.46.17",
4
4
  "description": "Composable and testable reactive data library.",
5
5
  "homepage": "https://atom.io.fyi",
6
6
  "sideEffects": false,
@@ -70,10 +70,10 @@
70
70
  "@types/node": "25.0.3",
71
71
  "@types/react": "19.2.7",
72
72
  "@types/tmp": "0.2.6",
73
- "@typescript-eslint/parser": "8.51.0",
74
- "@typescript-eslint/rule-tester": "8.51.0",
75
- "@typescript-eslint/utils": "8.51.0",
76
- "@typescript/native-preview": "7.0.0-dev.20260103.1",
73
+ "@typescript-eslint/parser": "8.52.0",
74
+ "@typescript-eslint/rule-tester": "8.52.0",
75
+ "@typescript-eslint/utils": "8.52.0",
76
+ "@typescript/native-preview": "7.0.0-dev.20260105.1",
77
77
  "@vitest/coverage-v8": "4.0.16",
78
78
  "@vitest/ui": "4.0.16",
79
79
  "arktype": "2.1.29",
@@ -83,7 +83,7 @@
83
83
  "eslint": "9.39.2",
84
84
  "happy-dom": "20.0.11",
85
85
  "http-proxy": "1.18.1",
86
- "motion": "12.23.26",
86
+ "motion": "12.24.0",
87
87
  "postgres": "3.4.7",
88
88
  "preact": "10.28.1",
89
89
  "react": "19.2.3",
@@ -98,8 +98,8 @@
98
98
  "vite": "7.3.0",
99
99
  "vite-tsconfig-paths": "6.0.3",
100
100
  "vitest": "4.0.16",
101
- "takua": "0.1.1",
102
- "break-check": "0.6.24"
101
+ "break-check": "0.6.24",
102
+ "takua": "0.1.1"
103
103
  },
104
104
  "main": "./dist/main/index.js",
105
105
  "files": [
@@ -1,7 +1,12 @@
1
- import type { ReadonlyPureSelectorToken, RegularAtomToken } from "atom.io"
1
+ import type {
2
+ ReadonlyPureSelectorToken,
3
+ RegularAtomToken,
4
+ ViewOf,
5
+ } from "atom.io"
2
6
  import { atom, getInternalRelations, selector } from "atom.io"
3
7
  import type { RoomKey, SocketKey, UserKey } from "atom.io/realtime"
4
- import { usersInRooms } from "atom.io/realtime"
8
+ import { ownersOfRooms, usersInRooms } from "atom.io/realtime"
9
+ import type { UList } from "atom.io/transceivers/u-list"
5
10
 
6
11
  export const mySocketKeyAtom: RegularAtomToken<SocketKey | undefined> = atom({
7
12
  key: `mySocketKey`,
@@ -42,3 +47,29 @@ export const myRoomKeySelector: ReadonlyPureSelectorToken<RoomKey | null> =
42
47
  return null
43
48
  },
44
49
  })
50
+
51
+ export const usersHereSelector: ReadonlyPureSelectorToken<
52
+ ViewOf<UList<UserKey> | null>
53
+ > = selector({
54
+ key: `usersHere`,
55
+ get: ({ get, find }) => {
56
+ const myRoomKey = get(myRoomKeySelector)
57
+ if (!myRoomKey) return null
58
+ const [usersInRoomsAtoms] = getInternalRelations(usersInRooms, `split`)
59
+ const users = get(find(usersInRoomsAtoms, myRoomKey))
60
+ return users
61
+ },
62
+ })
63
+
64
+ export const roomOwnerSelector: ReadonlyPureSelectorToken<UserKey | null> =
65
+ selector({
66
+ key: `roomOwner`,
67
+ get: ({ get }) => {
68
+ const myRoomKey = get(myRoomKeySelector)
69
+ if (!myRoomKey) return null
70
+ const [, ownerOfRoomsAtoms] = getInternalRelations(ownersOfRooms, `split`)
71
+ const owner = get(ownerOfRoomsAtoms, myRoomKey)
72
+ for (const userKey of owner) return userKey
73
+ return null
74
+ },
75
+ })
@@ -1,6 +1,7 @@
1
1
  import type { ChildProcessWithoutNullStreams } from "node:child_process"
2
2
  import { spawn } from "node:child_process"
3
3
 
4
+ import type { ReadableFamilyToken } from "atom.io"
4
5
  import type { RootStore } from "atom.io/internal"
5
6
  import {
6
7
  editRelationsInStore,
@@ -270,6 +271,7 @@ export function destroyRoom({
270
271
 
271
272
  export type ProvideRoomsConfig<RoomNames extends string> = {
272
273
  resolveRoomScript: (path: RoomNames) => [string, string[]]
274
+ roomAdminsToken: ReadableFamilyToken<boolean, UserKey>
273
275
  roomNames: RoomNames[]
274
276
  roomTimeLimit?: number
275
277
  userKey: UserKey
@@ -277,12 +279,14 @@ export type ProvideRoomsConfig<RoomNames extends string> = {
277
279
  socket: Socket
278
280
  }
279
281
  export function provideRooms<RoomNames extends string>({
280
- store = IMPLICIT.STORE,
281
- socket,
282
282
  resolveRoomScript,
283
+ roomAdminsToken,
283
284
  roomNames,
285
+ socket,
286
+ store = IMPLICIT.STORE,
284
287
  userKey,
285
288
  }: ProvideRoomsConfig<RoomNames>): () => void {
289
+ const isAdmin = getFromStore(store, roomAdminsToken, userKey)
286
290
  const roomSocket = guardSocket<RoomSocketInterface<RoomNames>>(
287
291
  socket,
288
292
  createRoomSocketGuard(roomNames),
@@ -330,11 +334,13 @@ export function provideRooms<RoomNames extends string>({
330
334
  enterRoom(userRoomKey)
331
335
  break
332
336
  }
333
- roomSocket.on(
334
- `createRoom`,
335
- spawnRoom({ store, socket, userKey, resolveRoomScript }),
336
- )
337
- roomSocket.on(`deleteRoom`, destroyRoom({ store, socket, userKey }))
337
+ if (isAdmin) {
338
+ roomSocket.on(
339
+ `createRoom`,
340
+ spawnRoom({ store, socket, userKey, resolveRoomScript }),
341
+ )
342
+ roomSocket.on(`deleteRoom`, destroyRoom({ store, socket, userKey }))
343
+ }
338
344
  return () => {
339
345
  unsubFromRoomKeys()
340
346
  unsubFromUsersInRooms()
@@ -22,10 +22,12 @@ export function realtimeAtomFamilyProvider({
22
22
  K extends Canonical,
23
23
  >(
24
24
  family: AtomIO.RegularAtomFamilyToken<J, K>,
25
- index: AtomIO.ReadableToken<Iterable<NoInfer<K>>> | Iterable<NoInfer<K>>,
25
+ index:
26
+ | AtomIO.ReadableToken<Iterable<NoInfer<K>> | null>
27
+ | Iterable<NoInfer<K>>,
26
28
  ): () => void {
27
29
  const [dynamicIndex, staticIndex]:
28
- | [AtomIO.ReadableToken<Iterable<NoInfer<K>>>, undefined]
30
+ | [AtomIO.ReadableToken<Iterable<NoInfer<K>> | null>, undefined]
29
31
  | [undefined, Iterable<NoInfer<K>>] = (() => {
30
32
  if (typeof index === `object` && `key` in index && `type` in index) {
31
33
  return [index, undefined] as const
@@ -105,13 +107,14 @@ export function realtimeAtomFamilyProvider({
105
107
  )
106
108
  coreSubscriptions.add(
107
109
  employSocket(socket, `sub:${family.key}`, (subKey: K) => {
108
- let exposedSubKeys: Iterable<K>
110
+ let exposedSubKeys: Iterable<K> | null
109
111
  if (dynamicIndex) {
110
112
  exposedSubKeys = getFromStore(store, dynamicIndex)
111
113
  } else {
112
114
  exposedSubKeys = staticIndex
113
115
  }
114
- const shouldExpose = isAvailable(exposedSubKeys, subKey)
116
+ const shouldExpose =
117
+ exposedSubKeys && isAvailable(exposedSubKeys, subKey)
115
118
  if (shouldExpose) {
116
119
  store.logger.info(
117
120
  `👀`,
@@ -150,6 +153,7 @@ export function realtimeAtomFamilyProvider({
150
153
  `has the following keys available for family "${family.key}"`,
151
154
  newExposedSubKeys,
152
155
  )
156
+ if (newExposedSubKeys === null) return
153
157
  for (const subKey of newExposedSubKeys) {
154
158
  if (familyMemberSubscriptionsWanted.has(stringifyJson(subKey))) {
155
159
  store.logger.info(
@@ -39,10 +39,12 @@ export function realtimeMutableFamilyProvider({
39
39
  K extends Canonical,
40
40
  >(
41
41
  family: AtomIO.MutableAtomFamilyToken<T, K>,
42
- index: AtomIO.ReadableToken<Iterable<NoInfer<K>>> | Iterable<NoInfer<K>>,
42
+ index:
43
+ | AtomIO.ReadableToken<Iterable<NoInfer<K>> | null>
44
+ | Iterable<NoInfer<K>>,
43
45
  ): () => void {
44
46
  const [dynamicIndex, staticIndex]:
45
- | [AtomIO.ReadableToken<Iterable<NoInfer<K>>>, undefined]
47
+ | [AtomIO.ReadableToken<Iterable<NoInfer<K>> | null>, undefined]
46
48
  | [undefined, Iterable<NoInfer<K>>] = (() => {
47
49
  if (typeof index === `object` && `key` in index && `type` in index) {
48
50
  return [index, undefined] as const
@@ -109,13 +111,14 @@ export function realtimeMutableFamilyProvider({
109
111
  )
110
112
  coreSubscriptions.add(
111
113
  employSocket(socket, `sub:${family.key}`, (subKey: K) => {
112
- let exposedSubKeys: Iterable<K>
114
+ let exposedSubKeys: Iterable<K> | null
113
115
  if (dynamicIndex) {
114
116
  exposedSubKeys = getFromStore(store, dynamicIndex)
115
117
  } else {
116
118
  exposedSubKeys = staticIndex
117
119
  }
118
- const shouldExpose = isAvailable(exposedSubKeys, subKey)
120
+ const shouldExpose =
121
+ exposedSubKeys && isAvailable(exposedSubKeys, subKey)
119
122
  if (shouldExpose) {
120
123
  store.logger.info(
121
124
  `👀`,
@@ -154,6 +157,7 @@ export function realtimeMutableFamilyProvider({
154
157
  `has the following keys available for family "${family.key}"`,
155
158
  newExposedSubKeys,
156
159
  )
160
+ if (newExposedSubKeys === null) return
157
161
  for (const subKey of newExposedSubKeys) {
158
162
  if (familyMemberSubscriptionsWanted.has(stringifyJson(subKey))) {
159
163
  store.logger.info(