@dxos/client-services 0.6.13 → 0.6.14-main.69511f5

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (138) hide show
  1. package/dist/lib/browser/{chunk-CRXXOI45.mjs → chunk-PK5RMXEO.mjs} +6462 -5230
  2. package/dist/lib/browser/chunk-PK5RMXEO.mjs.map +7 -0
  3. package/dist/lib/browser/index.mjs +7 -3
  4. package/dist/lib/browser/index.mjs.map +3 -3
  5. package/dist/lib/browser/meta.json +1 -1
  6. package/dist/lib/browser/testing/index.mjs +12 -8
  7. package/dist/lib/browser/testing/index.mjs.map +3 -3
  8. package/dist/lib/node/{chunk-PZ3JJJ3K.cjs → chunk-XDE6WELX.cjs} +6287 -5057
  9. package/dist/lib/node/chunk-XDE6WELX.cjs.map +7 -0
  10. package/dist/lib/node/index.cjs +50 -46
  11. package/dist/lib/node/index.cjs.map +3 -3
  12. package/dist/lib/node/meta.json +1 -1
  13. package/dist/lib/node/testing/index.cjs +18 -13
  14. package/dist/lib/node/testing/index.cjs.map +3 -3
  15. package/dist/lib/node-esm/chunk-S5DTGWTU.mjs +8956 -0
  16. package/dist/lib/node-esm/chunk-S5DTGWTU.mjs.map +7 -0
  17. package/dist/lib/node-esm/index.mjs +420 -0
  18. package/dist/lib/node-esm/index.mjs.map +7 -0
  19. package/dist/lib/node-esm/meta.json +1 -0
  20. package/dist/lib/node-esm/testing/index.mjs +424 -0
  21. package/dist/lib/node-esm/testing/index.mjs.map +7 -0
  22. package/dist/types/src/index.d.ts +1 -0
  23. package/dist/types/src/index.d.ts.map +1 -1
  24. package/dist/types/src/packlets/agents/edge-agent-manager.d.ts +35 -0
  25. package/dist/types/src/packlets/agents/edge-agent-manager.d.ts.map +1 -0
  26. package/dist/types/src/packlets/agents/edge-agent-service.d.ts +10 -0
  27. package/dist/types/src/packlets/agents/edge-agent-service.d.ts.map +1 -0
  28. package/dist/types/src/packlets/agents/index.d.ts +3 -0
  29. package/dist/types/src/packlets/agents/index.d.ts.map +1 -0
  30. package/dist/types/src/packlets/diagnostics/diagnostics-broadcast.d.ts.map +1 -1
  31. package/dist/types/src/packlets/identity/authenticator.d.ts.map +1 -1
  32. package/dist/types/src/packlets/identity/authenticator.node.test.d.ts +2 -0
  33. package/dist/types/src/packlets/identity/authenticator.node.test.d.ts.map +1 -0
  34. package/dist/types/src/packlets/identity/contacts-service.d.ts +1 -1
  35. package/dist/types/src/packlets/identity/contacts-service.d.ts.map +1 -1
  36. package/dist/types/src/packlets/identity/identity-manager.d.ts +28 -9
  37. package/dist/types/src/packlets/identity/identity-manager.d.ts.map +1 -1
  38. package/dist/types/src/packlets/identity/identity-recovery-manager.d.ts +18 -0
  39. package/dist/types/src/packlets/identity/identity-recovery-manager.d.ts.map +1 -0
  40. package/dist/types/src/packlets/identity/identity-service.d.ts +7 -2
  41. package/dist/types/src/packlets/identity/identity-service.d.ts.map +1 -1
  42. package/dist/types/src/packlets/identity/identity.d.ts +12 -3
  43. package/dist/types/src/packlets/identity/identity.d.ts.map +1 -1
  44. package/dist/types/src/packlets/invitations/device-invitation-protocol.d.ts.map +1 -1
  45. package/dist/types/src/packlets/invitations/edge-invitation-handler.d.ts +30 -0
  46. package/dist/types/src/packlets/invitations/edge-invitation-handler.d.ts.map +1 -0
  47. package/dist/types/src/packlets/invitations/invitation-guest-extenstion.d.ts +2 -1
  48. package/dist/types/src/packlets/invitations/invitation-guest-extenstion.d.ts.map +1 -1
  49. package/dist/types/src/packlets/invitations/invitation-host-extension.d.ts +2 -1
  50. package/dist/types/src/packlets/invitations/invitation-host-extension.d.ts.map +1 -1
  51. package/dist/types/src/packlets/invitations/invitation-state.d.ts +19 -0
  52. package/dist/types/src/packlets/invitations/invitation-state.d.ts.map +1 -0
  53. package/dist/types/src/packlets/invitations/invitations-handler.d.ts +8 -8
  54. package/dist/types/src/packlets/invitations/invitations-handler.d.ts.map +1 -1
  55. package/dist/types/src/packlets/invitations/space-invitation-protocol.d.ts.map +1 -1
  56. package/dist/types/src/packlets/services/service-context.d.ts +14 -9
  57. package/dist/types/src/packlets/services/service-context.d.ts.map +1 -1
  58. package/dist/types/src/packlets/services/service-host.d.ts +2 -0
  59. package/dist/types/src/packlets/services/service-host.d.ts.map +1 -1
  60. package/dist/types/src/packlets/spaces/data-space-manager.d.ts +7 -3
  61. package/dist/types/src/packlets/spaces/data-space-manager.d.ts.map +1 -1
  62. package/dist/types/src/packlets/spaces/data-space.d.ts +5 -3
  63. package/dist/types/src/packlets/spaces/data-space.d.ts.map +1 -1
  64. package/dist/types/src/packlets/spaces/edge-feed-replicator.d.ts +3 -0
  65. package/dist/types/src/packlets/spaces/edge-feed-replicator.d.ts.map +1 -1
  66. package/dist/types/src/packlets/spaces/edge-feed-replicator.test.d.ts +2 -0
  67. package/dist/types/src/packlets/spaces/edge-feed-replicator.test.d.ts.map +1 -0
  68. package/dist/types/src/packlets/spaces/epoch-migrations.d.ts +1 -1
  69. package/dist/types/src/packlets/spaces/epoch-migrations.d.ts.map +1 -1
  70. package/dist/types/src/packlets/spaces/notarization-plugin.d.ts +35 -6
  71. package/dist/types/src/packlets/spaces/notarization-plugin.d.ts.map +1 -1
  72. package/dist/types/src/packlets/spaces/spaces-service.d.ts +1 -1
  73. package/dist/types/src/packlets/spaces/spaces-service.d.ts.map +1 -1
  74. package/dist/types/src/packlets/storage/storage.d.ts.map +1 -1
  75. package/dist/types/src/packlets/testing/test-builder.d.ts +1 -2
  76. package/dist/types/src/packlets/testing/test-builder.d.ts.map +1 -1
  77. package/dist/types/src/packlets/worker/worker-runtime.d.ts.map +1 -1
  78. package/dist/types/src/testing/setup.d.ts +3 -0
  79. package/dist/types/src/testing/setup.d.ts.map +1 -0
  80. package/dist/types/src/version.d.ts +1 -1
  81. package/dist/types/src/version.d.ts.map +1 -1
  82. package/package.json +44 -45
  83. package/src/index.ts +1 -0
  84. package/src/packlets/agents/edge-agent-manager.ts +163 -0
  85. package/src/packlets/agents/edge-agent-service.ts +42 -0
  86. package/src/packlets/agents/index.ts +6 -0
  87. package/src/packlets/devices/devices-service.test.ts +4 -5
  88. package/src/packlets/diagnostics/diagnostics-broadcast.ts +1 -0
  89. package/src/packlets/identity/{authenticator.test.ts → authenticator.node.test.ts} +2 -3
  90. package/src/packlets/identity/authenticator.ts +5 -2
  91. package/src/packlets/identity/contacts-service.ts +1 -1
  92. package/src/packlets/identity/identity-manager.test.ts +31 -16
  93. package/src/packlets/identity/identity-manager.ts +76 -32
  94. package/src/packlets/identity/identity-recovery-manager.ts +95 -0
  95. package/src/packlets/identity/identity-service.test.ts +5 -8
  96. package/src/packlets/identity/identity-service.ts +11 -5
  97. package/src/packlets/identity/identity.test.ts +130 -239
  98. package/src/packlets/identity/identity.ts +60 -17
  99. package/src/packlets/invitations/device-invitation-protocol.test.ts +7 -4
  100. package/src/packlets/invitations/device-invitation-protocol.ts +8 -2
  101. package/src/packlets/invitations/edge-invitation-handler.ts +185 -0
  102. package/src/packlets/invitations/invitation-guest-extenstion.ts +8 -4
  103. package/src/packlets/invitations/invitation-host-extension.ts +8 -7
  104. package/src/packlets/invitations/invitation-state.ts +112 -0
  105. package/src/packlets/invitations/invitations-handler.test.ts +16 -9
  106. package/src/packlets/invitations/invitations-handler.ts +57 -98
  107. package/src/packlets/invitations/space-invitation-protocol.test.ts +4 -3
  108. package/src/packlets/invitations/space-invitation-protocol.ts +5 -0
  109. package/src/packlets/logging/logging.test.ts +1 -2
  110. package/src/packlets/network/network-service.test.ts +2 -3
  111. package/src/packlets/services/service-context.test.ts +3 -1
  112. package/src/packlets/services/service-context.ts +113 -35
  113. package/src/packlets/services/service-host.test.ts +8 -12
  114. package/src/packlets/services/service-host.ts +25 -7
  115. package/src/packlets/services/service-registry.test.ts +1 -2
  116. package/src/packlets/spaces/data-space-manager.test.ts +2 -2
  117. package/src/packlets/spaces/data-space-manager.ts +44 -7
  118. package/src/packlets/spaces/data-space.ts +37 -6
  119. package/src/packlets/spaces/edge-feed-replicator.test.ts +252 -0
  120. package/src/packlets/spaces/edge-feed-replicator.ts +80 -22
  121. package/src/packlets/spaces/epoch-migrations.ts +2 -2
  122. package/src/packlets/spaces/notarization-plugin.test.ts +10 -7
  123. package/src/packlets/spaces/notarization-plugin.ts +196 -29
  124. package/src/packlets/spaces/spaces-service.test.ts +5 -9
  125. package/src/packlets/spaces/spaces-service.ts +6 -1
  126. package/src/packlets/storage/storage.ts +0 -1
  127. package/src/packlets/system/system-service.test.ts +1 -2
  128. package/src/packlets/testing/test-builder.ts +7 -4
  129. package/src/packlets/worker/worker-runtime.ts +2 -2
  130. package/src/testing/setup.ts +11 -0
  131. package/src/version.ts +1 -5
  132. package/dist/lib/browser/chunk-CRXXOI45.mjs.map +0 -7
  133. package/dist/lib/node/chunk-PZ3JJJ3K.cjs.map +0 -7
  134. package/dist/types/src/packlets/identity/authenticator.test.d.ts +0 -2
  135. package/dist/types/src/packlets/identity/authenticator.test.d.ts.map +0 -1
  136. package/dist/types/src/packlets/services/automerge-host.test.d.ts +0 -2
  137. package/dist/types/src/packlets/services/automerge-host.test.d.ts.map +0 -1
  138. package/src/packlets/services/automerge-host.test.ts +0 -60
