@dxos/edge-client 0.8.4-main.7ace549 → 0.8.4-main.937b3ca

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.
@@ -867,6 +867,7 @@ import * as Effect2 from "effect/Effect";
867
867
  import * as Function from "effect/Function";
868
868
  import { sleep } from "@dxos/async";
869
869
  import { Context as Context3 } from "@dxos/context";
870
+ import { runAndForwardErrors } from "@dxos/effect";
870
871
  import { invariant as invariant4 } from "@dxos/invariant";
871
872
  import { log as log4 } from "@dxos/log";
872
873
  import { EdgeAuthChallengeError, EdgeCallFailedError } from "@dxos/protocols";
@@ -934,7 +935,7 @@ var EdgeHttpClient = class {
934
935
  url: this._baseUrl
935
936
  }, {
936
937
  F: __dxlog_file6,
937
- L: 102,
938
+ L: 107,
938
939
  S: this,
939
940
  C: (f, a) => f(...a)
940
941
  });
@@ -954,7 +955,8 @@ var EdgeHttpClient = class {
954
955
  async getStatus(args) {
955
956
  return this._call(new URL("/status", this.baseUrl), {
956
957
  ...args,
957
- method: "GET"
958
+ method: "GET",
959
+ auth: true
958
960
  });
959
961
  }
960
962
  //
@@ -1033,7 +1035,16 @@ var EdgeHttpClient = class {
1033
1035
  // Queues
1034
1036
  //
1035
1037
  async queryQueue(subspaceTag, spaceId, query, args) {
1036
- const { queueId } = query;
1038
+ const queueId = query.queueIds?.[0];
1039
+ invariant4(queueId, "queueId required", {
1040
+ F: __dxlog_file6,
1041
+ L: 216,
1042
+ S: this,
1043
+ A: [
1044
+ "queueId",
1045
+ "'queueId required'"
1046
+ ]
1047
+ });
1037
1048
  return this._call(createUrl(new URL(`/spaces/${subspaceTag}/${spaceId}/queue/${queueId}/query`, this.baseUrl), {
1038
1049
  after: query.after,
1039
1050
  before: query.before,
@@ -1162,21 +1173,21 @@ var EdgeHttpClient = class {
1162
1173
  // Internal
1163
1174
  //
1164
1175
  async _fetch(url, _args) {
1165
- return Function.pipe(HttpClient.get(url), withLogging, withRetryConfig, Effect2.provide(FetchHttpClient.layer), Effect2.provide(HttpConfig.default), Effect2.withSpan("EdgeHttpClient"), Effect2.runPromise);
1176
+ return Function.pipe(HttpClient.get(url), withLogging, withRetryConfig, Effect2.provide(FetchHttpClient.layer), Effect2.provide(HttpConfig.default), Effect2.withSpan("EdgeHttpClient"), runAndForwardErrors);
1166
1177
  }
1167
1178
  // TODO(burdon): Refactor with effect (see edge-http-client.test.ts).
1168
1179
  async _call(url, args) {
1169
1180
  const shouldRetry = createRetryHandler(args);
1170
1181
  const requestContext = args.context ?? Context3.default(void 0, {
1171
1182
  F: __dxlog_file6,
1172
- L: 400
1183
+ L: 408
1173
1184
  });
1174
1185
  log4("fetch", {
1175
1186
  url,
1176
1187
  request: args.body
1177
1188
  }, {
1178
1189
  F: __dxlog_file6,
1179
- L: 401,
1190
+ L: 409,
1180
1191
  S: this,
1181
1192
  C: (f, a) => f(...a)
1182
1193
  });
@@ -1198,7 +1209,7 @@ var EdgeHttpClient = class {
1198
1209
  authHeader: !!this._authHeader
1199
1210
  }, {
1200
1211
  F: __dxlog_file6,
1201
- L: 416,
1212
+ L: 424,
1202
1213
  S: this,
1203
1214
  C: (f, a) => f(...a)
1204
1215
  });
@@ -1207,7 +1218,7 @@ var EdgeHttpClient = class {
1207
1218
  const body2 = await response.clone().json();
1208
1219
  invariant4(body2, "Expected body to be present", {
1209
1220
  F: __dxlog_file6,
1210
- L: 421,
1221
+ L: 429,
1211
1222
  S: this,
1212
1223
  A: [
1213
1224
  "body",
@@ -1228,7 +1239,7 @@ var EdgeHttpClient = class {
1228
1239
  const body = response.headers.get("Content-Type") === "application/json" ? await response.clone().json() : void 0;
1229
1240
  invariant4(!body?.success, "Expected body to not be a failure response or undefined.", {
1230
1241
  F: __dxlog_file6,
1231
- L: 437,
1242
+ L: 445,
1232
1243
  S: this,
1233
1244
  A: [
1234
1245
  "!body?.success",
@@ -1242,7 +1253,7 @@ var EdgeHttpClient = class {
1242
1253
  } else {
1243
1254
  invariant4(!response.ok, "Expected response to not be ok.", {
1244
1255
  F: __dxlog_file6,
1245
- L: 444,
1256
+ L: 452,
1246
1257
  S: this,
1247
1258
  A: [
1248
1259
  "!response.ok",
@@ -1260,7 +1271,7 @@ var EdgeHttpClient = class {
1260
1271
  processingError
1261
1272
  }, {
1262
1273
  F: __dxlog_file6,
1263
- L: 452,
1274
+ L: 460,
1264
1275
  S: this,
1265
1276
  C: (f, a) => f(...a)
1266
1277
  });
@@ -1273,7 +1284,7 @@ var EdgeHttpClient = class {
1273
1284
  if (!this._edgeIdentity) {
1274
1285
  log4.warn("unauthorized response received before identity was set", void 0, {
1275
1286
  F: __dxlog_file6,
1276
- L: 461,
1287
+ L: 469,
1277
1288
  S: this,
1278
1289
  C: (f, a) => f(...a)
1279
1290
  });
@@ -1297,7 +1308,7 @@ var createRequest = ({ method, body, json = true }, authHeader) => {
1297
1308
  bodySize: requestBody.length
1298
1309
  }, {
1299
1310
  F: __dxlog_file6,
1300
- L: 485,
1311
+ L: 493,
1301
1312
  S: void 0,
1302
1313
  C: (f, a) => f(...a)
1303
1314
  });
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../../src/index.ts", "../../../src/auth.ts", "../../../src/edge-client.ts", "../../../src/edge-identity.ts", "../../../src/edge-ws-connection.ts", "../../../src/errors.ts", "../../../src/utils.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 './auth';\nexport * from './defs';\nexport * from './edge-client';\nexport * from './errors';\nexport * from './protocol';\nexport * from './edge-http-client';\nexport * from './edge-identity';\nexport * from './edge-ws-muxer';\nexport * from './http-client';\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 {\n Event,\n PersistentLifecycle,\n Trigger,\n TriggerState,\n scheduleMicroTask,\n scheduleTaskInterval,\n} 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\n// Refresh status every second: rtt, rate counters.\nconst STATUS_REFRESH_INTERVAL = 1000;\n\nexport type MessageListener = (message: Message) => void;\nexport type ReconnectListener = () => void;\n\nexport type MessengerConfig = {\n socketEndpoint: string;\n timeout?: number;\n protocol?: Protocol;\n disableAuth?: boolean;\n};\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 send(message: Message): Promise<void>;\n onMessage(listener: MessageListener): () => void;\n onReconnected(listener: ReconnectListener): () => void;\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 {\n state:\n Boolean(this._currentConnection) && this._ready.state === TriggerState.RESOLVED\n ? EdgeStatus.ConnectionState.CONNECTED\n : EdgeStatus.ConnectionState.NOT_CONNECTED,\n uptime: this._currentConnection?.uptime ?? 0,\n rtt: this._currentConnection?.rtt ?? 0,\n rateBytesUp: this._currentConnection?.uploadRate ?? 0,\n rateBytesDown: this._currentConnection?.downloadRate ?? 0,\n messagesSent: this._currentConnection?.messagesSent ?? 0,\n messagesReceived: this._currentConnection?.messagesReceived ?? 0,\n };\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) {\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 /**\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) {\n if (this._ready.state !== TriggerState.RESOLVED) {\n log('waiting for websocket');\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 public onMessage(listener: MessageListener) {\n this._messageListeners.add(listener);\n return () => this._messageListeners.delete(listener);\n }\n\n public onReconnected(listener: () => 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\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 // Notify about status changes (rtt, rate counters).\n scheduleTaskInterval(\n this._ctx,\n async () => {\n if (!this._currentConnection) {\n return;\n }\n this.statusChanged.emit(this.status);\n },\n STATUS_REFRESH_INTERVAL,\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 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 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 private _openTimestamp: number | undefined;\n\n // Latency tracking.\n private _pingTimestamp: number | undefined;\n private _rtt = 0;\n\n // Rate tracking with sliding window.\n private _uploadRate = 0;\n private _downloadRate = 0;\n private readonly _rateWindow = 10000; // 10 second sliding window.\n private readonly _rateUpdateInterval = 1000; // Update rates every second.\n private _bytesSamples: Array<{ timestamp: number; sent: number; received: number }> = [];\n\n private _messagesSent = 0;\n private _messagesReceived = 0;\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 get rtt(): number {\n return this._rtt;\n }\n\n public get uptime(): number {\n return this._openTimestamp ? (Date.now() - this._openTimestamp) / 1000 : 0;\n }\n\n public get uploadRate(): number {\n return this._uploadRate;\n }\n\n public get downloadRate(): number {\n return this._downloadRate;\n }\n\n public get messagesSent(): number {\n return this._messagesSent;\n }\n\n public get messagesReceived(): number {\n return this._messagesReceived;\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 this._messagesSent++;\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._recordBytes(binary.byteLength, 0);\n this._ws.send(binary);\n } else {\n // For muxer, we need to track the size of the message being sent.\n const binary = buf.toBinary(MessageSchema, message);\n this._recordBytes(binary.byteLength, 0);\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._openTimestamp = Date.now();\n this._callbacks.onConnected();\n this._scheduleHeartbeats();\n this._scheduleRateCalculation();\n } else {\n log.verbose('connected after becoming inactive', { currentIdentity: this._identity });\n }\n };\n this._ws.onclose = (event: WebSocket.CloseEvent) => {\n if (this.isOpen) {\n log.warn('server disconnected', { code: event.code, reason: event.reason });\n this._callbacks.onRestartRequired();\n muxer.destroy();\n }\n };\n this._ws.onerror = (event: WebSocket.ErrorEvent) => {\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: WebSocket.MessageEvent) => {\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 // Calculate latency.\n if (this._pingTimestamp) {\n this._rtt = Date.now() - this._pingTimestamp;\n this._pingTimestamp = undefined;\n }\n this._rescheduleHeartbeatTimeout();\n return;\n }\n const bytes = await toUint8Array(event.data);\n this._recordBytes(0, bytes.byteLength);\n if (!this.isOpen) {\n return;\n }\n\n this._messagesReceived++;\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._pingTimestamp = Date.now();\n this._ws?.send('__ping__');\n },\n SIGNAL_KEEPALIVE_INTERVAL,\n );\n this._pingTimestamp = Date.now();\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 private _recordBytes(sent: number, received: number): void {\n const now = Date.now();\n\n // Find if we have a sample for the current second.\n const currentSecond = Math.floor(now / 1000) * 1000;\n const existingSample = this._bytesSamples.find((s) => Math.floor(s.timestamp / 1000) * 1000 === currentSecond);\n\n if (existingSample) {\n existingSample.sent += sent;\n existingSample.received += received;\n } else {\n this._bytesSamples.push({ timestamp: now, sent, received });\n }\n }\n\n private _scheduleRateCalculation(): void {\n scheduleTaskInterval(\n this._ctx,\n async () => {\n this._calculateRates();\n },\n this._rateUpdateInterval,\n );\n // Calculate initial rates.\n this._calculateRates();\n }\n\n private _calculateRates(): void {\n const now = Date.now();\n const cutoff = now - this._rateWindow;\n\n // Remove old samples.\n this._bytesSamples = this._bytesSamples.filter((s) => s.timestamp > cutoff);\n\n if (this._bytesSamples.length === 0) {\n this._uploadRate = 0;\n this._downloadRate = 0;\n return;\n }\n\n // Calculate total bytes and time span.\n let totalSent = 0;\n let totalReceived = 0;\n const oldestTimestamp = Math.min(...this._bytesSamples.map((s) => s.timestamp));\n const timeSpan = (now - oldestTimestamp) / 1000; // Convert to seconds.\n\n for (const sample of this._bytesSamples) {\n totalSent += sample.sent;\n totalReceived += sample.received;\n }\n\n // Calculate rates (bytes per second).\n this._uploadRate = timeSpan > 0 ? Math.round(totalSent / timeSpan) : 0;\n this._downloadRate = timeSpan > 0 ? Math.round(totalReceived / timeSpan) : 0;\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 * as FetchHttpClient from '@effect/platform/FetchHttpClient';\nimport * as HttpClient from '@effect/platform/HttpClient';\nimport * as Effect from 'effect/Effect';\nimport * as Function from 'effect/Function';\n\nimport { sleep } from '@dxos/async';\nimport { Context } from '@dxos/context';\nimport { invariant } from '@dxos/invariant';\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 EdgeFailure,\n type EdgeStatus,\n type ExecuteWorkflowResponseBody,\n type ExportBundleRequest,\n type ExportBundleResponse,\n type GetAgentStatusResponseBody,\n type GetNotarizationResponseBody,\n type ImportBundleRequest,\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;\nconst WARNING_BODY_SIZE = 10 * 1024 * 1024; // 10MB\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 * @default true\n */\n json?: boolean;\n\n /**\n * Force authentication.\n * This should be used for requests with large bodies to avoid sending the body twice.\n * The client will call /auth endpoint to generate the auth header.\n */\n auth?: boolean;\n};\n\nexport type EdgeHttpGetArgs = Pick<EdgeHttpRequestArgs, 'context' | 'retry' | 'auth'>;\nexport type EdgeHttpPostArgs = Pick<EdgeHttpRequestArgs, 'context' | 'retry' | 'body' | 'auth'>;\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 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 formData = new FormData();\n formData.append('name', body.name ?? '');\n formData.append('version', body.version);\n formData.append('ownerPublicKey', body.ownerPublicKey);\n formData.append('entryPoint', body.entryPoint);\n body.runtime && formData.append('runtime', body.runtime);\n for (const [filename, content] of Object.entries(body.assets)) {\n formData.append(\n 'assets',\n new Blob([content as Uint8Array<ArrayBuffer>], { type: getFileMimeType(filename) }),\n filename,\n );\n }\n\n const path = ['functions', ...(pathParts.functionId ? [pathParts.functionId] : [])].join('/');\n return this._call(new URL(path, this.baseUrl), {\n ...args,\n body: formData,\n method: 'PUT',\n json: false,\n });\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 invokeFunction(\n params: {\n functionId: string;\n version?: string;\n spaceId?: SpaceId;\n cpuTimeLimit?: number;\n subrequestsLimit?: number;\n },\n input: unknown,\n args?: EdgeHttpGetArgs,\n ): Promise<any> {\n const url = new URL(`/functions/${params.functionId}`, this.baseUrl);\n if (params.version) {\n url.searchParams.set('version', params.version);\n }\n if (params.spaceId) {\n url.searchParams.set('spaceId', params.spaceId.toString());\n }\n if (params.cpuTimeLimit) {\n url.searchParams.set('cpuTimeLimit', params.cpuTimeLimit.toString());\n }\n if (params.subrequestsLimit) {\n url.searchParams.set('subrequestsLimit', params.subrequestsLimit.toString());\n }\n\n return this._call(url, {\n ...args,\n body: input,\n method: 'POST',\n });\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 // Triggers\n //\n\n public async getCronTriggers(spaceId: SpaceId) {\n return this._call(new URL(`/test/functions/${spaceId}/triggers/crons`, this.baseUrl), { method: 'GET' });\n }\n\n public async forceRunCronTrigger(spaceId: SpaceId, triggerId: ObjectId) {\n return this._call(new URL(`/test/functions/${spaceId}/triggers/crons/${triggerId}/run`, this.baseUrl), {\n method: 'POST',\n });\n }\n\n //\n // Import/Export space.\n //\n\n public async importBundle(\n spaceId: SpaceId, //\n body: ImportBundleRequest,\n args?: EdgeHttpGetArgs,\n ): Promise<void> {\n return this._call(new URL(`/spaces/${spaceId}/import`, this.baseUrl), { ...args, body, method: 'PUT' });\n }\n\n public async exportBundle(\n spaceId: SpaceId,\n body: ExportBundleRequest,\n args?: EdgeHttpGetArgs,\n ): Promise<ExportBundleResponse> {\n return this._call(new URL(`/spaces/${spaceId}/export`, this.baseUrl), {\n ...args,\n body,\n method: 'POST',\n });\n }\n\n //\n // Internal\n //\n\n private async _fetch<T>(url: URL, _args: EdgeHttpRequestArgs): Promise<T> {\n return Function.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 ?? Context.default();\n log('fetch', { url, request: args.body });\n\n let handledAuth = false;\n const tryCount = 1;\n while (true) {\n let processingError: EdgeCallFailedError | undefined = undefined;\n try {\n if (!this._authHeader && args.auth) {\n const response = await fetch(new URL(`/auth`, this.baseUrl));\n if (response.status === 401) {\n this._authHeader = await this._handleUnauthorized(response);\n }\n }\n\n const request = createRequest(args, this._authHeader);\n log('call edge', { url, tryCount, authHeader: !!this._authHeader });\n const response = await fetch(url, request);\n\n if (response.ok) {\n const body = await response.clone().json();\n invariant(body, 'Expected body to be present');\n if (!('success' in body)) {\n return body;\n }\n if (body.success) {\n return body.data;\n }\n } else if (response.status === 401 && !handledAuth) {\n this._authHeader = await this._handleUnauthorized(response);\n handledAuth = true;\n continue;\n }\n\n const body: EdgeFailure =\n response.headers.get('Content-Type') === 'application/json' ? await response.clone().json() : undefined;\n\n invariant(!body?.success, 'Expected body to not be a failure response or undefined.');\n\n if (body?.data?.type === 'auth_challenge' && typeof body?.data?.challenge === 'string') {\n processingError = new EdgeAuthChallengeError(body.data.challenge, body.data);\n } else if (body?.success === false) {\n processingError = EdgeCallFailedError.fromUnsuccessfulResponse(response, body);\n } else {\n invariant(!response.ok, 'Expected response to not be ok.');\n processingError = await EdgeCallFailedError.fromHttpFailure(response);\n }\n } catch (error: any) {\n processingError = EdgeCallFailedError.fromProcessingFailureCause(error);\n }\n\n if (processingError?.isRetryable && (await shouldRetry(requestContext, processingError.retryAfterMs))) {\n log.verbose('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 await EdgeCallFailedError.fromHttpFailure(response);\n }\n\n const challenge = await handleAuthChallenge(response, this._edgeIdentity);\n return encodeAuthHeader(challenge);\n }\n}\n\nconst createRequest = (\n { method, body, json = true }: EdgeHttpRequestArgs,\n authHeader: string | undefined,\n): RequestInit => {\n let requestBody: BodyInit | undefined;\n const headers: HeadersInit = {};\n\n if (json) {\n requestBody = body && JSON.stringify(body);\n headers['Content-Type'] = 'application/json';\n } else {\n requestBody = body;\n }\n\n if (typeof requestBody === 'string' && requestBody.length > WARNING_BODY_SIZE) {\n log.warn('Request with large body', { bodySize: requestBody.length });\n }\n\n if (authHeader) {\n headers['Authorization'] = authHeader;\n }\n\n return {\n method,\n body: requestBody,\n headers,\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\nconst getFileMimeType = (filename: string) =>\n ['.js', '.mjs'].some((codeExtension) => filename.endsWith(codeExtension))\n ? 'application/javascript+module'\n : filename.endsWith('.wasm')\n ? 'application/wasm'\n : 'application/octet-stream';\n", "//\n// Copyright 2025 DXOS.org\n//\n\nimport type * as HttpClient from '@effect/platform/HttpClient';\nimport type * as HttpClientError from '@effect/platform/HttpClientError';\nimport type * as HttpClientResponse from '@effect/platform/HttpClientResponse';\nimport * as Context from 'effect/Context';\nimport * as Duration from 'effect/Duration';\nimport * as Effect from 'effect/Effect';\nimport * as Layer from 'effect/Layer';\nimport * as Schedule from 'effect/Schedule';\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.HttpClientResponse, HttpClientError.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 = (\n effect: Effect.Effect<HttpClientResponse.HttpClientResponse, HttpClientError.HttpClientError, HttpClient.HttpClient>,\n) =>\n Effect.gen(function* () {\n const config = yield* HttpConfig;\n return yield* withRetry(effect, config);\n });\n\nexport const withLogging = <A extends HttpClientResponse.HttpClientResponse, E, R>(effect: Effect.Effect<A, E, R>) =>\n effect.pipe(\n Effect.tap((res) => {\n log.info('response', { status: res.status });\n }),\n );\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": ["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", "Event", "PersistentLifecycle", "Trigger", "TriggerState", "scheduleMicroTask", "scheduleTaskInterval", "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", "Date", "now", "_openTimestamp", "_pingTimestamp", "_rtt", "_uploadRate", "_downloadRate", "_rateWindow", "_rateUpdateInterval", "_bytesSamples", "_messagesSent", "_messagesReceived", "_identity", "_connectionInfo", "_callbacks", "info", "open", "isOpen", "identity", "identityKey", "device", "peerKey", "rtt", "uptime", "uploadRate", "downloadRate", "messagesSent", "messagesReceived", "send", "message", "invariant", "log", "payload", "protocol", "getPayloadType", "includes", "EdgeWebsocketProtocol", "V0", "binary", "buf", "toBinary", "MessageSchema", "length", "CLOUDFLARE_MESSAGE_MAX_BYTES", "error", "byteLength", "serviceId", "_recordBytes", "catch", "e", "_open", "baseProtocols", "Object", "values", "WebSocket", "url", "toString", "protocolHeader", "muxer", "WebSocketMuxer", "onopen", "onConnected", "_scheduleHeartbeats", "_scheduleRateCalculation", "verbose", "currentIdentity", "onclose", "event", "warn", "code", "reason", "onRestartRequired", "destroy", "onerror", "onmessage", "type", "data", "undefined", "_rescheduleHeartbeatTimeout", "bytes", "toUint8Array", "fromBinary", "receiveData", "from", "source", "onMessage", "_close", "dispose", "close", "err", "Error", "scheduleTaskInterval", "_ctx", "Context", "scheduleTask", "lastReceivedMessageTimestamp", "sent", "received", "currentSecond", "Math", "floor", "existingSample", "find", "s", "timestamp", "push", "_calculateRates", "cutoff", "filter", "totalSent", "totalReceived", "oldestTimestamp", "min", "map", "timeSpan", "sample", "round", "EdgeConnectionClosedError", "Error", "EdgeIdentityChangedError", "getEdgeUrlWithProtocol", "baseUrl", "protocol", "isSecure", "startsWith", "url", "URL", "toString", "DEFAULT_TIMEOUT", "STATUS_REFRESH_INTERVAL", "EdgeClient", "Resource", "statusChanged", "Event", "_persistentLifecycle", "PersistentLifecycle", "start", "_connect", "stop", "state", "_disconnect", "_messageListeners", "Set", "_reconnectListeners", "_baseWsUrl", "_baseHttpUrl", "_currentConnection", "undefined", "_ready", "Trigger", "_identity", "_config", "getEdgeUrlWithProtocol", "socketEndpoint", "info", "open", "isOpen", "status", "identity", "identityKey", "device", "peerKey", "Boolean", "TriggerState", "RESOLVED", "EdgeStatus", "ConnectionState", "CONNECTED", "NOT_CONNECTED", "uptime", "rtt", "rateBytesUp", "uploadRate", "rateBytesDown", "downloadRate", "messagesSent", "messagesReceived", "setIdentity", "log", "oldIdentity", "_closeCurrentConnection", "EdgeIdentityChangedError", "scheduleRestart", "send", "message", "wait", "timeout", "EdgeConnectionClosedError", "source", "onMessage", "listener", "add", "delete", "onReconnected", "scheduleMicroTask", "_ctx", "has", "error", "catch", "_open", "err", "warn", "scheduleTaskInterval", "emit", "_close", "close", "disposed", "path", "protocolHeader", "disableAuth", "_createAuthHeader", "restartRequired", "url", "URL", "toString", "connection", "EdgeWsConnection", "onConnected", "_isActive", "wake", "_notifyReconnected", "verbose", "onRestartRequired", "_notifyMessageReceived", "from", "type", "payload", "typeUrl", "Promise", "race", "throw", "reset", "protocol", "getPayloadType", "httpUrl", "response", "fetch", "method", "encodePresentationWsAuthHeader", "handleAuthChallenge", "statusText", "encodedPresentation", "encodedToken", "Buffer", "replace", "replaceAll", "FetchHttpClient", "HttpClient", "Effect", "Function", "sleep", "Context", "invariant", "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", "WARNING_BODY_SIZE", "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", "initiateOAuthFlow", "createSpace", "queryQueue", "subspaceTag", "query", "queueId", "createUrl", "after", "before", "limit", "reverse", "objectIds", "join", "insertIntoQueue", "objects", "deleteFromQueue", "ids", "uploadFunction", "pathParts", "formData", "FormData", "append", "name", "version", "ownerPublicKey", "entryPoint", "runtime", "filename", "content", "Object", "entries", "assets", "Blob", "type", "getFileMimeType", "path", "functionId", "json", "listFunctions", "invokeFunction", "params", "input", "searchParams", "set", "toString", "cpuTimeLimit", "subrequestsLimit", "executeWorkflow", "graphId", "getCronTriggers", "forceRunCronTrigger", "triggerId", "importBundle", "exportBundle", "_fetch", "_args", "pipe", "get", "withLogging", "withRetryConfig", "provide", "layer", "HttpConfig", "default", "withSpan", "runPromise", "shouldRetry", "createRetryHandler", "requestContext", "context", "Context", "handledAuth", "tryCount", "processingError", "auth", "response", "fetch", "status", "_handleUnauthorized", "createRequest", "authHeader", "ok", "clone", "invariant", "success", "data", "headers", "challenge", "EdgeAuthChallengeError", "EdgeCallFailedError", "fromUnsuccessfulResponse", "fromHttpFailure", "error", "fromProcessingFailureCause", "isRetryable", "retryAfterMs", "verbose", "warn", "handleAuthChallenge", "encodeAuthHeader", "requestBody", "JSON", "stringify", "length", "bodySize", "retry", "count", "retries", "maxRetries", "baseTimeout", "timeout", "jitter", "ctx", "retryAfter", "disposed", "sleep", "Math", "random", "some", "codeExtension", "endsWith"]
4
+ "sourcesContent": ["//\n// Copyright 2024 DXOS.org\n//\n\nexport * from '@dxos/protocols/buf/dxos/edge/messenger_pb';\n\nexport * from './auth';\nexport * from './defs';\nexport * from './edge-client';\nexport * from './errors';\nexport * from './protocol';\nexport * from './edge-http-client';\nexport * from './edge-identity';\nexport * from './edge-ws-muxer';\nexport * from './http-client';\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 {\n Event,\n PersistentLifecycle,\n Trigger,\n TriggerState,\n scheduleMicroTask,\n scheduleTaskInterval,\n} 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\n// Refresh status every second: rtt, rate counters.\nconst STATUS_REFRESH_INTERVAL = 1000;\n\nexport type MessageListener = (message: Message) => void;\nexport type ReconnectListener = () => void;\n\nexport type MessengerConfig = {\n socketEndpoint: string;\n timeout?: number;\n protocol?: Protocol;\n disableAuth?: boolean;\n};\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 send(message: Message): Promise<void>;\n onMessage(listener: MessageListener): () => void;\n onReconnected(listener: ReconnectListener): () => void;\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 {\n state:\n Boolean(this._currentConnection) && this._ready.state === TriggerState.RESOLVED\n ? EdgeStatus.ConnectionState.CONNECTED\n : EdgeStatus.ConnectionState.NOT_CONNECTED,\n uptime: this._currentConnection?.uptime ?? 0,\n rtt: this._currentConnection?.rtt ?? 0,\n rateBytesUp: this._currentConnection?.uploadRate ?? 0,\n rateBytesDown: this._currentConnection?.downloadRate ?? 0,\n messagesSent: this._currentConnection?.messagesSent ?? 0,\n messagesReceived: this._currentConnection?.messagesReceived ?? 0,\n };\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) {\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 /**\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) {\n if (this._ready.state !== TriggerState.RESOLVED) {\n log('waiting for websocket');\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 public onMessage(listener: MessageListener) {\n this._messageListeners.add(listener);\n return () => this._messageListeners.delete(listener);\n }\n\n public onReconnected(listener: () => 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\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 // Notify about status changes (rtt, rate counters).\n scheduleTaskInterval(\n this._ctx,\n async () => {\n if (!this._currentConnection) {\n return;\n }\n this.statusChanged.emit(this.status);\n },\n STATUS_REFRESH_INTERVAL,\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 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 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 private _openTimestamp: number | undefined;\n\n // Latency tracking.\n private _pingTimestamp: number | undefined;\n private _rtt = 0;\n\n // Rate tracking with sliding window.\n private _uploadRate = 0;\n private _downloadRate = 0;\n private readonly _rateWindow = 10000; // 10 second sliding window.\n private readonly _rateUpdateInterval = 1000; // Update rates every second.\n private _bytesSamples: Array<{ timestamp: number; sent: number; received: number }> = [];\n\n private _messagesSent = 0;\n private _messagesReceived = 0;\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 get rtt(): number {\n return this._rtt;\n }\n\n public get uptime(): number {\n return this._openTimestamp ? (Date.now() - this._openTimestamp) / 1000 : 0;\n }\n\n public get uploadRate(): number {\n return this._uploadRate;\n }\n\n public get downloadRate(): number {\n return this._downloadRate;\n }\n\n public get messagesSent(): number {\n return this._messagesSent;\n }\n\n public get messagesReceived(): number {\n return this._messagesReceived;\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 this._messagesSent++;\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._recordBytes(binary.byteLength, 0);\n this._ws.send(binary);\n } else {\n // For muxer, we need to track the size of the message being sent.\n const binary = buf.toBinary(MessageSchema, message);\n this._recordBytes(binary.byteLength, 0);\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._openTimestamp = Date.now();\n this._callbacks.onConnected();\n this._scheduleHeartbeats();\n this._scheduleRateCalculation();\n } else {\n log.verbose('connected after becoming inactive', { currentIdentity: this._identity });\n }\n };\n this._ws.onclose = (event: WebSocket.CloseEvent) => {\n if (this.isOpen) {\n log.warn('server disconnected', { code: event.code, reason: event.reason });\n this._callbacks.onRestartRequired();\n muxer.destroy();\n }\n };\n this._ws.onerror = (event: WebSocket.ErrorEvent) => {\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: WebSocket.MessageEvent) => {\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 // Calculate latency.\n if (this._pingTimestamp) {\n this._rtt = Date.now() - this._pingTimestamp;\n this._pingTimestamp = undefined;\n }\n this._rescheduleHeartbeatTimeout();\n return;\n }\n const bytes = await toUint8Array(event.data);\n this._recordBytes(0, bytes.byteLength);\n if (!this.isOpen) {\n return;\n }\n\n this._messagesReceived++;\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._pingTimestamp = Date.now();\n this._ws?.send('__ping__');\n },\n SIGNAL_KEEPALIVE_INTERVAL,\n );\n this._pingTimestamp = Date.now();\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 private _recordBytes(sent: number, received: number): void {\n const now = Date.now();\n\n // Find if we have a sample for the current second.\n const currentSecond = Math.floor(now / 1000) * 1000;\n const existingSample = this._bytesSamples.find((s) => Math.floor(s.timestamp / 1000) * 1000 === currentSecond);\n\n if (existingSample) {\n existingSample.sent += sent;\n existingSample.received += received;\n } else {\n this._bytesSamples.push({ timestamp: now, sent, received });\n }\n }\n\n private _scheduleRateCalculation(): void {\n scheduleTaskInterval(\n this._ctx,\n async () => {\n this._calculateRates();\n },\n this._rateUpdateInterval,\n );\n // Calculate initial rates.\n this._calculateRates();\n }\n\n private _calculateRates(): void {\n const now = Date.now();\n const cutoff = now - this._rateWindow;\n\n // Remove old samples.\n this._bytesSamples = this._bytesSamples.filter((s) => s.timestamp > cutoff);\n\n if (this._bytesSamples.length === 0) {\n this._uploadRate = 0;\n this._downloadRate = 0;\n return;\n }\n\n // Calculate total bytes and time span.\n let totalSent = 0;\n let totalReceived = 0;\n const oldestTimestamp = Math.min(...this._bytesSamples.map((s) => s.timestamp));\n const timeSpan = (now - oldestTimestamp) / 1000; // Convert to seconds.\n\n for (const sample of this._bytesSamples) {\n totalSent += sample.sent;\n totalReceived += sample.received;\n }\n\n // Calculate rates (bytes per second).\n this._uploadRate = timeSpan > 0 ? Math.round(totalSent / timeSpan) : 0;\n this._downloadRate = timeSpan > 0 ? Math.round(totalReceived / timeSpan) : 0;\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 * as FetchHttpClient from '@effect/platform/FetchHttpClient';\nimport * as HttpClient from '@effect/platform/HttpClient';\nimport * as Effect from 'effect/Effect';\nimport * as Function from 'effect/Function';\n\nimport { sleep } from '@dxos/async';\nimport { Context } from '@dxos/context';\nimport { runAndForwardErrors } from '@dxos/effect';\nimport { invariant } from '@dxos/invariant';\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 EdgeFailure,\n type EdgeStatus,\n type ExecuteWorkflowResponseBody,\n type ExportBundleRequest,\n type ExportBundleResponse,\n type GetAgentStatusResponseBody,\n type GetNotarizationResponseBody,\n type ImportBundleRequest,\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;\nconst WARNING_BODY_SIZE = 10 * 1024 * 1024; // 10MB\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 * @default true\n */\n json?: boolean;\n\n /**\n * Force authentication.\n * This should be used for requests with large bodies to avoid sending the body twice.\n * The client will call /auth endpoint to generate the auth header.\n */\n auth?: boolean;\n};\n\nexport type EdgeHttpGetArgs = Pick<EdgeHttpRequestArgs, 'context' | 'retry' | 'auth'>;\nexport type EdgeHttpPostArgs = Pick<EdgeHttpRequestArgs, 'context' | 'retry' | 'body' | 'auth'>;\n\nexport type GetCronTriggersResponse = {\n cronIds: string[];\n};\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', auth: true });\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 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.queueIds?.[0];\n invariant(queueId, 'queueId required');\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 formData = new FormData();\n formData.append('name', body.name ?? '');\n formData.append('version', body.version);\n formData.append('ownerPublicKey', body.ownerPublicKey);\n formData.append('entryPoint', body.entryPoint);\n body.runtime && formData.append('runtime', body.runtime);\n for (const [filename, content] of Object.entries(body.assets)) {\n formData.append(\n 'assets',\n new Blob([content as Uint8Array<ArrayBuffer>], { type: getFileMimeType(filename) }),\n filename,\n );\n }\n\n const path = ['functions', ...(pathParts.functionId ? [pathParts.functionId] : [])].join('/');\n return this._call(new URL(path, this.baseUrl), {\n ...args,\n body: formData,\n method: 'PUT',\n json: false,\n });\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 invokeFunction(\n params: {\n functionId: string;\n version?: string;\n spaceId?: SpaceId;\n cpuTimeLimit?: number;\n subrequestsLimit?: number;\n },\n input: unknown,\n args?: EdgeHttpGetArgs,\n ): Promise<any> {\n const url = new URL(`/functions/${params.functionId}`, this.baseUrl);\n if (params.version) {\n url.searchParams.set('version', params.version);\n }\n if (params.spaceId) {\n url.searchParams.set('spaceId', params.spaceId.toString());\n }\n if (params.cpuTimeLimit) {\n url.searchParams.set('cpuTimeLimit', params.cpuTimeLimit.toString());\n }\n if (params.subrequestsLimit) {\n url.searchParams.set('subrequestsLimit', params.subrequestsLimit.toString());\n }\n\n return this._call(url, {\n ...args,\n body: input,\n method: 'POST',\n });\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 // Triggers\n //\n\n public async getCronTriggers(spaceId: SpaceId): Promise<GetCronTriggersResponse> {\n return this._call<GetCronTriggersResponse>(new URL(`/test/functions/${spaceId}/triggers/crons`, this.baseUrl), {\n method: 'GET',\n });\n }\n\n public async forceRunCronTrigger(spaceId: SpaceId, triggerId: ObjectId) {\n return this._call(new URL(`/test/functions/${spaceId}/triggers/crons/${triggerId}/run`, this.baseUrl), {\n method: 'POST',\n });\n }\n\n //\n // Import/Export space.\n //\n\n public async importBundle(\n spaceId: SpaceId, //\n body: ImportBundleRequest,\n args?: EdgeHttpGetArgs,\n ): Promise<void> {\n return this._call(new URL(`/spaces/${spaceId}/import`, this.baseUrl), { ...args, body, method: 'PUT' });\n }\n\n public async exportBundle(\n spaceId: SpaceId,\n body: ExportBundleRequest,\n args?: EdgeHttpGetArgs,\n ): Promise<ExportBundleResponse> {\n return this._call(new URL(`/spaces/${spaceId}/export`, this.baseUrl), {\n ...args,\n body,\n method: 'POST',\n });\n }\n\n //\n // Internal\n //\n\n private async _fetch<T>(url: URL, _args: EdgeHttpRequestArgs): Promise<T> {\n return Function.pipe(\n HttpClient.get(url),\n withLogging,\n withRetryConfig,\n Effect.provide(FetchHttpClient.layer),\n Effect.provide(HttpConfig.default),\n Effect.withSpan('EdgeHttpClient'),\n runAndForwardErrors,\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 ?? Context.default();\n log('fetch', { url, request: args.body });\n\n let handledAuth = false;\n const tryCount = 1;\n while (true) {\n let processingError: EdgeCallFailedError | undefined = undefined;\n try {\n if (!this._authHeader && args.auth) {\n const response = await fetch(new URL(`/auth`, this.baseUrl));\n if (response.status === 401) {\n this._authHeader = await this._handleUnauthorized(response);\n }\n }\n\n const request = createRequest(args, this._authHeader);\n log('call edge', { url, tryCount, authHeader: !!this._authHeader });\n const response = await fetch(url, request);\n\n if (response.ok) {\n const body = await response.clone().json();\n invariant(body, 'Expected body to be present');\n if (!('success' in body)) {\n return body;\n }\n if (body.success) {\n return body.data;\n }\n } else if (response.status === 401 && !handledAuth) {\n this._authHeader = await this._handleUnauthorized(response);\n handledAuth = true;\n continue;\n }\n\n const body: EdgeFailure =\n response.headers.get('Content-Type') === 'application/json' ? await response.clone().json() : undefined;\n\n invariant(!body?.success, 'Expected body to not be a failure response or undefined.');\n\n if (body?.data?.type === 'auth_challenge' && typeof body?.data?.challenge === 'string') {\n processingError = new EdgeAuthChallengeError(body.data.challenge, body.data);\n } else if (body?.success === false) {\n processingError = EdgeCallFailedError.fromUnsuccessfulResponse(response, body);\n } else {\n invariant(!response.ok, 'Expected response to not be ok.');\n processingError = await EdgeCallFailedError.fromHttpFailure(response);\n }\n } catch (error: any) {\n processingError = EdgeCallFailedError.fromProcessingFailureCause(error);\n }\n\n if (processingError?.isRetryable && (await shouldRetry(requestContext, processingError.retryAfterMs))) {\n log.verbose('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 await EdgeCallFailedError.fromHttpFailure(response);\n }\n\n const challenge = await handleAuthChallenge(response, this._edgeIdentity);\n return encodeAuthHeader(challenge);\n }\n}\n\nconst createRequest = (\n { method, body, json = true }: EdgeHttpRequestArgs,\n authHeader: string | undefined,\n): RequestInit => {\n let requestBody: BodyInit | undefined;\n const headers: HeadersInit = {};\n\n if (json) {\n requestBody = body && JSON.stringify(body);\n headers['Content-Type'] = 'application/json';\n } else {\n requestBody = body;\n }\n\n if (typeof requestBody === 'string' && requestBody.length > WARNING_BODY_SIZE) {\n log.warn('Request with large body', { bodySize: requestBody.length });\n }\n\n if (authHeader) {\n headers['Authorization'] = authHeader;\n }\n\n return {\n method,\n body: requestBody,\n headers,\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\nconst getFileMimeType = (filename: string) =>\n ['.js', '.mjs'].some((codeExtension) => filename.endsWith(codeExtension))\n ? 'application/javascript+module'\n : filename.endsWith('.wasm')\n ? 'application/wasm'\n : 'application/octet-stream';\n", "//\n// Copyright 2025 DXOS.org\n//\n\nimport type * as HttpClient from '@effect/platform/HttpClient';\nimport type * as HttpClientError from '@effect/platform/HttpClientError';\nimport type * as HttpClientResponse from '@effect/platform/HttpClientResponse';\nimport * as Context from 'effect/Context';\nimport * as Duration from 'effect/Duration';\nimport * as Effect from 'effect/Effect';\nimport * as Layer from 'effect/Layer';\nimport * as Schedule from 'effect/Schedule';\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.HttpClientResponse, HttpClientError.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 = (\n effect: Effect.Effect<HttpClientResponse.HttpClientResponse, HttpClientError.HttpClientError, HttpClient.HttpClient>,\n) =>\n Effect.gen(function* () {\n const config = yield* HttpConfig;\n return yield* withRetry(effect, config);\n });\n\nexport const withLogging = <A extends HttpClientResponse.HttpClientResponse, E, R>(effect: Effect.Effect<A, E, R>) =>\n effect.pipe(\n Effect.tap((res) => {\n log.info('response', { status: res.status });\n }),\n );\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": ["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", "Event", "PersistentLifecycle", "Trigger", "TriggerState", "scheduleMicroTask", "scheduleTaskInterval", "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", "Date", "now", "_openTimestamp", "_pingTimestamp", "_rtt", "_uploadRate", "_downloadRate", "_rateWindow", "_rateUpdateInterval", "_bytesSamples", "_messagesSent", "_messagesReceived", "_identity", "_connectionInfo", "_callbacks", "info", "open", "isOpen", "identity", "identityKey", "device", "peerKey", "rtt", "uptime", "uploadRate", "downloadRate", "messagesSent", "messagesReceived", "send", "message", "invariant", "log", "payload", "protocol", "getPayloadType", "includes", "EdgeWebsocketProtocol", "V0", "binary", "buf", "toBinary", "MessageSchema", "length", "CLOUDFLARE_MESSAGE_MAX_BYTES", "error", "byteLength", "serviceId", "_recordBytes", "catch", "e", "_open", "baseProtocols", "Object", "values", "WebSocket", "url", "toString", "protocolHeader", "muxer", "WebSocketMuxer", "onopen", "onConnected", "_scheduleHeartbeats", "_scheduleRateCalculation", "verbose", "currentIdentity", "onclose", "event", "warn", "code", "reason", "onRestartRequired", "destroy", "onerror", "onmessage", "type", "data", "undefined", "_rescheduleHeartbeatTimeout", "bytes", "toUint8Array", "fromBinary", "receiveData", "from", "source", "onMessage", "_close", "dispose", "close", "err", "Error", "scheduleTaskInterval", "_ctx", "Context", "scheduleTask", "lastReceivedMessageTimestamp", "sent", "received", "currentSecond", "Math", "floor", "existingSample", "find", "s", "timestamp", "push", "_calculateRates", "cutoff", "filter", "totalSent", "totalReceived", "oldestTimestamp", "min", "map", "timeSpan", "sample", "round", "EdgeConnectionClosedError", "Error", "EdgeIdentityChangedError", "getEdgeUrlWithProtocol", "baseUrl", "protocol", "isSecure", "startsWith", "url", "URL", "toString", "DEFAULT_TIMEOUT", "STATUS_REFRESH_INTERVAL", "EdgeClient", "Resource", "statusChanged", "Event", "_persistentLifecycle", "PersistentLifecycle", "start", "_connect", "stop", "state", "_disconnect", "_messageListeners", "Set", "_reconnectListeners", "_baseWsUrl", "_baseHttpUrl", "_currentConnection", "undefined", "_ready", "Trigger", "_identity", "_config", "getEdgeUrlWithProtocol", "socketEndpoint", "info", "open", "isOpen", "status", "identity", "identityKey", "device", "peerKey", "Boolean", "TriggerState", "RESOLVED", "EdgeStatus", "ConnectionState", "CONNECTED", "NOT_CONNECTED", "uptime", "rtt", "rateBytesUp", "uploadRate", "rateBytesDown", "downloadRate", "messagesSent", "messagesReceived", "setIdentity", "log", "oldIdentity", "_closeCurrentConnection", "EdgeIdentityChangedError", "scheduleRestart", "send", "message", "wait", "timeout", "EdgeConnectionClosedError", "source", "onMessage", "listener", "add", "delete", "onReconnected", "scheduleMicroTask", "_ctx", "has", "error", "catch", "_open", "err", "warn", "scheduleTaskInterval", "emit", "_close", "close", "disposed", "path", "protocolHeader", "disableAuth", "_createAuthHeader", "restartRequired", "url", "URL", "toString", "connection", "EdgeWsConnection", "onConnected", "_isActive", "wake", "_notifyReconnected", "verbose", "onRestartRequired", "_notifyMessageReceived", "from", "type", "payload", "typeUrl", "Promise", "race", "throw", "reset", "protocol", "getPayloadType", "httpUrl", "response", "fetch", "method", "encodePresentationWsAuthHeader", "handleAuthChallenge", "statusText", "encodedPresentation", "encodedToken", "Buffer", "replace", "replaceAll", "FetchHttpClient", "HttpClient", "Effect", "Function", "sleep", "Context", "runAndForwardErrors", "invariant", "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", "WARNING_BODY_SIZE", "EdgeHttpClient", "_baseUrl", "_edgeIdentity", "_authHeader", "baseUrl", "getEdgeUrlWithProtocol", "log", "url", "setIdentity", "identity", "identityKey", "peerKey", "undefined", "getStatus", "args", "_call", "URL", "method", "auth", "createAgent", "body", "getAgentStatus", "request", "ownerIdentityKey", "toHex", "getCredentialsForNotarization", "spaceId", "notarizeCredentials", "recoverIdentity", "joinSpaceByInvitation", "initiateOAuthFlow", "createSpace", "queryQueue", "subspaceTag", "query", "queueId", "queueIds", "invariant", "createUrl", "after", "before", "limit", "reverse", "objectIds", "join", "insertIntoQueue", "objects", "deleteFromQueue", "ids", "uploadFunction", "pathParts", "formData", "FormData", "append", "name", "version", "ownerPublicKey", "entryPoint", "runtime", "filename", "content", "Object", "entries", "assets", "Blob", "type", "getFileMimeType", "path", "functionId", "json", "listFunctions", "invokeFunction", "params", "input", "searchParams", "set", "toString", "cpuTimeLimit", "subrequestsLimit", "executeWorkflow", "graphId", "getCronTriggers", "forceRunCronTrigger", "triggerId", "importBundle", "exportBundle", "_fetch", "_args", "pipe", "get", "withLogging", "withRetryConfig", "provide", "layer", "HttpConfig", "default", "withSpan", "runAndForwardErrors", "shouldRetry", "createRetryHandler", "requestContext", "context", "Context", "handledAuth", "tryCount", "processingError", "response", "fetch", "status", "_handleUnauthorized", "createRequest", "authHeader", "ok", "clone", "success", "data", "headers", "challenge", "EdgeAuthChallengeError", "EdgeCallFailedError", "fromUnsuccessfulResponse", "fromHttpFailure", "error", "fromProcessingFailureCause", "isRetryable", "retryAfterMs", "verbose", "warn", "handleAuthChallenge", "encodeAuthHeader", "requestBody", "JSON", "stringify", "length", "bodySize", "retry", "count", "retries", "maxRetries", "baseTimeout", "timeout", "jitter", "ctx", "retryAfter", "disposed", "sleep", "Math", "random", "some", "codeExtension", "endsWith"]
7
7
  }