@electric-sql/y-electric 0.1.0

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.
@@ -0,0 +1,415 @@
1
+ var __defProp = Object.defineProperty;
2
+ var __defProps = Object.defineProperties;
3
+ var __getOwnPropDescs = Object.getOwnPropertyDescriptors;
4
+ var __getOwnPropSymbols = Object.getOwnPropertySymbols;
5
+ var __hasOwnProp = Object.prototype.hasOwnProperty;
6
+ var __propIsEnum = Object.prototype.propertyIsEnumerable;
7
+ var __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value;
8
+ var __spreadValues = (a, b) => {
9
+ for (var prop in b || (b = {}))
10
+ if (__hasOwnProp.call(b, prop))
11
+ __defNormalProp(a, prop, b[prop]);
12
+ if (__getOwnPropSymbols)
13
+ for (var prop of __getOwnPropSymbols(b)) {
14
+ if (__propIsEnum.call(b, prop))
15
+ __defNormalProp(a, prop, b[prop]);
16
+ }
17
+ return a;
18
+ };
19
+ var __spreadProps = (a, b) => __defProps(a, __getOwnPropDescs(b));
20
+
21
+ // src/y-electric.ts
22
+ import * as encoding from "lib0/encoding";
23
+ import * as decoding from "lib0/decoding";
24
+ import * as awarenessProtocol from "y-protocols/awareness";
25
+ import { ObservableV2 } from "lib0/observable";
26
+ import * as env from "lib0/environment";
27
+ import * as Y from "yjs";
28
+ import {
29
+ isChangeMessage,
30
+ isControlMessage,
31
+ ShapeStream
32
+ } from "@electric-sql/client";
33
+ var ElectricProvider = class extends ObservableV2 {
34
+ /**
35
+ * Creates a new ElectricProvider instance that connects YJS documents to Electric SQL.
36
+ *
37
+ * @constructor
38
+ * @param {ElectricProviderOptions} options - Configuration options for the provider
39
+ * @param {Y.Doc} options.doc - The YJS document to be synchronized
40
+ * @param {Object} options.documentUpdates - Document updates configuration
41
+ * @param {ShapeStreamOptions} options.documentUpdates.shape - Options for the document updates shape stream
42
+ * @param {string|URL} options.documentUpdates.sendUrl - URL endpoint for sending document updates
43
+ * @param {Function} options.documentUpdates.getUpdateFromRow - Function to extract document update from row
44
+ * @param {SendErrorRetryHandler} [options.documentUpdates.sendErrorRetryHandler] - Error handler for retrying document updates
45
+ * @param {Object} [options.awarenessUpdates] - Awareness updates configuration (optional)
46
+ * @param {ShapeStreamOptions} options.awarenessUpdates.shape - Options for the awareness updates shape stream
47
+ * @param {string|URL} options.awarenessUpdates.sendUrl - URL endpoint for sending awareness updates
48
+ * @param {awarenessProtocol.Awareness} options.awarenessUpdates.protocol - Awareness protocol instance
49
+ * @param {Function} options.awarenessUpdates.getUpdateFromRow - Function to extract awareness update from row
50
+ * @param {SendErrorRetryHandler} [options.awarenessUpdates.sendErrorRetryHandler] - Error handler for retrying awareness updates
51
+ * @param {ResumeState} [options.resumeState] - Resume state for the provider
52
+ * @param {boolean} [options.connect=true] - Whether to automatically connect upon initialization
53
+ * @param {typeof fetch} [options.fetchClient] - Custom fetch implementation to use for HTTP requests
54
+ */
55
+ constructor({
56
+ doc,
57
+ documentUpdates: documentUpdatesConfig,
58
+ awarenessUpdates: awarenessUpdatesConfig,
59
+ resumeState,
60
+ connect = true,
61
+ fetchClient
62
+ }) {
63
+ var _a;
64
+ super();
65
+ this._connected = false;
66
+ this._synced = false;
67
+ this.sendingPendingChanges = false;
68
+ this.pendingChanges = null;
69
+ this.sendingAwarenessState = false;
70
+ this.pendingAwarenessUpdate = null;
71
+ this.doc = doc;
72
+ this.documentUpdates = documentUpdatesConfig;
73
+ this.awarenessUpdates = awarenessUpdatesConfig;
74
+ this.resumeState = resumeState != null ? resumeState : {};
75
+ this.fetchClient = fetchClient;
76
+ this.exitHandler = () => {
77
+ if (env.isNode && typeof process !== `undefined`) {
78
+ process.on(`exit`, this.destroy.bind(this));
79
+ }
80
+ };
81
+ this.documentUpdateHandler = this.doc.on(
82
+ `update`,
83
+ this.applyDocumentUpdate.bind(this)
84
+ );
85
+ if (this.awarenessUpdates) {
86
+ this.awarenessUpdateHandler = this.applyAwarenessUpdate.bind(this);
87
+ this.awarenessUpdates.protocol.on(`update`, this.awarenessUpdateHandler);
88
+ }
89
+ if ((_a = this.resumeState) == null ? void 0 : _a.stableStateVector) {
90
+ this.pendingChanges = Y.encodeStateAsUpdate(
91
+ this.doc,
92
+ this.resumeState.stableStateVector
93
+ );
94
+ }
95
+ if (connect) {
96
+ this.connect();
97
+ }
98
+ }
99
+ get synced() {
100
+ return this._synced;
101
+ }
102
+ set synced(state) {
103
+ if (this._synced !== state) {
104
+ this._synced = state;
105
+ this.emit(`synced`, [state]);
106
+ this.emit(`sync`, [state]);
107
+ }
108
+ }
109
+ set connected(state) {
110
+ if (this._connected !== state) {
111
+ this._connected = state;
112
+ if (state) {
113
+ this.sendOperations();
114
+ }
115
+ this.emit(`status`, [{ status: state ? `connected` : `disconnected` }]);
116
+ }
117
+ }
118
+ get connected() {
119
+ return this._connected;
120
+ }
121
+ batch(update) {
122
+ if (this.pendingChanges) {
123
+ this.pendingChanges = Y.mergeUpdates([this.pendingChanges, update]);
124
+ } else {
125
+ this.pendingChanges = update;
126
+ }
127
+ }
128
+ destroy() {
129
+ var _a;
130
+ this.disconnect();
131
+ this.doc.off(`update`, this.documentUpdateHandler);
132
+ (_a = this.awarenessUpdates) == null ? void 0 : _a.protocol.off(`update`, this.awarenessUpdateHandler);
133
+ if (env.isNode && typeof process !== `undefined`) {
134
+ process.off(`exit`, this.exitHandler);
135
+ }
136
+ super.destroy();
137
+ }
138
+ disconnect() {
139
+ var _a;
140
+ (_a = this.unsubscribeShapes) == null ? void 0 : _a.call(this);
141
+ if (!this.connected) {
142
+ return;
143
+ }
144
+ if (this.awarenessUpdates) {
145
+ awarenessProtocol.removeAwarenessStates(
146
+ this.awarenessUpdates.protocol,
147
+ Array.from(this.awarenessUpdates.protocol.getStates().keys()).filter(
148
+ (client) => client !== this.awarenessUpdates.protocol.clientID
149
+ ),
150
+ this
151
+ );
152
+ awarenessProtocol.removeAwarenessStates(
153
+ this.awarenessUpdates.protocol,
154
+ [this.awarenessUpdates.protocol.clientID],
155
+ `local`
156
+ );
157
+ this.awarenessUpdates.protocol.setLocalState({});
158
+ }
159
+ this.emit(`connection-close`, []);
160
+ this.pendingAwarenessUpdate = null;
161
+ this.connected = false;
162
+ this.synced = false;
163
+ }
164
+ connect() {
165
+ if (this.connected) {
166
+ return;
167
+ }
168
+ const abortController = new AbortController();
169
+ const operationsStream = new ShapeStream(__spreadProps(__spreadValues(__spreadValues({}, this.documentUpdates.shape), this.resumeState.document), {
170
+ signal: abortController.signal
171
+ }));
172
+ const operationsShapeUnsubscribe = operationsStream.subscribe(
173
+ (messages) => {
174
+ this.operationsShapeHandler(
175
+ messages,
176
+ operationsStream.lastOffset,
177
+ operationsStream.shapeHandle
178
+ );
179
+ }
180
+ );
181
+ let awarenessShapeUnsubscribe;
182
+ if (this.awarenessUpdates) {
183
+ const awarenessStream = new ShapeStream(__spreadProps(__spreadValues(__spreadValues({}, this.awarenessUpdates.shape), this.resumeState.awareness), {
184
+ signal: abortController.signal
185
+ }));
186
+ awarenessShapeUnsubscribe = awarenessStream.subscribe((messages) => {
187
+ this.awarenessShapeHandler(
188
+ messages,
189
+ awarenessStream.lastOffset,
190
+ awarenessStream.shapeHandle
191
+ );
192
+ });
193
+ }
194
+ this.unsubscribeShapes = () => {
195
+ abortController.abort();
196
+ operationsShapeUnsubscribe();
197
+ awarenessShapeUnsubscribe == null ? void 0 : awarenessShapeUnsubscribe();
198
+ this.unsubscribeShapes = void 0;
199
+ };
200
+ this.emit(`status`, [{ status: `connecting` }]);
201
+ }
202
+ operationsShapeHandler(messages, offset, handle) {
203
+ for (const message of messages) {
204
+ if (isChangeMessage(message)) {
205
+ const decoder = this.documentUpdates.getUpdateFromRow(message.value);
206
+ while (decoder.pos !== decoder.arr.length) {
207
+ const operation = decoding.readVarUint8Array(decoder);
208
+ Y.applyUpdate(this.doc, operation, `server`);
209
+ }
210
+ } else if (isControlMessage(message) && message.headers.control === `up-to-date`) {
211
+ this.resumeState.document = {
212
+ offset,
213
+ handle
214
+ };
215
+ if (!this.sendingPendingChanges) {
216
+ this.synced = true;
217
+ this.resumeState.stableStateVector = Y.encodeStateVector(this.doc);
218
+ }
219
+ this.emit(`resumeState`, [this.resumeState]);
220
+ this.connected = true;
221
+ }
222
+ }
223
+ }
224
+ // TODO: add an optional throttler that batches updates
225
+ // before pushing to the server
226
+ async applyDocumentUpdate(update, origin) {
227
+ if (origin === `server`) {
228
+ return;
229
+ }
230
+ this.batch(update);
231
+ this.sendOperations();
232
+ }
233
+ async sendOperations() {
234
+ var _a;
235
+ if (!this.connected || this.sendingPendingChanges) {
236
+ return;
237
+ }
238
+ try {
239
+ this.sendingPendingChanges = true;
240
+ while (this.pendingChanges && this.pendingChanges.length > 2 && this.connected) {
241
+ const sending = this.pendingChanges;
242
+ this.pendingChanges = null;
243
+ const encoder = encoding.createEncoder();
244
+ encoding.writeVarUint8Array(encoder, sending);
245
+ const success = await send(
246
+ encoder,
247
+ this.documentUpdates.sendUrl,
248
+ (_a = this.fetchClient) != null ? _a : fetch,
249
+ this.documentUpdates.sendErrorRetryHandler
250
+ );
251
+ if (!success) {
252
+ this.batch(sending);
253
+ this.disconnect();
254
+ }
255
+ }
256
+ this.resumeState.stableStateVector = Y.encodeStateVector(this.doc);
257
+ this.emit(`resumeState`, [this.resumeState]);
258
+ } finally {
259
+ this.sendingPendingChanges = false;
260
+ }
261
+ }
262
+ async applyAwarenessUpdate(awarenessUpdate, origin) {
263
+ var _a;
264
+ if (origin !== `local` || !this.connected) {
265
+ return;
266
+ }
267
+ this.pendingAwarenessUpdate = awarenessUpdate;
268
+ if (this.sendingAwarenessState) {
269
+ return;
270
+ }
271
+ this.sendingAwarenessState = true;
272
+ try {
273
+ while (this.pendingAwarenessUpdate && this.connected) {
274
+ const update = this.pendingAwarenessUpdate;
275
+ this.pendingAwarenessUpdate = null;
276
+ const { added, updated, removed } = update;
277
+ const changedClients = added.concat(updated).concat(removed);
278
+ const encoder = encoding.createEncoder();
279
+ encoding.writeVarUint8Array(
280
+ encoder,
281
+ awarenessProtocol.encodeAwarenessUpdate(
282
+ this.awarenessUpdates.protocol,
283
+ changedClients
284
+ )
285
+ );
286
+ const success = await send(
287
+ encoder,
288
+ this.awarenessUpdates.sendUrl,
289
+ (_a = this.fetchClient) != null ? _a : fetch,
290
+ this.awarenessUpdates.sendErrorRetryHandler
291
+ );
292
+ if (!success) {
293
+ this.disconnect();
294
+ }
295
+ }
296
+ } finally {
297
+ this.sendingAwarenessState = false;
298
+ }
299
+ }
300
+ awarenessShapeHandler(messages, offset, handle) {
301
+ for (const message of messages) {
302
+ if (isChangeMessage(message)) {
303
+ if (message.headers.operation === `delete`) {
304
+ awarenessProtocol.removeAwarenessStates(
305
+ this.awarenessUpdates.protocol,
306
+ [Number(message.value.client_id)],
307
+ `remote`
308
+ );
309
+ } else {
310
+ const decoder = this.awarenessUpdates.getUpdateFromRow(message.value);
311
+ awarenessProtocol.applyAwarenessUpdate(
312
+ this.awarenessUpdates.protocol,
313
+ decoding.readVarUint8Array(decoder),
314
+ this
315
+ );
316
+ }
317
+ } else if (isControlMessage(message) && message.headers.control === `up-to-date`) {
318
+ this.resumeState.awareness = {
319
+ offset,
320
+ handle
321
+ };
322
+ this.emit(`resumeState`, [this.resumeState]);
323
+ }
324
+ }
325
+ }
326
+ };
327
+ async function send(encoder, endpoint, fetchClient, retryHandler) {
328
+ var _a;
329
+ let response;
330
+ const op = encoding.toUint8Array(encoder);
331
+ try {
332
+ response = await fetchClient(endpoint, {
333
+ method: `PUT`,
334
+ headers: {
335
+ "Content-Type": `application/octet-stream`
336
+ },
337
+ body: op
338
+ });
339
+ if (!response.ok) {
340
+ throw new Error(`Server did not return 2xx`);
341
+ }
342
+ return true;
343
+ } catch (error) {
344
+ const shouldRetry = await ((_a = retryHandler == null ? void 0 : retryHandler({
345
+ response,
346
+ error
347
+ })) != null ? _a : false);
348
+ return shouldRetry;
349
+ }
350
+ }
351
+
352
+ // src/local-storage-resume-state.ts
353
+ import { ObservableV2 as ObservableV22 } from "lib0/observable.js";
354
+ import * as buffer from "lib0/buffer";
355
+ var LocalStorageResumeStateProvider = class extends ObservableV22 {
356
+ constructor(key) {
357
+ super();
358
+ this.key = key;
359
+ }
360
+ subscribeToResumeState(provider) {
361
+ const resumeStateHandler = provider.on(`resumeState`, this.save.bind(this));
362
+ return () => provider.off(`resumeState`, resumeStateHandler);
363
+ }
364
+ save(resumeState) {
365
+ const jsonPart = JSON.stringify({
366
+ operations: resumeState.document,
367
+ awareness: resumeState.awareness
368
+ });
369
+ localStorage.setItem(this.key, jsonPart);
370
+ if (resumeState.stableStateVector) {
371
+ const vectorBase64 = buffer.toBase64(resumeState.stableStateVector);
372
+ localStorage.setItem(`${this.key}_vector`, vectorBase64);
373
+ } else {
374
+ localStorage.removeItem(`${this.key}_vector`);
375
+ }
376
+ }
377
+ load() {
378
+ if (this.resumeState) {
379
+ return this.resumeState;
380
+ }
381
+ const jsonData = localStorage.getItem(this.key);
382
+ if (!jsonData) {
383
+ this.emit(`synced`, [{}]);
384
+ } else {
385
+ this.resumeState = JSON.parse(jsonData);
386
+ const vectorData = localStorage.getItem(`${this.key}_vector`);
387
+ if (vectorData) {
388
+ this.resumeState.stableStateVector = buffer.fromBase64(vectorData);
389
+ }
390
+ this.emit(`synced`, [this.resumeState]);
391
+ }
392
+ return this.resumeState;
393
+ }
394
+ };
395
+
396
+ // src/utils.ts
397
+ import * as decoding2 from "lib0/decoding";
398
+ var hexStringToUint8Array = (hexString) => {
399
+ const cleanHexString = hexString.startsWith(`\\x`) ? hexString.slice(2) : hexString;
400
+ return new Uint8Array(
401
+ cleanHexString.match(/.{1,2}/g).map((byte) => parseInt(byte, 16))
402
+ );
403
+ };
404
+ var parseToDecoder = {
405
+ bytea: (hexString) => {
406
+ const uint8Array = hexStringToUint8Array(hexString);
407
+ return decoding2.createDecoder(uint8Array);
408
+ }
409
+ };
410
+ export {
411
+ ElectricProvider,
412
+ LocalStorageResumeStateProvider,
413
+ parseToDecoder
414
+ };
415
+ //# sourceMappingURL=index.legacy-esm.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/y-electric.ts","../src/local-storage-resume-state.ts","../src/utils.ts"],"sourcesContent":["import * as encoding from 'lib0/encoding'\nimport * as decoding from 'lib0/decoding'\nimport * as awarenessProtocol from 'y-protocols/awareness'\nimport { ObservableV2 } from 'lib0/observable'\nimport * as env from 'lib0/environment'\nimport * as Y from 'yjs'\nimport {\n GetExtensions,\n isChangeMessage,\n isControlMessage,\n Message,\n Offset,\n Row,\n ShapeStream,\n ShapeStreamOptions,\n} from '@electric-sql/client'\nimport {\n YProvider,\n ResumeState,\n SendErrorRetryHandler,\n ElectricProviderOptions,\n} from './types'\n\ntype AwarenessUpdate = {\n added: number[]\n updated: number[]\n removed: number[]\n}\n\nexport class ElectricProvider<\n RowWithDocumentUpdate extends Row<decoding.Decoder> = never,\n RowWithAwarenessUpdate extends Row<decoding.Decoder> = never,\n> extends ObservableV2<YProvider> {\n private doc: Y.Doc\n\n private documentUpdates: {\n shape: ShapeStreamOptions<GetExtensions<RowWithDocumentUpdate>>\n sendUrl: string | URL\n getUpdateFromRow: (row: RowWithDocumentUpdate) => decoding.Decoder\n sendErrorRetryHandler?: SendErrorRetryHandler\n }\n\n private awarenessUpdates?: {\n shape: ShapeStreamOptions<GetExtensions<RowWithAwarenessUpdate>>\n sendUrl: string | URL\n protocol: awarenessProtocol.Awareness\n getUpdateFromRow: (row: RowWithAwarenessUpdate) => decoding.Decoder\n sendErrorRetryHandler?: SendErrorRetryHandler\n }\n\n private _connected: boolean = false\n private _synced: boolean = false\n\n private resumeState: ResumeState\n private sendingPendingChanges: boolean = false\n private pendingChanges: Uint8Array | null = null\n private sendingAwarenessState: boolean = false\n private pendingAwarenessUpdate: AwarenessUpdate | null = null\n\n private documentUpdateHandler: (\n update: Uint8Array,\n origin: unknown,\n doc: Y.Doc,\n transaction: Y.Transaction\n ) => void\n private awarenessUpdateHandler?: (\n update: AwarenessUpdate,\n origin: unknown\n ) => void\n\n private exitHandler: () => void\n private unsubscribeShapes?: () => void\n\n private fetchClient?: typeof fetch\n\n /**\n * Creates a new ElectricProvider instance that connects YJS documents to Electric SQL.\n *\n * @constructor\n * @param {ElectricProviderOptions} options - Configuration options for the provider\n * @param {Y.Doc} options.doc - The YJS document to be synchronized\n * @param {Object} options.documentUpdates - Document updates configuration\n * @param {ShapeStreamOptions} options.documentUpdates.shape - Options for the document updates shape stream\n * @param {string|URL} options.documentUpdates.sendUrl - URL endpoint for sending document updates\n * @param {Function} options.documentUpdates.getUpdateFromRow - Function to extract document update from row\n * @param {SendErrorRetryHandler} [options.documentUpdates.sendErrorRetryHandler] - Error handler for retrying document updates\n * @param {Object} [options.awarenessUpdates] - Awareness updates configuration (optional)\n * @param {ShapeStreamOptions} options.awarenessUpdates.shape - Options for the awareness updates shape stream\n * @param {string|URL} options.awarenessUpdates.sendUrl - URL endpoint for sending awareness updates\n * @param {awarenessProtocol.Awareness} options.awarenessUpdates.protocol - Awareness protocol instance\n * @param {Function} options.awarenessUpdates.getUpdateFromRow - Function to extract awareness update from row\n * @param {SendErrorRetryHandler} [options.awarenessUpdates.sendErrorRetryHandler] - Error handler for retrying awareness updates\n * @param {ResumeState} [options.resumeState] - Resume state for the provider\n * @param {boolean} [options.connect=true] - Whether to automatically connect upon initialization\n * @param {typeof fetch} [options.fetchClient] - Custom fetch implementation to use for HTTP requests\n */\n constructor({\n doc,\n documentUpdates: documentUpdatesConfig,\n awarenessUpdates: awarenessUpdatesConfig,\n resumeState,\n connect = true,\n fetchClient,\n }: ElectricProviderOptions<RowWithDocumentUpdate, RowWithAwarenessUpdate>) {\n super()\n\n this.doc = doc\n this.documentUpdates = documentUpdatesConfig\n this.awarenessUpdates = awarenessUpdatesConfig\n this.resumeState = resumeState ?? {}\n\n this.fetchClient = fetchClient\n\n this.exitHandler = () => {\n if (env.isNode && typeof process !== `undefined`) {\n process.on(`exit`, this.destroy.bind(this))\n }\n }\n\n this.documentUpdateHandler = this.doc.on(\n `update`,\n this.applyDocumentUpdate.bind(this)\n )\n if (this.awarenessUpdates) {\n this.awarenessUpdateHandler = this.applyAwarenessUpdate.bind(this)\n this.awarenessUpdates.protocol.on(`update`, this.awarenessUpdateHandler!)\n }\n\n // enqueue unsynced changes from document if the\n // resume state provides the document state vector\n if (this.resumeState?.stableStateVector) {\n this.pendingChanges = Y.encodeStateAsUpdate(\n this.doc,\n this.resumeState.stableStateVector\n )\n }\n\n if (connect) {\n this.connect()\n }\n }\n\n get synced() {\n return this._synced\n }\n\n set synced(state) {\n if (this._synced !== state) {\n this._synced = state\n this.emit(`synced`, [state])\n this.emit(`sync`, [state])\n }\n }\n\n set connected(state) {\n if (this._connected !== state) {\n this._connected = state\n if (state) {\n this.sendOperations()\n }\n this.emit(`status`, [{ status: state ? `connected` : `disconnected` }])\n }\n }\n\n get connected() {\n return this._connected\n }\n\n private batch(update: Uint8Array) {\n if (this.pendingChanges) {\n this.pendingChanges = Y.mergeUpdates([this.pendingChanges, update])\n } else {\n this.pendingChanges = update\n }\n }\n\n destroy() {\n this.disconnect()\n\n this.doc.off(`update`, this.documentUpdateHandler)\n this.awarenessUpdates?.protocol.off(`update`, this.awarenessUpdateHandler!)\n\n if (env.isNode && typeof process !== `undefined`) {\n process.off(`exit`, this.exitHandler!)\n }\n super.destroy()\n }\n\n disconnect() {\n this.unsubscribeShapes?.()\n\n if (!this.connected) {\n return\n }\n\n if (this.awarenessUpdates) {\n awarenessProtocol.removeAwarenessStates(\n this.awarenessUpdates.protocol,\n Array.from(this.awarenessUpdates.protocol.getStates().keys()).filter(\n (client) => client !== this.awarenessUpdates!.protocol.clientID\n ),\n this\n )\n\n // try to notifying other clients that we are disconnecting\n awarenessProtocol.removeAwarenessStates(\n this.awarenessUpdates.protocol,\n [this.awarenessUpdates.protocol.clientID],\n `local`\n )\n\n this.awarenessUpdates.protocol.setLocalState({})\n }\n\n // TODO: await for events before closing\n this.emit(`connection-close`, [])\n\n this.pendingAwarenessUpdate = null\n\n this.connected = false\n this.synced = false\n }\n\n connect() {\n if (this.connected) {\n return\n }\n const abortController = new AbortController()\n\n const operationsStream = new ShapeStream<RowWithDocumentUpdate>({\n ...this.documentUpdates.shape,\n ...this.resumeState.document,\n signal: abortController.signal,\n })\n\n const operationsShapeUnsubscribe = operationsStream.subscribe(\n (messages) => {\n this.operationsShapeHandler(\n messages,\n operationsStream.lastOffset,\n operationsStream.shapeHandle!\n )\n }\n )\n\n let awarenessShapeUnsubscribe: () => void | undefined\n if (this.awarenessUpdates) {\n const awarenessStream = new ShapeStream<RowWithAwarenessUpdate>({\n ...this.awarenessUpdates.shape,\n ...this.resumeState.awareness,\n signal: abortController.signal,\n })\n\n awarenessShapeUnsubscribe = awarenessStream.subscribe((messages) => {\n this.awarenessShapeHandler(\n messages,\n awarenessStream.lastOffset,\n awarenessStream.shapeHandle!\n )\n })\n }\n\n this.unsubscribeShapes = () => {\n abortController.abort()\n operationsShapeUnsubscribe()\n awarenessShapeUnsubscribe?.()\n this.unsubscribeShapes = undefined\n }\n\n this.emit(`status`, [{ status: `connecting` }])\n }\n\n private operationsShapeHandler(\n messages: Message<RowWithDocumentUpdate>[],\n offset: Offset,\n handle: string\n ) {\n for (const message of messages) {\n if (isChangeMessage(message)) {\n const decoder = this.documentUpdates.getUpdateFromRow(message.value)\n while (decoder.pos !== decoder.arr.length) {\n const operation = decoding.readVarUint8Array(decoder)\n Y.applyUpdate(this.doc, operation, `server`)\n }\n } else if (\n isControlMessage(message) &&\n message.headers.control === `up-to-date`\n ) {\n this.resumeState.document = {\n offset,\n handle,\n }\n\n if (!this.sendingPendingChanges) {\n this.synced = true\n this.resumeState.stableStateVector = Y.encodeStateVector(this.doc)\n }\n this.emit(`resumeState`, [this.resumeState])\n this.connected = true\n }\n }\n }\n\n // TODO: add an optional throttler that batches updates\n // before pushing to the server\n private async applyDocumentUpdate(update: Uint8Array, origin: unknown) {\n // don't re-send updates from electric\n if (origin === `server`) {\n return\n }\n\n this.batch(update)\n this.sendOperations()\n }\n\n private async sendOperations() {\n if (!this.connected || this.sendingPendingChanges) {\n return\n }\n\n try {\n this.sendingPendingChanges = true\n while (\n this.pendingChanges &&\n this.pendingChanges.length > 2 &&\n this.connected\n ) {\n const sending = this.pendingChanges\n this.pendingChanges = null\n\n const encoder = encoding.createEncoder()\n encoding.writeVarUint8Array(encoder, sending)\n\n const success = await send(\n encoder,\n this.documentUpdates.sendUrl,\n this.fetchClient ?? fetch,\n this.documentUpdates.sendErrorRetryHandler\n )\n if (!success) {\n this.batch(sending)\n this.disconnect()\n }\n }\n // no more pending changes, move stableStateVector forward\n this.resumeState.stableStateVector = Y.encodeStateVector(this.doc)\n this.emit(`resumeState`, [this.resumeState])\n } finally {\n this.sendingPendingChanges = false\n }\n }\n\n private async applyAwarenessUpdate(\n awarenessUpdate: AwarenessUpdate,\n origin: unknown\n ) {\n if (origin !== `local` || !this.connected) {\n return\n }\n\n this.pendingAwarenessUpdate = awarenessUpdate\n\n if (this.sendingAwarenessState) {\n return\n }\n\n this.sendingAwarenessState = true\n\n try {\n while (this.pendingAwarenessUpdate && this.connected) {\n const update = this.pendingAwarenessUpdate\n this.pendingAwarenessUpdate = null\n\n const { added, updated, removed } = update\n const changedClients = added.concat(updated).concat(removed)\n const encoder = encoding.createEncoder()\n\n encoding.writeVarUint8Array(\n encoder,\n awarenessProtocol.encodeAwarenessUpdate(\n this.awarenessUpdates!.protocol,\n changedClients\n )\n )\n const success = await send(\n encoder,\n this.awarenessUpdates!.sendUrl,\n this.fetchClient ?? fetch,\n this.awarenessUpdates!.sendErrorRetryHandler\n )\n if (!success) {\n this.disconnect()\n }\n }\n } finally {\n this.sendingAwarenessState = false\n }\n }\n\n private awarenessShapeHandler(\n messages: Message<RowWithAwarenessUpdate>[],\n offset: Offset,\n handle: string\n ) {\n for (const message of messages) {\n if (isChangeMessage(message)) {\n if (message.headers.operation === `delete`) {\n awarenessProtocol.removeAwarenessStates(\n this.awarenessUpdates!.protocol,\n [Number(message.value.client_id)],\n `remote`\n )\n } else {\n const decoder = this.awarenessUpdates!.getUpdateFromRow(message.value)\n awarenessProtocol.applyAwarenessUpdate(\n this.awarenessUpdates!.protocol,\n decoding.readVarUint8Array(decoder),\n this\n )\n }\n } else if (\n isControlMessage(message) &&\n message.headers.control === `up-to-date`\n ) {\n this.resumeState.awareness = {\n offset: offset,\n handle: handle,\n }\n this.emit(`resumeState`, [this.resumeState])\n }\n }\n }\n}\n\nasync function send(\n encoder: encoding.Encoder,\n endpoint: string | URL,\n fetchClient: typeof fetch,\n retryHandler?: SendErrorRetryHandler\n): Promise<boolean> {\n let response: Response | undefined\n const op = encoding.toUint8Array(encoder)\n\n try {\n response = await fetchClient(endpoint!, {\n method: `PUT`,\n headers: {\n 'Content-Type': `application/octet-stream`,\n },\n body: op,\n })\n\n if (!response.ok) {\n throw new Error(`Server did not return 2xx`)\n }\n\n return true\n } catch (error) {\n const shouldRetry = await (retryHandler?.({\n response,\n error,\n }) ?? false)\n return shouldRetry\n }\n}\n","import { ResumeState, ElectricResumeStateProvider } from './types'\nimport { ObservableV2 } from 'lib0/observable.js'\nimport { ElectricProvider } from './y-electric'\nimport * as buffer from 'lib0/buffer'\n\n/**\n * A ResumeStateProvider implementation using LocalStorage.\n * This is a reference implementation that can be used as a starting point\n * for implementing other ResumeStateProviders.\n */\nexport class LocalStorageResumeStateProvider extends ObservableV2<ElectricResumeStateProvider> {\n private key: string\n private resumeState?: ResumeState\n\n constructor(key: string) {\n super()\n this.key = key\n }\n\n subscribeToResumeState(provider: ElectricProvider): () => void {\n const resumeStateHandler = provider.on(`resumeState`, this.save.bind(this))\n return () => provider.off(`resumeState`, resumeStateHandler)\n }\n\n save(resumeState: ResumeState) {\n const jsonPart = JSON.stringify({\n operations: resumeState.document,\n awareness: resumeState.awareness,\n })\n localStorage.setItem(this.key, jsonPart)\n\n if (resumeState.stableStateVector) {\n const vectorBase64 = buffer.toBase64(resumeState.stableStateVector)\n localStorage.setItem(`${this.key}_vector`, vectorBase64)\n } else {\n // ensure vector is removed\n localStorage.removeItem(`${this.key}_vector`)\n }\n }\n\n load(): ResumeState {\n if (this.resumeState) {\n return this.resumeState\n }\n\n const jsonData = localStorage.getItem(this.key)\n if (!jsonData) {\n this.emit(`synced`, [{}])\n } else {\n this.resumeState = JSON.parse(jsonData)\n\n const vectorData = localStorage.getItem(`${this.key}_vector`)\n if (vectorData) {\n this.resumeState!.stableStateVector = buffer.fromBase64(vectorData)\n }\n\n this.emit(`synced`, [this.resumeState!])\n }\n\n return this.resumeState!\n }\n}\n","import * as decoding from 'lib0/decoding'\n\n/**\n * Convert a hex string from PostgreSQL's bytea format to a Uint8Array\n */\nconst hexStringToUint8Array = (hexString: string) => {\n const cleanHexString = hexString.startsWith(`\\\\x`)\n ? hexString.slice(2)\n : hexString\n return new Uint8Array(\n cleanHexString.match(/.{1,2}/g)!.map((byte) => parseInt(byte, 16))\n )\n}\n\n/**\n * Utility to parse hex string bytea data to a decoder for YJS operations\n */\nexport const parseToDecoder = {\n bytea: (hexString: string) => {\n const uint8Array = hexStringToUint8Array(hexString)\n return decoding.createDecoder(uint8Array)\n },\n}\n"],"mappings":"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","names":["ObservableV2","decoding"]}