atom.io 0.18.0 → 0.18.1

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.
Files changed (83) hide show
  1. package/dist/{chunk-OEVFAUPE.js → chunk-IZHOMSXA.js} +53 -11
  2. package/dist/chunk-IZHOMSXA.js.map +1 -0
  3. package/dist/chunk-JDUNWJFB.js +18 -0
  4. package/dist/chunk-JDUNWJFB.js.map +1 -0
  5. package/dist/index.cjs.map +1 -1
  6. package/dist/index.d.ts +5 -1
  7. package/dist/index.js.map +1 -1
  8. package/internal/dist/index.cjs +64 -34
  9. package/internal/dist/index.cjs.map +1 -1
  10. package/internal/dist/index.d.ts +1 -1
  11. package/internal/dist/index.js +64 -34
  12. package/internal/dist/index.js.map +1 -1
  13. package/internal/src/atom/delete-atom.ts +7 -6
  14. package/internal/src/caching.ts +6 -6
  15. package/internal/src/ingest-updates/ingest-atom-update.ts +6 -2
  16. package/internal/src/set-state/copy-mutable-if-needed.ts +5 -0
  17. package/internal/src/set-state/emit-update.ts +25 -11
  18. package/internal/src/set-state/set-atom.ts +4 -3
  19. package/internal/src/transaction/apply-transaction.ts +0 -1
  20. package/internal/src/transaction/set-epoch-number.ts +0 -1
  21. package/json/src/index.ts +3 -3
  22. package/package.json +241 -241
  23. package/react-devtools/dist/index.cjs.map +1 -1
  24. package/react-devtools/dist/index.js +1 -15
  25. package/react-devtools/dist/index.js.map +1 -1
  26. package/react-devtools/src/StateEditor.tsx +6 -6
  27. package/react-devtools/src/StateIndex.tsx +2 -2
  28. package/react-devtools/src/Updates.tsx +1 -1
  29. package/react-devtools/src/index.ts +3 -3
  30. package/realtime/dist/index.cjs +50 -2
  31. package/realtime/dist/index.cjs.map +1 -1
  32. package/realtime/dist/index.d.ts +110 -3
  33. package/realtime/dist/index.js +47 -4
  34. package/realtime/dist/index.js.map +1 -1
  35. package/realtime/src/index.ts +1 -0
  36. package/realtime/src/realtime-continuity.ts +14 -4
  37. package/realtime/src/shared-room-store.ts +48 -0
  38. package/realtime-client/dist/index.cjs +113 -200
  39. package/realtime-client/dist/index.cjs.map +1 -1
  40. package/realtime-client/dist/index.d.ts +2 -5
  41. package/realtime-client/dist/index.js +17 -161
  42. package/realtime-client/dist/index.js.map +1 -1
  43. package/realtime-client/src/index.ts +0 -2
  44. package/realtime-client/src/pull-mutable-atom-family-member.ts +5 -5
  45. package/realtime-client/src/realtime-client-stores/client-main-store.ts +10 -0
  46. package/realtime-client/src/sync-continuity.ts +56 -9
  47. package/realtime-react/dist/index.cjs +51 -26
  48. package/realtime-react/dist/index.cjs.map +1 -1
  49. package/realtime-react/dist/index.d.ts +2 -6
  50. package/realtime-react/dist/index.js +2 -17
  51. package/realtime-react/dist/index.js.map +1 -1
  52. package/realtime-react/src/index.ts +0 -2
  53. package/realtime-server/dist/index.cjs +399 -327
  54. package/realtime-server/dist/index.cjs.map +1 -1
  55. package/realtime-server/dist/index.d.ts +55 -60
  56. package/realtime-server/dist/index.js +394 -319
  57. package/realtime-server/dist/index.js.map +1 -1
  58. package/realtime-server/src/index.ts +2 -4
  59. package/realtime-server/src/ipc-sockets/child-socket.ts +135 -0
  60. package/realtime-server/src/ipc-sockets/custom-socket.ts +90 -0
  61. package/realtime-server/src/ipc-sockets/index.ts +3 -0
  62. package/realtime-server/src/ipc-sockets/parent-socket.ts +185 -0
  63. package/realtime-server/src/realtime-continuity-synchronizer.ts +225 -96
  64. package/realtime-server/src/realtime-server-stores/index.ts +2 -1
  65. package/realtime-server/src/realtime-server-stores/realtime-continuity-store.ts +50 -31
  66. package/realtime-server/src/realtime-server-stores/server-room-external-actions.ts +64 -0
  67. package/realtime-server/src/realtime-server-stores/server-room-external-store.ts +42 -0
  68. package/realtime-server/src/realtime-server-stores/server-sync-store.ts +49 -26
  69. package/realtime-testing/dist/index.cjs +8 -6
  70. package/realtime-testing/dist/index.cjs.map +1 -1
  71. package/realtime-testing/dist/index.d.ts +1 -0
  72. package/realtime-testing/dist/index.js +7 -6
  73. package/realtime-testing/dist/index.js.map +1 -1
  74. package/realtime-testing/src/setup-realtime-test.tsx +8 -6
  75. package/src/logger.ts +5 -1
  76. package/dist/chunk-OEVFAUPE.js.map +0 -1
  77. package/realtime-client/src/sync-server-action.ts +0 -168
  78. package/realtime-client/src/sync-state.ts +0 -19
  79. package/realtime-react/src/use-sync-server-action.ts +0 -17
  80. package/realtime-react/src/use-sync.ts +0 -17
  81. package/realtime-server/src/ipc-socket.ts +0 -230
  82. package/realtime-server/src/realtime-action-synchronizer.ts +0 -164
  83. package/realtime-server/src/realtime-server-stores/server-room-store.ts +0 -97
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/ipc-socket.ts","../src/realtime-continuity-synchronizer.ts","../src/realtime-server-stores/realtime-continuity-store.ts","../src/realtime-server-stores/server-room-store.ts","../src/realtime-server-stores/server-sync-store.ts","../src/realtime-server-stores/server-user-store.ts","../src/realtime-state-provider.ts","../src/realtime-state-synchronizer.ts","../src/realtime-family-provider.ts","../src/realtime-mutable-provider.ts","../src/realtime-mutable-family-provider.ts","../src/realtime-state-receiver.ts","../src/realtime-action-receiver.ts","../src/realtime-action-synchronizer.ts"],"names":["process","atomFamily","selectorFamily","completeUpdateAtoms","atom","join","SetRTX","transaction","IMPLICIT","getFromStore","subscribeToState","findInStore","getJsonToken","getUpdateToken","stringifyJson","setIntoStore","actUponStore","subscribeToTransaction"],"mappings":"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","sourcesContent":["import type { ChildProcessWithoutNullStreams } from \"child_process\"\n\nimport { Subject } from \"atom.io/internal\"\nimport { parseJson } from \"atom.io/json\"\nimport type { Json, Stringified } from \"atom.io/json\"\n\nimport type { Socket } from \".\"\n\nexport type Events = Json.Object<string, Json.Serializable[]>\n\nexport type StringifiedEvent<\n\tKey extends string,\n\tParams extends Json.Serializable[],\n> = Stringified<[Key, ...Params]>\n\nexport interface EventBuffer<\n\tKey extends string,\n\tParams extends Json.Serializable[],\n> extends Buffer {\n\ttoString(): StringifiedEvent<Key, Params>\n}\n\nexport class CustomSocket<I extends Events, O extends Events> implements Socket {\n\tprotected listeners: Map<keyof I, Set<(...args: Json.Array) => void>>\n\tprotected globalListeners: Set<(event: string, ...args: Json.Array) => void>\n\tprotected handleEvent<Event extends keyof I>(\n\t\tevent: string,\n\t\t...args: I[Event]\n\t): void {\n\t\tfor (const listener of this.globalListeners) {\n\t\t\tlistener(event, ...args)\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(...args)\n\t\t\t}\n\t\t}\n\t}\n\n\tpublic id = `no_id_retrieved`\n\n\tpublic constructor(\n\t\tpublic emit: <Event extends keyof O>(\n\t\t\tevent: Event,\n\t\t\t...args: O[Event]\n\t\t) => CustomSocket<I, O>,\n\t) {\n\t\tthis.listeners = new Map()\n\t\tthis.globalListeners = new Set()\n\t}\n\n\tpublic on<Event extends keyof I>(\n\t\tevent: Event,\n\t\tlistener: (...args: I[Event]) => void,\n\t): CustomSocket<I, O> {\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(\n\t\tlistener: (event: string, ...args: Json.Array) => void,\n\t): CustomSocket<I, O> {\n\t\tthis.globalListeners.add(listener)\n\t\treturn this\n\t}\n\n\tpublic off<Event extends keyof I>(\n\t\tevent: Event,\n\t\tlistener: (...args: I[Event]) => void,\n\t): CustomSocket<I, O> {\n\t\tconst listeners = this.listeners.get(event)\n\t\tif (listeners) {\n\t\t\tlisteners.delete(listener)\n\t\t}\n\t\treturn this\n\t}\n\n\tpublic offAny(\n\t\tlistener: (event: string, ...args: Json.Array) => void,\n\t): CustomSocket<I, O> {\n\t\tthis.globalListeners.delete(listener)\n\t\treturn this\n\t}\n}\n\nexport class ChildSocket<\n\tI extends Events,\n\tO extends Events & {\n\t\t/* eslint-disable quotes */\n\t\t\"setup-relay\": [string]\n\t\t/* eslint-enable quotes */\n\t},\n> extends CustomSocket<I, O> {\n\tprotected process: ChildProcessWithoutNullStreams\n\n\tpublic id = `no_id_retrieved`\n\n\tpublic constructor(process: ChildProcessWithoutNullStreams) {\n\t\tsuper((event, ...args) => {\n\t\t\tconst stringifiedEvent = JSON.stringify([event, ...args]) + `\\n`\n\t\t\tthis.process.stdin.write(stringifiedEvent)\n\t\t\treturn this\n\t\t})\n\t\tthis.process = process\n\t\tthis.process.stdout.on(\n\t\t\t`data`,\n\t\t\t<Event extends keyof I>(buffer: EventBuffer<string, I[Event]>) => {\n\t\t\t\tconst stringifiedEvent = buffer.toString()\n\t\t\t\tconst parsedEvent = parseJson(stringifiedEvent)\n\t\t\t\tthis.handleEvent(...parsedEvent)\n\t\t\t},\n\t\t)\n\t\tif (process.pid) {\n\t\t\tthis.id = process.pid.toString()\n\t\t}\n\t}\n}\n\nexport class SubjectSocket<\n\tI extends Events,\n\tO extends Events,\n> extends CustomSocket<I, O> {\n\tpublic in: Subject<[string, ...Json.Serializable[]]>\n\tpublic out: Subject<[string, ...Json.Serializable[]]>\n\tpublic id = `no_id_retrieved`\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 as [string, ...I[keyof I]]))\n\t\t})\n\t}\n}\n\nexport class ParentSocket<\n\tI extends Events & {\n\t\t[id in string as `relay:${id}`]: [string, ...Json.Serializable[]]\n\t} & {\n\t\t/* eslint-disable quotes */\n\t\t\"setup-relay\": [string]\n\t\t/* eslint-enable quotes */\n\t},\n\tO extends Events & {\n\t\t[id in string as `relay:${id}`]: [string, ...Json.Serializable[]]\n\t},\n> extends CustomSocket<I, O> {\n\tprotected queue: string[]\n\tprotected relays: Map<string, SubjectSocket<any, any>>\n\tprotected relayServices: ((\n\t\tsocket: SubjectSocket<any, any>,\n\t) => (() => void) | void)[]\n\tprotected process: NodeJS.Process\n\n\tpublic id = `no_id_retrieved`\n\n\tpublic constructor() {\n\t\tsuper((event, ...args) => {\n\t\t\tconst stringifiedEvent = JSON.stringify([event, ...args])\n\t\t\tthis.process.stdout.write(stringifiedEvent)\n\t\t\treturn this\n\t\t})\n\t\tthis.process = process\n\t\tthis.process.stdin.resume()\n\t\tthis.queue = []\n\t\tthis.relays = new Map()\n\t\tthis.relayServices = []\n\n\t\tthis.process.stdin.on(\n\t\t\t`data`,\n\t\t\t<Event extends keyof I>(chunk: EventBuffer<string, I[Event]>) => {\n\t\t\t\tconst buffer = chunk.toString()\n\t\t\t\tthis.queue.push(...buffer.split(`\\n`))\n\n\t\t\t\twhile (this.queue.length > 0) {\n\t\t\t\t\ttry {\n\t\t\t\t\t\tconst event = this.queue.shift() as StringifiedEvent<any, any>\n\t\t\t\t\t\tif (event === ``) continue\n\t\t\t\t\t\tconst parsedEvent = parseJson(event)\n\t\t\t\t\t\tthis.handleEvent(...(parsedEvent as [string, ...I[keyof I]]))\n\t\t\t\t\t} catch (error) {\n\t\t\t\t\t\tthis.process.stderr.write(`❌ ${error}\\n`)\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\n\t\t// process.stdin.on(`end`, () => process.exit(0))\n\t\tprocess.on(`SIGINT`, () => process.exit(0))\n\t\tif (process.pid) {\n\t\t\tthis.id = process.pid?.toString()\n\t\t}\n\n\t\tthis.on(`setup-relay`, (id: string) => {\n\t\t\tconst relay = new SubjectSocket(`relay:${id}`)\n\t\t\tthis.relays.set(id, relay)\n\t\t\tfor (const attachServices of this.relayServices) {\n\t\t\t\tattachServices(relay)\n\t\t\t}\n\t\t\tthis.on(`relay:${id}`, (...data) => {\n\t\t\t\trelay.in.next(data)\n\t\t\t})\n\t\t\trelay.out.subscribe(`socket`, (data) => {\n\t\t\t\tthis.emit(...(data as [string, ...O[keyof O]]))\n\t\t\t})\n\t\t})\n\t}\n\n\tpublic relay(\n\t\tattachServices: (socket: SubjectSocket<any, any>) => (() => void) | void,\n\t): void {\n\t\tthis.relayServices.push(attachServices)\n\t\tconst relays = this.relays.values()\n\t\tfor (const relay of relays) {\n\t\t\tattachServices(relay)\n\t\t}\n\t}\n}\n","import type * as AtomIO from \"atom.io\"\nimport {\n\tIMPLICIT,\n\tactUponStore,\n\tfindInStore,\n\tgetFromStore,\n\tisRootStore,\n\tsetIntoStore,\n\tsubscribeToState,\n\tsubscribeToTransaction,\n} from \"atom.io/internal\"\nimport type { Json, JsonIO } from \"atom.io/json\"\nimport type { ContinuityToken } from \"atom.io/realtime\"\n\nimport type { ServerConfig, Socket } from \".\"\nimport { socketAtoms, usersOfSockets } from \".\"\nimport { redactedPerspectiveUpdateSelectors } from \"./realtime-server-stores\"\nimport {\n\tcompleteUpdateAtoms,\n\tuserUnacknowledgedQueues,\n} from \"./realtime-server-stores\"\n\nexport type RealtimeContinuitySynchronizer = ReturnType<\n\ttypeof realtimeContinuitySynchronizer\n>\nexport function realtimeContinuitySynchronizer({\n\tsocket: initialSocket,\n\tstore = IMPLICIT.STORE,\n}: ServerConfig) {\n\treturn function synchronizer(continuity: ContinuityToken): () => void {\n\t\tlet socket: Socket | null = initialSocket\n\n\t\tconst continuityKey = continuity.key\n\t\tconst userKeyState = findInStore(\n\t\t\tusersOfSockets.states.userKeyOfSocket,\n\t\t\tsocket.id,\n\t\t\tstore,\n\t\t)\n\t\tconst userKey = getFromStore(userKeyState, store)\n\t\tif (!userKey) {\n\t\t\tstore.logger.error(\n\t\t\t\t`❌`,\n\t\t\t\t`continuity`,\n\t\t\t\tcontinuityKey,\n\t\t\t\t`Tried to create a synchronizer for a socket (${socket.id}) that is not connected to a user.`,\n\t\t\t)\n\t\t\treturn () => {}\n\t\t}\n\t\tconst socketKeyState = findInStore(\n\t\t\tusersOfSockets.states.socketKeyOfUser,\n\t\t\tuserKey,\n\t\t\tstore,\n\t\t)\n\t\tsubscribeToState(\n\t\t\tsocketKeyState,\n\t\t\t({ newValue: newSocketKey }) => {\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`seeing ${userKey} on new socket ${newSocketKey}`,\n\t\t\t\t)\n\t\t\t\tif (newSocketKey === null) {\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`Tried to create a synchronizer for a user (${userKey}) that is not connected to a socket.`,\n\t\t\t\t\t)\n\t\t\t\t\treturn\n\t\t\t\t}\n\t\t\t\tconst newSocketState = findInStore(socketAtoms, newSocketKey, store)\n\t\t\t\tconst newSocket = getFromStore(newSocketState, store)\n\t\t\t\tsocket = newSocket\n\t\t\t},\n\t\t\t`sync-continuity:${continuityKey}:${userKey}`,\n\t\t\tstore,\n\t\t)\n\n\t\tconst userUnacknowledgedQueue = findInStore(\n\t\t\tuserUnacknowledgedQueues,\n\t\t\tuserKey,\n\t\t\tstore,\n\t\t)\n\t\tconst userUnacknowledgedUpdates = getFromStore(\n\t\t\tuserUnacknowledgedQueue,\n\t\t\tstore,\n\t\t)\n\t\tconst unsubscribeFunctions: (() => void)[] = []\n\n\t\tconst sendInitialPayload = () => {\n\t\t\tconst initialPayload: Json.Serializable[] = []\n\t\t\tfor (const atom of continuity.globals) {\n\t\t\t\tinitialPayload.push(atom, getFromStore(atom, store))\n\t\t\t}\n\t\t\tfor (const { perspectiveAtoms } of continuity.perspectives) {\n\t\t\t\tconst perspectiveTokensState = findInStore(\n\t\t\t\t\tperspectiveAtoms,\n\t\t\t\t\tuserKey,\n\t\t\t\t\tstore,\n\t\t\t\t)\n\t\t\t\tconst perspectiveTokens = getFromStore(perspectiveTokensState, store)\n\t\t\t\tfor (const perspectiveToken of perspectiveTokens) {\n\t\t\t\t\tconst resource = getFromStore(perspectiveToken, store)\n\t\t\t\t\tinitialPayload.push(perspectiveToken, resource)\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tconst epoch = isRootStore(store)\n\t\t\t\t? store.transactionMeta.epoch.get(continuityKey) ?? null\n\t\t\t\t: null\n\t\t\tsocket?.emit(`continuity-init:${continuityKey}`, epoch, initialPayload)\n\n\t\t\tfor (const transaction of continuity.actions) {\n\t\t\t\tconst unsubscribeFromTransaction = subscribeToTransaction(\n\t\t\t\t\ttransaction,\n\t\t\t\t\t(update) => {\n\t\t\t\t\t\t// store.logger.info(`userId`, userKey)\n\t\t\t\t\t\tconst updateState = findInStore(\n\t\t\t\t\t\t\tcompleteUpdateAtoms,\n\t\t\t\t\t\t\tupdate.id,\n\t\t\t\t\t\t\tstore,\n\t\t\t\t\t\t)\n\t\t\t\t\t\tsetIntoStore(updateState, update, store)\n\t\t\t\t\t\tconst redactedUpdateKey = {\n\t\t\t\t\t\t\tuserId: userKey,\n\t\t\t\t\t\t\tsyncGroupKey: continuityKey,\n\t\t\t\t\t\t\tupdateId: update.id,\n\t\t\t\t\t\t}\n\t\t\t\t\t\tconst redactedUpdateState = findInStore(\n\t\t\t\t\t\t\tredactedPerspectiveUpdateSelectors,\n\t\t\t\t\t\t\tredactedUpdateKey,\n\t\t\t\t\t\t\tstore,\n\t\t\t\t\t\t)\n\n\t\t\t\t\t\tconst redactedUpdate = getFromStore(redactedUpdateState, store)\n\n\t\t\t\t\t\tsetIntoStore(\n\t\t\t\t\t\t\tuserUnacknowledgedQueue,\n\t\t\t\t\t\t\t(updates) => {\n\t\t\t\t\t\t\t\tif (redactedUpdate) {\n\t\t\t\t\t\t\t\t\tupdates.push(redactedUpdate)\n\t\t\t\t\t\t\t\t\tupdates.sort((a, b) => a.epoch - b.epoch)\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\treturn updates\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\tstore,\n\t\t\t\t\t\t)\n\n\t\t\t\t\t\tsocket?.emit(\n\t\t\t\t\t\t\t`tx-new:${continuityKey}`,\n\t\t\t\t\t\t\tredactedUpdate as Json.Serializable,\n\t\t\t\t\t\t)\n\t\t\t\t\t},\n\t\t\t\t\t`sync-continuity:${continuityKey}:${userKey}`,\n\t\t\t\t\tstore,\n\t\t\t\t)\n\t\t\t\tunsubscribeFunctions.push(unsubscribeFromTransaction)\n\t\t\t}\n\t\t}\n\t\tsocket.off(`get:${continuityKey}`, sendInitialPayload)\n\t\tsocket.on(`get:${continuityKey}`, sendInitialPayload)\n\n\t\tconst fillTransactionRequest = (\n\t\t\tupdate: Pick<AtomIO.TransactionUpdate<JsonIO>, `id` | `key` | `params`>,\n\t\t) => {\n\t\t\tconst transactionKey = update.key\n\t\t\tconst updateId = update.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\tactUponStore(\n\t\t\t\t{ type: `transaction`, key: transactionKey },\n\t\t\t\tupdateId,\n\t\t\t\tstore,\n\t\t\t)(...update.params)\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\tmetric.duration,\n\t\t\t)\n\t\t}\n\t\tsocket.off(`tx-run:${continuityKey}`, fillTransactionRequest)\n\t\tsocket.on(`tx-run:${continuityKey}`, fillTransactionRequest)\n\n\t\tlet i = 1\n\t\tlet next = 1\n\t\tconst retry = setInterval(() => {\n\t\t\tconst toEmit = userUnacknowledgedUpdates[0]\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`${store.config.name} retrying ${userKey} (${i}/${next})`,\n\t\t\t\tsocket?.id,\n\t\t\t\tuserUnacknowledgedUpdates,\n\t\t\t)\n\t\t\tif (toEmit && i === next) {\n\t\t\t\tsocket?.emit(`tx-new:${continuityKey}`, toEmit as Json.Serializable)\n\t\t\t\tnext *= 2\n\t\t\t}\n\n\t\t\ti++\n\t\t}, 250)\n\t\tconst trackClientAcknowledgement = (epoch: number) => {\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`${userKey} acknowledged epoch ${epoch}`,\n\t\t\t)\n\t\t\ti = 1\n\t\t\tnext = 1\n\t\t\tconst isUnacknowledged = userUnacknowledgedUpdates[0]?.epoch === epoch\n\n\t\t\tif (isUnacknowledged) {\n\t\t\t\tsetIntoStore(\n\t\t\t\t\tuserUnacknowledgedQueue,\n\t\t\t\t\t(updates) => {\n\t\t\t\t\t\tupdates.shift()\n\t\t\t\t\t\treturn updates\n\t\t\t\t\t},\n\t\t\t\t\tstore,\n\t\t\t\t)\n\t\t\t}\n\t\t}\n\t\tsocket.off(`ack:${continuityKey}`, trackClientAcknowledgement)\n\t\tsocket.on(`ack:${continuityKey}`, trackClientAcknowledgement)\n\n\t\treturn () => {\n\t\t\tclearInterval(retry)\n\t\t\tfor (const unsubscribe of unsubscribeFunctions) unsubscribe()\n\t\t\tsocket?.off(`ack:${continuityKey}`, trackClientAcknowledgement)\n\t\t\tsocket?.off(`get:${continuityKey}`, sendInitialPayload)\n\t\t\tsocket?.off(`tx-run:${continuityKey}`, fillTransactionRequest)\n\t\t}\n\t}\n}\n","import { selectorFamily } from \"atom.io\"\nimport type { TransactionUpdate } from \"atom.io\"\nimport type { JsonIO } from \"atom.io/json\"\nimport { SyncGroup } from \"atom.io/realtime\"\nimport { completeUpdateAtoms } from \"atom.io/realtime-server\"\n\nconst redactorAtoms = selectorFamily<\n\t(update: TransactionUpdate<any>) => TransactionUpdate<any>,\n\t{ userId: string; syncGroupKey: string }\n>({\n\tkey: `perspectiveRedactor`,\n\tget:\n\t\t({ userId, syncGroupKey }) =>\n\t\t({ get, find }) => {\n\t\t\tconst syncGroup = SyncGroup.existing.get(syncGroupKey)\n\t\t\tif (!syncGroup) {\n\t\t\t\tthrow new Error(\n\t\t\t\t\t`Tried to create a synchronizer for a sync group that does not exist.`,\n\t\t\t\t)\n\t\t\t}\n\n\t\t\tconst userPerspectiveTokens = syncGroup.perspectives.flatMap(\n\t\t\t\t({ perspectiveAtoms, resourceAtoms }) => {\n\t\t\t\t\tconst userPerspectiveToken = find(perspectiveAtoms, userId)\n\t\t\t\t\tconst userPerspective = get(userPerspectiveToken)\n\t\t\t\t\tconst visibleTokens = [...userPerspective].map((subKey) => {\n\t\t\t\t\t\tconst resourceToken = find(resourceAtoms, subKey)\n\t\t\t\t\t\treturn resourceToken.key\n\t\t\t\t\t})\n\n\t\t\t\t\treturn visibleTokens\n\t\t\t\t},\n\t\t\t)\n\n\t\t\tconst filterTransactionUpdate = (\n\t\t\t\tvisible: string[],\n\t\t\t\ttransactionUpdate: TransactionUpdate<any>,\n\t\t\t): TransactionUpdate<any> => {\n\t\t\t\tconst updates = transactionUpdate.updates\n\t\t\t\t\t.filter((update) => {\n\t\t\t\t\t\tif (`newValue` in update) {\n\t\t\t\t\t\t\treturn visible.includes(update.key)\n\t\t\t\t\t\t}\n\t\t\t\t\t\treturn true\n\t\t\t\t\t})\n\t\t\t\t\t.map((update) => {\n\t\t\t\t\t\tif (`updates` in update) {\n\t\t\t\t\t\t\treturn filterTransactionUpdate(visible, update)\n\t\t\t\t\t\t}\n\t\t\t\t\t\treturn update\n\t\t\t\t\t})\n\t\t\t\tconst filtered: TransactionUpdate<any> = {\n\t\t\t\t\t...transactionUpdate,\n\t\t\t\t\tupdates,\n\t\t\t\t}\n\t\t\t\treturn filtered\n\t\t\t}\n\t\t\tconst filter: (updates: TransactionUpdate<any>) => TransactionUpdate<any> =\n\t\t\t\t(update) => {\n\t\t\t\t\tconst visibleKeys: string[] = syncGroup.globals.map(\n\t\t\t\t\t\t(atomToken) => atomToken.key,\n\t\t\t\t\t)\n\t\t\t\t\tvisibleKeys.push(...userPerspectiveTokens)\n\t\t\t\t\treturn filterTransactionUpdate(visibleKeys, update)\n\t\t\t\t}\n\t\t\treturn filter\n\t\t},\n})\nexport const redactedPerspectiveUpdateSelectors = selectorFamily<\n\tPick<\n\t\tTransactionUpdate<JsonIO>,\n\t\t`epoch` | `id` | `key` | `output` | `updates`\n\t> | null,\n\t{ userId: string; syncGroupKey: string; updateId: string }\n>({\n\tkey: `redactedPerspectiveUpdate`,\n\tget:\n\t\t({ userId, syncGroupKey, updateId }) =>\n\t\t({ get, find }) => {\n\t\t\tconst updateState = find(completeUpdateAtoms, updateId)\n\t\t\tconst update = get(updateState)\n\t\t\tconst redactorKey = { userId, syncGroupKey }\n\t\t\tconst redactorState = find(redactorAtoms, redactorKey)\n\t\t\tconst redact = get(redactorState)\n\t\t\tif (update) {\n\t\t\t\treturn redact(update)\n\t\t\t}\n\t\t\treturn null\n\t\t},\n})\n","import type { ChildProcessWithoutNullStreams } from \"child_process\"\nimport { spawn } from \"child_process\"\n\nimport * as AtomIO from \"atom.io\"\nimport type { Loadable } from \"atom.io/data\"\nimport { join } from \"atom.io/data\"\nimport { SetRTX } from \"atom.io/transceivers/set-rtx\"\n\nexport type RoomArguments =\n\t| [script: string, options: string[]]\n\t| [script: string]\n\nexport const roomIndex = AtomIO.atom({\n\tkey: `roomIndex`,\n\tdefault: () => new SetRTX<string>(),\n\tmutable: true,\n\ttoJson: (set) => set.toJSON(),\n\tfromJson: (json) => SetRTX.fromJSON(json),\n})\n\nexport type UserInRoomMeta = {\n\tenteredAtEpoch: number\n}\nexport const DEFAULT_USER_IN_ROOM_META: UserInRoomMeta = {\n\tenteredAtEpoch: 0,\n}\nexport const usersInRooms = join(\n\t{\n\t\tkey: `usersInRooms`,\n\t\tbetween: [`room`, `user`],\n\t\tcardinality: `1:n`,\n\t},\n\tDEFAULT_USER_IN_ROOM_META,\n)\n\nexport const roomArgumentsAtoms = AtomIO.atomFamily<RoomArguments, string>({\n\tkey: `roomArguments`,\n\tdefault: [`echo Hello World!`],\n})\n\nexport const roomSelectors = AtomIO.selectorFamily<\n\tLoadable<ChildProcessWithoutNullStreams>,\n\tstring\n>({\n\tkey: `room`,\n\tget:\n\t\t(roomId) =>\n\t\t({ get, find }) => {\n\t\t\tconst argumentsState = find(roomArgumentsAtoms, roomId)\n\t\t\tconst args = get(argumentsState)\n\t\t\tconst [script, options] = args\n\t\t\treturn new Promise((resolve) => {\n\t\t\t\tconst room = spawn(script, options, { env: process.env })\n\t\t\t\tconst resolver = (data: Buffer) => {\n\t\t\t\t\tif (data.toString() === `✨`) {\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\nexport const createRoomTX = AtomIO.transaction<\n\t(\n\t\troomId: string,\n\t\tscript: string,\n\t\toptions?: string[],\n\t) => Loadable<ChildProcessWithoutNullStreams>\n>({\n\tkey: `createRoom`,\n\tdo: ({ get, set, find }, roomId, script, options) => {\n\t\tconst args: RoomArguments = options ? [script, options] : [script]\n\t\tconst roomArgumentsState = find(roomArgumentsAtoms, roomId)\n\t\tset(roomArgumentsState, args)\n\t\tset(roomIndex, (s) => s.add(roomId))\n\t\tconst roomState = find(roomSelectors, roomId)\n\t\tconst room = get(roomState)\n\t\treturn room\n\t},\n})\nexport type CreateRoomIO = AtomIO.TransactionIO<typeof createRoomTX>\n\nexport const joinRoomTX = AtomIO.transaction<\n\t(roomId: string, userId: string, enteredAtEpoch: number) => UserInRoomMeta\n>({\n\tkey: `joinRoom`,\n\tdo: (transactors, roomId, userId, enteredAtEpoch) => {\n\t\tconst meta = { enteredAtEpoch }\n\t\tusersInRooms.transact(transactors, ({ relations }) => {\n\t\t\trelations.set(roomId, userId, meta)\n\t\t})\n\t\treturn meta\n\t},\n})\nexport type JoinRoomIO = AtomIO.TransactionIO<typeof joinRoomTX>\n","import type { TransactionUpdate, TransactionUpdateContent } from \"atom.io\"\nimport { atomFamily, selectorFamily } from \"atom.io\"\n\nexport const completeUpdateAtoms = atomFamily<\n\tTransactionUpdate<any> | null,\n\tstring\n>({\n\tkey: `completeUpdate`,\n\tdefault: null,\n})\n\nexport const transactionRedactorAtoms = atomFamily<\n\t{\n\t\tfilter: (updates: TransactionUpdateContent[]) => TransactionUpdateContent[]\n\t},\n\tstring\n>({\n\tkey: `transactionRedactor`,\n\tdefault: { filter: (updates) => updates },\n})\nexport const redactedUpdateSelectors = selectorFamily<\n\tTransactionUpdate<any> | null,\n\t[transactionKey: string, updateId: string]\n>({\n\tkey: `redactedUpdate`,\n\tget:\n\t\t([transactionKey, updateId]) =>\n\t\t({ get, find }) => {\n\t\t\tconst update = get(find(completeUpdateAtoms, updateId))\n\t\t\tconst { filter } = get(find(transactionRedactorAtoms, transactionKey))\n\n\t\t\tif (update && filter) {\n\t\t\t\treturn { ...update, updates: filter(update.updates) }\n\t\t\t}\n\t\t\treturn null\n\t\t},\n})\n\nexport const userUnacknowledgedQueues = atomFamily<\n\tPick<TransactionUpdate<any>, `epoch` | `id` | `key` | `output` | `updates`>[],\n\tstring\n>({\n\tkey: `unacknowledgedUpdates`,\n\tdefault: () => [],\n})\n","import { atom, atomFamily } from \"atom.io\"\nimport { join } from \"atom.io/data\"\nimport { SetRTX } from \"atom.io/transceivers/set-rtx\"\n\nimport type { Socket } from \"..\"\n\nexport const socketAtoms = atomFamily<Socket | null, string>({\n\tkey: `sockets`,\n\tdefault: null,\n})\n\nexport const socketIndex = atom({\n\tkey: `socketsIndex`,\n\tmutable: true,\n\tdefault: () => new SetRTX<string>(),\n\ttoJson: (set) => set.toJSON(),\n\tfromJson: (json) => SetRTX.fromJSON(json),\n})\nexport const userIndex = atom({\n\tkey: `usersIndex`,\n\tmutable: true,\n\tdefault: () => new SetRTX<string>(),\n\ttoJson: (set) => set.toJSON(),\n\tfromJson: (json) => SetRTX.fromJSON(json),\n})\nexport const usersOfSockets = join({\n\tkey: `usersOfSockets`,\n\tbetween: [`user`, `socket`],\n\tcardinality: `1:1`,\n})\n","import type * as AtomIO from \"atom.io\"\nimport { IMPLICIT, getFromStore, subscribeToState } from \"atom.io/internal\"\nimport type { Json } from \"atom.io/json\"\n\nimport type { ServerConfig } from \".\"\n\nexport type StateProvider = ReturnType<typeof realtimeStateProvider>\nexport function realtimeStateProvider({\n\tsocket,\n\tstore = IMPLICIT.STORE,\n}: ServerConfig) {\n\treturn function stateProvider<J extends Json.Serializable>(\n\t\ttoken: AtomIO.WritableToken<J>,\n\t): () => void {\n\t\tlet unsubscribeFromStateUpdates: (() => void) | undefined\n\n\t\tconst fillSubRequest = () => {\n\t\t\tsocket.emit(`serve:${token.key}`, getFromStore(token, store))\n\n\t\t\tunsubscribeFromStateUpdates = subscribeToState(\n\t\t\t\ttoken,\n\t\t\t\t({ newValue }) => {\n\t\t\t\t\tsocket.emit(`serve:${token.key}`, newValue)\n\t\t\t\t},\n\t\t\t\t`expose-single:${socket.id}`,\n\t\t\t\tstore,\n\t\t\t)\n\n\t\t\tconst fillUnsubRequest = () => {\n\t\t\t\tsocket.off(`unsub:${token.key}`, fillUnsubRequest)\n\t\t\t\tif (unsubscribeFromStateUpdates) {\n\t\t\t\t\tunsubscribeFromStateUpdates()\n\t\t\t\t\tunsubscribeFromStateUpdates = undefined\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tsocket.on(`unsub:${token.key}`, fillUnsubRequest)\n\t\t}\n\n\t\tsocket.on(`sub:${token.key}`, fillSubRequest)\n\n\t\treturn () => {\n\t\t\tsocket.off(`sub:${token.key}`, fillSubRequest)\n\t\t\tif (unsubscribeFromStateUpdates) {\n\t\t\t\tunsubscribeFromStateUpdates()\n\t\t\t\tunsubscribeFromStateUpdates = undefined\n\t\t\t}\n\t\t}\n\t}\n}\n","import type * as AtomIO from \"atom.io\"\nimport { IMPLICIT, getFromStore } from \"atom.io/internal\"\nimport type { Json } from \"atom.io/json\"\n\nimport type { ServerConfig } from \".\"\n\nexport function realtimeStateSynchronizer({\n\tsocket,\n\tstore = IMPLICIT.STORE,\n}: ServerConfig) {\n\treturn function stateSynchronizer<J extends Json.Serializable>(\n\t\ttoken: AtomIO.WritableToken<J>,\n\t): () => void {\n\t\tconst fillGetRequest = () => {\n\t\t\tsocket.emit(`value:${token.key}`, getFromStore(token, store))\n\t\t}\n\n\t\tsocket.on(`get:${token.key}`, fillGetRequest)\n\t\treturn () => {\n\t\t\tsocket.off(`get:${token.key}`, fillGetRequest)\n\t\t}\n\t}\n}\n","import type * as AtomIO from \"atom.io\"\nimport {\n\tIMPLICIT,\n\tfindInStore,\n\tgetFromStore,\n\tsubscribeToState,\n} from \"atom.io/internal\"\nimport { type Json, stringifyJson } from \"atom.io/json\"\n\nimport type { ServerConfig } from \".\"\n\nexport type FamilyProvider = ReturnType<typeof realtimeAtomFamilyProvider>\nexport function realtimeAtomFamilyProvider({\n\tsocket,\n\tstore = IMPLICIT.STORE,\n}: ServerConfig) {\n\treturn function familyProvider<\n\t\tJ extends Json.Serializable,\n\t\tK extends Json.Serializable,\n\t>(\n\t\tfamily: AtomIO.RegularAtomFamilyToken<J, K>,\n\t\tindex: AtomIO.ReadableToken<Iterable<K>>,\n\t): () => void {\n\t\tconst unsubCallbacksByKey = new Map<string, () => void>()\n\n\t\tconst fillUnsubRequest = (key: string) => {\n\t\t\tsocket.off(`unsub:${key}`, fillUnsubRequest)\n\t\t\tconst unsub = unsubCallbacksByKey.get(key)\n\t\t\tif (unsub) {\n\t\t\t\tunsub()\n\t\t\t\tunsubCallbacksByKey.delete(key)\n\t\t\t}\n\t\t}\n\n\t\tconst fillSubRequest = (subKey: K) => {\n\t\t\tconst exposedSubKeys = getFromStore(index, store)\n\t\t\tfor (const exposedSubKey of exposedSubKeys) {\n\t\t\t\tif (stringifyJson(exposedSubKey) === stringifyJson(subKey)) {\n\t\t\t\t\tconst token = findInStore(family, subKey, store)\n\t\t\t\t\tsocket.emit(`serve:${token.key}`, getFromStore(token, store))\n\t\t\t\t\tconst unsubscribe = subscribeToState(\n\t\t\t\t\t\ttoken,\n\t\t\t\t\t\t({ newValue }) => {\n\t\t\t\t\t\t\tsocket.emit(`serve:${token.key}`, newValue)\n\t\t\t\t\t\t},\n\t\t\t\t\t\t`expose-family:${family.key}:${socket.id}`,\n\t\t\t\t\t\tstore,\n\t\t\t\t\t)\n\t\t\t\t\tunsubCallbacksByKey.set(token.key, unsubscribe)\n\t\t\t\t\tsocket.on(`unsub:${token.key}`, () => {\n\t\t\t\t\t\tfillUnsubRequest(token.key)\n\t\t\t\t\t})\n\t\t\t\t\tbreak\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tsocket.on(`sub:${family.key}`, fillSubRequest)\n\n\t\treturn () => {\n\t\t\tsocket.off(`sub:${family.key}`, fillSubRequest)\n\n\t\t\tfor (const [, unsub] of unsubCallbacksByKey) {\n\t\t\t\tunsub()\n\t\t\t}\n\t\t\tunsubCallbacksByKey.clear()\n\t\t}\n\t}\n}\n","import type * as AtomIO from \"atom.io\"\nimport {\n\tIMPLICIT,\n\tgetFromStore,\n\tgetJsonToken,\n\tgetUpdateToken,\n\tsubscribeToState,\n} from \"atom.io/internal\"\nimport type { Transceiver } from \"atom.io/internal\"\nimport type { Json } from \"atom.io/json\"\n\nimport type { ServerConfig } from \".\"\n\nexport type MutableProvider = ReturnType<typeof realtimeMutableProvider>\nexport function realtimeMutableProvider({\n\tsocket,\n\tstore = IMPLICIT.STORE,\n}: ServerConfig) {\n\treturn function mutableProvider<\n\t\tCore extends Transceiver<Json.Serializable>,\n\t\tSerializableCore extends Json.Serializable,\n\t>(token: AtomIO.MutableAtomToken<Core, SerializableCore>): () => void {\n\t\tlet unsubscribeFromStateUpdates: (() => void) | null = null\n\n\t\tconst jsonToken = getJsonToken(token)\n\t\tconst trackerToken = getUpdateToken(token)\n\n\t\tconst fillUnsubRequest = () => {\n\t\t\tsocket.off(`unsub:${token.key}`, fillUnsubRequest)\n\t\t\tunsubscribeFromStateUpdates?.()\n\t\t\tunsubscribeFromStateUpdates = null\n\t\t}\n\n\t\tconst fillSubRequest = () => {\n\t\t\tsocket.emit(`init:${token.key}`, getFromStore(jsonToken, store))\n\t\t\tunsubscribeFromStateUpdates = subscribeToState(\n\t\t\t\ttrackerToken,\n\t\t\t\t({ newValue }) => {\n\t\t\t\t\tsocket.emit(`next:${token.key}`, newValue)\n\t\t\t\t},\n\t\t\t\t`expose-single:${socket.id}`,\n\t\t\t\tstore,\n\t\t\t)\n\t\t\tsocket.on(`unsub:${token.key}`, fillUnsubRequest)\n\t\t}\n\n\t\tsocket.on(`sub:${token.key}`, fillSubRequest)\n\n\t\treturn () => {\n\t\t\tsocket.off(`sub:${token.key}`, fillSubRequest)\n\t\t\tunsubscribeFromStateUpdates?.()\n\t\t}\n\t}\n}\n","import type * as AtomIO from \"atom.io\"\nimport type { Transceiver } from \"atom.io/internal\"\nimport {\n\tIMPLICIT,\n\tfindInStore,\n\tgetFromStore,\n\tgetJsonToken,\n\tgetUpdateToken,\n\tsubscribeToState,\n} from \"atom.io/internal\"\nimport type { Json } from \"atom.io/json\"\nimport { stringifyJson } from \"atom.io/json\"\n\nimport type { ServerConfig } from \".\"\n\nexport type MutableFamilyProvider = ReturnType<\n\ttypeof realtimeMutableFamilyProvider\n>\nexport function realtimeMutableFamilyProvider({\n\tsocket,\n\tstore = IMPLICIT.STORE,\n}: ServerConfig) {\n\treturn function mutableFamilyProvider<\n\t\tT extends Transceiver<any>,\n\t\tJ extends Json.Serializable,\n\t\tK extends Json.Serializable,\n\t>(\n\t\tfamily: AtomIO.MutableAtomFamilyToken<T, J, K>,\n\t\tindex: AtomIO.ReadableToken<Iterable<K>>,\n\t): () => void {\n\t\tconst unsubCallbacksByKey = new Map<string, () => void>()\n\n\t\tconst fillUnsubRequest = (key: string) => {\n\t\t\tsocket.off(`unsub:${key}`, fillUnsubRequest)\n\t\t\tconst unsub = unsubCallbacksByKey.get(key)\n\t\t\tif (unsub) {\n\t\t\t\tunsub()\n\t\t\t\tunsubCallbacksByKey.delete(key)\n\t\t\t}\n\t\t}\n\n\t\tconst fillSubRequest = (subKey: K) => {\n\t\t\tconst exposedSubKeys = getFromStore(index, store)\n\t\t\tfor (const exposedSubKey of exposedSubKeys) {\n\t\t\t\tif (stringifyJson(exposedSubKey) === stringifyJson(subKey)) {\n\t\t\t\t\tconst token = findInStore(family, subKey, store)\n\t\t\t\t\tconst jsonToken = getJsonToken(token)\n\t\t\t\t\tconst updateToken = getUpdateToken(token)\n\t\t\t\t\tsocket.emit(`init:${token.key}`, getFromStore(jsonToken, store))\n\t\t\t\t\tconst unsubscribe = subscribeToState(\n\t\t\t\t\t\tupdateToken,\n\t\t\t\t\t\t({ newValue }) => {\n\t\t\t\t\t\t\tsocket.emit(`next:${token.key}`, newValue)\n\t\t\t\t\t\t},\n\t\t\t\t\t\t`expose-family:${family.key}:${socket.id}`,\n\t\t\t\t\t\tstore,\n\t\t\t\t\t)\n\t\t\t\t\tunsubCallbacksByKey.set(token.key, unsubscribe)\n\t\t\t\t\tsocket.on(`unsub:${token.key}`, () => {\n\t\t\t\t\t\tfillUnsubRequest(token.key)\n\t\t\t\t\t})\n\t\t\t\t\tbreak\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tsocket.on(`sub:${family.key}`, fillSubRequest)\n\n\t\treturn () => {\n\t\t\tsocket.off(`sub:${family.key}`, fillSubRequest)\n\t\t\tfor (const [, unsub] of unsubCallbacksByKey) {\n\t\t\t\tunsub()\n\t\t\t}\n\t\t\tunsubCallbacksByKey.clear()\n\t\t}\n\t}\n}\n","import type { WritableToken } from \"atom.io\"\nimport { IMPLICIT, setIntoStore } from \"atom.io/internal\"\nimport type { Json } from \"atom.io/json\"\n\nimport type { ServerConfig } from \".\"\n\nexport type StateReceiver = ReturnType<typeof realtimeStateReceiver>\nexport function realtimeStateReceiver({\n\tsocket,\n\tstore = IMPLICIT.STORE,\n}: ServerConfig) {\n\treturn function stateReceiver<J extends Json.Serializable>(\n\t\ttoken: WritableToken<J>,\n\t): () => void {\n\t\tconst publish = (newValue: J) => setIntoStore(token, newValue, store)\n\n\t\tconst fillPubUnclaim = () => {\n\t\t\tsocket.off(`pub:${token.key}`, publish)\n\t\t\tsocket.off(`unclaim:${token.key}`, fillPubUnclaim)\n\t\t}\n\t\tconst fillPubClaim = () => {\n\t\t\tsocket.on(`pub:${token.key}`, publish)\n\t\t\tsocket.on(`unclaim:${token.key}`, fillPubUnclaim)\n\t\t}\n\n\t\tsocket.on(`claim:${token.key}`, fillPubClaim)\n\n\t\treturn () => {\n\t\t\tsocket.off(`claim:${token.key}`, fillPubClaim)\n\t\t\tsocket.off(`pub:${token.key}`, publish)\n\t\t}\n\t}\n}\n","import type * as AtomIO from \"atom.io\"\nimport { IMPLICIT, actUponStore } from \"atom.io/internal\"\nimport type { JsonIO } from \"atom.io/json\"\n\nimport type { ServerConfig } from \".\"\n\nexport type ActionReceiver = ReturnType<typeof realtimeActionReceiver>\nexport function realtimeActionReceiver({\n\tsocket,\n\tstore = IMPLICIT.STORE,\n}: ServerConfig) {\n\treturn function actionReceiver<ƒ extends JsonIO>(\n\t\ttx: AtomIO.TransactionToken<ƒ>,\n\t): () => void {\n\t\tconst fillTransactionRequest = (\n\t\t\tupdate: Pick<AtomIO.TransactionUpdate<ƒ>, `id` | `params`>,\n\t\t) => {\n\t\t\tconst performanceKey = `tx-run:${tx.key}:${update.id}`\n\t\t\tconst performanceKeyStart = `${performanceKey}:start`\n\t\t\tconst performanceKeyEnd = `${performanceKey}:end`\n\t\t\tperformance.mark(performanceKeyStart)\n\t\t\tactUponStore<ƒ>(tx, update.id, store)(...update.params)\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(`🚀`, `transaction`, tx.key, update.id, metric.duration)\n\t\t}\n\t\tsocket.on(`tx-run:${tx.key}`, fillTransactionRequest)\n\n\t\treturn () => socket.off(`tx-run:${tx.key}`, fillTransactionRequest)\n\t}\n}\n","import type * as AtomIO from \"atom.io\"\nimport {\n\tIMPLICIT,\n\tactUponStore,\n\tassignTransactionToContinuity,\n\tfindInStore,\n\tgetFromStore,\n\tsetIntoStore,\n\tsubscribeToTransaction,\n} from \"atom.io/internal\"\nimport type { Json, JsonIO } from \"atom.io/json\"\n\nimport type { ServerConfig } from \".\"\nimport {\n\tcompleteUpdateAtoms,\n\tredactedUpdateSelectors,\n\ttransactionRedactorAtoms,\n\tuserUnacknowledgedQueues,\n\tusersOfSockets,\n} from \"./realtime-server-stores\"\n\nexport type ActionSynchronizer = ReturnType<typeof realtimeActionSynchronizer>\nexport function realtimeActionSynchronizer({\n\tsocket,\n\tstore = IMPLICIT.STORE,\n}: ServerConfig) {\n\treturn function actionSynchronizer<ƒ extends JsonIO>(\n\t\ttx: AtomIO.TransactionToken<ƒ>,\n\t\tfilter?: (\n\t\t\tupdate: AtomIO.TransactionUpdateContent[],\n\t\t) => AtomIO.TransactionUpdateContent[],\n\t): () => void {\n\t\tassignTransactionToContinuity(`default`, tx.key, store)\n\n\t\tconst userKeyState = findInStore(\n\t\t\tusersOfSockets.states.userKeyOfSocket,\n\t\t\tsocket.id,\n\t\t\tstore,\n\t\t)\n\t\tconst userKey = getFromStore(userKeyState, store)\n\t\tif (!userKey) {\n\t\t\tstore.logger.error(\n\t\t\t\t`❌`,\n\t\t\t\t`transaction`,\n\t\t\t\ttx.key,\n\t\t\t\t`Tried to create a synchronizer for a socket (${socket.id}) that is not connected to a user.`,\n\t\t\t)\n\t\t\treturn () => {}\n\t\t}\n\t\tconst userUnacknowledgedQueue = findInStore(\n\t\t\tuserUnacknowledgedQueues,\n\t\t\tuserKey,\n\t\t\tstore,\n\t\t)\n\t\tconst userUnacknowledgedUpdates = getFromStore(\n\t\t\tuserUnacknowledgedQueue,\n\t\t\tstore,\n\t\t)\n\t\tif (filter) {\n\t\t\tconst redactorState = findInStore(transactionRedactorAtoms, tx.key, store)\n\t\t\tsetIntoStore(redactorState, { filter }, store)\n\t\t}\n\n\t\tconst fillTransactionRequest = (\n\t\t\tupdate: Pick<AtomIO.TransactionUpdate<ƒ>, `id` | `params`>,\n\t\t) => {\n\t\t\tconst performanceKey = `tx-run:${tx.key}:${update.id}`\n\t\t\tconst performanceKeyStart = `${performanceKey}:start`\n\t\t\tconst performanceKeyEnd = `${performanceKey}:end`\n\t\t\tperformance.mark(performanceKeyStart)\n\t\t\tactUponStore<ƒ>(tx, update.id, store)(...update.params)\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(`🚀`, `transaction`, tx.key, update.id, metric.duration)\n\t\t}\n\t\tsocket.off(`tx-run:${tx.key}`, fillTransactionRequest)\n\t\tsocket.on(`tx-run:${tx.key}`, fillTransactionRequest)\n\n\t\tlet unsubscribeFromTransaction: (() => void) | undefined\n\t\tconst fillTransactionSubscriptionRequest = () => {\n\t\t\tunsubscribeFromTransaction = subscribeToTransaction(\n\t\t\t\ttx,\n\t\t\t\t(update) => {\n\t\t\t\t\tconst updateState = findInStore(completeUpdateAtoms, update.id, store)\n\t\t\t\t\tsetIntoStore(updateState, update, store)\n\t\t\t\t\tconst toEmit: Pick<\n\t\t\t\t\t\tAtomIO.TransactionUpdate<ƒ>,\n\t\t\t\t\t\t`epoch` | `id` | `key` | `output` | `updates`\n\t\t\t\t\t> | null = filter\n\t\t\t\t\t\t? getFromStore(\n\t\t\t\t\t\t\t\tfindInStore(redactedUpdateSelectors, [tx.key, update.id], store),\n\t\t\t\t\t\t\t\tstore,\n\t\t\t\t\t\t )\n\t\t\t\t\t\t: update\n\n\t\t\t\t\t// the problem is that only while a socket is connected can\n\t\t\t\t\t// updates be set in the queue for that socket's client.\n\t\t\t\t\t//\n\t\t\t\t\t// we need a client session that can persist between disconnects\n\t\t\t\t\tsetIntoStore(\n\t\t\t\t\t\tuserUnacknowledgedQueue,\n\t\t\t\t\t\t(updates) => {\n\t\t\t\t\t\t\tif (toEmit) {\n\t\t\t\t\t\t\t\tupdates.push(toEmit)\n\t\t\t\t\t\t\t\tupdates.sort((a, b) => a.epoch - b.epoch)\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\treturn updates\n\t\t\t\t\t\t},\n\t\t\t\t\t\tstore,\n\t\t\t\t\t)\n\n\t\t\t\t\tsocket.emit(`tx-new:${tx.key}`, toEmit as Json.Serializable)\n\t\t\t\t},\n\t\t\t\t`tx-sub:${tx.key}:${socket.id}`,\n\t\t\t\tstore,\n\t\t\t)\n\t\t\tsocket.on(`tx-unsub:${tx.key}`, unsubscribeFromTransaction)\n\t\t}\n\t\tsocket.on(`tx-sub:${tx.key}`, fillTransactionSubscriptionRequest)\n\n\t\tlet i = 1\n\t\tlet next = 1\n\t\tconst retry = setInterval(() => {\n\t\t\tconst toEmit = userUnacknowledgedUpdates[0]\n\t\t\tif (toEmit && i === next) {\n\t\t\t\tsocket.emit(`tx-new:${tx.key}`, toEmit as Json.Serializable)\n\t\t\t\tnext *= 2\n\t\t\t}\n\n\t\t\ti++\n\t\t}, 250)\n\n\t\tconst trackClientAcknowledgement = (epoch: number) => {\n\t\t\ti = 1\n\t\t\tnext = 1\n\t\t\t8\n\t\t\tif (userUnacknowledgedUpdates[0]?.epoch === epoch) {\n\t\t\t\tsetIntoStore(\n\t\t\t\t\tuserUnacknowledgedQueue,\n\t\t\t\t\t(updates) => {\n\t\t\t\t\t\tupdates.shift()\n\t\t\t\t\t\treturn updates\n\t\t\t\t\t},\n\t\t\t\t\tstore,\n\t\t\t\t)\n\t\t\t}\n\t\t}\n\t\tsocket.on(`tx-ack:${tx.key}`, trackClientAcknowledgement)\n\n\t\treturn () => {\n\t\t\tif (unsubscribeFromTransaction) {\n\t\t\t\tunsubscribeFromTransaction()\n\t\t\t\tunsubscribeFromTransaction = undefined\n\t\t\t}\n\t\t\tclearInterval(retry)\n\t\t\tsocket.off(`tx-run:${tx.key}`, fillTransactionRequest)\n\t\t\tsocket.off(`tx-sub:${tx.key}`, fillTransactionSubscriptionRequest)\n\t\t}\n\t}\n}\n"]}
1
+ {"version":3,"sources":["../src/ipc-sockets/child-socket.ts","../src/ipc-sockets/custom-socket.ts","../src/ipc-sockets/parent-socket.ts","../src/realtime-continuity-synchronizer.ts","../src/realtime-server-stores/realtime-continuity-store.ts","../src/realtime-server-stores/server-room-external-actions.ts","../src/realtime-server-stores/server-room-external-store.ts","../src/realtime-server-stores/server-sync-store.ts","../src/realtime-server-stores/server-user-store.ts","../src/realtime-state-provider.ts","../src/realtime-state-synchronizer.ts","../src/realtime-family-provider.ts","../src/realtime-mutable-provider.ts","../src/realtime-mutable-family-provider.ts","../src/realtime-state-receiver.ts","../src/realtime-action-receiver.ts"],"names":["process","parsedEvent","parsedLog","parseJson","IMPLICIT","getJsonToken","selectorFamily","atomFamily","SetRTX","unsubscribeFunctions","atom","transaction","getFromStore","subscribeToState","findInStore","stringifyJson","getUpdateToken","actUponStore"],"mappings":"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","sourcesContent":["import type { ChildProcessWithoutNullStreams } from \"child_process\"\n\nimport type { Json } from \"atom.io/json\"\nimport { parseJson } from \"atom.io/json\"\n\nimport type { EventBuffer, Events } from \"./custom-socket\"\nimport { CustomSocket } from \"./custom-socket\"\n\nexport class ChildSocket<\n\tI extends Events,\n\tO extends Events & {\n\t\t/* eslint-disable quotes */\n\t\t\"setup-relay\": [string]\n\t\t/* eslint-enable quotes */\n\t},\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\tprotected handleLog(arg: Json.Serializable): void {\n\t\tif (Array.isArray(arg)) {\n\t\t\tconst [level, ...rest] = arg\n\t\t\tswitch (level) {\n\t\t\t\tcase `i`:\n\t\t\t\t\tthis.logger.info(this.id, this.key, ...rest)\n\t\t\t\t\tbreak\n\t\t\t\tcase `w`:\n\t\t\t\t\tthis.logger.warn(this.id, this.key, ...rest)\n\t\t\t\t\tbreak\n\t\t\t\tcase `e`:\n\t\t\t\t\tthis.logger.error(this.id, this.key, ...rest)\n\t\t\t\t\tbreak\n\t\t\t}\n\t\t}\n\t}\n\n\tpublic constructor(\n\t\tpublic process: ChildProcessWithoutNullStreams,\n\t\tpublic key: string,\n\t\tpublic logger: {\n\t\t\tinfo: (prefix: string, message: string, ...args: unknown[]) => void\n\t\t\twarn: (prefix: string, message: string, ...args: unknown[]) => void\n\t\t\terror: (prefix: string, message: string, ...args: unknown[]) => void\n\t\t} = console,\n\t) {\n\t\tsuper((event, ...args) => {\n\t\t\tconst stringifiedEvent = JSON.stringify([event, ...args]) + `\\x03`\n\t\t\tconst errorHandler = (err) => {\n\t\t\t\tif (err.code === `EPIPE`) {\n\t\t\t\t\tconsole.error(`EPIPE error during write`, this.process.stdin)\n\t\t\t\t}\n\t\t\t\tthis.process.stdin.removeListener(`error`, errorHandler)\n\t\t\t}\n\n\t\t\tthis.process.stdin.once(`error`, errorHandler)\n\t\t\tthis.process.stdin.write(stringifiedEvent)\n\n\t\t\treturn this\n\t\t})\n\t\tthis.process = process\n\t\tthis.process.stdout.on(\n\t\t\t`data`,\n\t\t\t<Event extends keyof I>(buffer: EventBuffer<string, I[Event]>) => {\n\t\t\t\tconst chunk = buffer.toString()\n\n\t\t\t\tif (chunk === `✨`) {\n\t\t\t\t\t// console.log(chunk)\n\t\t\t\t\treturn\n\t\t\t\t}\n\t\t\t\tthis.unprocessedEvents.push(...chunk.split(`\\x03`))\n\t\t\t\t// console.log(`🤓`, chunk.length)\n\t\t\t\t// console.log(`🤓`, this.unprocessedEvents.length)\n\t\t\t\t// console.log(`🤓`, ...this.unprocessedEvents.map((x) => x.length))\n\t\t\t\tconst newInput = this.unprocessedEvents.shift()\n\t\t\t\tthis.incompleteData += newInput || ``\n\t\t\t\ttry {\n\t\t\t\t\tif (this.incompleteData.startsWith(`error`)) {\n\t\t\t\t\t\tconsole.log(`❗`, this.incompleteData)\n\t\t\t\t\t}\n\t\t\t\t\tconst parsedEvent = parseJson(this.incompleteData)\n\t\t\t\t\tthis.handleEvent(...(parsedEvent as [string, ...I[keyof I]]))\n\t\t\t\t\twhile (this.unprocessedEvents.length > 0) {\n\t\t\t\t\t\tconst event = this.unprocessedEvents.shift()\n\t\t\t\t\t\tif (event) {\n\t\t\t\t\t\t\tif (this.unprocessedEvents.length === 0) {\n\t\t\t\t\t\t\t\tthis.incompleteData = event\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\tconst parsedEvent = parseJson(event)\n\t\t\t\t\t\t\tthis.handleEvent(...(parsedEvent as [string, ...I[keyof I]]))\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t\tthis.incompleteData = ``\n\t\t\t\t} catch (error) {\n\t\t\t\t\tconsole.warn(`⚠️----------------⚠️`)\n\t\t\t\t\tconsole.warn(this.incompleteData)\n\t\t\t\t\tconsole.warn(`⚠️----------------⚠️`)\n\t\t\t\t\tconsole.error(error)\n\t\t\t\t}\n\t\t\t},\n\t\t)\n\t\tthis.process.stderr.on(`data`, (buf) => {\n\t\t\tconst chunk = buf.toString()\n\t\t\tthis.unprocessedLogs.push(...chunk.split(`\\x03`))\n\t\t\t// console.log(`🤫`, chunk.length)\n\t\t\t// console.log(`🤫`, this.unprocessedLogs.length)\n\t\t\t// console.log(`🤫`, ...this.unprocessedLogs.map((x) => x.length))\n\t\t\tconst newInput = this.unprocessedLogs.shift()\n\t\t\tthis.incompleteLog += newInput || ``\n\t\t\ttry {\n\t\t\t\tconst parsedLog = parseJson(this.incompleteLog)\n\t\t\t\t// console.log(`🤫`, parsedLog)\n\t\t\t\tthis.handleLog(parsedLog)\n\t\t\t\twhile (this.unprocessedLogs.length > 0) {\n\t\t\t\t\tthis.incompleteLog = this.unprocessedLogs.shift() ?? ``\n\t\t\t\t\tif (this.incompleteLog) {\n\t\t\t\t\t\tconst parsedLog = parseJson(this.incompleteLog)\n\t\t\t\t\t\tthis.handleLog(parsedLog)\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t} catch (error) {\n\t\t\t\tconsole.error(`❌❌❌`)\n\t\t\t\tconsole.error(this.incompleteLog)\n\t\t\t\tconsole.error(error)\n\t\t\t\tconsole.error(`❌❌❌️`)\n\t\t\t}\n\t\t})\n\t\tif (process.pid) {\n\t\t\tthis.id = process.pid.toString()\n\t\t}\n\t}\n}\n","import type { Json, Stringified } from \"atom.io/json\"\n\nimport type { Socket } from \"..\"\n\nexport type Events = Json.Object<string, Json.Serializable[]>\n\nexport type StringifiedEvent<\n\tKey extends string,\n\tParams extends Json.Serializable[],\n> = Stringified<[Key, ...Params]>\n\nexport interface EventBuffer<\n\tKey extends string,\n\tParams extends Json.Serializable[],\n> extends Buffer {\n\ttoString(): StringifiedEvent<Key, Params>\n}\n\nexport class CustomSocket<I extends Events, O extends Events> implements Socket {\n\tprotected listeners: Map<keyof I, Set<(...args: Json.Array) => void>>\n\tprotected globalListeners: Set<(event: string, ...args: Json.Array) => void>\n\tprotected handleEvent<Event extends keyof I>(\n\t\tevent: string,\n\t\t...args: I[Event]\n\t): void {\n\t\tfor (const listener of this.globalListeners) {\n\t\t\tlistener(event, ...args)\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(...args)\n\t\t\t}\n\t\t}\n\t}\n\n\tpublic id = `no_id_retrieved`\n\n\tpublic constructor(\n\t\tpublic emit: <Event extends keyof O>(\n\t\t\tevent: Event,\n\t\t\t...args: O[Event]\n\t\t) => CustomSocket<I, O>,\n\t) {\n\t\tthis.listeners = new Map()\n\t\tthis.globalListeners = new Set()\n\t}\n\n\tpublic on<Event extends keyof I>(\n\t\tevent: Event,\n\t\tlistener: (...args: I[Event]) => void,\n\t): CustomSocket<I, O> {\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(\n\t\tlistener: (event: string, ...args: Json.Array) => void,\n\t): CustomSocket<I, O> {\n\t\tthis.globalListeners.add(listener)\n\t\treturn this\n\t}\n\n\tpublic off<Event extends keyof I>(\n\t\tevent: Event,\n\t\tlistener?: (...args: I[Event]) => void,\n\t): CustomSocket<I, O> {\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(\n\t\tlistener: (event: string, ...args: Json.Array) => void,\n\t): CustomSocket<I, O> {\n\t\tthis.globalListeners.delete(listener)\n\t\treturn this\n\t}\n}\n","import { IMPLICIT, Subject } from \"atom.io/internal\"\nimport { parseJson, stringifyJson } from \"atom.io/json\"\nimport type { Json } from \"atom.io/json\"\n\nimport { SetRTX } from \"atom.io/transceivers/set-rtx\"\nimport { CustomSocket } from \"./custom-socket\"\nimport type { EventBuffer, Events } from \"./custom-socket\"\n\nexport class SubjectSocket<\n\tI extends Events,\n\tO extends Events,\n> extends CustomSocket<I, O> {\n\tpublic in: Subject<[string, ...Json.Serializable[]]>\n\tpublic out: Subject<[string, ...Json.Serializable[]]>\n\tpublic id = `no_id_retrieved`\n\tpublic disposalFunctions: (() => 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 as [string, ...I[keyof I]]))\n\t\t})\n\t}\n\n\tpublic dispose(): void {\n\t\tfor (const dispose of this.disposalFunctions) {\n\t\t\tdispose()\n\t\t}\n\t}\n}\n\nexport class ParentSocket<\n\tI extends Events & {\n\t\t[id in string as `user:${id}`]: [string, ...Json.Serializable[]]\n\t} & {\n\t\t/* eslint-disable quotes */\n\t\t\"user-joins\": [string]\n\t\t\"user-leaves\": [string]\n\t\t/* eslint-enable quotes */\n\t},\n\tO extends Events & {\n\t\t[id in string as `relay:${id}`]: [string, ...Json.Serializable[]]\n\t},\n> extends CustomSocket<I, O> {\n\tprotected incompleteData = ``\n\tprotected unprocessedEvents: string[] = []\n\tprotected relays: Map<string, SubjectSocket<any, any>>\n\tprotected relayServices: ((\n\t\tsocket: SubjectSocket<any, any>,\n\t) => (() => void) | void)[]\n\tprotected process: NodeJS.Process\n\n\tpublic id = `#####`\n\n\tprotected log(...args: any[]): void {\n\t\tthis.process.stderr.write(\n\t\t\tstringifyJson(\n\t\t\t\targs.map((arg) =>\n\t\t\t\t\targ instanceof SetRTX\n\t\t\t\t\t\t? `{ ${arg.toJSON().members.join(` | `)} }`\n\t\t\t\t\t\t: arg,\n\t\t\t\t),\n\t\t\t) + `\\x03`,\n\t\t)\n\t}\n\tpublic logger = {\n\t\tinfo: (...args: any[]): void => this.log(`i`, ...args),\n\t\twarn: (...args: any[]): void => this.log(`w`, ...args),\n\t\terror: (...args: any[]): void => this.log(`e`, ...args),\n\t}\n\n\tpublic constructor() {\n\t\tsuper((event, ...args) => {\n\t\t\tconst stringifiedEvent = JSON.stringify([event, ...args])\n\t\t\tthis.process.stdout.write(stringifiedEvent + `\\x03`)\n\t\t\treturn this\n\t\t})\n\t\tthis.process = process\n\t\tthis.process.stdin.resume()\n\t\tthis.relays = new Map()\n\t\tthis.relayServices = []\n\n\t\tthis.process.stdin.on(\n\t\t\t`data`,\n\t\t\t<Event extends keyof I>(buffer: EventBuffer<string, I[Event]>) => {\n\t\t\t\tconst chunk = buffer.toString()\n\t\t\t\tthis.unprocessedEvents.push(...chunk.split(`\\x03`))\n\t\t\t\tconst newInput = this.unprocessedEvents.shift()\n\t\t\t\tthis.incompleteData += newInput || ``\n\n\t\t\t\ttry {\n\t\t\t\t\tconst parsedEvent = parseJson(this.incompleteData)\n\t\t\t\t\tthis.logger.info(`🎰`, `received`, parsedEvent)\n\t\t\t\t\tthis.handleEvent(...(parsedEvent as [string, ...I[keyof I]]))\n\t\t\t\t\twhile (this.unprocessedEvents.length > 0) {\n\t\t\t\t\t\tconst event = this.unprocessedEvents.shift()\n\t\t\t\t\t\tif (event) {\n\t\t\t\t\t\t\tif (this.unprocessedEvents.length === 0) {\n\t\t\t\t\t\t\t\tthis.incompleteData = event\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\tconst parsedEvent = parseJson(event)\n\t\t\t\t\t\t\tthis.handleEvent(...(parsedEvent as [string, ...I[keyof I]]))\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t\tthis.incompleteData = ``\n\t\t\t\t} catch (thrown) {\n\t\t\t\t\tif (thrown instanceof Error) {\n\t\t\t\t\t\tthis.logger.error(`❗`, thrown.message, thrown.cause, thrown.stack)\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t},\n\t\t)\n\n\t\tthis.on(`exit`, () => {\n\t\t\tprocess.exit(0)\n\t\t})\n\t\tprocess.on(`exit`, () => {\n\t\t\tthis.logger.info(`🔥`, this.id, `exited`)\n\t\t\tprocess.exit(0)\n\t\t})\n\t\tprocess.on(`end`, () => {\n\t\t\tthis.logger.info(`🔥`, this.id, `ended`)\n\t\t\tprocess.exit(0)\n\t\t})\n\t\tprocess.on(`SIGTERM`, () => {\n\t\t\tthis.logger.error(`🔥`, this.id, `terminated`)\n\t\t\tprocess.exit(0)\n\t\t})\n\t\tprocess.on(`SIGINT`, () => {\n\t\t\tthis.logger.error(`🔥`, this.id, `interrupted`)\n\t\t\tprocess.exit(0)\n\t\t})\n\n\t\tif (process.pid) {\n\t\t\tthis.id = process.pid?.toString()\n\t\t}\n\n\t\tthis.on(`user-joins`, (username) => {\n\t\t\tthis.logger.info(`👤`, `user`, username, `joined`)\n\t\t\tconst relay = new SubjectSocket(`user:${username}`)\n\t\t\tthis.relays.set(username, relay)\n\t\t\tthis.logger.info(\n\t\t\t\t`🔗`,\n\t\t\t\t`attaching services:`,\n\t\t\t\t`[${[...this.relayServices.keys()].join(`, `)}]`,\n\t\t\t)\n\t\t\tfor (const attachServices of this.relayServices) {\n\t\t\t\tconst cleanup = attachServices(relay)\n\t\t\t\tif (cleanup) {\n\t\t\t\t\trelay.disposalFunctions.push(cleanup)\n\t\t\t\t}\n\t\t\t}\n\t\t\tthis.on(`user:${username}`, (...data) => {\n\t\t\t\trelay.in.next(data)\n\t\t\t})\n\t\t\trelay.out.subscribe(`socket`, (data) => {\n\t\t\t\tthis.emit(...(data as [string, ...O[keyof O]]))\n\t\t\t})\n\t\t})\n\n\t\tthis.on(`user-leaves`, (username) => {\n\t\t\tconst relay = this.relays.get(username)\n\t\t\tthis.off(`relay:${username}`)\n\t\t\tif (relay) {\n\t\t\t\trelay.dispose()\n\t\t\t\tthis.relays.delete(username)\n\t\t\t}\n\t\t})\n\n\t\tprocess.stdout.write(`✨`)\n\t}\n\n\tpublic relay(\n\t\tattachServices: (socket: SubjectSocket<any, any>) => (() => void) | void,\n\t): void {\n\t\tthis.logger.info(`🔗`, `running relay method`)\n\t\tthis.relayServices.push(attachServices)\n\t}\n}\n","import type * as AtomIO from \"atom.io\"\nimport {\n\tIMPLICIT,\n\tactUponStore,\n\tfindInStore,\n\tgetFromStore,\n\tgetJsonToken,\n\tisRootStore,\n\tsubscribeToState,\n\tsubscribeToTransaction,\n} from \"atom.io/internal\"\nimport type { Json, JsonIO } from \"atom.io/json\"\nimport type { ContinuityToken } from \"atom.io/realtime\"\n\nimport type { ServerConfig, Socket } from \".\"\nimport { socketAtoms, usersOfSockets } from \".\"\n\nimport {\n\tredactTransactionUpdateContent,\n\tuserUnacknowledgedQueues,\n} from \"./realtime-server-stores\"\n\nexport type RealtimeContinuitySynchronizer = ReturnType<\n\ttypeof realtimeContinuitySynchronizer\n>\nexport function realtimeContinuitySynchronizer({\n\tsocket: initialSocket,\n\tstore = IMPLICIT.STORE,\n}: ServerConfig) {\n\treturn function synchronizer(continuity: ContinuityToken): () => void {\n\t\tlet socket: Socket | null = initialSocket\n\n\t\tconst continuityKey = continuity.key\n\t\tconst userKeyState = findInStore(\n\t\t\tusersOfSockets.states.userKeyOfSocket,\n\t\t\tsocket.id,\n\t\t\tstore,\n\t\t)\n\t\tconst userKey = getFromStore(userKeyState, store)\n\t\tif (!userKey) {\n\t\t\tstore.logger.error(\n\t\t\t\t`❌`,\n\t\t\t\t`continuity`,\n\t\t\t\tcontinuityKey,\n\t\t\t\t`Tried to create a synchronizer for a socket (${socket.id}) that is not connected to a user.`,\n\t\t\t)\n\t\t\treturn () => {}\n\t\t}\n\t\tconst socketKeyState = findInStore(\n\t\t\tusersOfSockets.states.socketKeyOfUser,\n\t\t\tuserKey,\n\t\t\tstore,\n\t\t)\n\t\tsubscribeToState(\n\t\t\tsocketKeyState,\n\t\t\t({ newValue: newSocketKey }) => {\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`seeing ${userKey} on new socket ${newSocketKey}`,\n\t\t\t\t)\n\t\t\t\tif (newSocketKey === null) {\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`Tried to create a synchronizer for a user (${userKey}) that is not connected to a socket.`,\n\t\t\t\t\t)\n\t\t\t\t\treturn\n\t\t\t\t}\n\t\t\t\tconst newSocketState = findInStore(socketAtoms, newSocketKey, store)\n\t\t\t\tconst newSocket = getFromStore(newSocketState, store)\n\t\t\t\tsocket = newSocket\n\t\t\t},\n\t\t\t`sync-continuity:${continuityKey}:${userKey}`,\n\t\t\tstore,\n\t\t)\n\n\t\tconst userUnacknowledgedQueue = findInStore(\n\t\t\tuserUnacknowledgedQueues,\n\t\t\tuserKey,\n\t\t\tstore,\n\t\t)\n\t\tconst userUnacknowledgedUpdates = getFromStore(\n\t\t\tuserUnacknowledgedQueue,\n\t\t\tstore,\n\t\t)\n\t\tconst unsubscribeFunctions: (() => void)[] = []\n\n\t\tconst revealPerspectives = (): (() => void) => {\n\t\t\tconst unsubscribeFunctions: (() => void)[] = []\n\t\t\tfor (const perspective of continuity.perspectives) {\n\t\t\t\tconst { viewAtoms } = perspective\n\t\t\t\tconst userViewState = findInStore(viewAtoms, userKey, store)\n\t\t\t\tconst unsubscribe = subscribeToState(\n\t\t\t\t\tuserViewState,\n\t\t\t\t\t({ oldValue, newValue }) => {\n\t\t\t\t\t\tconst oldKeys = oldValue.map((token) => token.key)\n\t\t\t\t\t\tconst newKeys = newValue.map((token) => token.key)\n\t\t\t\t\t\tconst concealed = oldValue.filter(\n\t\t\t\t\t\t\t(token) => !newKeys.includes(token.key),\n\t\t\t\t\t\t)\n\t\t\t\t\t\tconst revealed = newValue\n\t\t\t\t\t\t\t.filter((token) => !oldKeys.includes(token.key))\n\t\t\t\t\t\t\t.flatMap((token) => {\n\t\t\t\t\t\t\t\tconst resourceToken =\n\t\t\t\t\t\t\t\t\ttoken.type === `mutable_atom` ? getJsonToken(token) : token\n\t\t\t\t\t\t\t\tconst resource = getFromStore(resourceToken, store)\n\t\t\t\t\t\t\t\treturn [resourceToken, resource]\n\t\t\t\t\t\t\t})\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`atom`,\n\t\t\t\t\t\t\tperspective.resourceAtoms.key,\n\t\t\t\t\t\t\t`${userKey} has a new perspective`,\n\t\t\t\t\t\t\t{ oldKeys, newKeys, revealed, concealed },\n\t\t\t\t\t\t)\n\t\t\t\t\t\tif (revealed.length > 0) {\n\t\t\t\t\t\t\tsocket?.emit(`reveal:${continuityKey}`, revealed)\n\t\t\t\t\t\t}\n\t\t\t\t\t\tif (concealed.length > 0) {\n\t\t\t\t\t\t\tsocket?.emit(`conceal:${continuityKey}`, concealed)\n\t\t\t\t\t\t}\n\t\t\t\t\t},\n\t\t\t\t\t`sync-continuity:${continuityKey}:${userKey}:perspective:${perspective.resourceAtoms.key}`,\n\t\t\t\t\tstore,\n\t\t\t\t)\n\t\t\t\tunsubscribeFunctions.push(unsubscribe)\n\t\t\t}\n\t\t\treturn () => {\n\t\t\t\tfor (const unsubscribe of unsubscribeFunctions) unsubscribe()\n\t\t\t}\n\t\t}\n\t\tconst unsubscribeFromPerspectives = revealPerspectives()\n\n\t\tconst sendInitialPayload = () => {\n\t\t\tconst initialPayload: Json.Serializable[] = []\n\t\t\tfor (const atom of continuity.globals) {\n\t\t\t\tconst resourceToken =\n\t\t\t\t\tatom.type === `mutable_atom` ? getJsonToken(atom) : atom\n\t\t\t\tinitialPayload.push(resourceToken, getFromStore(atom, store))\n\t\t\t}\n\t\t\tfor (const perspective of continuity.perspectives) {\n\t\t\t\tconst { viewAtoms, resourceAtoms } = perspective\n\t\t\t\tconst userViewState = findInStore(viewAtoms, userKey, store)\n\t\t\t\tconst userView = getFromStore(userViewState, store)\n\t\t\t\tstore.logger.info(`👁`, `atom`, resourceAtoms.key, `${userKey} can see`, {\n\t\t\t\t\tviewAtoms,\n\t\t\t\t\tresourceAtoms,\n\t\t\t\t\tuserView,\n\t\t\t\t})\n\t\t\t\tfor (const visibleToken of userView) {\n\t\t\t\t\tconst resourceToken =\n\t\t\t\t\t\tvisibleToken.type === `mutable_atom`\n\t\t\t\t\t\t\t? getJsonToken(visibleToken)\n\t\t\t\t\t\t\t: visibleToken\n\t\t\t\t\tconst resource = getFromStore(resourceToken, store)\n\n\t\t\t\t\tinitialPayload.push(resourceToken, resource)\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tconst epoch = isRootStore(store)\n\t\t\t\t? store.transactionMeta.epoch.get(continuityKey) ?? null\n\t\t\t\t: null\n\n\t\t\tsocket?.emit(`continuity-init:${continuityKey}`, epoch, initialPayload)\n\n\t\t\tfor (const transaction of continuity.actions) {\n\t\t\t\tconst unsubscribeFromTransaction = subscribeToTransaction(\n\t\t\t\t\ttransaction,\n\t\t\t\t\t(update) => {\n\t\t\t\t\t\ttry {\n\t\t\t\t\t\t\tconst visibleKeys = continuity.globals\n\t\t\t\t\t\t\t\t.map((atom) => atom.key)\n\t\t\t\t\t\t\t\t.concat(\n\t\t\t\t\t\t\t\t\tcontinuity.perspectives.flatMap((perspective) => {\n\t\t\t\t\t\t\t\t\t\tconst { viewAtoms } = perspective\n\t\t\t\t\t\t\t\t\t\tconst userPerspectiveTokenState = findInStore(\n\t\t\t\t\t\t\t\t\t\t\tviewAtoms,\n\t\t\t\t\t\t\t\t\t\t\tuserKey,\n\t\t\t\t\t\t\t\t\t\t\tstore,\n\t\t\t\t\t\t\t\t\t\t)\n\t\t\t\t\t\t\t\t\t\tconst visibleTokens = getFromStore(\n\t\t\t\t\t\t\t\t\t\t\tuserPerspectiveTokenState,\n\t\t\t\t\t\t\t\t\t\t\tstore,\n\t\t\t\t\t\t\t\t\t\t)\n\t\t\t\t\t\t\t\t\t\treturn visibleTokens.map((token) => {\n\t\t\t\t\t\t\t\t\t\t\tconst key =\n\t\t\t\t\t\t\t\t\t\t\t\ttoken.type === `mutable_atom`\n\t\t\t\t\t\t\t\t\t\t\t\t\t? `*` + token.key\n\t\t\t\t\t\t\t\t\t\t\t\t\t: token.key\n\t\t\t\t\t\t\t\t\t\t\treturn key\n\t\t\t\t\t\t\t\t\t\t})\n\t\t\t\t\t\t\t\t\t}),\n\t\t\t\t\t\t\t\t)\n\t\t\t\t\t\t\tconst redactedUpdates = redactTransactionUpdateContent(\n\t\t\t\t\t\t\t\tvisibleKeys,\n\t\t\t\t\t\t\t\tupdate.updates,\n\t\t\t\t\t\t\t)\n\t\t\t\t\t\t\tconst redactedUpdate = {\n\t\t\t\t\t\t\t\t...update,\n\t\t\t\t\t\t\t\tupdates: redactedUpdates,\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t// setIntoStore(\n\t\t\t\t\t\t\t// \tuserUnacknowledgedQueue,\n\t\t\t\t\t\t\t// \t(updates) => {\n\t\t\t\t\t\t\t// \t\tif (redactedUpdate) {\n\t\t\t\t\t\t\t// \t\t\tupdates.push(redactedUpdate)\n\t\t\t\t\t\t\t// \t\t\tupdates.sort((a, b) => a.epoch - b.epoch)\n\t\t\t\t\t\t\t// \t\t}\n\t\t\t\t\t\t\t// \t\treturn updates\n\t\t\t\t\t\t\t// \t},\n\t\t\t\t\t\t\t// \tstore,\n\t\t\t\t\t\t\t// )\n\n\t\t\t\t\t\t\tsocket?.emit(\n\t\t\t\t\t\t\t\t`tx-new:${continuityKey}`,\n\t\t\t\t\t\t\t\tredactedUpdate as Json.Serializable,\n\t\t\t\t\t\t\t)\n\t\t\t\t\t\t} catch (thrown) {\n\t\t\t\t\t\t\tif (thrown instanceof Error) {\n\t\t\t\t\t\t\t\tstore.logger.error(\n\t\t\t\t\t\t\t\t\t`❌`,\n\t\t\t\t\t\t\t\t\t`continuity`,\n\t\t\t\t\t\t\t\t\tcontinuityKey,\n\t\t\t\t\t\t\t\t\t`failed to send update from transaction ${transaction.key} to ${userKey}`,\n\t\t\t\t\t\t\t\t\tthrown.message,\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`sync-continuity:${continuityKey}:${userKey}`,\n\t\t\t\t\tstore,\n\t\t\t\t)\n\t\t\t\tunsubscribeFunctions.push(unsubscribeFromTransaction)\n\t\t\t}\n\t\t}\n\t\tsocket.off(`get:${continuityKey}`, sendInitialPayload)\n\t\tsocket.on(`get:${continuityKey}`, sendInitialPayload)\n\n\t\tconst fillTransactionRequest = (\n\t\t\tupdate: Pick<AtomIO.TransactionUpdate<JsonIO>, `id` | `key` | `params`>,\n\t\t) => {\n\t\t\tstore.logger.info(`🛎️`, `continuity`, continuityKey, `received`, update)\n\t\t\tconst transactionKey = update.key\n\t\t\tconst updateId = update.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\t{ type: `transaction`, key: transactionKey },\n\t\t\t\t\tupdateId,\n\t\t\t\t\tstore,\n\t\t\t\t)(...update.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} 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\tmetric.duration,\n\t\t\t)\n\n\t\t\tconst valuesOfCardsViewKey = `valuesOfCardsView(\"${userKey}\")`\n\t\t\tconst rootsOfCardValueView =\n\t\t\t\tstore.selectorAtoms.getRelatedKeys(valuesOfCardsViewKey)\n\t\t\tconst myCardValueView = store.valueMap.get(valuesOfCardsViewKey)\n\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`seeing ${userKey} card values`,\n\t\t\t\t{\n\t\t\t\t\tvaluesOfCardsViewKey,\n\t\t\t\t\trootsOfCardValueView,\n\t\t\t\t\tmyCardValueView,\n\t\t\t\t},\n\t\t\t)\n\t\t}\n\t\tsocket.off(`tx-run:${continuityKey}`, fillTransactionRequest)\n\t\tsocket.on(`tx-run:${continuityKey}`, fillTransactionRequest)\n\n\t\t// let i = 0\n\t\t// let n = 1\n\t\t// let retryTimeout: NodeJS.Timeout | undefined\n\t\t// const trackClientAcknowledgement = (epoch: number) => {\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`${userKey} acknowledged epoch ${epoch}`,\n\t\t// \t)\n\t\t// \tconst isUnacknowledged = userUnacknowledgedUpdates[0]?.epoch === epoch\n\t\t// \tif (isUnacknowledged) {\n\t\t// \t\tsetIntoStore(\n\t\t// \t\t\tuserUnacknowledgedQueue,\n\t\t// \t\t\t(updates) => {\n\t\t// \t\t\t\tupdates.shift()\n\t\t// \t\t\t\treturn updates\n\t\t// \t\t\t},\n\t\t// \t\t\tstore,\n\t\t// \t\t)\n\t\t// \t}\n\t\t// }\n\t\t// subscribeToState(\n\t\t// \tuserUnacknowledgedQueue,\n\t\t// \t({ newValue }) => {\n\t\t// \t\tif (newValue.length === 0) {\n\t\t// \t\t\tclearInterval(retryTimeout)\n\t\t// \t\t\tsocket?.off(`ack:${continuityKey}`, trackClientAcknowledgement)\n\t\t// \t\t\tretryTimeout = undefined\n\t\t// \t\t}\n\t\t// \t\tif (newValue.length > 0) {\n\t\t// \t\t\tif (retryTimeout) {\n\t\t// \t\t\t\treturn\n\t\t// \t\t\t}\n\n\t\t// \t\t\tsocket?.on(`ack:${continuityKey}`, trackClientAcknowledgement)\n\n\t\t// \t\t\tretryTimeout = setInterval(() => {\n\t\t// \t\t\t\ti++\n\t\t// \t\t\t\tif (i === n) {\n\t\t// \t\t\t\t\tn += i\n\t\t// \t\t\t\t\tconst toEmit = newValue[0]\n\t\t// \t\t\t\t\tif (!toEmit) return\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`${store.config.name} retrying ${userKey}`,\n\t\t// \t\t\t\t\t\tsocket?.id,\n\t\t// \t\t\t\t\t\tnewValue,\n\t\t// \t\t\t\t\t)\n\t\t// \t\t\t\t\tsocket?.emit(\n\t\t// \t\t\t\t\t\t`tx-new:${continuityKey}`,\n\t\t// \t\t\t\t\t\ttoEmit as Json.Serializable,\n\t\t// \t\t\t\t\t)\n\t\t// \t\t\t\t}\n\t\t// \t\t\t}, 250)\n\t\t// \t\t}\n\t\t// \t},\n\t\t// \t`sync-continuity:${continuityKey}:${userKey}`,\n\t\t// \tstore,\n\t\t// )\n\n\t\treturn () => {\n\t\t\t// clearInterval(retryTimeout)\n\t\t\tfor (const unsubscribe of unsubscribeFunctions) unsubscribe()\n\t\t\t// socket?.off(`ack:${continuityKey}`, trackClientAcknowledgement)\n\t\t\tunsubscribeFromPerspectives()\n\t\t\tsocket?.off(`get:${continuityKey}`, sendInitialPayload)\n\t\t\tsocket?.off(`tx-run:${continuityKey}`, fillTransactionRequest)\n\t\t}\n\t}\n}\n","import { selectorFamily } from \"atom.io\"\nimport type { TransactionUpdate } from \"atom.io\"\nimport { IMPLICIT, getJsonToken, getUpdateToken } from \"atom.io/internal\"\nimport type { JsonIO } from \"atom.io/json\"\nimport { SyncGroup } from \"atom.io/realtime\"\n// import { completeUpdateAtoms } from \"atom.io/realtime-server\"\n\nconst redactorAtoms = selectorFamily<\n\t(update: TransactionUpdate<any>) => TransactionUpdate<any>,\n\t{ userId: string; syncGroupKey: string }\n>({\n\tkey: `perspectiveRedactor`,\n\tget:\n\t\t({ userId, syncGroupKey }) =>\n\t\t({ get, find }) => {\n\t\t\tconst syncGroup = SyncGroup.existing.get(syncGroupKey)\n\t\t\tif (!syncGroup) {\n\t\t\t\tthrow new Error(\n\t\t\t\t\t`Tried to create a synchronizer for a sync group that does not exist.`,\n\t\t\t\t)\n\t\t\t}\n\n\t\t\tconst userPerspectiveTokens = syncGroup.perspectives.flatMap(\n\t\t\t\t({ viewAtoms }) => {\n\t\t\t\t\tconst userPerspectiveToken = find(viewAtoms, userId)\n\t\t\t\t\tconst userPerspective = get(userPerspectiveToken)\n\t\t\t\t\tconst visibleTokens = [...userPerspective].map((token) => {\n\t\t\t\t\t\treturn token.type === `mutable_atom`\n\t\t\t\t\t\t\t? getUpdateToken(token).key\n\t\t\t\t\t\t\t: token.key\n\t\t\t\t\t})\n\t\t\t\t\tIMPLICIT.STORE.logger.info(\n\t\t\t\t\t\t`🔭`,\n\t\t\t\t\t\t`continuity`,\n\t\t\t\t\t\tsyncGroupKey,\n\t\t\t\t\t\t`${userId} can see ${visibleTokens.length} tokens in ${viewAtoms.key}`,\n\t\t\t\t\t\tvisibleTokens,\n\t\t\t\t\t)\n\t\t\t\t\treturn visibleTokens\n\t\t\t\t},\n\t\t\t)\n\n\t\t\tconst filterTransactionUpdate = (\n\t\t\t\tvisible: string[],\n\t\t\t\ttransactionUpdate: TransactionUpdate<any>,\n\t\t\t): TransactionUpdate<any> => {\n\t\t\t\tIMPLICIT.STORE.logger.info(\n\t\t\t\t\t`🖌`,\n\t\t\t\t\t`continuity`,\n\t\t\t\t\tsyncGroupKey,\n\t\t\t\t\t`redacting updates from ${transactionUpdate.epoch}:${transactionUpdate.key}:${transactionUpdate.id}`,\n\t\t\t\t\tvisible,\n\t\t\t\t\ttransactionUpdate.updates,\n\t\t\t\t)\n\t\t\t\tconst updates = transactionUpdate.updates\n\t\t\t\t\t.filter((update) => {\n\t\t\t\t\t\tif (`newValue` in update) {\n\t\t\t\t\t\t\treturn visible.includes(update.key)\n\t\t\t\t\t\t}\n\t\t\t\t\t\treturn true\n\t\t\t\t\t})\n\t\t\t\t\t.map((update) => {\n\t\t\t\t\t\tif (`updates` in update) {\n\t\t\t\t\t\t\treturn filterTransactionUpdate(visible, update)\n\t\t\t\t\t\t}\n\t\t\t\t\t\treturn update\n\t\t\t\t\t})\n\t\t\t\tconst filtered: TransactionUpdate<any> = {\n\t\t\t\t\t...transactionUpdate,\n\t\t\t\t\tupdates,\n\t\t\t\t}\n\t\t\t\treturn filtered\n\t\t\t}\n\t\t\tconst filter: (updates: TransactionUpdate<any>) => TransactionUpdate<any> =\n\t\t\t\t(update) => {\n\t\t\t\t\tconst visibleKeys: string[] = syncGroup.globals.map((atomToken) =>\n\t\t\t\t\t\tatomToken.type === `mutable_atom`\n\t\t\t\t\t\t\t? getUpdateToken(atomToken).key\n\t\t\t\t\t\t\t: atomToken.key,\n\t\t\t\t\t)\n\t\t\t\t\tvisibleKeys.push(...userPerspectiveTokens)\n\t\t\t\t\treturn filterTransactionUpdate(visibleKeys, update)\n\t\t\t\t}\n\t\t\treturn filter\n\t\t},\n})\n// export const occludedUpdateSelectors = selectorFamily<\n// \tPick<\n// \t\tTransactionUpdate<JsonIO>,\n// \t\t`epoch` | `id` | `key` | `output` | `updates`\n// \t> | null,\n// \t{ userId: string; syncGroupKey: string; updateId: string }\n// >({\n// \tkey: `occludedUpdate`,\n// \tget:\n// \t\t({ userId, syncGroupKey, updateId }) =>\n// \t\t({ get, find }) => {\n// \t\t\tconst updateState = find(completeUpdateAtoms, updateId)\n// \t\t\tconst update = get(updateState)\n// \t\t\tconst redactorKey = { userId, syncGroupKey }\n// \t\t\tconst redactorState = find(redactorAtoms, redactorKey)\n// \t\t\tconst redact = get(redactorState)\n// \t\t\tif (update) {\n// \t\t\t\t// return redact(update)\n// \t\t\t\treturn update\n// \t\t\t}\n// \t\t\treturn null\n// \t\t},\n// })\n","import * as AtomIO from \"atom.io\"\nimport type { Loadable } from \"atom.io/data\"\nimport type { UserInRoomMeta } from \"atom.io/realtime\"\nimport { roomIndex, usersInRooms } from \"atom.io/realtime\"\n\nimport type { ChildSocket } from \"../ipc-sockets\"\nimport type { RoomArguments } from \"./server-room-external-store\"\nimport { roomArgumentsAtoms, roomSelectors } from \"./server-room-external-store\"\n\nexport const createRoomTX = AtomIO.transaction<\n\t(\n\t\troomId: string,\n\t\tscript: string,\n\t\toptions?: string[],\n\t) => Loadable<ChildSocket<any, any>>\n>({\n\tkey: `createRoom`,\n\tdo: ({ get, set, find }, roomId, script, options) => {\n\t\tconst args: RoomArguments = options ? [script, options] : [script]\n\t\tconst roomArgumentsState = find(roomArgumentsAtoms, roomId)\n\t\tset(roomArgumentsState, args)\n\t\tset(roomIndex, (s) => s.add(roomId))\n\t\tconst roomState = find(roomSelectors, roomId)\n\t\tconst room = get(roomState)\n\t\treturn room\n\t},\n})\nexport type CreateRoomIO = AtomIO.TransactionIO<typeof createRoomTX>\n\nexport const joinRoomTX = AtomIO.transaction<\n\t(roomId: string, userId: string, enteredAtEpoch: number) => UserInRoomMeta\n>({\n\tkey: `joinRoom`,\n\tdo: (transactors, roomId, userId, enteredAtEpoch) => {\n\t\tconst meta = { enteredAtEpoch }\n\t\tusersInRooms.transact(transactors, ({ relations }) => {\n\t\t\trelations.set(roomId, userId, meta)\n\t\t})\n\t\treturn meta\n\t},\n})\nexport type JoinRoomIO = AtomIO.TransactionIO<typeof joinRoomTX>\n\nexport const leaveRoomTX = AtomIO.transaction<\n\t(roomId: string, userId: string) => void\n>({\n\tkey: `leaveRoom`,\n\tdo: (transactors, roomId, userId) => {\n\t\tusersInRooms.transact(transactors, ({ relations }) => {\n\t\t\trelations.delete({ room: roomId, user: userId })\n\t\t})\n\t},\n})\nexport type LeaveRoomIO = AtomIO.TransactionIO<typeof leaveRoomTX>\n\nexport const destroyRoomTX = AtomIO.transaction<(roomId: string) => void>({\n\tkey: `destroyRoom`,\n\tdo: (transactors, roomId) => {\n\t\tusersInRooms.transact(transactors, ({ relations }) => {\n\t\t\trelations.delete({ room: roomId })\n\t\t})\n\t\ttransactors.set(roomIndex, (s) => (s.delete(roomId), s))\n\t},\n})\n","import type { ChildProcessWithoutNullStreams } from \"child_process\"\nimport { spawn } from \"child_process\"\n\nimport { atomFamily, selectorFamily } from \"atom.io\"\nimport type { Loadable } from \"atom.io/data\"\nimport { ChildSocket } from \"../ipc-sockets\"\n\nexport type RoomArguments =\n\t| [script: string, options: string[]]\n\t| [script: string]\n\nexport const roomArgumentsAtoms = atomFamily<RoomArguments, string>({\n\tkey: `roomArguments`,\n\tdefault: [`echo`, [`Hello World!`]],\n})\n\nexport const roomSelectors = selectorFamily<\n\tLoadable<ChildSocket<any, any>>,\n\tstring\n>({\n\tkey: `room`,\n\tget:\n\t\t(roomId) =>\n\t\tasync ({ get, find }) => {\n\t\t\tconst argumentsState = find(roomArgumentsAtoms, roomId)\n\t\t\tconst args = get(argumentsState)\n\t\t\tconst [script, options] = args\n\t\t\tconst child = await new Promise<ChildProcessWithoutNullStreams>(\n\t\t\t\t(resolve) => {\n\t\t\t\t\tconst room = spawn(script, options, { env: process.env })\n\t\t\t\t\tconst resolver = (data: Buffer) => {\n\t\t\t\t\t\tif (data.toString() === `✨`) {\n\t\t\t\t\t\t\troom.stdout.off(`data`, resolver)\n\t\t\t\t\t\t\tresolve(room)\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t\troom.stdout.on(`data`, resolver)\n\t\t\t\t},\n\t\t\t)\n\t\t\treturn new ChildSocket(child, roomId)\n\t\t},\n})\n","import type { TransactionUpdate, TransactionUpdateContent } from \"atom.io\"\nimport { atomFamily, selectorFamily } from \"atom.io\"\n\n// export const completeUpdateAtoms = atomFamily<\n// \tTransactionUpdate<any> | null,\n// \tstring\n// >({\n// \tkey: `completeUpdate`,\n// \tdefault: null,\n// })\n\nexport function redactTransactionUpdateContent(\n\tvisibleStateKeys: string[],\n\tupdates: TransactionUpdateContent[],\n): TransactionUpdateContent[] {\n\treturn updates\n\t\t.map((update): TransactionUpdateContent => {\n\t\t\tif (`newValue` in update) {\n\t\t\t\treturn update\n\t\t\t}\n\t\t\tconst redacted = redactTransactionUpdateContent(\n\t\t\t\tvisibleStateKeys,\n\t\t\t\tupdate.updates,\n\t\t\t)\n\t\t\treturn { ...update, updates: redacted }\n\t\t})\n\t\t.filter((update) => {\n\t\t\tif (`newValue` in update) {\n\t\t\t\treturn visibleStateKeys.includes(update.key)\n\t\t\t}\n\t\t\treturn true\n\t\t})\n}\n\nexport const actionOcclusionAtoms = atomFamily<\n\t{\n\t\tocclude: (updates: TransactionUpdateContent[]) => TransactionUpdateContent[]\n\t},\n\tstring\n>({\n\tkey: `transactionRedactor`,\n\tdefault: { occlude: (updates) => updates },\n})\n// export const redactedUpdateSelectors = selectorFamily<\n// \tTransactionUpdate<any> | null,\n// \t[transactionKey: string, updateId: string]\n// >({\n// \tkey: `redactedUpdate`,\n// \tget:\n// \t\t([transactionKey, updateId]) =>\n// \t\t({ get, find }) => {\n// \t\t\tconst update = get(find(completeUpdateAtoms, updateId))\n// \t\t\tconst { filter } = get(find(transactionRedactorAtoms, transactionKey))\n\n// \t\t\tif (update && filter) {\n// \t\t\t\treturn { ...update, updates: filter(update.updates) }\n// \t\t\t}\n// \t\t\treturn null\n// \t\t},\n// })\n\nexport const userUnacknowledgedQueues = atomFamily<\n\tPick<TransactionUpdate<any>, `epoch` | `id` | `key` | `output` | `updates`>[],\n\tstring\n>({\n\tkey: `unacknowledgedUpdates`,\n\tdefault: () => [],\n})\n","import { atom, atomFamily } from \"atom.io\"\nimport { join } from \"atom.io/data\"\nimport { SetRTX } from \"atom.io/transceivers/set-rtx\"\n\nimport type { Socket } from \"..\"\n\nexport const socketAtoms = atomFamily<Socket | null, string>({\n\tkey: `sockets`,\n\tdefault: null,\n})\n\nexport const socketIndex = atom({\n\tkey: `socketsIndex`,\n\tmutable: true,\n\tdefault: () => new SetRTX<string>(),\n\ttoJson: (set) => set.toJSON(),\n\tfromJson: (json) => SetRTX.fromJSON(json),\n})\nexport const userIndex = atom({\n\tkey: `usersIndex`,\n\tmutable: true,\n\tdefault: () => new SetRTX<string>(),\n\ttoJson: (set) => set.toJSON(),\n\tfromJson: (json) => SetRTX.fromJSON(json),\n})\nexport const usersOfSockets = join({\n\tkey: `usersOfSockets`,\n\tbetween: [`user`, `socket`],\n\tcardinality: `1:1`,\n})\n","import type * as AtomIO from \"atom.io\"\nimport { IMPLICIT, getFromStore, subscribeToState } from \"atom.io/internal\"\nimport type { Json } from \"atom.io/json\"\n\nimport type { ServerConfig } from \".\"\n\nexport type StateProvider = ReturnType<typeof realtimeStateProvider>\nexport function realtimeStateProvider({\n\tsocket,\n\tstore = IMPLICIT.STORE,\n}: ServerConfig) {\n\treturn function stateProvider<J extends Json.Serializable>(\n\t\ttoken: AtomIO.WritableToken<J>,\n\t): () => void {\n\t\tlet unsubscribeFromStateUpdates: (() => void) | undefined\n\n\t\tconst fillSubRequest = () => {\n\t\t\tsocket.emit(`serve:${token.key}`, getFromStore(token, store))\n\n\t\t\tunsubscribeFromStateUpdates = subscribeToState(\n\t\t\t\ttoken,\n\t\t\t\t({ newValue }) => {\n\t\t\t\t\tsocket.emit(`serve:${token.key}`, newValue)\n\t\t\t\t},\n\t\t\t\t`expose-single:${socket.id}`,\n\t\t\t\tstore,\n\t\t\t)\n\n\t\t\tconst fillUnsubRequest = () => {\n\t\t\t\tsocket.off(`unsub:${token.key}`, fillUnsubRequest)\n\t\t\t\tif (unsubscribeFromStateUpdates) {\n\t\t\t\t\tunsubscribeFromStateUpdates()\n\t\t\t\t\tunsubscribeFromStateUpdates = undefined\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tsocket.on(`unsub:${token.key}`, fillUnsubRequest)\n\t\t}\n\n\t\tsocket.on(`sub:${token.key}`, fillSubRequest)\n\n\t\treturn () => {\n\t\t\tsocket.off(`sub:${token.key}`, fillSubRequest)\n\t\t\tif (unsubscribeFromStateUpdates) {\n\t\t\t\tunsubscribeFromStateUpdates()\n\t\t\t\tunsubscribeFromStateUpdates = undefined\n\t\t\t}\n\t\t}\n\t}\n}\n","import type * as AtomIO from \"atom.io\"\nimport { IMPLICIT, getFromStore } from \"atom.io/internal\"\nimport type { Json } from \"atom.io/json\"\n\nimport type { ServerConfig } from \".\"\n\nexport function realtimeStateSynchronizer({\n\tsocket,\n\tstore = IMPLICIT.STORE,\n}: ServerConfig) {\n\treturn function stateSynchronizer<J extends Json.Serializable>(\n\t\ttoken: AtomIO.WritableToken<J>,\n\t): () => void {\n\t\tconst fillGetRequest = () => {\n\t\t\tsocket.emit(`value:${token.key}`, getFromStore(token, store))\n\t\t}\n\n\t\tsocket.on(`get:${token.key}`, fillGetRequest)\n\t\treturn () => {\n\t\t\tsocket.off(`get:${token.key}`, fillGetRequest)\n\t\t}\n\t}\n}\n","import type * as AtomIO from \"atom.io\"\nimport {\n\tIMPLICIT,\n\tfindInStore,\n\tgetFromStore,\n\tsubscribeToState,\n} from \"atom.io/internal\"\nimport { type Json, stringifyJson } from \"atom.io/json\"\n\nimport type { ServerConfig } from \".\"\n\nexport type FamilyProvider = ReturnType<typeof realtimeAtomFamilyProvider>\nexport function realtimeAtomFamilyProvider({\n\tsocket,\n\tstore = IMPLICIT.STORE,\n}: ServerConfig) {\n\treturn function familyProvider<\n\t\tJ extends Json.Serializable,\n\t\tK extends Json.Serializable,\n\t>(\n\t\tfamily: AtomIO.RegularAtomFamilyToken<J, K>,\n\t\tindex: AtomIO.ReadableToken<Iterable<K>>,\n\t): () => void {\n\t\tconst unsubCallbacksByKey = new Map<string, () => void>()\n\n\t\tconst fillUnsubRequest = (key: string) => {\n\t\t\tsocket.off(`unsub:${key}`, fillUnsubRequest)\n\t\t\tconst unsub = unsubCallbacksByKey.get(key)\n\t\t\tif (unsub) {\n\t\t\t\tunsub()\n\t\t\t\tunsubCallbacksByKey.delete(key)\n\t\t\t}\n\t\t}\n\n\t\tconst fillSubRequest = (subKey: K) => {\n\t\t\tconst exposedSubKeys = getFromStore(index, store)\n\t\t\tfor (const exposedSubKey of exposedSubKeys) {\n\t\t\t\tif (stringifyJson(exposedSubKey) === stringifyJson(subKey)) {\n\t\t\t\t\tconst token = findInStore(family, subKey, store)\n\t\t\t\t\tsocket.emit(`serve:${token.key}`, getFromStore(token, store))\n\t\t\t\t\tconst unsubscribe = subscribeToState(\n\t\t\t\t\t\ttoken,\n\t\t\t\t\t\t({ newValue }) => {\n\t\t\t\t\t\t\tsocket.emit(`serve:${token.key}`, newValue)\n\t\t\t\t\t\t},\n\t\t\t\t\t\t`expose-family:${family.key}:${socket.id}`,\n\t\t\t\t\t\tstore,\n\t\t\t\t\t)\n\t\t\t\t\tunsubCallbacksByKey.set(token.key, unsubscribe)\n\t\t\t\t\tsocket.on(`unsub:${token.key}`, () => {\n\t\t\t\t\t\tfillUnsubRequest(token.key)\n\t\t\t\t\t})\n\t\t\t\t\tbreak\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tsocket.on(`sub:${family.key}`, fillSubRequest)\n\n\t\treturn () => {\n\t\t\tsocket.off(`sub:${family.key}`, fillSubRequest)\n\n\t\t\tfor (const [, unsub] of unsubCallbacksByKey) {\n\t\t\t\tunsub()\n\t\t\t}\n\t\t\tunsubCallbacksByKey.clear()\n\t\t}\n\t}\n}\n","import type * as AtomIO from \"atom.io\"\nimport {\n\tIMPLICIT,\n\tgetFromStore,\n\tgetJsonToken,\n\tgetUpdateToken,\n\tsubscribeToState,\n} from \"atom.io/internal\"\nimport type { Transceiver } from \"atom.io/internal\"\nimport type { Json } from \"atom.io/json\"\n\nimport type { ServerConfig } from \".\"\n\nexport type MutableProvider = ReturnType<typeof realtimeMutableProvider>\nexport function realtimeMutableProvider({\n\tsocket,\n\tstore = IMPLICIT.STORE,\n}: ServerConfig) {\n\treturn function mutableProvider<\n\t\tCore extends Transceiver<Json.Serializable>,\n\t\tSerializableCore extends Json.Serializable,\n\t>(token: AtomIO.MutableAtomToken<Core, SerializableCore>): () => void {\n\t\tlet unsubscribeFromStateUpdates: (() => void) | null = null\n\n\t\tconst jsonToken = getJsonToken(token)\n\t\tconst trackerToken = getUpdateToken(token)\n\n\t\tconst fillUnsubRequest = () => {\n\t\t\tsocket.off(`unsub:${token.key}`, fillUnsubRequest)\n\t\t\tunsubscribeFromStateUpdates?.()\n\t\t\tunsubscribeFromStateUpdates = null\n\t\t}\n\n\t\tconst fillSubRequest = () => {\n\t\t\tsocket.emit(`init:${token.key}`, getFromStore(jsonToken, store))\n\t\t\tunsubscribeFromStateUpdates = subscribeToState(\n\t\t\t\ttrackerToken,\n\t\t\t\t({ newValue }) => {\n\t\t\t\t\tsocket.emit(`next:${token.key}`, newValue)\n\t\t\t\t},\n\t\t\t\t`expose-single:${socket.id}`,\n\t\t\t\tstore,\n\t\t\t)\n\t\t\tsocket.on(`unsub:${token.key}`, fillUnsubRequest)\n\t\t}\n\n\t\tsocket.on(`sub:${token.key}`, fillSubRequest)\n\n\t\treturn () => {\n\t\t\tsocket.off(`sub:${token.key}`, fillSubRequest)\n\t\t\tunsubscribeFromStateUpdates?.()\n\t\t}\n\t}\n}\n","import type * as AtomIO from \"atom.io\"\nimport type { Transceiver } from \"atom.io/internal\"\nimport {\n\tIMPLICIT,\n\tfindInStore,\n\tgetFromStore,\n\tgetJsonToken,\n\tgetUpdateToken,\n\tsubscribeToState,\n} from \"atom.io/internal\"\nimport type { Json } from \"atom.io/json\"\nimport { stringifyJson } from \"atom.io/json\"\n\nimport type { ServerConfig } from \".\"\n\nexport type MutableFamilyProvider = ReturnType<\n\ttypeof realtimeMutableFamilyProvider\n>\nexport function realtimeMutableFamilyProvider({\n\tsocket,\n\tstore = IMPLICIT.STORE,\n}: ServerConfig) {\n\treturn function mutableFamilyProvider<\n\t\tT extends Transceiver<any>,\n\t\tJ extends Json.Serializable,\n\t\tK extends Json.Serializable,\n\t>(\n\t\tfamily: AtomIO.MutableAtomFamilyToken<T, J, K>,\n\t\tindex: AtomIO.ReadableToken<Iterable<K>>,\n\t): () => void {\n\t\tconst unsubCallbacksByKey = new Map<string, () => void>()\n\n\t\tconst fillUnsubRequest = (key: string) => {\n\t\t\tsocket.off(`unsub:${key}`, fillUnsubRequest)\n\t\t\tconst unsub = unsubCallbacksByKey.get(key)\n\t\t\tif (unsub) {\n\t\t\t\tunsub()\n\t\t\t\tunsubCallbacksByKey.delete(key)\n\t\t\t}\n\t\t}\n\n\t\tconst fillSubRequest = (subKey: K) => {\n\t\t\tconst exposedSubKeys = getFromStore(index, store)\n\t\t\tfor (const exposedSubKey of exposedSubKeys) {\n\t\t\t\tif (stringifyJson(exposedSubKey) === stringifyJson(subKey)) {\n\t\t\t\t\tconst token = findInStore(family, subKey, store)\n\t\t\t\t\tconst jsonToken = getJsonToken(token)\n\t\t\t\t\tconst updateToken = getUpdateToken(token)\n\t\t\t\t\tsocket.emit(`init:${token.key}`, getFromStore(jsonToken, store))\n\t\t\t\t\tconst unsubscribe = subscribeToState(\n\t\t\t\t\t\tupdateToken,\n\t\t\t\t\t\t({ newValue }) => {\n\t\t\t\t\t\t\tsocket.emit(`next:${token.key}`, newValue)\n\t\t\t\t\t\t},\n\t\t\t\t\t\t`expose-family:${family.key}:${socket.id}`,\n\t\t\t\t\t\tstore,\n\t\t\t\t\t)\n\t\t\t\t\tunsubCallbacksByKey.set(token.key, unsubscribe)\n\t\t\t\t\tsocket.on(`unsub:${token.key}`, () => {\n\t\t\t\t\t\tfillUnsubRequest(token.key)\n\t\t\t\t\t})\n\t\t\t\t\tbreak\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tsocket.on(`sub:${family.key}`, fillSubRequest)\n\n\t\treturn () => {\n\t\t\tsocket.off(`sub:${family.key}`, fillSubRequest)\n\t\t\tfor (const [, unsub] of unsubCallbacksByKey) {\n\t\t\t\tunsub()\n\t\t\t}\n\t\t\tunsubCallbacksByKey.clear()\n\t\t}\n\t}\n}\n","import type { WritableToken } from \"atom.io\"\nimport { IMPLICIT, setIntoStore } from \"atom.io/internal\"\nimport type { Json } from \"atom.io/json\"\n\nimport type { ServerConfig } from \".\"\n\nexport type StateReceiver = ReturnType<typeof realtimeStateReceiver>\nexport function realtimeStateReceiver({\n\tsocket,\n\tstore = IMPLICIT.STORE,\n}: ServerConfig) {\n\treturn function stateReceiver<J extends Json.Serializable>(\n\t\ttoken: WritableToken<J>,\n\t): () => void {\n\t\tconst publish = (newValue: J) => setIntoStore(token, newValue, store)\n\n\t\tconst fillPubUnclaim = () => {\n\t\t\tsocket.off(`pub:${token.key}`, publish)\n\t\t\tsocket.off(`unclaim:${token.key}`, fillPubUnclaim)\n\t\t}\n\t\tconst fillPubClaim = () => {\n\t\t\tsocket.on(`pub:${token.key}`, publish)\n\t\t\tsocket.on(`unclaim:${token.key}`, fillPubUnclaim)\n\t\t}\n\n\t\tsocket.on(`claim:${token.key}`, fillPubClaim)\n\n\t\treturn () => {\n\t\t\tsocket.off(`claim:${token.key}`, fillPubClaim)\n\t\t\tsocket.off(`pub:${token.key}`, publish)\n\t\t}\n\t}\n}\n","import type * as AtomIO from \"atom.io\"\nimport { IMPLICIT, actUponStore } from \"atom.io/internal\"\nimport type { JsonIO } from \"atom.io/json\"\n\nimport type { ServerConfig } from \".\"\n\nexport type ActionReceiver = ReturnType<typeof realtimeActionReceiver>\nexport function realtimeActionReceiver({\n\tsocket,\n\tstore = IMPLICIT.STORE,\n}: ServerConfig) {\n\treturn function actionReceiver<ƒ extends JsonIO>(\n\t\ttx: AtomIO.TransactionToken<ƒ>,\n\t): () => void {\n\t\tconst fillTransactionRequest = (\n\t\t\tupdate: Pick<AtomIO.TransactionUpdate<ƒ>, `id` | `params`>,\n\t\t) => {\n\t\t\tconst performanceKey = `tx-run:${tx.key}:${update.id}`\n\t\t\tconst performanceKeyStart = `${performanceKey}:start`\n\t\t\tconst performanceKeyEnd = `${performanceKey}:end`\n\t\t\tperformance.mark(performanceKeyStart)\n\t\t\tactUponStore<ƒ>(tx, update.id, store)(...update.params)\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(`🚀`, `transaction`, tx.key, update.id, metric.duration)\n\t\t}\n\t\tsocket.on(`tx-run:${tx.key}`, fillTransactionRequest)\n\n\t\treturn () => socket.off(`tx-run:${tx.key}`, fillTransactionRequest)\n\t}\n}\n"]}
@@ -1,10 +1,8 @@
1
1
  import type { Store } from "atom.io/internal"
