@dxos/edge-client 0.8.2-main.12df754 → 0.8.2-main.36232bc

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 (38) hide show
  1. package/dist/lib/browser/chunk-XS3TKGM4.mjs +545 -0
  2. package/dist/lib/browser/chunk-XS3TKGM4.mjs.map +7 -0
  3. package/dist/lib/browser/index.mjs +60 -282
  4. package/dist/lib/browser/index.mjs.map +4 -4
  5. package/dist/lib/browser/meta.json +1 -1
  6. package/dist/lib/browser/testing/index.mjs +32 -20
  7. package/dist/lib/browser/testing/index.mjs.map +3 -3
  8. package/dist/lib/node/chunk-ZURVCY7K.cjs +577 -0
  9. package/dist/lib/node/chunk-ZURVCY7K.cjs.map +7 -0
  10. package/dist/lib/node/index.cjs +50 -281
  11. package/dist/lib/node/index.cjs.map +4 -4
  12. package/dist/lib/node/meta.json +1 -1
  13. package/dist/lib/node/testing/index.cjs +31 -21
  14. package/dist/lib/node/testing/index.cjs.map +3 -3
  15. package/dist/lib/node-esm/chunk-HNRMNQPG.mjs +547 -0
  16. package/dist/lib/node-esm/chunk-HNRMNQPG.mjs.map +7 -0
  17. package/dist/lib/node-esm/index.mjs +60 -282
  18. package/dist/lib/node-esm/index.mjs.map +4 -4
  19. package/dist/lib/node-esm/meta.json +1 -1
  20. package/dist/lib/node-esm/testing/index.mjs +32 -20
  21. package/dist/lib/node-esm/testing/index.mjs.map +3 -3
  22. package/dist/types/src/edge-ws-connection.d.ts +5 -0
  23. package/dist/types/src/edge-ws-connection.d.ts.map +1 -1
  24. package/dist/types/src/edge-ws-muxer.d.ts +23 -0
  25. package/dist/types/src/edge-ws-muxer.d.ts.map +1 -0
  26. package/dist/types/src/testing/test-utils.d.ts +6 -2
  27. package/dist/types/src/testing/test-utils.d.ts.map +1 -1
  28. package/package.json +14 -14
  29. package/src/edge-client.ts +2 -2
  30. package/src/edge-ws-connection.ts +38 -15
  31. package/src/edge-ws-muxer.ts +187 -0
  32. package/src/testing/test-utils.ts +33 -26
  33. package/dist/lib/browser/chunk-ZWJXA37R.mjs +0 -113
  34. package/dist/lib/browser/chunk-ZWJXA37R.mjs.map +0 -7
  35. package/dist/lib/node/chunk-ANV2HBEH.cjs +0 -136
  36. package/dist/lib/node/chunk-ANV2HBEH.cjs.map +0 -7
  37. package/dist/lib/node-esm/chunk-HNVT57AU.mjs +0 -115
  38. package/dist/lib/node-esm/chunk-HNVT57AU.mjs.map +0 -7