@@ -2,14 +2,18 @@
2
2
  // Copyright 2023 DXOS.org
3
3
  //
4
4
 
5
- import { DeferredTask, Event, scheduleTask, sleep, TimeoutError, Trigger } from '@dxos/async';
6
- import { Context, rejectOnDispose } from '@dxos/context';
7
- import { type CredentialProcessor } from '@dxos/credentials';
5
+ import { DeferredTask, Event, scheduleTask, sleep, TimeoutError, Trigger, scheduleMicroTask } from '@dxos/async';
6
+ import { type Context, rejectOnDispose, Resource } from '@dxos/context';
7
+ import { type CredentialProcessor, verifyCredential } from '@dxos/credentials';
8
+ import { type EdgeHttpClient } from '@dxos/edge-client';
8
9
  import { type FeedWriter } from '@dxos/feed-store';
9
10
  import { invariant } from '@dxos/invariant';
10
11
  import { PublicKey } from '@dxos/keys';
11
- import { log } from '@dxos/log';
12
+ import { type SpaceId } from '@dxos/keys';
13
+ import { logInfo, log } from '@dxos/log';
14
+ import { EdgeCallFailedError } from '@dxos/protocols';
12
15
  import { schema } from '@dxos/protocols/proto';
16
+ import { type Runtime } from '@dxos/protocols/proto/dxos/config';
13
17
  import { type Credential } from '@dxos/protocols/proto/dxos/halo/credentials';
