@dxos/edge-client 0.8.4-main.c1de068 → 0.8.4-main.fd6878d

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 (31) hide show
  1. package/dist/lib/browser/{chunk-LMP5TVOP.mjs → chunk-SUXH7FH6.mjs} +8 -1
  2. package/dist/lib/browser/{chunk-LMP5TVOP.mjs.map → chunk-SUXH7FH6.mjs.map} +3 -3
  3. package/dist/lib/browser/edge-ws-muxer.mjs +1 -1
  4. package/dist/lib/browser/index.mjs +31 -9
  5. package/dist/lib/browser/index.mjs.map +3 -3
  6. package/dist/lib/browser/meta.json +1 -1
  7. package/dist/lib/browser/testing/index.mjs +1 -1
  8. package/dist/lib/browser/testing/index.mjs.map +2 -2
  9. package/dist/lib/node-esm/{chunk-X7J46ISZ.mjs → chunk-R6K4IIBW.mjs} +8 -1
  10. package/dist/lib/node-esm/{chunk-X7J46ISZ.mjs.map → chunk-R6K4IIBW.mjs.map} +3 -3
  11. package/dist/lib/node-esm/edge-ws-muxer.mjs +1 -1
  12. package/dist/lib/node-esm/index.mjs +31 -9
  13. package/dist/lib/node-esm/index.mjs.map +3 -3
  14. package/dist/lib/node-esm/meta.json +1 -1
  15. package/dist/lib/node-esm/testing/index.mjs +1 -1
  16. package/dist/lib/node-esm/testing/index.mjs.map +2 -2
  17. package/dist/types/src/edge-client.d.ts +1 -1
  18. package/dist/types/src/edge-client.d.ts.map +1 -1
  19. package/dist/types/src/edge-http-client.d.ts +1 -1
  20. package/dist/types/src/edge-http-client.d.ts.map +1 -1
  21. package/dist/types/src/edge-ws-connection.d.ts.map +1 -1
  22. package/dist/types/src/edge-ws-muxer.d.ts.map +1 -1
  23. package/dist/types/src/testing/test-utils.d.ts.map +1 -1
  24. package/dist/types/tsconfig.tsbuildinfo +1 -1
  25. package/package.json +18 -15
  26. package/src/edge-client.ts +2 -2
  27. package/src/edge-http-client.ts +6 -6
  28. package/src/edge-ws-connection.ts +1 -1
  29. package/src/edge-ws-muxer.ts +1 -1
  30. package/src/testing/test-utils.ts +1 -1
  31. package/src/websocket.test.ts +1 -1
@@ -16,6 +16,7 @@ import { bufferToArray } from "@dxos/util";
16
16
  var __dxlog_file = "/__w/dxos/dxos/packages/core/mesh/edge-client/src/protocol.ts";
17
17
  var getTypename = (typeName) => `type.googleapis.com/${typeName}`;