@@ -0,0 +1,577 @@
1
+ "use strict";
2
+ var __create = Object.create;
3
+ var __defProp = Object.defineProperty;
4
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
5
+ var __getOwnPropNames = Object.getOwnPropertyNames;
6
+ var __getProtoOf = Object.getPrototypeOf;
7
+ var __hasOwnProp = Object.prototype.hasOwnProperty;
8
+ var __export = (target, all) => {
9
+ for (var name in all)
10
+ __defProp(target, name, { get: all[name], enumerable: true });
11
+ };
12
+ var __copyProps = (to, from, except, desc) => {
13
+ if (from && typeof from === "object" || typeof from === "function") {
14
+ for (let key of __getOwnPropNames(from))
15
+ if (!__hasOwnProp.call(to, key) && key !== except)
16
+ __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
17
+ }
18
+ return to;
19
+ };
20
+ var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
21
+ // If the importer is in node compatibility mode or this is not an ESM
22
+ // file that has been converted to a CommonJS file using a Babel-
23
+ // compatible transform (i.e. "__esModule" has not been set), then set
24
+ // "default" to the CommonJS "module.exports" for node compatibility.
25
+ isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
26
+ mod
27
+ ));
28
+ var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
29
+ var chunk_ZURVCY7K_exports = {};
30
+ __export(chunk_ZURVCY7K_exports, {
31
+ EDGE_WEBSOCKET_PROTOCOL_V1: () => EDGE_WEBSOCKET_PROTOCOL_V1,
32
+ EdgeWsConnection: () => EdgeWsConnection,
33
+ Protocol: () => Protocol,
34
+ WebSocketMuxer: () => WebSocketMuxer,
35
+ getTypename: () => getTypename,
36
+ protocol: () => protocol,
37
+ toUint8Array: () => toUint8Array
38
+ });
39
+ module.exports = __toCommonJS(chunk_ZURVCY7K_exports);
40
+ var import_invariant = require("@dxos/invariant");
41
+ var import_buf = require("@dxos/protocols/buf");
42
+ var import_messenger_pb = require("@dxos/protocols/buf/dxos/edge/messenger_pb");
43
+ var import_util = require("@dxos/util");
44
+ var import_buf2 = require("@dxos/protocols/buf");
45
+ var import_messenger_pb2 = require("@dxos/protocols/buf/dxos/edge/messenger_pb");
46
+ var import_isomorphic_ws = __toESM(require("isomorphic-ws"));
47
+ var import_async = require("@dxos/async");
48
+ var import_log = require("@dxos/log");
49
+ var import_buf3 = require("@dxos/protocols/buf");
50
+ var import_messenger_pb3 = require("@dxos/protocols/buf/dxos/edge/messenger_pb");
51
+ var import_isomorphic_ws2 = __toESM(require("isomorphic-ws"));
52
+ var import_async2 = require("@dxos/async");
53
+ var import_context = require("@dxos/context");
54
+ var import_invariant2 = require("@dxos/invariant");
55
+ var import_log2 = require("@dxos/log");
56
+ var import_buf4 = require("@dxos/protocols/buf");
57
+ var import_messenger_pb4 = require("@dxos/protocols/buf/dxos/edge/messenger_pb");
58
+ var __dxlog_file = "/home/runner/work/dxos/dxos/packages/core/mesh/edge-client/src/protocol.ts";
59
+ var getTypename = (typeName) => `type.googleapis.com/${typeName}`;
60
+ var Protocol = class {
61
+ constructor(types) {
62
+ this._typeRegistry = import_buf.buf.createRegistry(...types);
63
+ }
64
+ get typeRegistry() {
65
+ return this._typeRegistry;
66
+ }
67
+ toJson(message) {
68
+ try {
69
+ return import_buf.buf.toJson(import_messenger_pb.MessageSchema, message, {
70
+ registry: this.typeRegistry
71
+ });
72
+ } catch (err) {
73
+ return {
74
+ type: this.getPayloadType(message)
75
+ };
76
+ }
77
+ }
78
+ /**
79
+ * Return the payload with the given type.
80
+ */
81
+ getPayload(message, type) {
82
+ (0, import_invariant.invariant)(message.payload, void 0, {
83
+ F: __dxlog_file,
84
+ L: 40,
85
+ S: this,
86
+ A: [
87
+ "message.payload",
88
+ ""
89
+ ]
90
+ });
91
+ const payloadTypename = this.getPayloadType(message);
92
+ if (type && type.typeName !== payloadTypename) {
93
+ throw new Error(`Unexpected payload type: ${payloadTypename}; expected ${type.typeName}`);
94
+ }
95
+ (0, import_invariant.invariant)(import_buf.bufWkt.anyIs(message.payload, type), `Unexpected payload type: ${payloadTypename}}`, {
96
+ F: __dxlog_file,
97
+ L: 46,
98
+ S: this,
99
+ A: [
100
+ "bufWkt.anyIs(message.payload, type)",
101
+ "`Unexpected payload type: ${payloadTypename}}`"
102
+ ]
103
+ });
104
+ const payload = import_buf.bufWkt.anyUnpack(message.payload, this.typeRegistry);
105
+ (0, import_invariant.invariant)(payload, `Empty payload: ${payloadTypename}}`, {
106
+ F: __dxlog_file,
107
+ L: 48,
108
+ S: this,
109
+ A: [
110
+ "payload",
111
+ "`Empty payload: ${payloadTypename}}`"
112
+ ]
113
+ });
114
+ return payload;
115
+ }
116
+ /**
117
+ * Get the payload type.
118
+ */
119
+ getPayloadType(message) {
120
+ if (!message.payload) {
121
+ return void 0;
122
+ }
123
+ const [, type] = message.payload.typeUrl.split("/");
124
+ return type;
125
+ }
126
+ /**
127
+ * Create a packed message.
128
+ */
129
+ createMessage(type, { source, target, payload, serviceId }) {
130
+ return import_buf.buf.create(import_messenger_pb.MessageSchema, {
131
+ timestamp: (/* @__PURE__ */ new Date()).toISOString(),
132
+ source,
133
+ target,
134
+ serviceId,
135
+ payload: payload ? import_buf.bufWkt.anyPack(type, import_buf.buf.create(type, payload)) : void 0
136
+ });
137
+ }
138
+ };
139
+ var toUint8Array = async (data) => {
140
+ if (data instanceof Buffer) {
141
+ return (0, import_util.bufferToArray)(data);
142
+ }
143
+ if (data instanceof Blob) {
144
+ return new Uint8Array(await data.arrayBuffer());
145
+ }
146
+ throw new Error(`Unexpected datatype: ${data}`);
147
+ };
148
+ var protocol = new Protocol([
149
+ import_messenger_pb2.SwarmRequestSchema,
150
+ import_messenger_pb2.SwarmResponseSchema,
151
+ import_messenger_pb2.TextMessageSchema,
152
+ import_buf2.bufWkt.AnySchema
153
+ ]);
154
+ var __dxlog_file2 = "/home/runner/work/dxos/dxos/packages/core/mesh/edge-client/src/edge-ws-muxer.ts";
155
+ var FLAG_SEGMENT_SEQ = 1;
156
+ var FLAG_SEGMENT_SEQ_TERMINATED = 1 << 1;
157
+ var CLOUDFLARE_MESSAGE_LENGTH_LIMIT = 1024 * 1024;
158
+ var MAX_CHUNK_LENGTH = 16384;
159
+ var MAX_BUFFERED_AMOUNT = CLOUDFLARE_MESSAGE_LENGTH_LIMIT;
160
+ var BUFFER_FULL_BACKOFF_TIMEOUT = 100;
161
+ var WebSocketMuxer = class {
162
+ constructor(_ws) {
163
+ this._ws = _ws;
164
+ this._incomingMessageAccumulator = /* @__PURE__ */ new Map();
165
+ this._outgoingMessageChunks = /* @__PURE__ */ new Map();
166
+ this._serviceToChannel = /* @__PURE__ */ new Map();
167
+ }
168
+ /**
169
+ * Resolves when all the message chunks get enqueued for sending.
170
+ */
171
+ async send(message) {
172
+ const binary = import_buf3.buf.toBinary(import_messenger_pb3.MessageSchema, message);
173
+ const channelId = this._resolveChannel(message);
174
+ if (channelId == null && binary.length > CLOUDFLARE_MESSAGE_LENGTH_LIMIT) {
175
+ import_log.log.error("Large message dropped because channel resolution failed.", {
176
+ byteLength: binary.byteLength,
177
+ serviceId: message.serviceId,
178
+ payload: protocol.getPayloadType(message)
179
+ }, {
180
+ F: __dxlog_file2,
181
+ L: 52,
182
+ S: this,
183
+ C: (f, a) => f(...a)
184
+ });
185
+ return;
186
+ }
187
+ if (channelId == null || binary.length < MAX_CHUNK_LENGTH) {
188
+ const flags = Buffer.from([
189
+ 0
190
+ ]);
191
+ this._ws.send(Buffer.concat([
192
+ flags,
193
+ binary
194
+ ]));
195
+ return;
196
+ }
197
+ const terminatorSentTrigger = new import_async.Trigger();
198
+ const messageChunks = [];
199
+ for (let i = 0; i < binary.length; i += MAX_CHUNK_LENGTH) {
200
+ const chunk = binary.slice(i, i + MAX_CHUNK_LENGTH);
201
+ const isLastChunk = i + MAX_CHUNK_LENGTH < binary.length;
202
+ if (isLastChunk) {
203
+ const flags = Buffer.from([
204
+ FLAG_SEGMENT_SEQ | FLAG_SEGMENT_SEQ_TERMINATED,
205
+ channelId
206
+ ]);
207
+ messageChunks.push({
208
+ payload: Buffer.concat([
209
+ flags,
210
+ chunk
211
+ ]),
212
+ trigger: terminatorSentTrigger
213
+ });
214
+ } else {
215
+ const flags = Buffer.from([
216
+ FLAG_SEGMENT_SEQ
217
+ ]);
218
+ messageChunks.push({
219
+ payload: Buffer.concat([
220
+ flags,
221
+ chunk
222
+ ])
223
+ });
224
+ }
225
+ }
226
+ const queuedMessages = this._outgoingMessageChunks.get(channelId);
227
+ if (queuedMessages) {
228
+ queuedMessages.push(...messageChunks);
229
+ } else {
230
+ this._outgoingMessageChunks.set(channelId, messageChunks);
231
+ }
232
+ this._sendChunkedMessages();
233
+ return terminatorSentTrigger.wait();
234
+ }
235
+ receiveData(data) {
236
+ if ((data[0] & FLAG_SEGMENT_SEQ) === 0) {
237
+ return import_buf3.buf.fromBinary(import_messenger_pb3.MessageSchema, data.slice(1));
238
+ }
239
+ const [flags, channelId, ...payload] = data;
240
+ let chunkAccumulator = this._incomingMessageAccumulator.get(channelId);
241
+ if (chunkAccumulator) {
242
+ chunkAccumulator.push(Buffer.from(payload));
243
+ } else {
244
+ chunkAccumulator = [
245
+ Buffer.from(payload)
246
+ ];
247
+ this._incomingMessageAccumulator.set(channelId, chunkAccumulator);
248
+ }
249
+ if ((flags & FLAG_SEGMENT_SEQ_TERMINATED) === 0) {
250
+ return void 0;
251
+ }
252
+ const message = import_buf3.buf.fromBinary(import_messenger_pb3.MessageSchema, Buffer.concat(chunkAccumulator));
253
+ this._incomingMessageAccumulator.delete(channelId);
254
+ return message;
255
+ }
256
+ destroy() {
257
+ if (this._sendTimeout) {
258
+ clearTimeout(this._sendTimeout);
259
+ this._sendTimeout = void 0;
260
+ }
261
+ for (const channelChunks of this._outgoingMessageChunks.values()) {
262
+ channelChunks.forEach((chunk) => chunk.trigger?.wake());
263
+ }
264
+ this._outgoingMessageChunks.clear();
265
+ this._incomingMessageAccumulator.clear();
266
+ this._serviceToChannel.clear();
267
+ }
268
+ _sendChunkedMessages() {
269
+ if (this._sendTimeout) {
270
+ return;
271
+ }
272
+ const send = () => {
273
+ if (this._ws.readyState === import_isomorphic_ws.default.CLOSING || this._ws.readyState === import_isomorphic_ws.default.CLOSED) {
274
+ import_log.log.warn("send called for closed websocket", void 0, {
275
+ F: __dxlog_file2,
276
+ L: 135,
277
+ S: this,
278
+ C: (f, a) => f(...a)
279
+ });
280
+ this._sendTimeout = void 0;
281
+ return;
282
+ }
283
+ let timeout = 0;
284
+ const emptyChannels = [];
285
+ for (const [channelId, messages] of this._outgoingMessageChunks.entries()) {
286
+ if (this._ws.bufferedAmount + MAX_CHUNK_LENGTH > MAX_BUFFERED_AMOUNT) {
287
+ timeout = BUFFER_FULL_BACKOFF_TIMEOUT;
288
+ break;
289
+ }
290
+ const nextMessage = messages.shift();
291
+ if (nextMessage) {
292
+ this._ws.send(nextMessage.payload);
293
+ nextMessage.trigger?.wake();
294
+ } else {
295
+ emptyChannels.push(channelId);
296
+ }
297
+ }
298
+ emptyChannels.forEach((channelId) => this._outgoingMessageChunks.delete(channelId));
299
+ if (this._outgoingMessageChunks.size > 0) {
300
+ this._sendTimeout = setTimeout(send, timeout);
301
+ } else {
302
+ this._sendTimeout = void 0;
303
+ }
304
+ };
305
+ this._sendTimeout = setTimeout(send);
306
+ }
307
+ _resolveChannel(message) {
308
+ if (!message.serviceId) {
309
+ return void 0;
310
+ }
311
+ let id = this._serviceToChannel.get(message.serviceId);
312
+ if (!id) {
313
+ id = this._serviceToChannel.size + 1;
314
+ this._serviceToChannel.set(message.serviceId, id);
315
+ }
316
+ return id;
317
+ }
318
+ };
319
+ function _ts_decorate(decorators, target, key, desc) {
320
+ var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
321
+ if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
322
+ else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
323
+ return c > 3 && r && Object.defineProperty(target, key, r), r;
324
+ }
325
+ var __dxlog_file3 = "/home/runner/work/dxos/dxos/packages/core/mesh/edge-client/src/edge-ws-connection.ts";
326
+ var SIGNAL_KEEPALIVE_INTERVAL = 4e3;
327
+ var SIGNAL_KEEPALIVE_TIMEOUT = 12e3;
328
+ var EDGE_WEBSOCKET_PROTOCOL_V0 = "edge-ws-v0";
329
+ var EDGE_WEBSOCKET_PROTOCOL_V1 = "edge-ws-v1";
330
+ var EdgeWsConnection = class extends import_context.Resource {
331
+ constructor(_identity, _connectionInfo, _callbacks) {
332
+ super();
333
+ this._identity = _identity;
334
+ this._connectionInfo = _connectionInfo;
335
+ this._callbacks = _callbacks;
336
+ }
337
+ get info() {
338
+ return {
339
+ open: this.isOpen,
340
+ identity: this._identity.identityKey,
341
+ device: this._identity.peerKey
342
+ };
343
+ }
344
+ send(message) {
345
+ (0, import_invariant2.invariant)(this._ws, void 0, {
346
+ F: __dxlog_file3,
347
+ L: 57,
348
+ S: this,
349
+ A: [
350
+ "this._ws",
351
+ ""
352
+ ]
353
+ });
354
+ (0, import_invariant2.invariant)(this._wsMuxer, void 0, {
355
+ F: __dxlog_file3,
356
+ L: 58,
357
+ S: this,
358
+ A: [
359
+ "this._wsMuxer",
360
+ ""
361
+ ]
362
+ });
363
+ (0, import_log2.log)("sending...", {
364
+ peerKey: this._identity.peerKey,
365
+ payload: protocol.getPayloadType(message)
366
+ }, {
367
+ F: __dxlog_file3,
368
+ L: 59,
369
+ S: this,
370
+ C: (f, a) => f(...a)
371
+ });
372
+ if (this._ws?.protocol.includes(EDGE_WEBSOCKET_PROTOCOL_V0)) {
373
+ const binary = import_buf4.buf.toBinary(import_messenger_pb4.MessageSchema, message);
374
+ if (binary.length > CLOUDFLARE_MESSAGE_LENGTH_LIMIT) {
375
+ import_log2.log.error("Message dropped because it was too large (>1MB).", {
376
+ byteLength: binary.byteLength,
377
+ serviceId: message.serviceId,
378
+ payload: protocol.getPayloadType(message)
379
+ }, {
380
+ F: __dxlog_file3,
381
+ L: 63,
382
+ S: this,
383
+ C: (f, a) => f(...a)
384
+ });
385
+ return;
386
+ }
387
+ this._ws.send(binary);
388
+ } else {
389
+ this._wsMuxer.send(message).catch((e) => import_log2.log.catch(e, void 0, {
390
+ F: __dxlog_file3,
391
+ L: 72,
392
+ S: this,
393
+ C: (f, a) => f(...a)
394
+ }));
395
+ }
396
+ }
397
+ async _open() {
398
+ const baseProtocols = [
399
+ EDGE_WEBSOCKET_PROTOCOL_V0,
400
+ EDGE_WEBSOCKET_PROTOCOL_V1
401
+ ];
402
+ this._ws = new import_isomorphic_ws2.default(this._connectionInfo.url.toString(), this._connectionInfo.protocolHeader ? [
403
+ ...baseProtocols,
404
+ this._connectionInfo.protocolHeader
405
+ ] : [
406
+ ...baseProtocols
407
+ ]);
408
+ const muxer = new WebSocketMuxer(this._ws);
409
+ this._wsMuxer = muxer;
410
+ this._ws.onopen = () => {
411
+ if (this.isOpen) {
412
+ (0, import_log2.log)("connected", void 0, {
413
+ F: __dxlog_file3,
414
+ L: 89,
415
+ S: this,
416
+ C: (f, a) => f(...a)
417
+ });
418
+ this._callbacks.onConnected();
419
+ this._scheduleHeartbeats();
420
+ } else {
421
+ import_log2.log.verbose("connected after becoming inactive", {
422
+ currentIdentity: this._identity
423
+ }, {
424
+ F: __dxlog_file3,
425
+ L: 93,
426
+ S: this,
427
+ C: (f, a) => f(...a)
428
+ });
429
+ }
430
+ };
431
+ this._ws.onclose = (event) => {
432
+ if (this.isOpen) {
433
+ import_log2.log.warn("disconnected while being open", {
434
+ code: event.code,
435
+ reason: event.reason
436
+ }, {
437
+ F: __dxlog_file3,
438
+ L: 98,
439
+ S: this,
440
+ C: (f, a) => f(...a)
441
+ });
442
+ this._callbacks.onRestartRequired();
443
+ muxer.destroy();
444
+ }
445
+ };
446
+ this._ws.onerror = (event) => {
447
+ if (this.isOpen) {
448
+ import_log2.log.warn("edge connection socket error", {
449
+ error: event.error,
450
+ info: event.message
451
+ }, {
452
+ F: __dxlog_file3,
453
+ L: 105,
454
+ S: this,
455
+ C: (f, a) => f(...a)
456
+ });
457
+ this._callbacks.onRestartRequired();
458
+ } else {
459
+ import_log2.log.verbose("error ignored on closed connection", {
460
+ error: event.error
461
+ }, {
462
+ F: __dxlog_file3,
463
+ L: 108,
464
+ S: this,
465
+ C: (f, a) => f(...a)
466
+ });
467
+ }
468
+ };
469
+ this._ws.onmessage = async (event) => {
470
+ if (!this.isOpen) {
471
+ import_log2.log.verbose("message ignored on closed connection", {
472
+ event: event.type
473
+ }, {
474
+ F: __dxlog_file3,
475
+ L: 116,
476
+ S: this,
477
+ C: (f, a) => f(...a)
478
+ });
479
+ return;
480
+ }
481
+ if (event.data === "__pong__") {
482
+ this._rescheduleHeartbeatTimeout();
483
+ return;
484
+ }
485
+ const bytes = await toUint8Array(event.data);
486
+ if (!this.isOpen) {
487
+ return;
488
+ }
489
+ const message = this._ws?.protocol?.includes(EDGE_WEBSOCKET_PROTOCOL_V0) ? import_buf4.buf.fromBinary(import_messenger_pb4.MessageSchema, bytes) : muxer.receiveData(bytes);
490
+ if (message) {
491
+ (0, import_log2.log)("received", {
492
+ from: message.source,
493
+ payload: protocol.getPayloadType(message)
494
+ }, {
495
+ F: __dxlog_file3,
496
+ L: 133,
497
+ S: this,
498
+ C: (f, a) => f(...a)
499
+ });
500
+ this._callbacks.onMessage(message);
501
+ }
502
+ };
503
+ }
504
+ async _close() {
505
+ void this._inactivityTimeoutCtx?.dispose().catch(() => {
506
+ });
507
+ try {
508
+ this._ws?.close();
509
+ this._ws = void 0;
510
+ this._wsMuxer?.destroy();
511
+ this._wsMuxer = void 0;
512
+ } catch (err) {
513
+ if (err instanceof Error && err.message.includes("WebSocket is closed before the connection is established.")) {
514
+ return;
515
+ }
516
+ import_log2.log.warn("Error closing websocket", {
517
+ err
518
+ }, {
519
+ F: __dxlog_file3,
520
+ L: 151,
521
+ S: this,
522
+ C: (f, a) => f(...a)
523
+ });
524
+ }
525
+ }
526
+ _scheduleHeartbeats() {
527
+ (0, import_invariant2.invariant)(this._ws, void 0, {
528
+ F: __dxlog_file3,
529
+ L: 156,
530
+ S: this,
531
+ A: [
532
+ "this._ws",
533
+ ""
534
+ ]
535
+ });
536
+ (0, import_async2.scheduleTaskInterval)(this._ctx, async () => {
537
+ this._ws?.send("__ping__");
538
+ }, SIGNAL_KEEPALIVE_INTERVAL);
539
+ this._ws.send("__ping__");
540
+ this._rescheduleHeartbeatTimeout();
541
+ }
542
+ _rescheduleHeartbeatTimeout() {
543
+ if (!this.isOpen) {
544
+ return;
545
+ }
546
+ void this._inactivityTimeoutCtx?.dispose();
547
+ this._inactivityTimeoutCtx = new import_context.Context(void 0, {
548
+ F: __dxlog_file3,
549
+ L: 175
550
+ });
551
+ (0, import_async2.scheduleTask)(this._inactivityTimeoutCtx, () => {
552
+ if (this.isOpen) {
553
+ import_log2.log.warn("restart due to inactivity timeout", void 0, {
554
+ F: __dxlog_file3,
555
+ L: 180,
556
+ S: this,
557
+ C: (f, a) => f(...a)
558
+ });
559
+ this._callbacks.onRestartRequired();
560
+ }
561
+ }, SIGNAL_KEEPALIVE_TIMEOUT);
562
+ }
563
+ };
564
+ _ts_decorate([
565
+ import_log2.logInfo
566
+ ], EdgeWsConnection.prototype, "info", null);
567
+ // Annotate the CommonJS export names for ESM import in node:
568
+ 0 && (module.exports = {
569
+ EDGE_WEBSOCKET_PROTOCOL_V1,
570
+ EdgeWsConnection,
571
+ Protocol,
572
+ WebSocketMuxer,
573
+ getTypename,
574
+ protocol,
575
+ toUint8Array
576
+ });
577
+ //# sourceMappingURL=chunk-ZURVCY7K.cjs.map
@@ -0,0 +1,7 @@
1
+ {
2
+ "version": 3,
3
+ "sources": ["../../../src/protocol.ts", "../../../src/defs.ts", "../../../src/edge-ws-muxer.ts", "../../../src/edge-ws-connection.ts"],
4
+ "sourcesContent": ["//\n// Copyright 2024 DXOS.org\n//\n\nimport { invariant } from '@dxos/invariant';\nimport { buf, bufWkt } from '@dxos/protocols/buf';\nimport { type Message, MessageSchema, type PeerSchema } from '@dxos/protocols/buf/dxos/edge/messenger_pb';\nimport { bufferToArray } from '@dxos/util';\n\nexport type PeerData = buf.MessageInitShape<typeof PeerSchema>;\n\nexport const getTypename = (typeName: string) => `type.googleapis.com/${typeName}`;\n\n/**\n * NOTE: The type registry should be extended with all message types.\n */\nexport class Protocol {\n private readonly _typeRegistry: buf.Registry;\n\n constructor(types: buf.DescMessage[]) {\n this._typeRegistry = buf.createRegistry(...types);\n }\n\n get typeRegistry(): buf.Registry {\n return this._typeRegistry;\n }\n\n toJson(message: Message): any {\n try {\n return buf.toJson(MessageSchema, message, { registry: this.typeRegistry });\n } catch (err) {\n return { type: this.getPayloadType(message) };\n }\n }\n\n /**\n * Return the payload with the given type.\n */\n getPayload<Desc extends buf.DescMessage>(message: Message, type: Desc): buf.MessageShape<Desc> {\n invariant(message.payload);\n const payloadTypename = this.getPayloadType(message);\n if (type && type.typeName !== payloadTypename) {\n throw new Error(`Unexpected payload type: ${payloadTypename}; expected ${type.typeName}`);\n }\n\n invariant(bufWkt.anyIs(message.payload, type), `Unexpected payload type: ${payloadTypename}}`);\n const payload = bufWkt.anyUnpack(message.payload, this.typeRegistry) as buf.MessageShape<Desc>;\n invariant(payload, `Empty payload: ${payloadTypename}}`);\n return payload;\n }\n\n /**\n * Get the payload type.\n */\n getPayloadType(message: Message): string | undefined {\n if (!message.payload) {\n return undefined;\n }\n\n const [, type] = message.payload.typeUrl.split('/');\n return type;\n }\n\n /**\n * Create a packed message.\n */\n createMessage<Desc extends buf.DescMessage>(\n type: Desc,\n {\n source,\n target,\n payload,\n serviceId,\n }: {\n source?: PeerData;\n target?: PeerData[];\n payload?: buf.MessageInitShape<Desc>;\n serviceId?: string;\n },\n ) {\n return buf.create(MessageSchema, {\n timestamp: new Date().toISOString(),\n source,\n target,\n serviceId,\n payload: payload ? bufWkt.anyPack(type, buf.create(type, payload)) : undefined,\n });\n }\n}\n\n/**\n * Convert websocket data to Uint8Array.\n */\nexport const toUint8Array = async (data: any): Promise<Uint8Array> => {\n // Node.\n if (data instanceof Buffer) {\n return bufferToArray(data);\n }\n\n // Browser.\n if (data instanceof Blob) {\n return new Uint8Array(await (data as Blob).arrayBuffer());\n }\n\n throw new Error(`Unexpected datatype: ${data}`);\n};\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport { bufWkt } from '@dxos/protocols/buf';\nimport { SwarmRequestSchema, SwarmResponseSchema, TextMessageSchema } from '@dxos/protocols/buf/dxos/edge/messenger_pb';\n\nimport { Protocol } from './protocol';\n\nexport const protocol = new Protocol([SwarmRequestSchema, SwarmResponseSchema, TextMessageSchema, bufWkt.AnySchema]);\n", "//\n// Copyright 2025 DXOS.org\n//\nimport WebSocket from 'isomorphic-ws';\n\nimport { Trigger } from '@dxos/async';\nimport { log } from '@dxos/log';\nimport { buf } from '@dxos/protocols/buf';\nimport { MessageSchema, type Message } from '@dxos/protocols/buf/dxos/edge/messenger_pb';\n\nimport { protocol } from './defs';\n\n/**\n * 0000 0001 - message contains a part of segmented message chunk sequence.\n * The next byte defines a channel id and the rest of the message contains a part of Message proto binary.\n * Messages from different channels might interleave.\n * When the flag is NOT set the rest of the message should be interpreted as the valid Message proto binary.\n */\nconst FLAG_SEGMENT_SEQ = 1;\n/**\n * 0000 0010 - message terminates a segmented message chunk sequence.\n * All the chunks accumulated for the channel specified by the second byte can be concatenated\n * and interpreted as a valid Message proto binary.\n */\nconst FLAG_SEGMENT_SEQ_TERMINATED = 1 << 1;\n\n/**\n * 1MB websocket message limit: https://developers.cloudflare.com/durable-objects/platform/limits/\n */\nexport const CLOUDFLARE_MESSAGE_LENGTH_LIMIT = 1024 * 1024;\n\nconst MAX_CHUNK_LENGTH = 16384;\nconst MAX_BUFFERED_AMOUNT = CLOUDFLARE_MESSAGE_LENGTH_LIMIT;\nconst BUFFER_FULL_BACKOFF_TIMEOUT = 100;\n\nexport class WebSocketMuxer {\n private readonly _incomingMessageAccumulator = new Map<number, Buffer[]>();\n private readonly _outgoingMessageChunks = new Map<number, MessageChunk[]>();\n private readonly _serviceToChannel = new Map<string, number>();\n\n private _sendTimeout: any | undefined;\n\n constructor(private readonly _ws: WebSocket) {}\n\n /**\n * Resolves when all the message chunks get enqueued for sending.\n */\n public async send(message: Message): Promise<void> {\n const binary = buf.toBinary(MessageSchema, message);\n const channelId = this._resolveChannel(message);\n if (channelId == null && binary.length > CLOUDFLARE_MESSAGE_LENGTH_LIMIT) {\n log.error('Large message dropped because channel resolution failed.', {\n byteLength: binary.byteLength,\n serviceId: message.serviceId,\n payload: protocol.getPayloadType(message),\n });\n return;\n }\n\n if (channelId == null || binary.length < MAX_CHUNK_LENGTH) {\n const flags = Buffer.from([0]);\n this._ws.send(Buffer.concat([flags, binary]));\n return;\n }\n\n const terminatorSentTrigger = new Trigger();\n const messageChunks: MessageChunk[] = [];\n for (let i = 0; i < binary.length; i += MAX_CHUNK_LENGTH) {\n const chunk = binary.slice(i, i + MAX_CHUNK_LENGTH);\n const isLastChunk = i + MAX_CHUNK_LENGTH < binary.length;\n if (isLastChunk) {\n const flags = Buffer.from([FLAG_SEGMENT_SEQ | FLAG_SEGMENT_SEQ_TERMINATED, channelId]);\n messageChunks.push({ payload: Buffer.concat([flags, chunk]), trigger: terminatorSentTrigger });\n } else {\n const flags = Buffer.from([FLAG_SEGMENT_SEQ]);\n messageChunks.push({ payload: Buffer.concat([flags, chunk]) });\n }\n }\n\n const queuedMessages = this._outgoingMessageChunks.get(channelId);\n if (queuedMessages) {\n queuedMessages.push(...messageChunks);\n } else {\n this._outgoingMessageChunks.set(channelId, messageChunks);\n }\n\n this._sendChunkedMessages();\n\n return terminatorSentTrigger.wait();\n }\n\n public receiveData(data: Uint8Array): Message | undefined {\n if ((data[0] & FLAG_SEGMENT_SEQ) === 0) {\n return buf.fromBinary(MessageSchema, data.slice(1));\n }\n\n const [flags, channelId, ...payload] = data;\n let chunkAccumulator = this._incomingMessageAccumulator.get(channelId);\n if (chunkAccumulator) {\n chunkAccumulator.push(Buffer.from(payload));\n } else {\n chunkAccumulator = [Buffer.from(payload)];\n this._incomingMessageAccumulator.set(channelId, chunkAccumulator);\n }\n\n if ((flags & FLAG_SEGMENT_SEQ_TERMINATED) === 0) {\n return undefined;\n }\n\n const message = buf.fromBinary(MessageSchema, Buffer.concat(chunkAccumulator));\n this._incomingMessageAccumulator.delete(channelId);\n return message;\n }\n\n public destroy() {\n if (this._sendTimeout) {\n clearTimeout(this._sendTimeout);\n this._sendTimeout = undefined;\n }\n for (const channelChunks of this._outgoingMessageChunks.values()) {\n channelChunks.forEach((chunk) => chunk.trigger?.wake());\n }\n this._outgoingMessageChunks.clear();\n this._incomingMessageAccumulator.clear();\n this._serviceToChannel.clear();\n }\n\n private _sendChunkedMessages() {\n if (this._sendTimeout) {\n return;\n }\n\n const send = () => {\n if (this._ws.readyState === WebSocket.CLOSING || this._ws.readyState === WebSocket.CLOSED) {\n log.warn('send called for closed websocket');\n this._sendTimeout = undefined;\n return;\n }\n\n let timeout = 0;\n const emptyChannels: number[] = [];\n for (const [channelId, messages] of this._outgoingMessageChunks.entries()) {\n if (this._ws.bufferedAmount + MAX_CHUNK_LENGTH > MAX_BUFFERED_AMOUNT) {\n timeout = BUFFER_FULL_BACKOFF_TIMEOUT;\n break;\n }\n\n const nextMessage = messages.shift();\n if (nextMessage) {\n this._ws.send(nextMessage.payload);\n nextMessage.trigger?.wake();\n } else {\n emptyChannels.push(channelId);\n }\n }\n\n emptyChannels.forEach((channelId) => this._outgoingMessageChunks.delete(channelId));\n\n if (this._outgoingMessageChunks.size > 0) {\n this._sendTimeout = setTimeout(send, timeout);\n } else {\n this._sendTimeout = undefined;\n }\n };\n this._sendTimeout = setTimeout(send);\n }\n\n private _resolveChannel(message: Message): number | undefined {\n if (!message.serviceId) {\n return undefined;\n }\n let id = this._serviceToChannel.get(message.serviceId);\n if (!id) {\n id = this._serviceToChannel.size + 1;\n this._serviceToChannel.set(message.serviceId, id);\n }\n return id;\n }\n}\n\ntype MessageChunk = {\n payload: Buffer;\n /**\n * Wakes when the payload is enqueued by WebSocket.\n */\n trigger?: Trigger;\n};\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport WebSocket from 'isomorphic-ws';\n\nimport { scheduleTask, scheduleTaskInterval } from '@dxos/async';\nimport { Context, Resource } from '@dxos/context';\nimport { invariant } from '@dxos/invariant';\nimport { log, logInfo } from '@dxos/log';\nimport { buf } from '@dxos/protocols/buf';\nimport { MessageSchema, type Message } from '@dxos/protocols/buf/dxos/edge/messenger_pb';\n\nimport { protocol } from './defs';\nimport { type EdgeIdentity } from './edge-identity';\nimport { CLOUDFLARE_MESSAGE_LENGTH_LIMIT, WebSocketMuxer } from './edge-ws-muxer';\nimport { toUint8Array } from './protocol';\n\nconst SIGNAL_KEEPALIVE_INTERVAL = 4_000;\nconst SIGNAL_KEEPALIVE_TIMEOUT = 12_000;\n\nconst EDGE_WEBSOCKET_PROTOCOL_V0 = 'edge-ws-v0';\n/**\n * Supports message segmentation and muxing.\n */\nexport const EDGE_WEBSOCKET_PROTOCOL_V1 = 'edge-ws-v1';\n\nexport type EdgeWsConnectionCallbacks = {\n onConnected: () => void;\n onMessage: (message: Message) => void;\n onRestartRequired: () => void;\n};\n\nexport class EdgeWsConnection extends Resource {\n private _inactivityTimeoutCtx: Context | undefined;\n private _ws: WebSocket | undefined;\n private _wsMuxer: WebSocketMuxer | undefined;\n\n constructor(\n private readonly _identity: EdgeIdentity,\n private readonly _connectionInfo: { url: URL; protocolHeader?: string },\n private readonly _callbacks: EdgeWsConnectionCallbacks,\n ) {\n super();\n }\n\n @logInfo\n public get info() {\n return {\n open: this.isOpen,\n identity: this._identity.identityKey,\n device: this._identity.peerKey,\n };\n }\n\n public send(message: Message) {\n invariant(this._ws);\n invariant(this._wsMuxer);\n log('sending...', { peerKey: this._identity.peerKey, payload: protocol.getPayloadType(message) });\n if (this._ws?.protocol.includes(EDGE_WEBSOCKET_PROTOCOL_V0)) {\n const binary = buf.toBinary(MessageSchema, message);\n if (binary.length > CLOUDFLARE_MESSAGE_LENGTH_LIMIT) {\n log.error('Message dropped because it was too large (>1MB).', {\n byteLength: binary.byteLength,\n serviceId: message.serviceId,\n payload: protocol.getPayloadType(message),\n });\n return;\n }\n this._ws.send(binary);\n } else {\n this._wsMuxer.send(message).catch((e) => log.catch(e));\n }\n }\n\n protected override async _open() {\n const baseProtocols = [EDGE_WEBSOCKET_PROTOCOL_V0, EDGE_WEBSOCKET_PROTOCOL_V1];\n this._ws = new WebSocket(\n this._connectionInfo.url.toString(),\n this._connectionInfo.protocolHeader\n ? [...baseProtocols, this._connectionInfo.protocolHeader]\n : [...baseProtocols],\n );\n const muxer = new WebSocketMuxer(this._ws);\n this._wsMuxer = muxer;\n\n this._ws.onopen = () => {\n if (this.isOpen) {\n log('connected');\n this._callbacks.onConnected();\n this._scheduleHeartbeats();\n } else {\n log.verbose('connected after becoming inactive', { currentIdentity: this._identity });\n }\n };\n this._ws.onclose = (event) => {\n if (this.isOpen) {\n log.warn('disconnected while being open', { code: event.code, reason: event.reason });\n this._callbacks.onRestartRequired();\n muxer.destroy();\n }\n };\n this._ws.onerror = (event) => {\n if (this.isOpen) {\n log.warn('edge connection socket error', { error: event.error, info: event.message });\n this._callbacks.onRestartRequired();\n } else {\n log.verbose('error ignored on closed connection', { error: event.error });\n }\n };\n /**\n * https://developer.mozilla.org/en-US/docs/Web/API/MessageEvent/data\n */\n this._ws.onmessage = async (event) => {\n if (!this.isOpen) {\n log.verbose('message ignored on closed connection', { event: event.type });\n return;\n }\n if (event.data === '__pong__') {\n this._rescheduleHeartbeatTimeout();\n return;\n }\n const bytes = await toUint8Array(event.data);\n if (!this.isOpen) {\n return;\n }\n\n const message = this._ws?.protocol?.includes(EDGE_WEBSOCKET_PROTOCOL_V0)\n ? buf.fromBinary(MessageSchema, bytes)\n : muxer.receiveData(bytes);\n\n if (message) {\n log('received', { from: message.source, payload: protocol.getPayloadType(message) });\n this._callbacks.onMessage(message);\n }\n };\n }\n\n protected override async _close() {\n void this._inactivityTimeoutCtx?.dispose().catch(() => {});\n\n try {\n this._ws?.close();\n this._ws = undefined;\n this._wsMuxer?.destroy();\n this._wsMuxer = undefined;\n } catch (err) {\n if (err instanceof Error && err.message.includes('WebSocket is closed before the connection is established.')) {\n return;\n }\n log.warn('Error closing websocket', { err });\n }\n }\n\n private _scheduleHeartbeats() {\n invariant(this._ws);\n scheduleTaskInterval(\n this._ctx,\n async () => {\n // TODO(mykola): use RFC6455 ping/pong once implemented in the browser?\n // Cloudflare's worker responds to this `without interrupting hibernation`. https://developers.cloudflare.com/durable-objects/api/websockets/#setwebsocketautoresponse\n this._ws?.send('__ping__');\n },\n SIGNAL_KEEPALIVE_INTERVAL,\n );\n this._ws.send('__ping__');\n this._rescheduleHeartbeatTimeout();\n }\n\n private _rescheduleHeartbeatTimeout() {\n if (!this.isOpen) {\n return;\n }\n void this._inactivityTimeoutCtx?.dispose();\n this._inactivityTimeoutCtx = new Context();\n scheduleTask(\n this._inactivityTimeoutCtx,\n () => {\n if (this.isOpen) {\n log.warn('restart due to inactivity timeout');\n this._callbacks.onRestartRequired();\n }\n },\n SIGNAL_KEEPALIVE_TIMEOUT,\n );\n }\n}\n"],
5
+ "mappings": "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",
6
+ "names": ["import_buf", "import_messenger_pb", "import_isomorphic_ws", "import_async", "import_invariant", "import_log", "getTypename", "typeName", "Protocol", "constructor", "types", "_typeRegistry", "buf", "createRegistry", "typeRegistry", "toJson", "message", "MessageSchema", "registry", "err", "type", "getPayloadType", "getPayload", "invariant", "payload", "payloadTypename", "Error", "bufWkt", "anyIs", "anyUnpack", "undefined", "typeUrl", "split", "createMessage", "source", "target", "serviceId", "create", "timestamp", "Date", "toISOString", "anyPack", "toUint8Array", "data", "Buffer", "bufferToArray", "Blob", "Uint8Array", "arrayBuffer", "protocol", "SwarmRequestSchema", "SwarmResponseSchema", "TextMessageSchema", "AnySchema", "FLAG_SEGMENT_SEQ", "FLAG_SEGMENT_SEQ_TERMINATED", "CLOUDFLARE_MESSAGE_LENGTH_LIMIT", "MAX_CHUNK_LENGTH", "MAX_BUFFERED_AMOUNT", "BUFFER_FULL_BACKOFF_TIMEOUT", "WebSocketMuxer", "_ws", "_incomingMessageAccumulator", "Map", "_outgoingMessageChunks", "_serviceToChannel", "send", "binary", "toBinary", "channelId", "_resolveChannel", "length", "log", "error", "byteLength", "flags", "from", "concat", "terminatorSentTrigger", "Trigger", "messageChunks", "i", "chunk", "slice", "isLastChunk", "push", "trigger", "queuedMessages", "get", "set", "_sendChunkedMessages", "wait", "receiveData", "fromBinary", "chunkAccumulator", "delete", "destroy", "_sendTimeout", "clearTimeout", "channelChunks", "values", "forEach", "wake", "clear", "readyState", "WebSocket", "CLOSING", "CLOSED", "warn", "timeout", "emptyChannels", "messages", "entries", "bufferedAmount", "nextMessage", "shift", "size", "setTimeout", "id", "SIGNAL_KEEPALIVE_INTERVAL", "SIGNAL_KEEPALIVE_TIMEOUT", "EDGE_WEBSOCKET_PROTOCOL_V0", "EDGE_WEBSOCKET_PROTOCOL_V1", "EdgeWsConnection", "Resource", "_identity", "_connectionInfo", "_callbacks", "info", "open", "isOpen", "identity", "identityKey", "device", "peerKey", "_wsMuxer", "includes", "catch", "e", "_open", "baseProtocols", "url", "toString", "protocolHeader", "muxer", "onopen", "onConnected", "_scheduleHeartbeats", "verbose", "currentIdentity", "onclose", "event", "code", "reason", "onRestartRequired", "onerror", "onmessage", "_rescheduleHeartbeatTimeout", "bytes", "onMessage", "_close", "_inactivityTimeoutCtx", "dispose", "close", "scheduleTaskInterval", "_ctx", "Context", "scheduleTask", "logInfo"]
7
+ }