14
18
  import { type NotarizationService, type NotarizeRequest } from '@dxos/protocols/proto/dxos/mesh/teleport/notarization';
15
19
  import { type ExtensionContext, RpcExtension } from '@dxos/teleport';
@@ -21,8 +25,21 @@ const DEFAULT_SUCCESS_DELAY = 1_000;
21
25
 
22
26
  const DEFAULT_NOTARIZE_TIMEOUT = 10_000;
23
27
 
28
+ const DEFAULT_ACTIVE_EDGE_POLLING_INTERVAL = 3_000;
29
+
30
+ const MAX_EDGE_RETRIES = 2;
31
+
24
32
  const WRITER_NOT_SET_ERROR_CODE = 'WRITER_NOT_SET';
25
33
 
34
+ const credentialCodec = schema.getCodecForType('dxos.halo.credentials.Credential');
35
+
36
+ export type NotarizationPluginParams = {
37
+ spaceId: SpaceId;
38
+ edgeClient?: EdgeHttpClient;
39
+ edgeFeatures?: Runtime.Client.EdgeFeatures;
40
+ activeEdgePollingInterval?: number;
41
+ };
42
+
26
43
  export type NotarizeParams = {
27
44
  /**
28
45
  * For cancellation.
@@ -53,13 +70,17 @@ export type NotarizeParams = {
53
70
  * @default {@link DEFAULT_SUCCESS_DELAY}
54
71
  */
55
72
  successDelay?: number;
73
+
74
+ /**
75
+ * A random amount of time before making or retrying an edge request to help prevent large bursts of requests.
76
+ */
77
+ edgeRetryJitter?: number;
56
78
  };