18
18
  var Protocol = class {
19
+ _typeRegistry;
19
20
  constructor(types) {
20
21
  this._typeRegistry = buf.createRegistry(...types);
21
22
  }
@@ -122,6 +123,12 @@ var MAX_CHUNK_LENGTH = 16384;
122
123
  var MAX_BUFFERED_AMOUNT = CLOUDFLARE_MESSAGE_MAX_BYTES;
123
124
  var BUFFER_FULL_BACKOFF_TIMEOUT = 100;
124
125
  var WebSocketMuxer = class {
126
+ _ws;
127
+ _inMessageAccumulator;
128
+ _outMessageChunks;
129
+ _outMessageChannelByService;
130
+ _sendTimeout;
131
+ _maxChunkLength;
125
132
  constructor(_ws, config) {
126
133
  this._ws = _ws;
127
134
  this._inMessageAccumulator = /* @__PURE__ */ new Map();
@@ -294,4 +301,4 @@ export {
294
301
  CLOUDFLARE_RPC_MAX_BYTES,
295
302
  WebSocketMuxer
296
303
  };
297
- //# sourceMappingURL=chunk-LMP5TVOP.mjs.map
304
+ //# sourceMappingURL=chunk-SUXH7FH6.mjs.map
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../src/edge-ws-muxer.ts", "../../../src/defs.ts", "../../../src/protocol.ts"],
4
- "sourcesContent": ["//\n// Copyright 2025 DXOS.org\n//\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 * https://developers.cloudflare.com/durable-objects/platform/limits/\n */\nexport const CLOUDFLARE_MESSAGE_MAX_BYTES = 1000 * 1000; // 1MB\nexport const CLOUDFLARE_RPC_MAX_BYTES = 32 * 1000 * 1000; // 32MB\n\nconst MAX_CHUNK_LENGTH = 16384;\nconst MAX_BUFFERED_AMOUNT = CLOUDFLARE_MESSAGE_MAX_BYTES;\nconst BUFFER_FULL_BACKOFF_TIMEOUT = 100;\n\nexport class WebSocketMuxer {\n private readonly _inMessageAccumulator = new Map<number, Buffer[]>();\n private readonly _outMessageChunks = new Map<number, MessageChunk[]>();\n private readonly _outMessageChannelByService = new Map<string, number>();\n\n private _sendTimeout: any | undefined;\n\n private readonly _maxChunkLength: number;\n\n constructor(\n private readonly _ws: WebSocketCompat,\n config?: { maxChunkLength: number },\n ) {\n this._maxChunkLength = config?.maxChunkLength ?? MAX_CHUNK_LENGTH;\n }\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 (\n (channelId == null && binary.byteLength > CLOUDFLARE_MESSAGE_MAX_BYTES) ||\n binary.byteLength > CLOUDFLARE_RPC_MAX_BYTES\n ) {\n log.error('Large message dropped', {\n byteLength: binary.byteLength,\n serviceId: message.serviceId,\n payload: protocol.getPayloadType(message),\n channelId,\n });\n return;\n }\n\n if (channelId == null || binary.length < this._maxChunkLength) {\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 += this._maxChunkLength) {\n const chunk = binary.slice(i, i + this._maxChunkLength);\n const isLastChunk = i + this._maxChunkLength >= 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, channelId]);\n messageChunks.push({ payload: Buffer.concat([flags, chunk]) });\n }\n }\n\n const queuedMessages = this._outMessageChunks.get(channelId);\n if (queuedMessages) {\n queuedMessages.push(...messageChunks);\n } else {\n this._outMessageChunks.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._inMessageAccumulator.get(channelId);\n if (chunkAccumulator) {\n chunkAccumulator.push(Buffer.from(payload));\n } else {\n chunkAccumulator = [Buffer.from(payload)];\n this._inMessageAccumulator.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._inMessageAccumulator.delete(channelId);\n return message;\n }\n\n public destroy(): void {\n if (this._sendTimeout) {\n clearTimeout(this._sendTimeout);\n this._sendTimeout = undefined;\n }\n for (const channelChunks of this._outMessageChunks.values()) {\n channelChunks.forEach((chunk) => chunk.trigger?.wake());\n }\n this._outMessageChunks.clear();\n this._inMessageAccumulator.clear();\n this._outMessageChannelByService.clear();\n }\n\n private _sendChunkedMessages(): void {\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._outMessageChunks.entries()) {\n if (this._ws.bufferedAmount != null) {\n if (this._ws.bufferedAmount + MAX_CHUNK_LENGTH > MAX_BUFFERED_AMOUNT) {\n timeout = BUFFER_FULL_BACKOFF_TIMEOUT;\n break;\n }\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._outMessageChunks.delete(channelId));\n\n if (this._outMessageChunks.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._outMessageChannelByService.get(message.serviceId);\n if (!id) {\n id = this._outMessageChannelByService.size + 1;\n this._outMessageChannelByService.set(message.serviceId, id);\n }\n return id;\n }\n}\n\ntype WebSocketCompat = {\n readonly readyState: number;\n /**\n * Not available in workerd.\n */\n bufferedAmount?: number;\n send(message: (ArrayBuffer | ArrayBufferView) | string): void;\n};\n\ntype MessageChunk = {\n payload: Buffer;\n /**\n * Wakes when the payload is enqueued by WebSocket.\n */\n trigger?: Trigger;\n};\n\n/**\n * To avoid using isomorphic-ws on edge.\n */\nenum WebSocket {\n CLOSING = 2,\n CLOSED = 3,\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 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 ): Message {\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"],
5
- "mappings": ";AAIA,SAASA,eAAe;AACxB,SAASC,WAAW;AACpB,SAASC,OAAAA,YAAW;AACpB,SAASC,iBAAAA,sBAAmC;;;ACH5C,SAASC,UAAAA,eAAc;AACvB,SAASC,oBAAoBC,qBAAqBC,yBAAyB;;;ACD3E,SAASC,iBAAiB;AAC1B,SAASC,KAAKC,cAAc;AAC5B,SAAuBC,qBAAsC;AAC7D,SAASC,qBAAqB;;AAIvB,IAAMC,cAAc,CAACC,aAAqB,uBAAuBA,QAAAA;AAKjE,IAAMC,WAAN,MAAMA;EAGX,YAAYC,OAA0B;AACpC,SAAKC,gBAAgBR,IAAIS,eAAc,GAAIF,KAAAA;EAC7C;EAEA,IAAIG,eAA6B;AAC/B,WAAO,KAAKF;EACd;EAEAG,OAAOC,SAAuB;AAC5B,QAAI;AACF,aAAOZ,IAAIW,OAAOT,eAAeU,SAAS;QAAEC,UAAU,KAAKH;MAAa,CAAA;IAC1E,SAASI,KAAK;AACZ,aAAO;QAAEC,MAAM,KAAKC,eAAeJ,OAAAA;MAAS;IAC9C;EACF;;;;EAKAK,WAAyCL,SAAkBG,MAAoC;AAC7FhB,cAAUa,QAAQM,SAAO,QAAA;;;;;;;;;AACzB,UAAMC,kBAAkB,KAAKH,eAAeJ,OAAAA;AAC5C,QAAIG,QAAQA,KAAKV,aAAac,iBAAiB;AAC7C,YAAM,IAAIC,MAAM,4BAA4BD,eAAAA,cAA6BJ,KAAKV,QAAQ,EAAE;IAC1F;AAEAN,cAAUE,OAAOoB,MAAMT,QAAQM,SAASH,IAAAA,GAAO,4BAA4BI,eAAAA,KAAkB;;;;;;;;;AAC7F,UAAMD,UAAUjB,OAAOqB,UAAUV,QAAQM,SAAS,KAAKR,YAAY;AACnEX,cAAUmB,SAAS,kBAAkBC,eAAAA,KAAkB;;;;;;;;;AACvD,WAAOD;EACT;;;;EAKAF,eAAeJ,SAAsC;AACnD,QAAI,CAACA,QAAQM,SAAS;AACpB,aAAOK;IACT;AAEA,UAAM,CAAA,EAAGR,IAAAA,IAAQH,QAAQM,QAAQM,QAAQC,MAAM,GAAA;AAC/C,WAAOV;EACT;;;;EAKAW,cACEX,MACA,EACEY,QACAC,QACAV,SACAW,UAAS,GAOF;AACT,WAAO7B,IAAI8B,OAAO5B,eAAe;MAC/B6B,YAAW,oBAAIC,KAAAA,GAAOC,YAAW;MACjCN;MACAC;MACAC;MACAX,SAASA,UAAUjB,OAAOiC,QAAQnB,MAAMf,IAAI8B,OAAOf,MAAMG,OAAAA,CAAAA,IAAYK;IACvE,CAAA;EACF;AACF;AAKO,IAAMY,eAAe,OAAOC,SAAAA;AAEjC,MAAIA,gBAAgBC,QAAQ;AAC1B,WAAOlC,cAAciC,IAAAA;EACvB;AAGA,MAAIA,gBAAgBE,MAAM;AACxB,WAAO,IAAIC,WAAW,MAAOH,KAAcI,YAAW,CAAA;EACxD;AAEA,QAAM,IAAIpB,MAAM,wBAAwBgB,IAAAA,EAAM;AAChD;;;ADhGO,IAAMK,WAAW,IAAIC,SAAS;EAACC;EAAoBC;EAAqBC;EAAmBC,QAAOC;CAAU;;;;ADQnH,IAAMC,mBAAmB;AAMzB,IAAMC,8BAA8B,KAAK;AAKlC,IAAMC,+BAA+B,MAAO;AAC5C,IAAMC,2BAA2B,KAAK,MAAO;AAEpD,IAAMC,mBAAmB;AACzB,IAAMC,sBAAsBH;AAC5B,IAAMI,8BAA8B;AAE7B,IAAMC,iBAAN,MAAMA;EASX,YACmBC,KACjBC,QACA;SAFiBD,MAAAA;SATFE,wBAAwB,oBAAIC,IAAAA;SAC5BC,oBAAoB,oBAAID,IAAAA;SACxBE,8BAA8B,oBAAIF,IAAAA;AAUjD,SAAKG,kBAAkBL,QAAQM,kBAAkBX;EACnD;;;;EAKA,MAAaY,KAAKC,SAAiC;AACjD,UAAMC,SAASC,KAAIC,SAASC,gBAAeJ,OAAAA;AAC3C,UAAMK,YAAY,KAAKC,gBAAgBN,OAAAA;AACvC,QACGK,aAAa,QAAQJ,OAAOM,aAAatB,gCAC1CgB,OAAOM,aAAarB,0BACpB;AACAsB,UAAIC,MAAM,yBAAyB;QACjCF,YAAYN,OAAOM;QACnBG,WAAWV,QAAQU;QACnBC,SAASC,SAASC,eAAeb,OAAAA;QACjCK;MACF,GAAA;;;;;;AACA;IACF;AAEA,QAAIA,aAAa,QAAQJ,OAAOa,SAAS,KAAKjB,iBAAiB;AAC7D,YAAMkB,QAAQC,OAAOC,KAAK;QAAC;OAAE;AAC7B,WAAK1B,IAAIQ,KAAKiB,OAAOE,OAAO;QAACH;QAAOd;OAAO,CAAA;AAC3C;IACF;AAEA,UAAMkB,wBAAwB,IAAIC,QAAAA;AAClC,UAAMC,gBAAgC,CAAA;AACtC,aAASC,IAAI,GAAGA,IAAIrB,OAAOa,QAAQQ,KAAK,KAAKzB,iBAAiB;AAC5D,YAAM0B,QAAQtB,OAAOuB,MAAMF,GAAGA,IAAI,KAAKzB,eAAe;AACtD,YAAM4B,cAAcH,IAAI,KAAKzB,mBAAmBI,OAAOa;AACvD,UAAIW,aAAa;AACf,cAAMV,QAAQC,OAAOC,KAAK;UAAClC,mBAAmBC;UAA6BqB;SAAU;AACrFgB,sBAAcK,KAAK;UAAEf,SAASK,OAAOE,OAAO;YAACH;YAAOQ;WAAM;UAAGI,SAASR;QAAsB,CAAA;MAC9F,OAAO;AACL,cAAMJ,QAAQC,OAAOC,KAAK;UAAClC;UAAkBsB;SAAU;AACvDgB,sBAAcK,KAAK;UAAEf,SAASK,OAAOE,OAAO;YAACH;YAAOQ;WAAM;QAAE,CAAA;MAC9D;IACF;AAEA,UAAMK,iBAAiB,KAAKjC,kBAAkBkC,IAAIxB,SAAAA;AAClD,QAAIuB,gBAAgB;AAClBA,qBAAeF,KAAI,GAAIL,aAAAA;IACzB,OAAO;AACL,WAAK1B,kBAAkBmC,IAAIzB,WAAWgB,aAAAA;IACxC;AAEA,SAAKU,qBAAoB;AAEzB,WAAOZ,sBAAsBa,KAAI;EACnC;EAEOC,YAAYC,MAAuC;AACxD,SAAKA,KAAK,CAAA,IAAKnD,sBAAsB,GAAG;AACtC,aAAOmB,KAAIiC,WAAW/B,gBAAe8B,KAAKV,MAAM,CAAA,CAAA;IAClD;AAEA,UAAM,CAACT,OAAOV,WAAW,GAAGM,OAAAA,IAAWuB;AACvC,QAAIE,mBAAmB,KAAK3C,sBAAsBoC,IAAIxB,SAAAA;AACtD,QAAI+B,kBAAkB;AACpBA,uBAAiBV,KAAKV,OAAOC,KAAKN,OAAAA,CAAAA;IACpC,OAAO;AACLyB,yBAAmB;QAACpB,OAAOC,KAAKN,OAAAA;;AAChC,WAAKlB,sBAAsBqC,IAAIzB,WAAW+B,gBAAAA;IAC5C;AAEA,SAAKrB,QAAQ/B,iCAAiC,GAAG;AAC/C,aAAOqD;IACT;AAEA,UAAMrC,UAAUE,KAAIiC,WAAW/B,gBAAeY,OAAOE,OAAOkB,gBAAAA,CAAAA;AAC5D,SAAK3C,sBAAsB6C,OAAOjC,SAAAA;AAClC,WAAOL;EACT;EAEOuC,UAAgB;AACrB,QAAI,KAAKC,cAAc;AACrBC,mBAAa,KAAKD,YAAY;AAC9B,WAAKA,eAAeH;IACtB;AACA,eAAWK,iBAAiB,KAAK/C,kBAAkBgD,OAAM,GAAI;AAC3DD,oBAAcE,QAAQ,CAACrB,UAAUA,MAAMI,SAASkB,KAAAA,CAAAA;IAClD;AACA,SAAKlD,kBAAkBmD,MAAK;AAC5B,SAAKrD,sBAAsBqD,MAAK;AAChC,SAAKlD,4BAA4BkD,MAAK;EACxC;EAEQf,uBAA6B;AACnC,QAAI,KAAKS,cAAc;AACrB;IACF;AAEA,UAAMzC,OAAO,MAAA;AACX,UAAI,KAAKR,IAAIwD,eAAU,KAA0B,KAAKxD,IAAIwD,eAAU,GAAuB;AACzFvC,YAAIwC,KAAK,oCAAA,QAAA;;;;;;AACT,aAAKR,eAAeH;AACpB;MACF;AAEA,UAAIY,UAAU;AACd,YAAMC,gBAA0B,CAAA;AAChC,iBAAW,CAAC7C,WAAW8C,QAAAA,KAAa,KAAKxD,kBAAkByD,QAAO,GAAI;AACpE,YAAI,KAAK7D,IAAI8D,kBAAkB,MAAM;AACnC,cAAI,KAAK9D,IAAI8D,iBAAiBlE,mBAAmBC,qBAAqB;AACpE6D,sBAAU5D;AACV;UACF;QACF;AAEA,cAAMiE,cAAcH,SAASI,MAAK;AAClC,YAAID,aAAa;AACf,eAAK/D,IAAIQ,KAAKuD,YAAY3C,OAAO;AACjC2C,sBAAY3B,SAASkB,KAAAA;QACvB,OAAO;AACLK,wBAAcxB,KAAKrB,SAAAA;QACrB;MACF;AAEA6C,oBAAcN,QAAQ,CAACvC,cAAc,KAAKV,kBAAkB2C,OAAOjC,SAAAA,CAAAA;AAEnE,UAAI,KAAKV,kBAAkB6D,OAAO,GAAG;AACnC,aAAKhB,eAAeiB,WAAW1D,MAAMkD,OAAAA;MACvC,OAAO;AACL,aAAKT,eAAeH;MACtB;IACF;AACA,SAAKG,eAAeiB,WAAW1D,IAAAA;EACjC;EAEQO,gBAAgBN,SAAsC;AAC5D,QAAI,CAACA,QAAQU,WAAW;AACtB,aAAO2B;IACT;AACA,QAAIqB,KAAK,KAAK9D,4BAA4BiC,IAAI7B,QAAQU,SAAS;AAC/D,QAAI,CAACgD,IAAI;AACPA,WAAK,KAAK9D,4BAA4B4D,OAAO;AAC7C,WAAK5D,4BAA4BkC,IAAI9B,QAAQU,WAAWgD,EAAAA;IAC1D;AACA,WAAOA;EACT;AACF;",
6
- "names": ["Trigger", "log", "buf", "MessageSchema", "bufWkt", "SwarmRequestSchema", "SwarmResponseSchema", "TextMessageSchema", "invariant", "buf", "bufWkt", "MessageSchema", "bufferToArray", "getTypename", "typeName", "Protocol", "types", "_typeRegistry", "createRegistry", "typeRegistry", "toJson", "message", "registry", "err", "type", "getPayloadType", "getPayload", "payload", "payloadTypename", "Error", "anyIs", "anyUnpack", "undefined", "typeUrl", "split", "createMessage", "source", "target", "serviceId", "create", "timestamp", "Date", "toISOString", "anyPack", "toUint8Array", "data", "Buffer", "Blob", "Uint8Array", "arrayBuffer", "protocol", "Protocol", "SwarmRequestSchema", "SwarmResponseSchema", "TextMessageSchema", "bufWkt", "AnySchema", "FLAG_SEGMENT_SEQ", "FLAG_SEGMENT_SEQ_TERMINATED", "CLOUDFLARE_MESSAGE_MAX_BYTES", "CLOUDFLARE_RPC_MAX_BYTES", "MAX_CHUNK_LENGTH", "MAX_BUFFERED_AMOUNT", "BUFFER_FULL_BACKOFF_TIMEOUT", "WebSocketMuxer", "_ws", "config", "_inMessageAccumulator", "Map", "_outMessageChunks", "_outMessageChannelByService", "_maxChunkLength", "maxChunkLength", "send", "message", "binary", "buf", "toBinary", "MessageSchema", "channelId", "_resolveChannel", "byteLength", "log", "error", "serviceId", "payload", "protocol", "getPayloadType", "length", "flags", "Buffer", "from", "concat", "terminatorSentTrigger", "Trigger", "messageChunks", "i", "chunk", "slice", "isLastChunk", "push", "trigger", "queuedMessages", "get", "set", "_sendChunkedMessages", "wait", "receiveData", "data", "fromBinary", "chunkAccumulator", "undefined", "delete", "destroy", "_sendTimeout", "clearTimeout", "channelChunks", "values", "forEach", "wake", "clear", "readyState", "warn", "timeout", "emptyChannels", "messages", "entries", "bufferedAmount", "nextMessage", "shift", "size", "setTimeout", "id"]
4
+ "sourcesContent": ["//\n// Copyright 2025 DXOS.org\n//\n\nimport { Trigger } from '@dxos/async';\nimport { log } from '@dxos/log';\nimport { buf } from '@dxos/protocols/buf';\nimport { type Message, MessageSchema } 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 * https://developers.cloudflare.com/durable-objects/platform/limits/\n */\nexport const CLOUDFLARE_MESSAGE_MAX_BYTES = 1000 * 1000; // 1MB\nexport const CLOUDFLARE_RPC_MAX_BYTES = 32 * 1000 * 1000; // 32MB\n\nconst MAX_CHUNK_LENGTH = 16384;\nconst MAX_BUFFERED_AMOUNT = CLOUDFLARE_MESSAGE_MAX_BYTES;\nconst BUFFER_FULL_BACKOFF_TIMEOUT = 100;\n\nexport class WebSocketMuxer {\n private readonly _inMessageAccumulator = new Map<number, Buffer[]>();\n private readonly _outMessageChunks = new Map<number, MessageChunk[]>();\n private readonly _outMessageChannelByService = new Map<string, number>();\n\n private _sendTimeout: any | undefined;\n\n private readonly _maxChunkLength: number;\n\n constructor(\n private readonly _ws: WebSocketCompat,\n config?: { maxChunkLength: number },\n ) {\n this._maxChunkLength = config?.maxChunkLength ?? MAX_CHUNK_LENGTH;\n }\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 (\n (channelId == null && binary.byteLength > CLOUDFLARE_MESSAGE_MAX_BYTES) ||\n binary.byteLength > CLOUDFLARE_RPC_MAX_BYTES\n ) {\n log.error('Large message dropped', {\n byteLength: binary.byteLength,\n serviceId: message.serviceId,\n payload: protocol.getPayloadType(message),\n channelId,\n });\n return;\n }\n\n if (channelId == null || binary.length < this._maxChunkLength) {\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 += this._maxChunkLength) {\n const chunk = binary.slice(i, i + this._maxChunkLength);\n const isLastChunk = i + this._maxChunkLength >= 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, channelId]);\n messageChunks.push({ payload: Buffer.concat([flags, chunk]) });\n }\n }\n\n const queuedMessages = this._outMessageChunks.get(channelId);\n if (queuedMessages) {\n queuedMessages.push(...messageChunks);\n } else {\n this._outMessageChunks.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._inMessageAccumulator.get(channelId);\n if (chunkAccumulator) {\n chunkAccumulator.push(Buffer.from(payload));\n } else {\n chunkAccumulator = [Buffer.from(payload)];\n this._inMessageAccumulator.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._inMessageAccumulator.delete(channelId);\n return message;\n }\n\n public destroy(): void {\n if (this._sendTimeout) {\n clearTimeout(this._sendTimeout);\n this._sendTimeout = undefined;\n }\n for (const channelChunks of this._outMessageChunks.values()) {\n channelChunks.forEach((chunk) => chunk.trigger?.wake());\n }\n this._outMessageChunks.clear();\n this._inMessageAccumulator.clear();\n this._outMessageChannelByService.clear();\n }\n\n private _sendChunkedMessages(): void {\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._outMessageChunks.entries()) {\n if (this._ws.bufferedAmount != null) {\n if (this._ws.bufferedAmount + MAX_CHUNK_LENGTH > MAX_BUFFERED_AMOUNT) {\n timeout = BUFFER_FULL_BACKOFF_TIMEOUT;\n break;\n }\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._outMessageChunks.delete(channelId));\n\n if (this._outMessageChunks.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._outMessageChannelByService.get(message.serviceId);\n if (!id) {\n id = this._outMessageChannelByService.size + 1;\n this._outMessageChannelByService.set(message.serviceId, id);\n }\n return id;\n }\n}\n\ntype WebSocketCompat = {\n readonly readyState: number;\n /**\n * Not available in workerd.\n */\n bufferedAmount?: number;\n send(message: (ArrayBuffer | ArrayBufferView) | string): void;\n};\n\ntype MessageChunk = {\n payload: Buffer;\n /**\n * Wakes when the payload is enqueued by WebSocket.\n */\n trigger?: Trigger;\n};\n\n/**\n * To avoid using isomorphic-ws on edge.\n */\nenum WebSocket {\n CLOSING = 2,\n CLOSED = 3,\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 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 ): Message {\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"],
5
+ "mappings": "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",
6
+ "names": ["Trigger", "log", "buf", "MessageSchema", "bufWkt", "SwarmRequestSchema", "SwarmResponseSchema", "TextMessageSchema", "invariant", "buf", "bufWkt", "MessageSchema", "bufferToArray", "getTypename", "typeName", "Protocol", "_typeRegistry", "types", "createRegistry", "typeRegistry", "toJson", "message", "registry", "err", "type", "getPayloadType", "getPayload", "payload", "payloadTypename", "Error", "anyIs", "anyUnpack", "undefined", "typeUrl", "split", "createMessage", "source", "target", "serviceId", "create", "timestamp", "Date", "toISOString", "anyPack", "toUint8Array", "data", "Buffer", "Blob", "Uint8Array", "arrayBuffer", "protocol", "Protocol", "SwarmRequestSchema", "SwarmResponseSchema", "TextMessageSchema", "bufWkt", "AnySchema", "FLAG_SEGMENT_SEQ", "FLAG_SEGMENT_SEQ_TERMINATED", "CLOUDFLARE_MESSAGE_MAX_BYTES", "CLOUDFLARE_RPC_MAX_BYTES", "MAX_CHUNK_LENGTH", "MAX_BUFFERED_AMOUNT", "BUFFER_FULL_BACKOFF_TIMEOUT", "WebSocketMuxer", "_inMessageAccumulator", "_outMessageChunks", "_outMessageChannelByService", "_sendTimeout", "_maxChunkLength", "_ws", "config", "Map", "maxChunkLength", "send", "message", "binary", "buf", "toBinary", "MessageSchema", "channelId", "_resolveChannel", "byteLength", "log", "error", "serviceId", "payload", "protocol", "getPayloadType", "length", "flags", "Buffer", "from", "concat", "terminatorSentTrigger", "Trigger", "messageChunks", "i", "chunk", "slice", "isLastChunk", "push", "trigger", "queuedMessages", "get", "set", "_sendChunkedMessages", "wait", "receiveData", "data", "fromBinary", "chunkAccumulator", "undefined", "delete", "destroy", "clearTimeout", "channelChunks", "values", "forEach", "wake", "clear", "readyState", "warn", "timeout", "emptyChannels", "messages", "entries", "bufferedAmount", "nextMessage", "shift", "size", "setTimeout", "id"]
7
7
  }
@@ -2,7 +2,7 @@ import {
2
2
  CLOUDFLARE_MESSAGE_MAX_BYTES,
3
3
  CLOUDFLARE_RPC_MAX_BYTES,
4
4
  WebSocketMuxer
5
- } from "./chunk-LMP5TVOP.mjs";
5
+ } from "./chunk-SUXH7FH6.mjs";
6
6
  export {
7
7
  CLOUDFLARE_MESSAGE_MAX_BYTES,
8
8
  CLOUDFLARE_RPC_MAX_BYTES,
@@ -6,13 +6,13 @@ import {
6
6
  getTypename,
7
7
  protocol,
8
8
  toUint8Array
9
- } from "./chunk-LMP5TVOP.mjs";
9
+ } from "./chunk-SUXH7FH6.mjs";
10
10
 
11
11
  // src/index.ts
12
12
  export * from "@dxos/protocols/buf/dxos/edge/messenger_pb";
13
13
 
14
14
  // src/edge-client.ts
15
- import { Trigger, scheduleMicroTask, TriggerState, PersistentLifecycle, Event } from "@dxos/async";
15
+ import { Event, PersistentLifecycle, Trigger, TriggerState, scheduleMicroTask } from "@dxos/async";
16
16
  import { Resource as Resource2 } from "@dxos/context";
17
17
  import { log as log2, logInfo as logInfo2 } from "@dxos/log";
18
18
  import { EdgeStatus } from "@dxos/protocols/proto/dxos/client/services";
@@ -76,6 +76,13 @@ var __dxlog_file2 = "/__w/dxos/dxos/packages/core/mesh/edge-client/src/edge-ws-c
76
76
  var SIGNAL_KEEPALIVE_INTERVAL = 4e3;
77
77
  var SIGNAL_KEEPALIVE_TIMEOUT = 12e3;
78
78
  var EdgeWsConnection = class extends Resource {
79
+ _identity;
80
+ _connectionInfo;
81
+ _callbacks;
82
+ _inactivityTimeoutCtx;
83
+ _ws;
84
+ _wsMuxer;
85
+ _lastReceivedMessageTimestamp;
79
86
  constructor(_identity, _connectionInfo, _callbacks) {
80
87
  super(), this._identity = _identity, this._connectionInfo = _connectionInfo, this._callbacks = _callbacks, this._lastReceivedMessageTimestamp = Date.now();
81
88
  }
@@ -346,6 +353,16 @@ function _ts_decorate2(decorators, target, key, desc) {
346
353
  var __dxlog_file3 = "/__w/dxos/dxos/packages/core/mesh/edge-client/src/edge-client.ts";
347
354
  var DEFAULT_TIMEOUT = 1e4;
348
355
  var EdgeClient = class extends Resource2 {
356
+ _identity;
357
+ _config;
358
+ statusChanged;
359
+ _persistentLifecycle;
360
+ _messageListeners;
361
+ _reconnectListeners;
362
+ _baseWsUrl;
363
+ _baseHttpUrl;
364
+ _currentConnection;
365
+ _ready;
349
366
  constructor(_identity, _config) {
350
367
  super(), this._identity = _identity, this._config = _config, this.statusChanged = new Event(), this._persistentLifecycle = new PersistentLifecycle({
351
368
  start: async () => this._connect(),
@@ -622,6 +639,7 @@ var EdgeClient = class extends Resource2 {
622
639
  return void 0;
623
640
  }
624
641
  }
642
+ _isActive;
625
643
  };
626
644
  _ts_decorate2([
627
645
  logInfo2
@@ -756,13 +774,11 @@ import { Context as Context2, Duration, Effect, Layer, Schedule } from "effect";
756
774
  import { log as log3 } from "@dxos/log";
757
775
  var __dxlog_file5 = "/__w/dxos/dxos/packages/core/mesh/edge-client/src/http-client.ts";
758
776
  var HttpConfig = class _HttpConfig extends Context2.Tag("HttpConfig")() {
759
- static {
760
- this.default = Layer.succeed(_HttpConfig, {
761
- timeout: Duration.millis(1e3),
762
- retryTimes: 3,
763
- retryBaseDelay: Duration.millis(1e3)
764
- });
765
- }
777
+ static default = Layer.succeed(_HttpConfig, {
778
+ timeout: Duration.millis(1e3),
779
+ retryTimes: 3,
780
+ retryBaseDelay: Duration.millis(1e3)
781
+ });
766
782
  };
767
783
  var withRetry = (effect, { timeout = Duration.millis(1e3), retryBaseDelay = Duration.millis(1e3), retryTimes = 3 } = {}) => {
768
784
  return effect.pipe(Effect.flatMap((res) => (
@@ -796,6 +812,12 @@ var DEFAULT_RETRY_TIMEOUT = 1500;
796
812
  var DEFAULT_RETRY_JITTER = 500;
797
813
  var DEFAULT_MAX_RETRIES_COUNT = 3;
798
814
  var EdgeHttpClient = class {
815
+ _baseUrl;
816
+ _edgeIdentity;
817
+ /**
818
+ * Auth header is cached until receiving the next 401 from EDGE, at which point it gets refreshed.
819
+ */
820
+ _authHeader;
799
821
  constructor(baseUrl) {
800
822
  this._baseUrl = getEdgeUrlWithProtocol(baseUrl, "http");
801
823
  log4("created", {
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../src/index.ts", "../../../src/edge-client.ts", "../../../src/edge-identity.ts", "../../../src/edge-ws-connection.ts", "../../../src/errors.ts", "../../../src/utils.ts", "../../../src/auth.ts", "../../../src/edge-http-client.ts", "../../../src/http-client.ts"],
4
- "sourcesContent": ["//\n// Copyright 2024 DXOS.org\n//\n\nexport * from '@dxos/protocols/buf/dxos/edge/messenger_pb';\n\nexport * from './edge-client';\nexport * from './defs';\nexport * from './protocol';\nexport * from './errors';\nexport * from './auth';\nexport * from './edge-http-client';\nexport * from './edge-identity';\nexport * from './edge-ws-muxer';\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport { Trigger, scheduleMicroTask, TriggerState, PersistentLifecycle, Event } from '@dxos/async';\nimport { Resource, type Lifecycle } from '@dxos/context';\nimport { log, logInfo } from '@dxos/log';\nimport { type Message } from '@dxos/protocols/buf/dxos/edge/messenger_pb';\nimport { EdgeStatus } from '@dxos/protocols/proto/dxos/client/services';\n\nimport { protocol } from './defs';\nimport { type EdgeIdentity, handleAuthChallenge } from './edge-identity';\nimport { EdgeWsConnection } from './edge-ws-connection';\nimport { EdgeConnectionClosedError, EdgeIdentityChangedError } from './errors';\nimport { type Protocol } from './protocol';\nimport { getEdgeUrlWithProtocol } from './utils';\n\nconst DEFAULT_TIMEOUT = 10_000;\n\nexport type MessageListener = (message: Message) => void;\nexport type ReconnectListener = () => void;\n\nexport interface EdgeConnection extends Required<Lifecycle> {\n statusChanged: Event<EdgeStatus>;\n get info(): any;\n get identityKey(): string;\n get peerKey(): string;\n get isOpen(): boolean;\n get status(): EdgeStatus;\n setIdentity(identity: EdgeIdentity): void;\n onMessage(listener: MessageListener): () => void;\n onReconnected(listener: ReconnectListener): () => void;\n send(message: Message): Promise<void>;\n}\n\nexport type MessengerConfig = {\n socketEndpoint: string;\n timeout?: number;\n protocol?: Protocol;\n disableAuth?: boolean;\n};\n\n/**\n * Messenger client for EDGE:\n * - While open, uses PersistentLifecycle to keep an open EdgeWsConnection, reconnecting on failures.\n * - Manages identity and re-create EdgeWsConnection when identity changes.\n * - Dispatches connection state and message notifications.\n */\nexport class EdgeClient extends Resource implements EdgeConnection {\n public readonly statusChanged = new Event<EdgeStatus>();\n\n private readonly _persistentLifecycle = new PersistentLifecycle<EdgeWsConnection>({\n start: async () => this._connect(),\n stop: async (state: EdgeWsConnection) => this._disconnect(state),\n });\n\n private readonly _messageListeners = new Set<MessageListener>();\n private readonly _reconnectListeners = new Set<ReconnectListener>();\n private readonly _baseWsUrl: string;\n private readonly _baseHttpUrl: string;\n private _currentConnection?: EdgeWsConnection = undefined;\n private _ready = new Trigger();\n\n constructor(\n private _identity: EdgeIdentity,\n private readonly _config: MessengerConfig,\n ) {\n super();\n this._baseWsUrl = getEdgeUrlWithProtocol(_config.socketEndpoint, 'ws');\n this._baseHttpUrl = getEdgeUrlWithProtocol(_config.socketEndpoint, 'http');\n }\n\n @logInfo\n public get info() {\n return {\n open: this.isOpen,\n status: this.status,\n identity: this._identity.identityKey,\n device: this._identity.peerKey,\n };\n }\n\n get status(): EdgeStatus {\n return Boolean(this._currentConnection) && this._ready.state === TriggerState.RESOLVED\n ? EdgeStatus.CONNECTED\n : EdgeStatus.NOT_CONNECTED;\n }\n\n get identityKey() {\n return this._identity.identityKey;\n }\n\n get peerKey() {\n return this._identity.peerKey;\n }\n\n setIdentity(identity: EdgeIdentity): void {\n if (identity.identityKey !== this._identity.identityKey || identity.peerKey !== this._identity.peerKey) {\n log('Edge identity changed', { identity, oldIdentity: this._identity });\n this._identity = identity;\n this._closeCurrentConnection(new EdgeIdentityChangedError());\n void this._persistentLifecycle.scheduleRestart();\n }\n }\n\n public onMessage(listener: MessageListener): () => void {\n this._messageListeners.add(listener);\n return () => this._messageListeners.delete(listener);\n }\n\n public onReconnected(listener: () => void): () => void {\n this._reconnectListeners.add(listener);\n if (this._ready.state === TriggerState.RESOLVED) {\n // Microtask so that listener is always called asynchronously, no matter the state of the ready trigger\n // at the moment of registration.\n scheduleMicroTask(this._ctx, () => {\n if (this._reconnectListeners.has(listener)) {\n try {\n listener();\n } catch (error) {\n log.catch(error);\n }\n }\n });\n }\n return () => this._reconnectListeners.delete(listener);\n }\n\n /**\n * Open connection to messaging service.\n */\n protected override async _open(): Promise<void> {\n log('opening...', { info: this.info });\n this._persistentLifecycle.open().catch((err) => {\n log.warn('Error while opening connection', { err });\n });\n }\n\n /**\n * Close connection and free resources.\n */\n protected override async _close(): Promise<void> {\n log('closing...', { peerKey: this._identity.peerKey });\n this._closeCurrentConnection();\n await this._persistentLifecycle.close();\n }\n\n private async _connect(): Promise<EdgeWsConnection | undefined> {\n if (this._ctx.disposed) {\n return undefined;\n }\n\n const identity = this._identity;\n const path = `/ws/${identity.identityKey}/${identity.peerKey}`;\n const protocolHeader = this._config.disableAuth ? undefined : await this._createAuthHeader(path);\n if (this._identity !== identity) {\n log('identity changed during auth header request');\n return undefined;\n }\n\n const restartRequired = new Trigger();\n const url = new URL(path, this._baseWsUrl);\n log('Opening websocket', { url: url.toString(), protocolHeader });\n const connection = new EdgeWsConnection(\n identity,\n { url, protocolHeader },\n {\n onConnected: () => {\n if (this._isActive(connection)) {\n this._ready.wake();\n this._notifyReconnected();\n } else {\n log.verbose('connected callback ignored, because connection is not active');\n }\n },\n onRestartRequired: () => {\n if (this._isActive(connection)) {\n this._closeCurrentConnection();\n void this._persistentLifecycle.scheduleRestart();\n } else {\n log.verbose('restart requested by inactive connection');\n }\n restartRequired.wake();\n },\n onMessage: (message) => {\n if (this._isActive(connection)) {\n this._notifyMessageReceived(message);\n } else {\n log.verbose('ignored a message on inactive connection', {\n from: message.source,\n type: message.payload?.typeUrl,\n });\n }\n },\n },\n );\n this._currentConnection = connection;\n\n await connection.open();\n // Race with restartRequired so that restart is not blocked by _connect execution.\n // Wait on ready to attempt a reconnect if it times out.\n await Promise.race([this._ready.wait({ timeout: this._config.timeout ?? DEFAULT_TIMEOUT }), restartRequired]);\n\n return connection;\n }\n\n private async _disconnect(state: EdgeWsConnection): Promise<void> {\n await state.close();\n this.statusChanged.emit(this.status);\n }\n\n private _closeCurrentConnection(error: Error = new EdgeConnectionClosedError()): void {\n this._currentConnection = undefined;\n this._ready.throw(error);\n this._ready.reset();\n this.statusChanged.emit(this.status);\n }\n\n private _notifyReconnected(): void {\n this.statusChanged.emit(this.status);\n for (const listener of this._reconnectListeners) {\n try {\n listener();\n } catch (err) {\n log.error('ws reconnect listener failed', { err });\n }\n }\n }\n\n private _notifyMessageReceived(message: Message): void {\n for (const listener of this._messageListeners) {\n try {\n listener(message);\n } catch (err) {\n log.error('ws incoming message processing failed', { err, payload: protocol.getPayloadType(message) });\n }\n }\n }\n\n /**\n * Send message.\n * NOTE: The message is guaranteed to be delivered but the service must respond with a message to confirm processing.\n */\n public async send(message: Message): Promise<void> {\n if (this._ready.state !== TriggerState.RESOLVED) {\n log('waiting for websocket to become ready');\n await this._ready.wait({ timeout: this._config.timeout ?? DEFAULT_TIMEOUT });\n }\n\n if (!this._currentConnection) {\n throw new EdgeConnectionClosedError();\n }\n\n if (\n message.source &&\n (message.source.peerKey !== this._identity.peerKey || message.source.identityKey !== this.identityKey)\n ) {\n throw new EdgeIdentityChangedError();\n }\n\n this._currentConnection.send(message);\n }\n\n private async _createAuthHeader(path: string): Promise<string | undefined> {\n const httpUrl = new URL(path, this._baseHttpUrl);\n httpUrl.protocol = getEdgeUrlWithProtocol(this._baseWsUrl.toString(), 'http');\n const response = await fetch(httpUrl, { method: 'GET' });\n if (response.status === 401) {\n return encodePresentationWsAuthHeader(await handleAuthChallenge(response, this._identity));\n } else {\n log.warn('no auth challenge from edge', { status: response.status, statusText: response.statusText });\n return undefined;\n }\n }\n\n private _isActive = (connection: EdgeWsConnection) => connection === this._currentConnection;\n}\n\nconst encodePresentationWsAuthHeader = (encodedPresentation: Uint8Array): string => {\n // = and / characters are not allowed in the WebSocket subprotocol header.\n const encodedToken = Buffer.from(encodedPresentation).toString('base64').replace(/=*$/, '').replaceAll('/', '|');\n return `base64url.bearer.authorization.dxos.org.${encodedToken}`;\n};\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport { invariant } from '@dxos/invariant';\nimport { schema } from '@dxos/protocols/proto';\nimport { type Presentation } from '@dxos/protocols/proto/dxos/halo/credentials';\n\nexport interface EdgeIdentity {\n peerKey: string;\n identityKey: string;\n /**\n * Returns credential presentation issued by the identity key.\n * Presentation must have the provided challenge.\n * Presentation may include ServiceAccess credentials.\n */\n presentCredentials({ challenge }: { challenge: Uint8Array }): Promise<Presentation>;\n}\n\nexport const handleAuthChallenge = async (failedResponse: Response, identity: EdgeIdentity): Promise<Uint8Array> => {\n invariant(failedResponse.status === 401);\n\n const headerValue = failedResponse.headers.get('Www-Authenticate');\n invariant(headerValue?.startsWith('VerifiablePresentation challenge='));\n\n const challenge = headerValue?.slice('VerifiablePresentation challenge='.length);\n invariant(challenge);\n\n const presentation = await identity.presentCredentials({ challenge: Buffer.from(challenge, 'base64') });\n return schema.getCodecForType('dxos.halo.credentials.Presentation').encode(presentation);\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 { EdgeWebsocketProtocol } from '@dxos/protocols';\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_MAX_BYTES, WebSocketMuxer } from './edge-ws-muxer';\nimport { toUint8Array } from './protocol';\n\nconst SIGNAL_KEEPALIVE_INTERVAL = 4_000;\nconst SIGNAL_KEEPALIVE_TIMEOUT = 12_000;\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 private _lastReceivedMessageTimestamp = Date.now();\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): void {\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(EdgeWebsocketProtocol.V0)) {\n const binary = buf.toBinary(MessageSchema, message);\n if (binary.length > CLOUDFLARE_MESSAGE_MAX_BYTES) {\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(): Promise<void> {\n const baseProtocols = [...Object.values(EdgeWebsocketProtocol)];\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 this._lastReceivedMessageTimestamp = Date.now();\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(EdgeWebsocketProtocol.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(): Promise<void> {\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(): void {\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(): void {\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 if (Date.now() - this._lastReceivedMessageTimestamp > SIGNAL_KEEPALIVE_TIMEOUT) {\n log.warn('restart due to inactivity timeout', {\n lastReceivedMessageTimestamp: this._lastReceivedMessageTimestamp,\n });\n this._callbacks.onRestartRequired();\n } else {\n this._rescheduleHeartbeatTimeout();\n }\n }\n },\n SIGNAL_KEEPALIVE_TIMEOUT,\n );\n }\n}\n", "//\n// Copyright 2024 DXOS.org\n//\n\nexport class EdgeConnectionClosedError extends Error {\n constructor() {\n super('Edge connection closed.');\n }\n}\n\nexport class EdgeIdentityChangedError extends Error {\n constructor() {\n super('Edge identity changed.');\n }\n}\n", "//\n// Copyright 2024 DXOS.org\n//\n\nexport const getEdgeUrlWithProtocol = (baseUrl: string, protocol: 'http' | 'ws') => {\n const isSecure = baseUrl.startsWith('https') || baseUrl.startsWith('wss');\n const url = new URL(baseUrl);\n url.protocol = protocol + (isSecure ? 's' : '');\n return url.toString();\n};\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport { createCredential, signPresentation } from '@dxos/credentials';\nimport { type Signer } from '@dxos/crypto';\nimport { invariant } from '@dxos/invariant';\nimport { Keyring } from '@dxos/keyring';\nimport { PublicKey } from '@dxos/keys';\nimport { type Chain, type Credential } from '@dxos/protocols/proto/dxos/halo/credentials';\n\nimport type { EdgeIdentity } from './edge-identity';\n\n/**\n * Edge identity backed by a device key without a credential chain.\n */\nexport const createDeviceEdgeIdentity = async (signer: Signer, key: PublicKey): Promise<EdgeIdentity> => {\n return {\n identityKey: key.toHex(),\n peerKey: key.toHex(),\n presentCredentials: async ({ challenge }) => {\n return signPresentation({\n presentation: {\n credentials: [\n // Verifier requires at least one credential in the presentation to establish the subject.\n await createCredential({\n assertion: {\n '@type': 'dxos.halo.credentials.Auth',\n },\n issuer: key,\n subject: key,\n signer,\n }),\n ],\n },\n signer,\n signerKey: key,\n nonce: challenge,\n });\n },\n };\n};\n\n/**\n * Edge identity backed by a chain of credentials.\n */\nexport const createChainEdgeIdentity = async (\n signer: Signer,\n identityKey: PublicKey,\n peerKey: PublicKey,\n chain: Chain | undefined,\n credentials: Credential[],\n): Promise<EdgeIdentity> => {\n const credentialsToSign =\n credentials.length > 0\n ? credentials\n : [\n await createCredential({\n assertion: {\n '@type': 'dxos.halo.credentials.Auth',\n },\n issuer: identityKey,\n subject: identityKey,\n signer,\n chain,\n signingKey: peerKey,\n }),\n ];\n\n return {\n identityKey: identityKey.toHex(),\n peerKey: peerKey.toHex(),\n presentCredentials: async ({ challenge }) => {\n // TODO: make chain required after device invitation flow update release\n invariant(chain);\n return signPresentation({\n presentation: {\n credentials: credentialsToSign,\n },\n signer,\n nonce: challenge,\n signerKey: peerKey,\n chain,\n });\n },\n };\n};\n\n/**\n * Edge identity backed by a random ephemeral key without HALO.\n */\nexport const createEphemeralEdgeIdentity = async (): Promise<EdgeIdentity> => {\n const keyring = new Keyring();\n const key = await keyring.createKey();\n return createDeviceEdgeIdentity(keyring, key);\n};\n\n/**\n * Creates a HALO chain of credentials to act as an edge identity.\n */\nexport const createTestHaloEdgeIdentity = async (\n signer: Signer,\n identityKey: PublicKey,\n deviceKey: PublicKey,\n): Promise<EdgeIdentity> => {\n const deviceAdmission = await createCredential({\n assertion: {\n '@type': 'dxos.halo.credentials.AuthorizedDevice',\n deviceKey,\n identityKey,\n },\n issuer: identityKey,\n subject: deviceKey,\n signer,\n });\n return createChainEdgeIdentity(signer, identityKey, deviceKey, { credential: deviceAdmission }, [\n await createCredential({\n assertion: {\n '@type': 'dxos.halo.credentials.Auth',\n },\n issuer: identityKey,\n subject: identityKey,\n signer,\n }),\n ]);\n};\n\nexport const createStubEdgeIdentity = (): EdgeIdentity => {\n const identityKey = PublicKey.random();\n const deviceKey = PublicKey.random();\n return {\n identityKey: identityKey.toHex(),\n peerKey: deviceKey.toHex(),\n presentCredentials: async () => {\n throw new Error('Stub identity does not support authentication.');\n },\n };\n};\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport { FetchHttpClient, HttpClient } from '@effect/platform';\nimport { Effect, pipe } from 'effect';\n\nimport { sleep } from '@dxos/async';\nimport { Context } from '@dxos/context';\nimport { type PublicKey, type SpaceId } from '@dxos/keys';\nimport { log } from '@dxos/log';\nimport {\n type CreateAgentResponseBody,\n type CreateAgentRequestBody,\n type CreateSpaceRequest,\n type CreateSpaceResponseBody,\n EdgeAuthChallengeError,\n EdgeCallFailedError,\n type EdgeHttpResponse,\n type ExecuteWorkflowResponseBody,\n type GetAgentStatusResponseBody,\n type GetNotarizationResponseBody,\n type InitiateOAuthFlowRequest,\n type InitiateOAuthFlowResponse,\n type JoinSpaceRequest,\n type JoinSpaceResponseBody,\n type RecoverIdentityRequest,\n type RecoverIdentityResponseBody,\n type ObjectId,\n type PostNotarizationRequestBody,\n type QueueQuery,\n type QueryResult,\n type UploadFunctionRequest,\n type UploadFunctionResponseBody,\n type EdgeStatus,\n} from '@dxos/protocols';\nimport { createUrl } from '@dxos/util';\n\nimport { type EdgeIdentity, handleAuthChallenge } from './edge-identity';\nimport { encodeAuthHeader, HttpConfig, withLogging, withRetryConfig } from './http-client';\nimport { getEdgeUrlWithProtocol } from './utils';\n\nconst DEFAULT_RETRY_TIMEOUT = 1500;\nconst DEFAULT_RETRY_JITTER = 500;\nconst DEFAULT_MAX_RETRIES_COUNT = 3;\n\nexport type RetryConfig = {\n /**\n * A number of call retries, not counting the initial request.\n */\n count: number;\n /**\n * Delay before retries in ms.\n */\n timeout?: number;\n /**\n * A random amount of time before retrying to help prevent large bursts of requests.\n */\n jitter?: number;\n};\n\ntype EdgeHttpRequestArgs = {\n method: string;\n context?: Context;\n retry?: RetryConfig;\n body?: any;\n};\n\nexport type EdgeHttpGetArgs = Pick<EdgeHttpRequestArgs, 'context' | 'retry'>;\nexport type EdgeHttpPostArgs = Pick<EdgeHttpRequestArgs, 'context' | 'retry' | 'body'>;\n\nexport class EdgeHttpClient {\n private readonly _baseUrl: string;\n\n private _edgeIdentity: EdgeIdentity | undefined;\n\n /**\n * Auth header is cached until receiving the next 401 from EDGE, at which point it gets refreshed.\n */\n private _authHeader: string | undefined;\n\n constructor(baseUrl: string) {\n this._baseUrl = getEdgeUrlWithProtocol(baseUrl, 'http');\n log('created', { url: this._baseUrl });\n }\n\n get baseUrl() {\n return this._baseUrl;\n }\n\n setIdentity(identity: EdgeIdentity): void {\n if (this._edgeIdentity?.identityKey !== identity.identityKey || this._edgeIdentity?.peerKey !== identity.peerKey) {\n this._edgeIdentity = identity;\n this._authHeader = undefined;\n }\n }\n\n //\n // Status\n //\n\n public async getStatus(args?: EdgeHttpGetArgs): Promise<EdgeStatus> {\n return this._call(new URL('/status', this.baseUrl), { ...args, method: 'GET' });\n }\n\n //\n // Agents\n //\n\n public createAgent(body: CreateAgentRequestBody, args?: EdgeHttpGetArgs): Promise<CreateAgentResponseBody> {\n return this._call(new URL('/agents/create', this.baseUrl), { ...args, method: 'POST', body });\n }\n\n public getAgentStatus(\n request: { ownerIdentityKey: PublicKey },\n args?: EdgeHttpGetArgs,\n ): Promise<GetAgentStatusResponseBody> {\n return this._call(new URL(`/users/${request.ownerIdentityKey.toHex()}/agent/status`, this.baseUrl), {\n ...args,\n method: 'GET',\n });\n }\n\n //\n // Credentials\n //\n\n public getCredentialsForNotarization(spaceId: SpaceId, args?: EdgeHttpGetArgs): Promise<GetNotarizationResponseBody> {\n return this._call(new URL(`/spaces/${spaceId}/notarization`, this.baseUrl), { ...args, method: 'GET' });\n }\n\n public async notarizeCredentials(\n spaceId: SpaceId,\n body: PostNotarizationRequestBody,\n args?: EdgeHttpGetArgs,\n ): Promise<void> {\n await this._call(new URL(`/spaces/${spaceId}/notarization`, this.baseUrl), { ...args, body, method: 'POST' });\n }\n\n //\n // Identity\n //\n\n public async recoverIdentity(\n body: RecoverIdentityRequest,\n args?: EdgeHttpGetArgs,\n ): Promise<RecoverIdentityResponseBody> {\n return this._call(new URL('/identity/recover', this.baseUrl), { ...args, body, method: 'POST' });\n }\n\n //\n // Invitations\n //\n\n public async joinSpaceByInvitation(\n spaceId: SpaceId,\n body: JoinSpaceRequest,\n args?: EdgeHttpGetArgs,\n ): Promise<JoinSpaceResponseBody> {\n return this._call(new URL(`/spaces/${spaceId}/join`, this.baseUrl), { ...args, body, method: 'POST' });\n }\n\n //\n // OAuth and credentials\n //\n\n public async listFunctions(args?: EdgeHttpGetArgs): Promise<any> {\n return this._call(new URL('/functions', this.baseUrl), { ...args, method: 'GET' });\n }\n\n public async initiateOAuthFlow(\n body: InitiateOAuthFlowRequest,\n args?: EdgeHttpGetArgs,\n ): Promise<InitiateOAuthFlowResponse> {\n return this._call(new URL('/oauth/initiate', this.baseUrl), { ...args, body, method: 'POST' });\n }\n\n //\n // Spaces\n //\n\n async createSpace(body: CreateSpaceRequest, args?: EdgeHttpGetArgs): Promise<CreateSpaceResponseBody> {\n return this._call(new URL('/spaces/create', this.baseUrl), { ...args, body, method: 'POST' });\n }\n\n //\n // Queues\n //\n\n public async queryQueue(\n subspaceTag: string,\n spaceId: SpaceId,\n query: QueueQuery,\n args?: EdgeHttpGetArgs,\n ): Promise<QueryResult> {\n const { queueId } = query;\n return this._call(\n createUrl(new URL(`/spaces/${subspaceTag}/${spaceId}/queue/${queueId}/query`, this.baseUrl), {\n after: query.after,\n before: query.before,\n limit: query.limit,\n reverse: query.reverse,\n objectIds: query.objectIds?.join(','),\n }),\n {\n ...args,\n method: 'GET',\n },\n );\n }\n\n public async insertIntoQueue(\n subspaceTag: string,\n spaceId: SpaceId,\n queueId: ObjectId,\n objects: unknown[],\n args?: EdgeHttpGetArgs,\n ): Promise<void> {\n return this._call(new URL(`/spaces/${subspaceTag}/${spaceId}/queue/${queueId}`, this.baseUrl), {\n ...args,\n body: { objects },\n method: 'POST',\n });\n }\n\n public async deleteFromQueue(\n subspaceTag: string,\n spaceId: SpaceId,\n queueId: ObjectId,\n objectIds: ObjectId[],\n args?: EdgeHttpGetArgs,\n ): Promise<void> {\n return this._call(\n createUrl(new URL(`/spaces/${subspaceTag}/${spaceId}/queue/${queueId}`, this.baseUrl), {\n ids: objectIds.join(','),\n }),\n {\n ...args,\n method: 'DELETE',\n },\n );\n }\n\n //\n // Functions\n //\n\n public async uploadFunction(\n pathParts: { functionId?: string },\n body: UploadFunctionRequest,\n args?: EdgeHttpGetArgs,\n ): Promise<UploadFunctionResponseBody> {\n const path = ['functions', ...(pathParts.functionId ? [pathParts.functionId] : [])].join('/');\n return this._call(new URL(path, this.baseUrl), { ...args, body, method: 'PUT' });\n }\n\n //\n // Workflows\n //\n\n public async executeWorkflow(\n spaceId: SpaceId,\n graphId: ObjectId,\n input: any,\n args?: EdgeHttpGetArgs,\n ): Promise<ExecuteWorkflowResponseBody> {\n return this._call(new URL(`/workflows/${spaceId}/${graphId}`, this.baseUrl), {\n ...args,\n body: input,\n method: 'POST',\n });\n }\n\n //\n // Internal\n //\n\n private async _fetch<T>(url: URL, args: EdgeHttpRequestArgs): Promise<T> {\n return pipe(\n HttpClient.get(url),\n withLogging,\n withRetryConfig,\n Effect.provide(FetchHttpClient.layer),\n Effect.provide(HttpConfig.default),\n Effect.withSpan('EdgeHttpClient'),\n Effect.runPromise,\n ) as T;\n }\n\n // TODO(burdon): Refactor with effect (see edge-http-client.test.ts).\n private async _call<T>(url: URL, args: EdgeHttpRequestArgs): Promise<T> {\n const shouldRetry = createRetryHandler(args);\n const requestContext = args.context ?? new Context();\n log('fetch', { url, request: args.body });\n\n let handledAuth = false;\n while (true) {\n let processingError: EdgeCallFailedError;\n let retryAfterHeaderValue: number = Number.NaN;\n try {\n const request = createRequest(args, this._authHeader);\n const response = await fetch(url, request);\n retryAfterHeaderValue = Number(response.headers.get('Retry-After'));\n if (response.ok) {\n const body = (await response.json()) as EdgeHttpResponse<T>;\n if (body.success) {\n return body.data;\n }\n\n log.warn('unsuccessful edge response', { url, body });\n if (body.errorData?.type === 'auth_challenge' && typeof body.errorData?.challenge === 'string') {\n processingError = new EdgeAuthChallengeError(body.errorData.challenge, body.errorData);\n } else {\n processingError = EdgeCallFailedError.fromUnsuccessfulResponse(response, body);\n }\n } else if (response.status === 401 && !handledAuth) {\n this._authHeader = await this._handleUnauthorized(response);\n handledAuth = true;\n continue;\n } else {\n processingError = EdgeCallFailedError.fromHttpFailure(response);\n }\n } catch (error: any) {\n processingError = EdgeCallFailedError.fromProcessingFailureCause(error);\n }\n\n if (processingError.isRetryable && (await shouldRetry(requestContext, retryAfterHeaderValue))) {\n log('retrying edge request', { url, processingError });\n } else {\n throw processingError;\n }\n }\n }\n\n private async _handleUnauthorized(response: Response): Promise<string> {\n if (!this._edgeIdentity) {\n log.warn('unauthorized response received before identity was set');\n throw EdgeCallFailedError.fromHttpFailure(response);\n }\n\n const challenge = await handleAuthChallenge(response, this._edgeIdentity);\n return encodeAuthHeader(challenge);\n }\n}\n\nconst createRequest = ({ method, body }: EdgeHttpRequestArgs, authHeader: string | undefined): RequestInit => {\n return {\n method,\n body: body && JSON.stringify(body),\n headers: authHeader ? { Authorization: authHeader } : undefined,\n };\n};\n\n/**\n * @deprecated\n */\nconst createRetryHandler = ({ retry }: EdgeHttpRequestArgs) => {\n if (!retry || retry.count < 1) {\n return async () => false;\n }\n\n let retries = 0;\n const maxRetries = retry.count ?? DEFAULT_MAX_RETRIES_COUNT;\n const baseTimeout = retry.timeout ?? DEFAULT_RETRY_TIMEOUT;\n const jitter = retry.jitter ?? DEFAULT_RETRY_JITTER;\n return async (ctx: Context, retryAfter: number) => {\n if (++retries > maxRetries || ctx.disposed) {\n return false;\n }\n\n if (retryAfter) {\n await sleep(retryAfter);\n } else {\n const timeout = baseTimeout + Math.random() * jitter;\n await sleep(timeout);\n }\n\n return true;\n };\n};\n", "//\n// Copyright 2025 DXOS.org\n//\n\nimport { type HttpClient } from '@effect/platform';\nimport { type HttpClientError } from '@effect/platform/HttpClientError';\nimport { type HttpClientResponse } from '@effect/platform/HttpClientResponse';\nimport { Context, Duration, Effect, Layer, Schedule } from 'effect';\n\nimport { log } from '@dxos/log';\n\n// TODO(burdon): Factor out.\n\nexport type RetryOptions = {\n timeout: Duration.Duration;\n retryTimes: number;\n retryBaseDelay: Duration.Duration;\n};\n\n// Layer pattern.\nexport class HttpConfig extends Context.Tag('HttpConfig')<HttpConfig, RetryOptions>() {\n static default = Layer.succeed(HttpConfig, {\n timeout: Duration.millis(1_000),\n retryTimes: 3,\n retryBaseDelay: Duration.millis(1_000),\n });\n}\n\n// HOC pattern.\nexport const withRetry = (\n effect: Effect.Effect<HttpClientResponse, HttpClientError, HttpClient.HttpClient>,\n {\n timeout = Duration.millis(1_000),\n retryBaseDelay = Duration.millis(1_000),\n retryTimes = 3,\n }: Partial<RetryOptions> = {},\n) => {\n return effect.pipe(\n Effect.flatMap((res) =>\n // Treat 500 errors as retryable?\n res.status === 500 ? Effect.fail(new Error(res.status.toString())) : res.json,\n ),\n Effect.timeout(timeout),\n Effect.retry({\n schedule: Schedule.exponential(retryBaseDelay).pipe(Schedule.jittered),\n times: retryTimes,\n }),\n );\n};\n\nexport const withRetryConfig = (effect: Effect.Effect<HttpClientResponse, HttpClientError, HttpClient.HttpClient>) =>\n Effect.gen(function* () {\n const config = yield* HttpConfig;\n return yield* withRetry(effect, config);\n });\n\nexport const withLogging = <A extends HttpClientResponse, E, R>(effect: Effect.Effect<A, E, R>) =>\n effect.pipe(Effect.tap((res) => log.info('response', { status: res.status })));\n\n/**\n *\n */\n// TODO(burdon): Document.\nexport const encodeAuthHeader = (challenge: Uint8Array) => {\n const encodedChallenge = Buffer.from(challenge).toString('base64');\n return `VerifiablePresentation pb;base64,${encodedChallenge}`;\n};\n"],
5
- "mappings": "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",
6
- "names": ["Trigger", "scheduleMicroTask", "TriggerState", "PersistentLifecycle", "Event", "Resource", "log", "logInfo", "EdgeStatus", "invariant", "schema", "handleAuthChallenge", "failedResponse", "identity", "status", "headerValue", "headers", "get", "startsWith", "challenge", "slice", "length", "presentation", "presentCredentials", "Buffer", "from", "getCodecForType", "encode", "WebSocket", "scheduleTask", "scheduleTaskInterval", "Context", "Resource", "invariant", "log", "logInfo", "EdgeWebsocketProtocol", "buf", "MessageSchema", "SIGNAL_KEEPALIVE_INTERVAL", "SIGNAL_KEEPALIVE_TIMEOUT", "EdgeWsConnection", "Resource", "_identity", "_connectionInfo", "_callbacks", "_lastReceivedMessageTimestamp", "Date", "now", "info", "open", "isOpen", "identity", "identityKey", "device", "peerKey", "send", "message", "invariant", "_ws", "_wsMuxer", "log", "payload", "protocol", "getPayloadType", "includes", "EdgeWebsocketProtocol", "V0", "binary", "buf", "toBinary", "MessageSchema", "length", "CLOUDFLARE_MESSAGE_MAX_BYTES", "error", "byteLength", "serviceId", "catch", "e", "_open", "baseProtocols", "Object", "values", "WebSocket", "url", "toString", "protocolHeader", "muxer", "WebSocketMuxer", "onopen", "onConnected", "_scheduleHeartbeats", "verbose", "currentIdentity", "onclose", "event", "warn", "code", "reason", "onRestartRequired", "destroy", "onerror", "onmessage", "type", "data", "_rescheduleHeartbeatTimeout", "bytes", "toUint8Array", "fromBinary", "receiveData", "from", "source", "onMessage", "_close", "_inactivityTimeoutCtx", "dispose", "close", "undefined", "err", "Error", "scheduleTaskInterval", "_ctx", "Context", "scheduleTask", "lastReceivedMessageTimestamp", "EdgeConnectionClosedError", "Error", "EdgeIdentityChangedError", "getEdgeUrlWithProtocol", "baseUrl", "protocol", "isSecure", "startsWith", "url", "URL", "toString", "DEFAULT_TIMEOUT", "EdgeClient", "Resource", "_identity", "_config", "statusChanged", "Event", "_persistentLifecycle", "PersistentLifecycle", "start", "_connect", "stop", "state", "_disconnect", "_messageListeners", "Set", "_reconnectListeners", "_currentConnection", "undefined", "_ready", "Trigger", "_isActive", "connection", "_baseWsUrl", "getEdgeUrlWithProtocol", "socketEndpoint", "_baseHttpUrl", "info", "open", "isOpen", "status", "identity", "identityKey", "device", "peerKey", "Boolean", "TriggerState", "RESOLVED", "EdgeStatus", "CONNECTED", "NOT_CONNECTED", "setIdentity", "log", "oldIdentity", "_closeCurrentConnection", "EdgeIdentityChangedError", "scheduleRestart", "onMessage", "listener", "add", "delete", "onReconnected", "scheduleMicroTask", "_ctx", "has", "error", "catch", "_open", "err", "warn", "_close", "close", "disposed", "path", "protocolHeader", "disableAuth", "_createAuthHeader", "restartRequired", "url", "URL", "toString", "EdgeWsConnection", "onConnected", "wake", "_notifyReconnected", "verbose", "onRestartRequired", "message", "_notifyMessageReceived", "from", "source", "type", "payload", "typeUrl", "Promise", "race", "wait", "timeout", "emit", "EdgeConnectionClosedError", "throw", "reset", "protocol", "getPayloadType", "send", "httpUrl", "response", "fetch", "method", "encodePresentationWsAuthHeader", "handleAuthChallenge", "statusText", "encodedPresentation", "encodedToken", "Buffer", "replace", "replaceAll", "createCredential", "signPresentation", "invariant", "Keyring", "PublicKey", "createDeviceEdgeIdentity", "signer", "key", "identityKey", "toHex", "peerKey", "presentCredentials", "challenge", "presentation", "credentials", "assertion", "issuer", "subject", "signerKey", "nonce", "createChainEdgeIdentity", "chain", "credentialsToSign", "length", "signingKey", "createEphemeralEdgeIdentity", "keyring", "createKey", "createTestHaloEdgeIdentity", "deviceKey", "deviceAdmission", "credential", "createStubEdgeIdentity", "random", "Error", "FetchHttpClient", "HttpClient", "Effect", "pipe", "sleep", "Context", "log", "EdgeAuthChallengeError", "EdgeCallFailedError", "createUrl", "Context", "Duration", "Effect", "Layer", "Schedule", "log", "HttpConfig", "Tag", "default", "succeed", "timeout", "millis", "retryTimes", "retryBaseDelay", "withRetry", "effect", "pipe", "flatMap", "res", "status", "fail", "Error", "toString", "json", "retry", "schedule", "exponential", "jittered", "times", "withRetryConfig", "gen", "config", "withLogging", "tap", "info", "encodeAuthHeader", "challenge", "encodedChallenge", "Buffer", "from", "DEFAULT_RETRY_TIMEOUT", "DEFAULT_RETRY_JITTER", "DEFAULT_MAX_RETRIES_COUNT", "EdgeHttpClient", "baseUrl", "_baseUrl", "getEdgeUrlWithProtocol", "log", "url", "setIdentity", "identity", "_edgeIdentity", "identityKey", "peerKey", "_authHeader", "undefined", "getStatus", "args", "_call", "URL", "method", "createAgent", "body", "getAgentStatus", "request", "ownerIdentityKey", "toHex", "getCredentialsForNotarization", "spaceId", "notarizeCredentials", "recoverIdentity", "joinSpaceByInvitation", "listFunctions", "initiateOAuthFlow", "createSpace", "queryQueue", "subspaceTag", "query", "queueId", "createUrl", "after", "before", "limit", "reverse", "objectIds", "join", "insertIntoQueue", "objects", "deleteFromQueue", "ids", "uploadFunction", "pathParts", "path", "functionId", "executeWorkflow", "graphId", "input", "_fetch", "pipe", "HttpClient", "get", "withLogging", "withRetryConfig", "Effect", "provide", "FetchHttpClient", "layer", "HttpConfig", "default", "withSpan", "runPromise", "shouldRetry", "createRetryHandler", "requestContext", "context", "Context", "handledAuth", "processingError", "retryAfterHeaderValue", "Number", "NaN", "createRequest", "response", "fetch", "headers", "ok", "json", "success", "data", "warn", "errorData", "type", "challenge", "EdgeAuthChallengeError", "EdgeCallFailedError", "fromUnsuccessfulResponse", "status", "_handleUnauthorized", "fromHttpFailure", "error", "fromProcessingFailureCause", "isRetryable", "handleAuthChallenge", "encodeAuthHeader", "authHeader", "JSON", "stringify", "Authorization", "retry", "count", "retries", "maxRetries", "baseTimeout", "timeout", "jitter", "ctx", "retryAfter", "disposed", "sleep", "Math", "random"]
4
+ "sourcesContent": ["//\n// Copyright 2024 DXOS.org\n//\n\nexport * from '@dxos/protocols/buf/dxos/edge/messenger_pb';\n\nexport * from './edge-client';\nexport * from './defs';\nexport * from './protocol';\nexport * from './errors';\nexport * from './auth';\nexport * from './edge-http-client';\nexport * from './edge-identity';\nexport * from './edge-ws-muxer';\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport { Event, PersistentLifecycle, Trigger, TriggerState, scheduleMicroTask } from '@dxos/async';\nimport { type Lifecycle, Resource } from '@dxos/context';\nimport { log, logInfo } from '@dxos/log';\nimport { type Message } from '@dxos/protocols/buf/dxos/edge/messenger_pb';\nimport { EdgeStatus } from '@dxos/protocols/proto/dxos/client/services';\n\nimport { protocol } from './defs';\nimport { type EdgeIdentity, handleAuthChallenge } from './edge-identity';\nimport { EdgeWsConnection } from './edge-ws-connection';\nimport { EdgeConnectionClosedError, EdgeIdentityChangedError } from './errors';\nimport { type Protocol } from './protocol';\nimport { getEdgeUrlWithProtocol } from './utils';\n\nconst DEFAULT_TIMEOUT = 10_000;\n\nexport type MessageListener = (message: Message) => void;\nexport type ReconnectListener = () => void;\n\nexport interface EdgeConnection extends Required<Lifecycle> {\n statusChanged: Event<EdgeStatus>;\n get info(): any;\n get identityKey(): string;\n get peerKey(): string;\n get isOpen(): boolean;\n get status(): EdgeStatus;\n setIdentity(identity: EdgeIdentity): void;\n onMessage(listener: MessageListener): () => void;\n onReconnected(listener: ReconnectListener): () => void;\n send(message: Message): Promise<void>;\n}\n\nexport type MessengerConfig = {\n socketEndpoint: string;\n timeout?: number;\n protocol?: Protocol;\n disableAuth?: boolean;\n};\n\n/**\n * Messenger client for EDGE:\n * - While open, uses PersistentLifecycle to keep an open EdgeWsConnection, reconnecting on failures.\n * - Manages identity and re-create EdgeWsConnection when identity changes.\n * - Dispatches connection state and message notifications.\n */\nexport class EdgeClient extends Resource implements EdgeConnection {\n public readonly statusChanged = new Event<EdgeStatus>();\n\n private readonly _persistentLifecycle = new PersistentLifecycle<EdgeWsConnection>({\n start: async () => this._connect(),\n stop: async (state: EdgeWsConnection) => this._disconnect(state),\n });\n\n private readonly _messageListeners = new Set<MessageListener>();\n private readonly _reconnectListeners = new Set<ReconnectListener>();\n private readonly _baseWsUrl: string;\n private readonly _baseHttpUrl: string;\n private _currentConnection?: EdgeWsConnection = undefined;\n private _ready = new Trigger();\n\n constructor(\n private _identity: EdgeIdentity,\n private readonly _config: MessengerConfig,\n ) {\n super();\n this._baseWsUrl = getEdgeUrlWithProtocol(_config.socketEndpoint, 'ws');\n this._baseHttpUrl = getEdgeUrlWithProtocol(_config.socketEndpoint, 'http');\n }\n\n @logInfo\n public get info() {\n return {\n open: this.isOpen,\n status: this.status,\n identity: this._identity.identityKey,\n device: this._identity.peerKey,\n };\n }\n\n get status(): EdgeStatus {\n return Boolean(this._currentConnection) && this._ready.state === TriggerState.RESOLVED\n ? EdgeStatus.CONNECTED\n : EdgeStatus.NOT_CONNECTED;\n }\n\n get identityKey() {\n return this._identity.identityKey;\n }\n\n get peerKey() {\n return this._identity.peerKey;\n }\n\n setIdentity(identity: EdgeIdentity): void {\n if (identity.identityKey !== this._identity.identityKey || identity.peerKey !== this._identity.peerKey) {\n log('Edge identity changed', { identity, oldIdentity: this._identity });\n this._identity = identity;\n this._closeCurrentConnection(new EdgeIdentityChangedError());\n void this._persistentLifecycle.scheduleRestart();\n }\n }\n\n public onMessage(listener: MessageListener): () => void {\n this._messageListeners.add(listener);\n return () => this._messageListeners.delete(listener);\n }\n\n public onReconnected(listener: () => void): () => void {\n this._reconnectListeners.add(listener);\n if (this._ready.state === TriggerState.RESOLVED) {\n // Microtask so that listener is always called asynchronously, no matter the state of the ready trigger\n // at the moment of registration.\n scheduleMicroTask(this._ctx, () => {\n if (this._reconnectListeners.has(listener)) {\n try {\n listener();\n } catch (error) {\n log.catch(error);\n }\n }\n });\n }\n return () => this._reconnectListeners.delete(listener);\n }\n\n /**\n * Open connection to messaging service.\n */\n protected override async _open(): Promise<void> {\n log('opening...', { info: this.info });\n this._persistentLifecycle.open().catch((err) => {\n log.warn('Error while opening connection', { err });\n });\n }\n\n /**\n * Close connection and free resources.\n */\n protected override async _close(): Promise<void> {\n log('closing...', { peerKey: this._identity.peerKey });\n this._closeCurrentConnection();\n await this._persistentLifecycle.close();\n }\n\n private async _connect(): Promise<EdgeWsConnection | undefined> {\n if (this._ctx.disposed) {\n return undefined;\n }\n\n const identity = this._identity;\n const path = `/ws/${identity.identityKey}/${identity.peerKey}`;\n const protocolHeader = this._config.disableAuth ? undefined : await this._createAuthHeader(path);\n if (this._identity !== identity) {\n log('identity changed during auth header request');\n return undefined;\n }\n\n const restartRequired = new Trigger();\n const url = new URL(path, this._baseWsUrl);\n log('Opening websocket', { url: url.toString(), protocolHeader });\n const connection = new EdgeWsConnection(\n identity,\n { url, protocolHeader },\n {\n onConnected: () => {\n if (this._isActive(connection)) {\n this._ready.wake();\n this._notifyReconnected();\n } else {\n log.verbose('connected callback ignored, because connection is not active');\n }\n },\n onRestartRequired: () => {\n if (this._isActive(connection)) {\n this._closeCurrentConnection();\n void this._persistentLifecycle.scheduleRestart();\n } else {\n log.verbose('restart requested by inactive connection');\n }\n restartRequired.wake();\n },\n onMessage: (message) => {\n if (this._isActive(connection)) {\n this._notifyMessageReceived(message);\n } else {\n log.verbose('ignored a message on inactive connection', {\n from: message.source,\n type: message.payload?.typeUrl,\n });\n }\n },\n },\n );\n this._currentConnection = connection;\n\n await connection.open();\n // Race with restartRequired so that restart is not blocked by _connect execution.\n // Wait on ready to attempt a reconnect if it times out.\n await Promise.race([this._ready.wait({ timeout: this._config.timeout ?? DEFAULT_TIMEOUT }), restartRequired]);\n\n return connection;\n }\n\n private async _disconnect(state: EdgeWsConnection): Promise<void> {\n await state.close();\n this.statusChanged.emit(this.status);\n }\n\n private _closeCurrentConnection(error: Error = new EdgeConnectionClosedError()): void {\n this._currentConnection = undefined;\n this._ready.throw(error);\n this._ready.reset();\n this.statusChanged.emit(this.status);\n }\n\n private _notifyReconnected(): void {\n this.statusChanged.emit(this.status);\n for (const listener of this._reconnectListeners) {\n try {\n listener();\n } catch (err) {\n log.error('ws reconnect listener failed', { err });\n }\n }\n }\n\n private _notifyMessageReceived(message: Message): void {\n for (const listener of this._messageListeners) {\n try {\n listener(message);\n } catch (err) {\n log.error('ws incoming message processing failed', { err, payload: protocol.getPayloadType(message) });\n }\n }\n }\n\n /**\n * Send message.\n * NOTE: The message is guaranteed to be delivered but the service must respond with a message to confirm processing.\n */\n public async send(message: Message): Promise<void> {\n if (this._ready.state !== TriggerState.RESOLVED) {\n log('waiting for websocket to become ready');\n await this._ready.wait({ timeout: this._config.timeout ?? DEFAULT_TIMEOUT });\n }\n\n if (!this._currentConnection) {\n throw new EdgeConnectionClosedError();\n }\n\n if (\n message.source &&\n (message.source.peerKey !== this._identity.peerKey || message.source.identityKey !== this.identityKey)\n ) {\n throw new EdgeIdentityChangedError();\n }\n\n this._currentConnection.send(message);\n }\n\n private async _createAuthHeader(path: string): Promise<string | undefined> {\n const httpUrl = new URL(path, this._baseHttpUrl);\n httpUrl.protocol = getEdgeUrlWithProtocol(this._baseWsUrl.toString(), 'http');\n const response = await fetch(httpUrl, { method: 'GET' });\n if (response.status === 401) {\n return encodePresentationWsAuthHeader(await handleAuthChallenge(response, this._identity));\n } else {\n log.warn('no auth challenge from edge', { status: response.status, statusText: response.statusText });\n return undefined;\n }\n }\n\n private _isActive = (connection: EdgeWsConnection) => connection === this._currentConnection;\n}\n\nconst encodePresentationWsAuthHeader = (encodedPresentation: Uint8Array): string => {\n // = and / characters are not allowed in the WebSocket subprotocol header.\n const encodedToken = Buffer.from(encodedPresentation).toString('base64').replace(/=*$/, '').replaceAll('/', '|');\n return `base64url.bearer.authorization.dxos.org.${encodedToken}`;\n};\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport { invariant } from '@dxos/invariant';\nimport { schema } from '@dxos/protocols/proto';\nimport { type Presentation } from '@dxos/protocols/proto/dxos/halo/credentials';\n\nexport interface EdgeIdentity {\n peerKey: string;\n identityKey: string;\n /**\n * Returns credential presentation issued by the identity key.\n * Presentation must have the provided challenge.\n * Presentation may include ServiceAccess credentials.\n */\n presentCredentials({ challenge }: { challenge: Uint8Array }): Promise<Presentation>;\n}\n\nexport const handleAuthChallenge = async (failedResponse: Response, identity: EdgeIdentity): Promise<Uint8Array> => {\n invariant(failedResponse.status === 401);\n\n const headerValue = failedResponse.headers.get('Www-Authenticate');\n invariant(headerValue?.startsWith('VerifiablePresentation challenge='));\n\n const challenge = headerValue?.slice('VerifiablePresentation challenge='.length);\n invariant(challenge);\n\n const presentation = await identity.presentCredentials({ challenge: Buffer.from(challenge, 'base64') });\n return schema.getCodecForType('dxos.halo.credentials.Presentation').encode(presentation);\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 { EdgeWebsocketProtocol } from '@dxos/protocols';\nimport { buf } from '@dxos/protocols/buf';\nimport { type Message, MessageSchema } from '@dxos/protocols/buf/dxos/edge/messenger_pb';\n\nimport { protocol } from './defs';\nimport { type EdgeIdentity } from './edge-identity';\nimport { CLOUDFLARE_MESSAGE_MAX_BYTES, WebSocketMuxer } from './edge-ws-muxer';\nimport { toUint8Array } from './protocol';\n\nconst SIGNAL_KEEPALIVE_INTERVAL = 4_000;\nconst SIGNAL_KEEPALIVE_TIMEOUT = 12_000;\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 private _lastReceivedMessageTimestamp = Date.now();\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): void {\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(EdgeWebsocketProtocol.V0)) {\n const binary = buf.toBinary(MessageSchema, message);\n if (binary.length > CLOUDFLARE_MESSAGE_MAX_BYTES) {\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(): Promise<void> {\n const baseProtocols = [...Object.values(EdgeWebsocketProtocol)];\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 this._lastReceivedMessageTimestamp = Date.now();\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(EdgeWebsocketProtocol.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(): Promise<void> {\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(): void {\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(): void {\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 if (Date.now() - this._lastReceivedMessageTimestamp > SIGNAL_KEEPALIVE_TIMEOUT) {\n log.warn('restart due to inactivity timeout', {\n lastReceivedMessageTimestamp: this._lastReceivedMessageTimestamp,\n });\n this._callbacks.onRestartRequired();\n } else {\n this._rescheduleHeartbeatTimeout();\n }\n }\n },\n SIGNAL_KEEPALIVE_TIMEOUT,\n );\n }\n}\n", "//\n// Copyright 2024 DXOS.org\n//\n\nexport class EdgeConnectionClosedError extends Error {\n constructor() {\n super('Edge connection closed.');\n }\n}\n\nexport class EdgeIdentityChangedError extends Error {\n constructor() {\n super('Edge identity changed.');\n }\n}\n", "//\n// Copyright 2024 DXOS.org\n//\n\nexport const getEdgeUrlWithProtocol = (baseUrl: string, protocol: 'http' | 'ws') => {\n const isSecure = baseUrl.startsWith('https') || baseUrl.startsWith('wss');\n const url = new URL(baseUrl);\n url.protocol = protocol + (isSecure ? 's' : '');\n return url.toString();\n};\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport { createCredential, signPresentation } from '@dxos/credentials';\nimport { type Signer } from '@dxos/crypto';\nimport { invariant } from '@dxos/invariant';\nimport { Keyring } from '@dxos/keyring';\nimport { PublicKey } from '@dxos/keys';\nimport { type Chain, type Credential } from '@dxos/protocols/proto/dxos/halo/credentials';\n\nimport type { EdgeIdentity } from './edge-identity';\n\n/**\n * Edge identity backed by a device key without a credential chain.\n */\nexport const createDeviceEdgeIdentity = async (signer: Signer, key: PublicKey): Promise<EdgeIdentity> => {\n return {\n identityKey: key.toHex(),\n peerKey: key.toHex(),\n presentCredentials: async ({ challenge }) => {\n return signPresentation({\n presentation: {\n credentials: [\n // Verifier requires at least one credential in the presentation to establish the subject.\n await createCredential({\n assertion: {\n '@type': 'dxos.halo.credentials.Auth',\n },\n issuer: key,\n subject: key,\n signer,\n }),\n ],\n },\n signer,\n signerKey: key,\n nonce: challenge,\n });\n },\n };\n};\n\n/**\n * Edge identity backed by a chain of credentials.\n */\nexport const createChainEdgeIdentity = async (\n signer: Signer,\n identityKey: PublicKey,\n peerKey: PublicKey,\n chain: Chain | undefined,\n credentials: Credential[],\n): Promise<EdgeIdentity> => {\n const credentialsToSign =\n credentials.length > 0\n ? credentials\n : [\n await createCredential({\n assertion: {\n '@type': 'dxos.halo.credentials.Auth',\n },\n issuer: identityKey,\n subject: identityKey,\n signer,\n chain,\n signingKey: peerKey,\n }),\n ];\n\n return {\n identityKey: identityKey.toHex(),\n peerKey: peerKey.toHex(),\n presentCredentials: async ({ challenge }) => {\n // TODO: make chain required after device invitation flow update release\n invariant(chain);\n return signPresentation({\n presentation: {\n credentials: credentialsToSign,\n },\n signer,\n nonce: challenge,\n signerKey: peerKey,\n chain,\n });\n },\n };\n};\n\n/**\n * Edge identity backed by a random ephemeral key without HALO.\n */\nexport const createEphemeralEdgeIdentity = async (): Promise<EdgeIdentity> => {\n const keyring = new Keyring();\n const key = await keyring.createKey();\n return createDeviceEdgeIdentity(keyring, key);\n};\n\n/**\n * Creates a HALO chain of credentials to act as an edge identity.\n */\nexport const createTestHaloEdgeIdentity = async (\n signer: Signer,\n identityKey: PublicKey,\n deviceKey: PublicKey,\n): Promise<EdgeIdentity> => {\n const deviceAdmission = await createCredential({\n assertion: {\n '@type': 'dxos.halo.credentials.AuthorizedDevice',\n deviceKey,\n identityKey,\n },\n issuer: identityKey,\n subject: deviceKey,\n signer,\n });\n return createChainEdgeIdentity(signer, identityKey, deviceKey, { credential: deviceAdmission }, [\n await createCredential({\n assertion: {\n '@type': 'dxos.halo.credentials.Auth',\n },\n issuer: identityKey,\n subject: identityKey,\n signer,\n }),\n ]);\n};\n\nexport const createStubEdgeIdentity = (): EdgeIdentity => {\n const identityKey = PublicKey.random();\n const deviceKey = PublicKey.random();\n return {\n identityKey: identityKey.toHex(),\n peerKey: deviceKey.toHex(),\n presentCredentials: async () => {\n throw new Error('Stub identity does not support authentication.');\n },\n };\n};\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport { FetchHttpClient, HttpClient } from '@effect/platform';\nimport { Effect, pipe } from 'effect';\n\nimport { sleep } from '@dxos/async';\nimport { Context } from '@dxos/context';\nimport { type PublicKey, type SpaceId } from '@dxos/keys';\nimport { log } from '@dxos/log';\nimport {\n type CreateAgentRequestBody,\n type CreateAgentResponseBody,\n type CreateSpaceRequest,\n type CreateSpaceResponseBody,\n EdgeAuthChallengeError,\n EdgeCallFailedError,\n type EdgeHttpResponse,\n type EdgeStatus,\n type ExecuteWorkflowResponseBody,\n type GetAgentStatusResponseBody,\n type GetNotarizationResponseBody,\n type InitiateOAuthFlowRequest,\n type InitiateOAuthFlowResponse,\n type JoinSpaceRequest,\n type JoinSpaceResponseBody,\n type ObjectId,\n type PostNotarizationRequestBody,\n type QueryResult,\n type QueueQuery,\n type RecoverIdentityRequest,\n type RecoverIdentityResponseBody,\n type UploadFunctionRequest,\n type UploadFunctionResponseBody,\n} from '@dxos/protocols';\nimport { createUrl } from '@dxos/util';\n\nimport { type EdgeIdentity, handleAuthChallenge } from './edge-identity';\nimport { HttpConfig, encodeAuthHeader, withLogging, withRetryConfig } from './http-client';\nimport { getEdgeUrlWithProtocol } from './utils';\n\nconst DEFAULT_RETRY_TIMEOUT = 1500;\nconst DEFAULT_RETRY_JITTER = 500;\nconst DEFAULT_MAX_RETRIES_COUNT = 3;\n\nexport type RetryConfig = {\n /**\n * A number of call retries, not counting the initial request.\n */\n count: number;\n /**\n * Delay before retries in ms.\n */\n timeout?: number;\n /**\n * A random amount of time before retrying to help prevent large bursts of requests.\n */\n jitter?: number;\n};\n\ntype EdgeHttpRequestArgs = {\n method: string;\n context?: Context;\n retry?: RetryConfig;\n body?: any;\n};\n\nexport type EdgeHttpGetArgs = Pick<EdgeHttpRequestArgs, 'context' | 'retry'>;\nexport type EdgeHttpPostArgs = Pick<EdgeHttpRequestArgs, 'context' | 'retry' | 'body'>;\n\nexport class EdgeHttpClient {\n private readonly _baseUrl: string;\n\n private _edgeIdentity: EdgeIdentity | undefined;\n\n /**\n * Auth header is cached until receiving the next 401 from EDGE, at which point it gets refreshed.\n */\n private _authHeader: string | undefined;\n\n constructor(baseUrl: string) {\n this._baseUrl = getEdgeUrlWithProtocol(baseUrl, 'http');\n log('created', { url: this._baseUrl });\n }\n\n get baseUrl() {\n return this._baseUrl;\n }\n\n setIdentity(identity: EdgeIdentity): void {\n if (this._edgeIdentity?.identityKey !== identity.identityKey || this._edgeIdentity?.peerKey !== identity.peerKey) {\n this._edgeIdentity = identity;\n this._authHeader = undefined;\n }\n }\n\n //\n // Status\n //\n\n public async getStatus(args?: EdgeHttpGetArgs): Promise<EdgeStatus> {\n return this._call(new URL('/status', this.baseUrl), { ...args, method: 'GET' });\n }\n\n //\n // Agents\n //\n\n public createAgent(body: CreateAgentRequestBody, args?: EdgeHttpGetArgs): Promise<CreateAgentResponseBody> {\n return this._call(new URL('/agents/create', this.baseUrl), { ...args, method: 'POST', body });\n }\n\n public getAgentStatus(\n request: { ownerIdentityKey: PublicKey },\n args?: EdgeHttpGetArgs,\n ): Promise<GetAgentStatusResponseBody> {\n return this._call(new URL(`/users/${request.ownerIdentityKey.toHex()}/agent/status`, this.baseUrl), {\n ...args,\n method: 'GET',\n });\n }\n\n //\n // Credentials\n //\n\n public getCredentialsForNotarization(spaceId: SpaceId, args?: EdgeHttpGetArgs): Promise<GetNotarizationResponseBody> {\n return this._call(new URL(`/spaces/${spaceId}/notarization`, this.baseUrl), { ...args, method: 'GET' });\n }\n\n public async notarizeCredentials(\n spaceId: SpaceId,\n body: PostNotarizationRequestBody,\n args?: EdgeHttpGetArgs,\n ): Promise<void> {\n await this._call(new URL(`/spaces/${spaceId}/notarization`, this.baseUrl), { ...args, body, method: 'POST' });\n }\n\n //\n // Identity\n //\n\n public async recoverIdentity(\n body: RecoverIdentityRequest,\n args?: EdgeHttpGetArgs,\n ): Promise<RecoverIdentityResponseBody> {\n return this._call(new URL('/identity/recover', this.baseUrl), { ...args, body, method: 'POST' });\n }\n\n //\n // Invitations\n //\n\n public async joinSpaceByInvitation(\n spaceId: SpaceId,\n body: JoinSpaceRequest,\n args?: EdgeHttpGetArgs,\n ): Promise<JoinSpaceResponseBody> {\n return this._call(new URL(`/spaces/${spaceId}/join`, this.baseUrl), { ...args, body, method: 'POST' });\n }\n\n //\n // OAuth and credentials\n //\n\n public async listFunctions(args?: EdgeHttpGetArgs): Promise<any> {\n return this._call(new URL('/functions', this.baseUrl), { ...args, method: 'GET' });\n }\n\n public async initiateOAuthFlow(\n body: InitiateOAuthFlowRequest,\n args?: EdgeHttpGetArgs,\n ): Promise<InitiateOAuthFlowResponse> {\n return this._call(new URL('/oauth/initiate', this.baseUrl), { ...args, body, method: 'POST' });\n }\n\n //\n // Spaces\n //\n\n async createSpace(body: CreateSpaceRequest, args?: EdgeHttpGetArgs): Promise<CreateSpaceResponseBody> {\n return this._call(new URL('/spaces/create', this.baseUrl), { ...args, body, method: 'POST' });\n }\n\n //\n // Queues\n //\n\n public async queryQueue(\n subspaceTag: string,\n spaceId: SpaceId,\n query: QueueQuery,\n args?: EdgeHttpGetArgs,\n ): Promise<QueryResult> {\n const { queueId } = query;\n return this._call(\n createUrl(new URL(`/spaces/${subspaceTag}/${spaceId}/queue/${queueId}/query`, this.baseUrl), {\n after: query.after,\n before: query.before,\n limit: query.limit,\n reverse: query.reverse,\n objectIds: query.objectIds?.join(','),\n }),\n {\n ...args,\n method: 'GET',\n },\n );\n }\n\n public async insertIntoQueue(\n subspaceTag: string,\n spaceId: SpaceId,\n queueId: ObjectId,\n objects: unknown[],\n args?: EdgeHttpGetArgs,\n ): Promise<void> {\n return this._call(new URL(`/spaces/${subspaceTag}/${spaceId}/queue/${queueId}`, this.baseUrl), {\n ...args,\n body: { objects },\n method: 'POST',\n });\n }\n\n public async deleteFromQueue(\n subspaceTag: string,\n spaceId: SpaceId,\n queueId: ObjectId,\n objectIds: ObjectId[],\n args?: EdgeHttpGetArgs,\n ): Promise<void> {\n return this._call(\n createUrl(new URL(`/spaces/${subspaceTag}/${spaceId}/queue/${queueId}`, this.baseUrl), {\n ids: objectIds.join(','),\n }),\n {\n ...args,\n method: 'DELETE',\n },\n );\n }\n\n //\n // Functions\n //\n\n public async uploadFunction(\n pathParts: { functionId?: string },\n body: UploadFunctionRequest,\n args?: EdgeHttpGetArgs,\n ): Promise<UploadFunctionResponseBody> {\n const path = ['functions', ...(pathParts.functionId ? [pathParts.functionId] : [])].join('/');\n return this._call(new URL(path, this.baseUrl), { ...args, body, method: 'PUT' });\n }\n\n //\n // Workflows\n //\n\n public async executeWorkflow(\n spaceId: SpaceId,\n graphId: ObjectId,\n input: any,\n args?: EdgeHttpGetArgs,\n ): Promise<ExecuteWorkflowResponseBody> {\n return this._call(new URL(`/workflows/${spaceId}/${graphId}`, this.baseUrl), {\n ...args,\n body: input,\n method: 'POST',\n });\n }\n\n //\n // Internal\n //\n\n private async _fetch<T>(url: URL, args: EdgeHttpRequestArgs): Promise<T> {\n return pipe(\n HttpClient.get(url),\n withLogging,\n withRetryConfig,\n Effect.provide(FetchHttpClient.layer),\n Effect.provide(HttpConfig.default),\n Effect.withSpan('EdgeHttpClient'),\n Effect.runPromise,\n ) as T;\n }\n\n // TODO(burdon): Refactor with effect (see edge-http-client.test.ts).\n private async _call<T>(url: URL, args: EdgeHttpRequestArgs): Promise<T> {\n const shouldRetry = createRetryHandler(args);\n const requestContext = args.context ?? new Context();\n log('fetch', { url, request: args.body });\n\n let handledAuth = false;\n while (true) {\n let processingError: EdgeCallFailedError;\n let retryAfterHeaderValue: number = Number.NaN;\n try {\n const request = createRequest(args, this._authHeader);\n const response = await fetch(url, request);\n retryAfterHeaderValue = Number(response.headers.get('Retry-After'));\n if (response.ok) {\n const body = (await response.json()) as EdgeHttpResponse<T>;\n if (body.success) {\n return body.data;\n }\n\n log.warn('unsuccessful edge response', { url, body });\n if (body.errorData?.type === 'auth_challenge' && typeof body.errorData?.challenge === 'string') {\n processingError = new EdgeAuthChallengeError(body.errorData.challenge, body.errorData);\n } else {\n processingError = EdgeCallFailedError.fromUnsuccessfulResponse(response, body);\n }\n } else if (response.status === 401 && !handledAuth) {\n this._authHeader = await this._handleUnauthorized(response);\n handledAuth = true;\n continue;\n } else {\n processingError = EdgeCallFailedError.fromHttpFailure(response);\n }\n } catch (error: any) {\n processingError = EdgeCallFailedError.fromProcessingFailureCause(error);\n }\n\n if (processingError.isRetryable && (await shouldRetry(requestContext, retryAfterHeaderValue))) {\n log('retrying edge request', { url, processingError });\n } else {\n throw processingError;\n }\n }\n }\n\n private async _handleUnauthorized(response: Response): Promise<string> {\n if (!this._edgeIdentity) {\n log.warn('unauthorized response received before identity was set');\n throw EdgeCallFailedError.fromHttpFailure(response);\n }\n\n const challenge = await handleAuthChallenge(response, this._edgeIdentity);\n return encodeAuthHeader(challenge);\n }\n}\n\nconst createRequest = ({ method, body }: EdgeHttpRequestArgs, authHeader: string | undefined): RequestInit => {\n return {\n method,\n body: body && JSON.stringify(body),\n headers: authHeader ? { Authorization: authHeader } : undefined,\n };\n};\n\n/**\n * @deprecated\n */\nconst createRetryHandler = ({ retry }: EdgeHttpRequestArgs) => {\n if (!retry || retry.count < 1) {\n return async () => false;\n }\n\n let retries = 0;\n const maxRetries = retry.count ?? DEFAULT_MAX_RETRIES_COUNT;\n const baseTimeout = retry.timeout ?? DEFAULT_RETRY_TIMEOUT;\n const jitter = retry.jitter ?? DEFAULT_RETRY_JITTER;\n return async (ctx: Context, retryAfter: number) => {\n if (++retries > maxRetries || ctx.disposed) {\n return false;\n }\n\n if (retryAfter) {\n await sleep(retryAfter);\n } else {\n const timeout = baseTimeout + Math.random() * jitter;\n await sleep(timeout);\n }\n\n return true;\n };\n};\n", "//\n// Copyright 2025 DXOS.org\n//\n\nimport { type HttpClient } from '@effect/platform';\nimport { type HttpClientError } from '@effect/platform/HttpClientError';\nimport { type HttpClientResponse } from '@effect/platform/HttpClientResponse';\nimport { Context, Duration, Effect, Layer, Schedule } from 'effect';\n\nimport { log } from '@dxos/log';\n\n// TODO(burdon): Factor out.\n\nexport type RetryOptions = {\n timeout: Duration.Duration;\n retryTimes: number;\n retryBaseDelay: Duration.Duration;\n};\n\n// Layer pattern.\nexport class HttpConfig extends Context.Tag('HttpConfig')<HttpConfig, RetryOptions>() {\n static default = Layer.succeed(HttpConfig, {\n timeout: Duration.millis(1_000),\n retryTimes: 3,\n retryBaseDelay: Duration.millis(1_000),\n });\n}\n\n// HOC pattern.\nexport const withRetry = (\n effect: Effect.Effect<HttpClientResponse, HttpClientError, HttpClient.HttpClient>,\n {\n timeout = Duration.millis(1_000),\n retryBaseDelay = Duration.millis(1_000),\n retryTimes = 3,\n }: Partial<RetryOptions> = {},\n) => {\n return effect.pipe(\n Effect.flatMap((res) =>\n // Treat 500 errors as retryable?\n res.status === 500 ? Effect.fail(new Error(res.status.toString())) : res.json,\n ),\n Effect.timeout(timeout),\n Effect.retry({\n schedule: Schedule.exponential(retryBaseDelay).pipe(Schedule.jittered),\n times: retryTimes,\n }),\n );\n};\n\nexport const withRetryConfig = (effect: Effect.Effect<HttpClientResponse, HttpClientError, HttpClient.HttpClient>) =>\n Effect.gen(function* () {\n const config = yield* HttpConfig;\n return yield* withRetry(effect, config);\n });\n\nexport const withLogging = <A extends HttpClientResponse, E, R>(effect: Effect.Effect<A, E, R>) =>\n effect.pipe(Effect.tap((res) => log.info('response', { status: res.status })));\n\n/**\n *\n */\n// TODO(burdon): Document.\nexport const encodeAuthHeader = (challenge: Uint8Array) => {\n const encodedChallenge = Buffer.from(challenge).toString('base64');\n return `VerifiablePresentation pb;base64,${encodedChallenge}`;\n};\n"],
5
+ "mappings": "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",
6
+ "names": ["Event", "PersistentLifecycle", "Trigger", "TriggerState", "scheduleMicroTask", "Resource", "log", "logInfo", "EdgeStatus", "invariant", "schema", "handleAuthChallenge", "failedResponse", "identity", "status", "headerValue", "headers", "get", "startsWith", "challenge", "slice", "length", "presentation", "presentCredentials", "Buffer", "from", "getCodecForType", "encode", "WebSocket", "scheduleTask", "scheduleTaskInterval", "Context", "Resource", "invariant", "log", "logInfo", "EdgeWebsocketProtocol", "buf", "MessageSchema", "SIGNAL_KEEPALIVE_INTERVAL", "SIGNAL_KEEPALIVE_TIMEOUT", "EdgeWsConnection", "Resource", "_inactivityTimeoutCtx", "_ws", "_wsMuxer", "_lastReceivedMessageTimestamp", "_identity", "_connectionInfo", "_callbacks", "Date", "now", "info", "open", "isOpen", "identity", "identityKey", "device", "peerKey", "send", "message", "invariant", "log", "payload", "protocol", "getPayloadType", "includes", "EdgeWebsocketProtocol", "V0", "binary", "buf", "toBinary", "MessageSchema", "length", "CLOUDFLARE_MESSAGE_MAX_BYTES", "error", "byteLength", "serviceId", "catch", "e", "_open", "baseProtocols", "Object", "values", "WebSocket", "url", "toString", "protocolHeader", "muxer", "WebSocketMuxer", "onopen", "onConnected", "_scheduleHeartbeats", "verbose", "currentIdentity", "onclose", "event", "warn", "code", "reason", "onRestartRequired", "destroy", "onerror", "onmessage", "type", "data", "_rescheduleHeartbeatTimeout", "bytes", "toUint8Array", "fromBinary", "receiveData", "from", "source", "onMessage", "_close", "dispose", "close", "undefined", "err", "Error", "scheduleTaskInterval", "_ctx", "Context", "scheduleTask", "lastReceivedMessageTimestamp", "EdgeConnectionClosedError", "Error", "EdgeIdentityChangedError", "getEdgeUrlWithProtocol", "baseUrl", "protocol", "isSecure", "startsWith", "url", "URL", "toString", "DEFAULT_TIMEOUT", "EdgeClient", "Resource", "statusChanged", "_persistentLifecycle", "_messageListeners", "_reconnectListeners", "_baseWsUrl", "_baseHttpUrl", "_currentConnection", "_ready", "_identity", "_config", "Event", "PersistentLifecycle", "start", "_connect", "stop", "state", "_disconnect", "Set", "undefined", "Trigger", "_isActive", "connection", "getEdgeUrlWithProtocol", "socketEndpoint", "info", "open", "isOpen", "status", "identity", "identityKey", "device", "peerKey", "Boolean", "TriggerState", "RESOLVED", "EdgeStatus", "CONNECTED", "NOT_CONNECTED", "setIdentity", "log", "oldIdentity", "_closeCurrentConnection", "EdgeIdentityChangedError", "scheduleRestart", "onMessage", "listener", "add", "delete", "onReconnected", "scheduleMicroTask", "_ctx", "has", "error", "catch", "_open", "err", "warn", "_close", "close", "disposed", "path", "protocolHeader", "disableAuth", "_createAuthHeader", "restartRequired", "url", "URL", "toString", "EdgeWsConnection", "onConnected", "wake", "_notifyReconnected", "verbose", "onRestartRequired", "message", "_notifyMessageReceived", "from", "source", "type", "payload", "typeUrl", "Promise", "race", "wait", "timeout", "emit", "EdgeConnectionClosedError", "throw", "reset", "protocol", "getPayloadType", "send", "httpUrl", "response", "fetch", "method", "encodePresentationWsAuthHeader", "handleAuthChallenge", "statusText", "encodedPresentation", "encodedToken", "Buffer", "replace", "replaceAll", "createCredential", "signPresentation", "invariant", "Keyring", "PublicKey", "createDeviceEdgeIdentity", "signer", "key", "identityKey", "toHex", "peerKey", "presentCredentials", "challenge", "presentation", "credentials", "assertion", "issuer", "subject", "signerKey", "nonce", "createChainEdgeIdentity", "chain", "credentialsToSign", "length", "signingKey", "createEphemeralEdgeIdentity", "keyring", "createKey", "createTestHaloEdgeIdentity", "deviceKey", "deviceAdmission", "credential", "createStubEdgeIdentity", "random", "Error", "FetchHttpClient", "HttpClient", "Effect", "pipe", "sleep", "Context", "log", "EdgeAuthChallengeError", "EdgeCallFailedError", "createUrl", "Context", "Duration", "Effect", "Layer", "Schedule", "log", "HttpConfig", "Tag", "default", "succeed", "timeout", "millis", "retryTimes", "retryBaseDelay", "withRetry", "effect", "pipe", "flatMap", "res", "status", "fail", "Error", "toString", "json", "retry", "schedule", "exponential", "jittered", "times", "withRetryConfig", "gen", "config", "withLogging", "tap", "info", "encodeAuthHeader", "challenge", "encodedChallenge", "Buffer", "from", "DEFAULT_RETRY_TIMEOUT", "DEFAULT_RETRY_JITTER", "DEFAULT_MAX_RETRIES_COUNT", "EdgeHttpClient", "_baseUrl", "_edgeIdentity", "_authHeader", "baseUrl", "getEdgeUrlWithProtocol", "log", "url", "setIdentity", "identity", "identityKey", "peerKey", "undefined", "getStatus", "args", "_call", "URL", "method", "createAgent", "body", "getAgentStatus", "request", "ownerIdentityKey", "toHex", "getCredentialsForNotarization", "spaceId", "notarizeCredentials", "recoverIdentity", "joinSpaceByInvitation", "listFunctions", "initiateOAuthFlow", "createSpace", "queryQueue", "subspaceTag", "query", "queueId", "createUrl", "after", "before", "limit", "reverse", "objectIds", "join", "insertIntoQueue", "objects", "deleteFromQueue", "ids", "uploadFunction", "pathParts", "path", "functionId", "executeWorkflow", "graphId", "input", "_fetch", "pipe", "HttpClient", "get", "withLogging", "withRetryConfig", "Effect", "provide", "FetchHttpClient", "layer", "HttpConfig", "default", "withSpan", "runPromise", "shouldRetry", "createRetryHandler", "requestContext", "context", "Context", "handledAuth", "processingError", "retryAfterHeaderValue", "Number", "NaN", "createRequest", "response", "fetch", "headers", "ok", "json", "success", "data", "warn", "errorData", "type", "challenge", "EdgeAuthChallengeError", "EdgeCallFailedError", "fromUnsuccessfulResponse", "status", "_handleUnauthorized", "fromHttpFailure", "error", "fromProcessingFailureCause", "isRetryable", "handleAuthChallenge", "encodeAuthHeader", "authHeader", "JSON", "stringify", "Authorization", "retry", "count", "retries", "maxRetries", "baseTimeout", "timeout", "jitter", "ctx", "retryAfter", "disposed", "sleep", "Math", "random"]
7
7
  }