2
2
  import type { Json } from "atom.io/json"
3
- import { off } from "npmlog"
4
3
 
5
- export * from "./ipc-socket"
4
+ export * from "./ipc-sockets"
6
5
  export * from "./realtime-continuity-synchronizer"
7
- export * from "./realtime-server-stores/server-room-store"
8
6
  export * from "./realtime-server-stores"
9
7
  export * from "./realtime-state-provider"
10
8
  export * from "./realtime-state-synchronizer"
@@ -13,7 +11,7 @@ export * from "./realtime-mutable-provider"
13
11
  export * from "./realtime-mutable-family-provider"
14
12
  export * from "./realtime-state-receiver"
15
13
  export * from "./realtime-action-receiver"
16
- export * from "./realtime-action-synchronizer"
14
+ // export * from "./realtime-action-synchronizer.txt"
17
15
 
18
16
  export type Socket = {
19
17
  id: string
@@ -0,0 +1,135 @@
1
+ import type { ChildProcessWithoutNullStreams } from "child_process"
2
+
3
+ import type { Json } from "atom.io/json"
4
+ import { parseJson } from "atom.io/json"
5
+
6
+ import type { EventBuffer, Events } from "./custom-socket"
7
+ import { CustomSocket } from "./custom-socket"
8
+
9
+ export class ChildSocket<
10
+ I extends Events,
11
+ O extends Events & {
12
+ /* eslint-disable quotes */
13
+ "setup-relay": [string]
14
+ /* eslint-enable quotes */
15
+ },
16
+ > extends CustomSocket<I, O> {
17
+ protected incompleteData = ``
18
+ protected unprocessedEvents: string[] = []
19
+ protected incompleteLog = ``
20
+ protected unprocessedLogs: string[] = []
21
+
22
+ public id = `#####`
23
+
24
+ protected handleLog(arg: Json.Serializable): void {
25
+ if (Array.isArray(arg)) {
26
+ const [level, ...rest] = arg
27
+ switch (level) {
28
+ case `i`:
29
+ this.logger.info(this.id, this.key, ...rest)
30
+ break
31
+ case `w`:
32
+ this.logger.warn(this.id, this.key, ...rest)
33
+ break
34
+ case `e`:
35
+ this.logger.error(this.id, this.key, ...rest)
36
+ break
37
+ }
38
+ }
39
+ }
40
+
41
+ public constructor(
42
+ public process: ChildProcessWithoutNullStreams,
43
+ public key: string,
44
+ public logger: {
45
+ info: (prefix: string, message: string, ...args: unknown[]) => void
46
+ warn: (prefix: string, message: string, ...args: unknown[]) => void
47
+ error: (prefix: string, message: string, ...args: unknown[]) => void
48
+ } = console,
49
+ ) {
50
+ super((event, ...args) => {
51
+ const stringifiedEvent = JSON.stringify([event, ...args]) + `\x03`
52
+ const errorHandler = (err) => {
53
+ if (err.code === `EPIPE`) {
54
+ console.error(`EPIPE error during write`, this.process.stdin)
55
+ }
56
+ this.process.stdin.removeListener(`error`, errorHandler)
57
+ }
58
+
59
+ this.process.stdin.once(`error`, errorHandler)
60
+ this.process.stdin.write(stringifiedEvent)
61
+
62
+ return this
63
+ })
64
+ this.process = process
65
+ this.process.stdout.on(
66
+ `data`,
67
+ <Event extends keyof I>(buffer: EventBuffer<string, I[Event]>) => {
68
+ const chunk = buffer.toString()
69
+
70
+ if (chunk === `✨`) {
71
+ // console.log(chunk)
72
+ return
73
+ }
74
+ this.unprocessedEvents.push(...chunk.split(`\x03`))
75
+ // console.log(`🤓`, chunk.length)
76
+ // console.log(`🤓`, this.unprocessedEvents.length)
77
+ // console.log(`🤓`, ...this.unprocessedEvents.map((x) => x.length))
78
+ const newInput = this.unprocessedEvents.shift()
79
+ this.incompleteData += newInput || ``
80
+ try {
81
+ if (this.incompleteData.startsWith(`error`)) {
82
+ console.log(`❗`, this.incompleteData)
83
+ }
84
+ const parsedEvent = parseJson(this.incompleteData)
85
+ this.handleEvent(...(parsedEvent as [string, ...I[keyof I]]))
86
+ while (this.unprocessedEvents.length > 0) {
87
+ const event = this.unprocessedEvents.shift()
88
+ if (event) {
89
+ if (this.unprocessedEvents.length === 0) {
90
+ this.incompleteData = event
91
+ }
92
+ const parsedEvent = parseJson(event)
93
+ this.handleEvent(...(parsedEvent as [string, ...I[keyof I]]))
94
+ }
95
+ }
96
+ this.incompleteData = ``
97
+ } catch (error) {
98
+ console.warn(`⚠️----------------⚠️`)
99
+ console.warn(this.incompleteData)
100
+ console.warn(`⚠️----------------⚠️`)
101
+ console.error(error)
102
+ }
103
+ },
104
+ )
105
+ this.process.stderr.on(`data`, (buf) => {
106
+ const chunk = buf.toString()
107
+ this.unprocessedLogs.push(...chunk.split(`\x03`))
108
+ // console.log(`🤫`, chunk.length)
109
+ // console.log(`🤫`, this.unprocessedLogs.length)
110
+ // console.log(`🤫`, ...this.unprocessedLogs.map((x) => x.length))
111
+ const newInput = this.unprocessedLogs.shift()
112
+ this.incompleteLog += newInput || ``
113
+ try {
114
+ const parsedLog = parseJson(this.incompleteLog)
115
+ // console.log(`🤫`, parsedLog)
116
+ this.handleLog(parsedLog)
117
+ while (this.unprocessedLogs.length > 0) {
118
+ this.incompleteLog = this.unprocessedLogs.shift() ?? ``
119
+ if (this.incompleteLog) {
120
+ const parsedLog = parseJson(this.incompleteLog)
121
+ this.handleLog(parsedLog)
122
+ }
123
+ }
124
+ } catch (error) {
125
+ console.error(`❌❌❌`)
126
+ console.error(this.incompleteLog)
127
+ console.error(error)
128
+ console.error(`❌❌❌️`)
129
+ }
130
+ })
131
+ if (process.pid) {
132
+ this.id = process.pid.toString()
133
+ }
134
+ }
135
+ }
@@ -0,0 +1,90 @@
1
+ import type { Json, Stringified } from "atom.io/json"
2
+
3
+ import type { Socket } from ".."
4
+
5
+ export type Events = Json.Object<string, Json.Serializable[]>
6
+
7
+ export type StringifiedEvent<
8
+ Key extends string,
9
+ Params extends Json.Serializable[],
10
+ > = Stringified<[Key, ...Params]>
11
+
12
+ export interface EventBuffer<
13
+ Key extends string,
14
+ Params extends Json.Serializable[],
15
+ > extends Buffer {
16
+ toString(): StringifiedEvent<Key, Params>
17
+ }
18
+
19
+ export class CustomSocket<I extends Events, O extends Events> implements Socket {
20
+ protected listeners: Map<keyof I, Set<(...args: Json.Array) => void>>
21
+ protected globalListeners: Set<(event: string, ...args: Json.Array) => void>
22
+ protected handleEvent<Event extends keyof I>(
23
+ event: string,
24
+ ...args: I[Event]
25
+ ): void {
26
+ for (const listener of this.globalListeners) {
27
+ listener(event, ...args)
28
+ }
29
+ const listeners = this.listeners.get(event)
30
+ if (listeners) {
31
+ for (const listener of listeners) {
32
+ listener(...args)
33
+ }
34
+ }
35
+ }
36
+
37
+ public id = `no_id_retrieved`
38
+
39
+ public constructor(
40
+ public emit: <Event extends keyof O>(
41
+ event: Event,
42
+ ...args: O[Event]
43
+ ) => CustomSocket<I, O>,
44
+ ) {
45
+ this.listeners = new Map()
46
+ this.globalListeners = new Set()
47
+ }
48
+
49
+ public on<Event extends keyof I>(
50
+ event: Event,
51
+ listener: (...args: I[Event]) => void,
52
+ ): CustomSocket<I, O> {
53
+ const listeners = this.listeners.get(event)
54
+ if (listeners) {
55
+ listeners.add(listener)
56
+ } else {
57
+ this.listeners.set(event, new Set([listener]))
58
+ }
59
+ return this
60
+ }
61
+
62
+ public onAny(
63
+ listener: (event: string, ...args: Json.Array) => void,
64
+ ): CustomSocket<I, O> {
65
+ this.globalListeners.add(listener)
66
+ return this
67
+ }
68
+
69
+ public off<Event extends keyof I>(
70
+ event: Event,
71
+ listener?: (...args: I[Event]) => void,
72
+ ): CustomSocket<I, O> {
73
+ const listeners = this.listeners.get(event)
74
+ if (listeners) {
75
+ if (listener) {
76
+ listeners.delete(listener)
77
+ } else {
78
+ this.listeners.delete(event)
79
+ }
80
+ }
81
+ return this
82
+ }
83
+
84
+ public offAny(
85
+ listener: (event: string, ...args: Json.Array) => void,
86
+ ): CustomSocket<I, O> {
87
+ this.globalListeners.delete(listener)
88
+ return this
89
+ }
90
+ }
@@ -0,0 +1,3 @@
1
+ export * from "./child-socket"
2
+ export * from "./custom-socket"
3
+ export * from "./parent-socket"