57
79
 
58
80
  /**
59
81
  * See NotarizationService proto.
60
82
  */
61
- export class NotarizationPlugin implements CredentialProcessor {
62
- private readonly _ctx = new Context();
83
+ export class NotarizationPlugin extends Resource implements CredentialProcessor {
63
84
  private readonly _extensionOpened = new Event();
64
85
 
65
86
  private _writer: FeedWriter<Credential> | undefined;
@@ -67,13 +88,54 @@ export class NotarizationPlugin implements CredentialProcessor {
67
88
  private readonly _processedCredentials = new ComplexSet<PublicKey>(PublicKey.hash);
68
89
  private readonly _processCredentialsTriggers = new ComplexMap<PublicKey, Trigger>(PublicKey.hash);
69
90
 
91
+ private _activeEdgePollingIntervalHandle: any | undefined = undefined;
92
+ private readonly _activeEdgePollingInterval: number;
93
+
94
+ @logInfo
95
+ private readonly _spaceId: SpaceId;
96
+
97
+ private readonly _edgeClient: EdgeHttpClient | undefined;
98
+
99
+ constructor(params: NotarizationPluginParams) {
100
+ super();
101
+ this._spaceId = params.spaceId;
102
+ this._activeEdgePollingInterval = params.activeEdgePollingInterval ?? DEFAULT_ACTIVE_EDGE_POLLING_INTERVAL;
103
+ if (params.edgeClient && params.edgeFeatures?.feedReplicator) {
104
+ this._edgeClient = params.edgeClient;
105
+ }
106
+ }
107
+
108
+ setActiveEdgePollingEnabled(enabled: boolean) {
109
+ invariant(this.isOpen);
110
+ const client = this._edgeClient;
111
+ invariant(client);
112
+ if (enabled && !this._activeEdgePollingIntervalHandle) {
113
+ this._activeEdgePollingIntervalHandle = setInterval(() => {
114
+ if (this._writer) {
115
+ this._notarizePendingEdgeCredentials(client, this._writer);
116
+ }
117
+ }, this._activeEdgePollingInterval);
118
+ } else if (!enabled && this._activeEdgePollingIntervalHandle) {
119
+ clearInterval(this._activeEdgePollingIntervalHandle);
120
+ this._activeEdgePollingIntervalHandle = undefined;
121
+ }
122
+ }
123
+
70
124
  get hasWriter() {
71
125
  return !!this._writer;
72
126
  }
73
127
 
74
- async open() {}
128
+ protected override async _open() {
129
+ if (this._edgeClient && this._writer) {
130
+ this._notarizePendingEdgeCredentials(this._edgeClient, this._writer);
131
+ }
132
+ }
75
133
 
76
- async close() {
134
+ protected override async _close() {
135
+ if (this._activeEdgePollingIntervalHandle) {
136
+ clearInterval(this._activeEdgePollingIntervalHandle);
137
+ this._activeEdgePollingIntervalHandle = undefined;
138
+ }
77
139
  await this._ctx.dispose();
78
140
  }
79
141
 
@@ -86,6 +148,7 @@ export class NotarizationPlugin implements CredentialProcessor {
86
148
  timeout = DEFAULT_NOTARIZE_TIMEOUT,
87
149
  retryTimeout = DEFAULT_RETRY_TIMEOUT,
88
150
  successDelay = DEFAULT_SUCCESS_DELAY,
151
+ edgeRetryJitter,
89
152
  }: NotarizeParams) {
90
153
  log('notarize', { credentials });
91
154
  invariant(
@@ -103,24 +166,35 @@ export class NotarizationPlugin implements CredentialProcessor {
103
166
  });
104
167
  opCtx?.onDispose(() => ctx.dispose());
105
168
 
106
- // Timeout/
107
169
  if (timeout !== 0) {
108
- scheduleTask(
109
- ctx,
110
- () => {
111
- log.warn('Notarization timeout', {
112
- timeout,
113
- peers: Array.from(this._extensions).map((extension) => extension.remotePeerId),
114
- });
115
- void ctx.dispose();
116
- errors.throw(new TimeoutError(timeout, 'Notarization timed out'));
117
- },
118
- timeout,
119
- );
170
+ this._scheduleTimeout(ctx, errors, timeout);
120
171
  }
121
172
 
122
173
  const allNotarized = Promise.all(credentials.map((credential) => this._waitUntilProcessed(credential.id!)));
123
174
 
175
+ this._tryNotarizeCredentialsWithPeers(ctx, credentials, { retryTimeout, successDelay });
176
+
177
+ if (this._edgeClient) {
178
+ this._tryNotarizeCredentialsWithEdge(ctx, this._edgeClient, credentials, {
179
+ retryTimeout,
180
+ successDelay,
181
+ jitter: edgeRetryJitter,
182
+ });
183
+ }
184
+
185
+ try {
186
+ await Promise.race([rejectOnDispose(ctx), allNotarized, errors.wait()]);
187
+ log('done');
188
+ } finally {
189
+ await ctx.dispose();
190
+ }
191
+ }
192
+
193
+ private _tryNotarizeCredentialsWithPeers(
194
+ ctx: Context,
195
+ credentials: Credential[],
196
+ { retryTimeout, successDelay }: NotarizationTimeouts,
197
+ ) {
124
198
  const peersTried = new Set<NotarizationTeleportExtension>();
125
199
 
126
200
  // Repeatable task that tries to notarize credentials with one of the available peers.
@@ -145,6 +219,7 @@ export class NotarizationPlugin implements CredentialProcessor {
145
219
  credentials: credentials.filter((credential) => !this._processedCredentials.has(credential.id!)),
146
220
  });
147
221
  log('success');
222
+
148
223
  await sleep(successDelay); // wait before trying with a new peer
149
224
  } catch (err: any) {
150
225
  if (!ctx.disposed && !err.message.includes(WRITER_NOT_SET_ERROR_CODE)) {
@@ -156,13 +231,31 @@ export class NotarizationPlugin implements CredentialProcessor {
156
231
 
157
232
  notarizeTask.schedule();
158
233
  this._extensionOpened.on(ctx, () => notarizeTask.schedule());
234
+ }
159
235
 
160
- try {
161
- await Promise.race([rejectOnDispose(ctx), allNotarized, errors.wait()]);
162
- log('done');
163
- } finally {
164
- await ctx.dispose();
165
- }
236
+ private _tryNotarizeCredentialsWithEdge(
237
+ ctx: Context,
238
+ client: EdgeHttpClient,
239
+ credentials: Credential[],
240
+ timeouts: NotarizationTimeouts & { jitter?: number },
241
+ ) {
242
+ const encodedCredentials = credentials.map((credential) => {
243
+ const binary = credentialCodec.encode(credential);
244
+ return Buffer.from(binary).toString('base64');
245
+ });
246
+ scheduleTask(ctx, async () => {
247
+ try {
248
+ await client.notarizeCredentials(
249
+ this._spaceId,
250
+ { credentials: encodedCredentials },
251
+ { retry: { count: MAX_EDGE_RETRIES, timeout: timeouts.retryTimeout, jitter: timeouts.jitter } },
252
+ );
253
+
254
+ log('edge notarization success');
255
+ } catch (error: any) {
256
+ handleEdgeError(error);
257
+ }
258
+ });
166
259
  }
167
260
 
168
261
  /**
@@ -180,6 +273,44 @@ export class NotarizationPlugin implements CredentialProcessor {
180
273
  setWriter(writer: FeedWriter<Credential>) {
181
274
  invariant(!this._writer, 'Writer already set.');
182
275
  this._writer = writer;
276
+ if (this._edgeClient) {
277
+ this._notarizePendingEdgeCredentials(this._edgeClient, writer);
278
+ }
279
+ }
280
+
281
+ /**
282
+ * The method is used only for adding agent feeds to spaces.
283
+ * When an agent is created we can admit them into all the existing spaces. In case the operation fails
284
+ * this method will fix it on the next space open.
285
+ * Given how rarely this happens there's no need to poll the endpoint.
286
+ */
287
+ private _notarizePendingEdgeCredentials(client: EdgeHttpClient, writer: FeedWriter<Credential>) {
288
+ scheduleMicroTask(this._ctx, async () => {
289
+ try {
290
+ const response = await client.getCredentialsForNotarization(this._spaceId, {
291
+ retry: { count: MAX_EDGE_RETRIES },
292
+ });
293
+
294
+ const credentials = response.awaitingNotarization.credentials;
295
+ if (!credentials.length) {
296
+ log('edge did not return credentials for notarization');
297
+ return;
298
+ }
299
+
300
+ log('got edge credentials for notarization', { count: credentials.length });
301
+
302
+ const decodedCredentials = credentials.map((credential) => {
303
+ const binary = Buffer.from(credential, 'base64');
304
+ return credentialCodec.decode(binary);
305
+ });
306
+
307
+ await this._notarizeCredentials(writer, decodedCredentials);
308
+
309
+ log.info('notarized edge credentials', { count: decodedCredentials.length });
310
+ } catch (error: any) {
311
+ handleEdgeError(error);
312
+ }
313
+ });
183
314
  }
184
315
 
185
316
  private async _waitUntilProcessed(id: PublicKey) {
@@ -196,12 +327,20 @@ export class NotarizationPlugin implements CredentialProcessor {
196
327
  if (!this._writer) {
197
328
  throw new Error(WRITER_NOT_SET_ERROR_CODE);
198
329
  }
199
- for (const credential of request.credentials ?? []) {
330
+ await this._notarizeCredentials(this._writer, request.credentials ?? []);
331
+ }
332
+
333
+ private async _notarizeCredentials(writer: FeedWriter<Credential>, credentials: Credential[]) {
334
+ for (const credential of credentials) {
200
335
  invariant(credential.id, 'Credential must have an id');
201
336
  if (this._processedCredentials.has(credential.id)) {
202
337
  continue;
203
338
  }
204
- await this._writer.write(credential);
339
+ const verificationResult = await verifyCredential(credential);
340
+ if (verificationResult.kind === 'fail') {
341
+ throw new Error(`Credential verification failed: ${verificationResult.errors.join('\n')}.`);
342
+ }
343
+ await writer.write(credential);
205
344
  }
206
345
  }
207
346
 
@@ -220,8 +359,31 @@ export class NotarizationPlugin implements CredentialProcessor {
220
359
  });
221
360
  return extension;
222
361
  }
362
+
363
+ private _scheduleTimeout(ctx: Context, errors: Trigger, timeout: number) {
364
+ scheduleTask(
365
+ ctx,
366
+ () => {
367
+ log.warn('Notarization timeout', {
368
+ timeout,
369
+ peers: Array.from(this._extensions).map((extension) => extension.remotePeerId),
370
+ });
371
+ void ctx.dispose();
372
+ errors.throw(new TimeoutError(timeout, 'Notarization timed out'));
373
+ },
374
+ timeout,
375
+ );
376
+ }
223
377
  }
224
378
 
379
+ const handleEdgeError = (error: any) => {
380
+ if (!(error instanceof EdgeCallFailedError) || error.errorData) {
381
+ log.catch(error);
382
+ } else {
383
+ log.info('Edge notarization failure', { reason: error.reason });
384
+ }
385
+ };
386
+
225
387
  export type NotarizationTeleportExtensionParams = {
226
388
  onOpen: () => Promise<void>;
227
389
  onClose: () => Promise<void>;
@@ -261,6 +423,11 @@ export class NotarizationTeleportExtension extends RpcExtension<Services, Servic
261
423
  }
262
424
  }
263
425
 
426
+ type NotarizationTimeouts = {
427
+ retryTimeout: number;
428
+ successDelay: number;
429
+ };
430
+
264
431
  type Services = {
265
432
  NotarizationService: NotarizationService;
266
433
  };
@@ -2,21 +2,17 @@
2
2
  // Copyright 2023 DXOS.org
3
3
  //
4
4
 
5
- import chai, { expect } from 'chai';
6
- import chaiAsPromised from 'chai-as-promised';
5
+ import { afterEach, onTestFinished, beforeEach, describe, expect, test } from 'vitest';
7
6
 
8
7
  import { Trigger } from '@dxos/async';
9
8
  import { Context } from '@dxos/context';
10
9
  import { PublicKey } from '@dxos/keys';
11
10
  import { type Space, type SpacesService } from '@dxos/protocols/proto/dxos/client/services';
12
- import { afterEach, afterTest, beforeEach, describe, test } from '@dxos/test';
13
11
 
14
12
  import { SpacesServiceImpl } from './spaces-service';
15
13
  import { type ServiceContext } from '../services';
16
14
  import { createServiceContext } from '../testing';
17
15
 
18
- chai.use(chaiAsPromised);
19
-
20
16
  describe('SpacesService', () => {
21
17
  let serviceContext: ServiceContext;
22
18
  let spacesService: SpacesService;
@@ -36,7 +32,7 @@ describe('SpacesService', () => {
36
32
 
37
33
  describe('createSpace', () => {
38
34
  test('fails if no identity is available', async () => {
39
- await expect(spacesService.createSpace()).to.be.rejectedWith();
35
+ await expect(spacesService.createSpace()).rejects.toBeInstanceOf(Error);
40
36
  });
41
37
 
42
38
  test('creates a new space', async () => {
@@ -56,7 +52,7 @@ describe('SpacesService', () => {
56
52
  query.subscribe(({ spaces }) => {
57
53
  result.wake(spaces);
58
54
  });
59
- afterTest(() => query.close());
55
+ onTestFinished(() => query.close());
60
56
  expect(await result.wait()).to.be.length(0);
61
57
  });
62
58
 
@@ -73,7 +69,7 @@ describe('SpacesService', () => {
73
69
  query.subscribe(({ spaces }) => {
74
70
  result.wake(spaces);
75
71
  });
76
- afterTest(() => query.close());
72
+ onTestFinished(() => query.close());
77
73
 
78
74
  const spaces = await result.wait();
79
75
  expect(spaces).to.be.length(3);
@@ -87,7 +83,7 @@ describe('SpacesService', () => {
87
83
  query.subscribe(({ spaces }) => {
88
84
  result.wake(spaces);
89
85
  });
90
- afterTest(() => query.close());
86
+ onTestFinished(() => query.close());
91
87
  expect(await result.wait()).to.be.length(0);
92
88
 
93
89
  result.reset();
@@ -60,7 +60,7 @@ export class SpacesServiceImpl implements SpacesService {
60
60
  return this._serializeSpace(space);
61
61
  }
62
62
 
63
- async updateSpace({ spaceKey, state }: UpdateSpaceRequest) {
63
+ async updateSpace({ spaceKey, state, edgeReplication }: UpdateSpaceRequest) {
64
64
  const dataSpaceManager = await this._getDataSpaceManager();
65
65
  const space = dataSpaceManager.spaces.get(spaceKey) ?? raise(new SpaceNotFoundError(spaceKey));
66
66
 
@@ -77,6 +77,10 @@ export class SpacesServiceImpl implements SpacesService {
77
77
  throw new ApiError('Invalid space state');
78
78
  }
79
79
  }
80
+
81
+ if (edgeReplication !== undefined) {
82
+ await dataSpaceManager.setSpaceEdgeReplicationSetting(spaceKey, edgeReplication);
83
+ }
80
84
  }
81
85
 
82
86
  async updateMemberRole(request: UpdateMemberRoleRequest): Promise<void> {
@@ -308,6 +312,7 @@ export class SpacesServiceImpl implements SpacesService {
308
312
  creator: space.inner.spaceState.creator?.key,
309
313
  cache: space.cache,
310
314
  metrics: space.metrics,
315
+ edgeReplication: space.getEdgeReplicationSetting(),
311
316
  };
312
317
  }
313
318
 
@@ -14,7 +14,6 @@ import { getRootPath } from './util';
14
14
  // TODO(burdon): Factor out.
15
15
  export const createStorageObjects = (config: Runtime.Client.Storage) => {
16
16
  const { persistent = false, keyStore, dataStore } = config ?? {};
17
-
18
17
  if (persistent && dataStore === StorageDriver.RAM) {
19
18
  throw new InvalidConfigError('RAM storage cannot be used in persistent mode.');
20
19
  }
@@ -2,12 +2,11 @@
2
2
  // Copyright 2023 DXOS.org
3
3
  //
4
4
 
5
- import { expect } from 'chai';
5
+ import { beforeEach, describe, expect, test } from 'vitest';
6
6
 
7
7
  import { Event, Trigger } from '@dxos/async';
8
8
  import { Config } from '@dxos/config';
9
9
  import { type SystemService, SystemStatus, type QueryStatusResponse } from '@dxos/protocols/proto/dxos/client/services';
10
- import { beforeEach, describe, test } from '@dxos/test';
11
10
 
12
11
  import { SystemServiceImpl } from './system-service';
13
12
 
@@ -6,8 +6,7 @@ import { type Config } from '@dxos/config';
6
6
  import { Context } from '@dxos/context';
7
7
  import { createCredentialSignerWithChain, CredentialGenerator } from '@dxos/credentials';
8
8
  import { failUndefined } from '@dxos/debug';
9
- import { EchoHost } from '@dxos/echo-db';
10
- import { MetadataStore, SpaceManager, valueEncoding, MeshEchoReplicator } from '@dxos/echo-pipeline';
9
+ import { EchoHost, MetadataStore, SpaceManager, valueEncoding, MeshEchoReplicator } from '@dxos/echo-pipeline';
11
10
  import { FeedFactory, FeedStore } from '@dxos/feed-store';
12
11
  import { Keyring } from '@dxos/keyring';
13
12
  import { type LevelDB } from '@dxos/kv-store';
@@ -54,8 +53,8 @@ export const createServiceContext = async ({
54
53
  const level = createTestLevel();
55
54
  await level.open();
56
55
 
57
- return new ServiceContext(storage, level, networkManager, signalManager, undefined, {
58
- invitationConnectionDefaultParams: { controlHeartbeatInterval: 200 },
56
+ return new ServiceContext(storage, level, networkManager, signalManager, undefined, undefined, {
57
+ invitationConnectionDefaultParams: { teleport: { controlHeartbeatInterval: 200 } },
59
58
  ...runtimeParams,
60
59
  });
61
60
  };
@@ -219,6 +218,10 @@ export class TestPeer {
219
218
 
220
219
  async createIdentity() {
221
220
  this._props.signingContext ??= await createSigningContext(this.keyring);
221
+ this.networkManager.setPeerInfo({
222
+ identityKey: this._props.signingContext.identityKey.toHex(),
223
+ peerKey: this._props.signingContext.deviceKey.toHex(),
224
+ });
222
225
  }
223
226
 
224
227
  async destroy() {
@@ -14,7 +14,7 @@ import {
14
14
  WebsocketSignalManager,
15
15
  setIdentityTags,
16
16
  } from '@dxos/messaging';
17
- import { SimplePeerTransportProxyFactory } from '@dxos/network-manager';
17
+ import { RtcTransportProxyFactory } from '@dxos/network-manager';
18
18
  import { type RpcPort } from '@dxos/rpc';
19
19
  import { type MaybePromise } from '@dxos/util';
20
20
 
@@ -46,7 +46,7 @@ export class WorkerRuntime {
46
46
  private readonly _acquireLock: () => Promise<void>;
47
47
  private readonly _releaseLock: () => void;
48
48
  private readonly _onStop?: () => Promise<void>;
49
- private readonly _transportFactory = new SimplePeerTransportProxyFactory();
49
+ private readonly _transportFactory = new RtcTransportProxyFactory();
50
50
  private readonly _ready = new Trigger<Error | undefined>();
51
51
  private readonly _sessions = new Set<WorkerSession>();
52
52
  private readonly _clientServices!: ClientServicesHost;
@@ -0,0 +1,11 @@
1
+ //
2
+ // Copyright 2024 DXOS.org
3
+ //
4
+
5
+ import { runTestSignalServer } from '@dxos/signal';
6
+
7
+ const port = 12004;
8
+
9
+ export default async () => {
10
+ await runTestSignalServer({ port, killExisting: true });
11
+ };
package/src/version.ts CHANGED
@@ -1,5 +1 @@
1
- //
2
- // Copyright 2023 DXOS.org
3
- //
4
-
5
- export const DXOS_VERSION = '0.6.13'; // {x-release-please-version}
1
+ export const DXOS_VERSION = "0.6.14-main.69511f5";