@dxos/client-services 0.3.11-main.7ff9773 → 0.3.11-main.80db41b

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 (53) hide show
  1. package/dist/lib/browser/{chunk-5EFJSN6A.mjs → chunk-7JFKC33Q.mjs} +198 -91
  2. package/dist/lib/browser/chunk-7JFKC33Q.mjs.map +7 -0
  3. package/dist/lib/browser/index.mjs +1 -1
  4. package/dist/lib/browser/meta.json +1 -1
  5. package/dist/lib/browser/packlets/testing/index.mjs +4 -11
  6. package/dist/lib/browser/packlets/testing/index.mjs.map +3 -3
  7. package/dist/lib/node/{chunk-34H4H2HZ.cjs → chunk-OLPJJZBC.cjs} +341 -234
  8. package/dist/lib/node/chunk-OLPJJZBC.cjs.map +7 -0
  9. package/dist/lib/node/index.cjs +37 -37
  10. package/dist/lib/node/meta.json +1 -1
  11. package/dist/lib/node/packlets/testing/index.cjs +11 -18
  12. package/dist/lib/node/packlets/testing/index.cjs.map +3 -3
  13. package/dist/types/src/packlets/invitations/device-invitation-protocol.d.ts +2 -0
  14. package/dist/types/src/packlets/invitations/device-invitation-protocol.d.ts.map +1 -1
  15. package/dist/types/src/packlets/invitations/invitation-protocol.d.ts +28 -3
  16. package/dist/types/src/packlets/invitations/invitation-protocol.d.ts.map +1 -1
  17. package/dist/types/src/packlets/invitations/invitations-handler.d.ts.map +1 -1
  18. package/dist/types/src/packlets/invitations/space-invitation-protocol.d.ts +2 -0
  19. package/dist/types/src/packlets/invitations/space-invitation-protocol.d.ts.map +1 -1
  20. package/dist/types/src/packlets/network/network-service.d.ts.map +1 -1
  21. package/dist/types/src/packlets/services/diagnostics.d.ts +32 -16
  22. package/dist/types/src/packlets/services/diagnostics.d.ts.map +1 -1
  23. package/dist/types/src/packlets/services/platform.d.ts +1 -14
  24. package/dist/types/src/packlets/services/platform.d.ts.map +1 -1
  25. package/dist/types/src/packlets/spaces/automerge-space-state.d.ts +2 -0
  26. package/dist/types/src/packlets/spaces/automerge-space-state.d.ts.map +1 -1
  27. package/dist/types/src/packlets/spaces/data-space-manager.d.ts.map +1 -1
  28. package/dist/types/src/packlets/spaces/data-space.d.ts +2 -0
  29. package/dist/types/src/packlets/spaces/data-space.d.ts.map +1 -1
  30. package/dist/types/src/packlets/system/system-service.d.ts +3 -2
  31. package/dist/types/src/packlets/system/system-service.d.ts.map +1 -1
  32. package/dist/types/src/packlets/testing/invitation-utils.d.ts.map +1 -1
  33. package/dist/types/src/version.d.ts +1 -1
  34. package/package.json +35 -35
  35. package/src/packlets/identity/identity-manager.ts +1 -1
  36. package/src/packlets/invitations/device-invitation-protocol.test.ts +14 -0
  37. package/src/packlets/invitations/device-invitation-protocol.ts +14 -0
  38. package/src/packlets/invitations/invitation-protocol.ts +44 -6
  39. package/src/packlets/invitations/invitations-handler.ts +19 -17
  40. package/src/packlets/invitations/space-invitation-protocol.test.ts +28 -0
  41. package/src/packlets/invitations/space-invitation-protocol.ts +11 -0
  42. package/src/packlets/network/network-service.ts +5 -1
  43. package/src/packlets/services/diagnostics.ts +52 -16
  44. package/src/packlets/services/platform.ts +7 -19
  45. package/src/packlets/spaces/automerge-space-state.ts +4 -0
  46. package/src/packlets/spaces/data-space-manager.ts +5 -1
  47. package/src/packlets/spaces/data-space.ts +37 -4
  48. package/src/packlets/spaces/spaces-service.ts +3 -3
  49. package/src/packlets/system/system-service.ts +7 -1
  50. package/src/packlets/testing/invitation-utils.ts +2 -10
  51. package/src/version.ts +1 -1
  52. package/dist/lib/browser/chunk-5EFJSN6A.mjs.map +0 -7
  53. package/dist/lib/node/chunk-34H4H2HZ.cjs.map +0 -7
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@dxos/client-services",
3
- "version": "0.3.11-main.7ff9773",
3
+ "version": "0.3.11-main.80db41b",
4
4
  "description": "DXOS client services implementation",
5
5
  "homepage": "https://dxos.org",
6
6
  "bugs": "https://github.com/dxos/dxos/issues",
@@ -22,44 +22,44 @@
22
22
  ],
23
23
  "dependencies": {
24
24
  "platform": "^1.3.6",
25
- "@dxos/codec-protobuf": "0.3.11-main.7ff9773",
26
- "@dxos/async": "0.3.11-main.7ff9773",
27
- "@dxos/config": "0.3.11-main.7ff9773",
28
- "@dxos/context": "0.3.11-main.7ff9773",
29
- "@dxos/crypto": "0.3.11-main.7ff9773",
30
- "@dxos/document-model": "0.3.11-main.7ff9773",
31
- "@dxos/debug": "0.3.11-main.7ff9773",
32
- "@dxos/credentials": "0.3.11-main.7ff9773",
33
- "@dxos/echo-db": "0.3.11-main.7ff9773",
34
- "@dxos/echo-pipeline": "0.3.11-main.7ff9773",
35
- "@dxos/echo-schema": "0.3.11-main.7ff9773",
36
- "@dxos/feed-store": "0.3.11-main.7ff9773",
37
- "@dxos/invariant": "0.3.11-main.7ff9773",
38
- "@dxos/keyring": "0.3.11-main.7ff9773",
39
- "@dxos/lock-file": "0.3.11-main.7ff9773",
40
- "@dxos/keys": "0.3.11-main.7ff9773",
41
- "@dxos/client-protocol": "0.3.11-main.7ff9773",
42
- "@dxos/messaging": "0.3.11-main.7ff9773",
43
- "@dxos/log": "0.3.11-main.7ff9773",
44
- "@dxos/model-factory": "0.3.11-main.7ff9773",
45
- "@dxos/network-manager": "0.3.11-main.7ff9773",
46
- "@dxos/protocols": "0.3.11-main.7ff9773",
47
- "@dxos/random-access-storage": "0.3.11-main.7ff9773",
48
- "@dxos/node-std": "0.3.11-main.7ff9773",
49
- "@dxos/rpc": "0.3.11-main.7ff9773",
50
- "@dxos/teleport": "0.3.11-main.7ff9773",
51
- "@dxos/teleport-extension-gossip": "0.3.11-main.7ff9773",
52
- "@dxos/teleport-extension-object-sync": "0.3.11-main.7ff9773",
53
- "@dxos/text-model": "0.3.11-main.7ff9773",
54
- "@dxos/timeframe": "0.3.11-main.7ff9773",
55
- "@dxos/tracing": "0.3.11-main.7ff9773",
56
- "@dxos/util": "0.3.11-main.7ff9773",
57
- "@dxos/websocket-rpc": "0.3.11-main.7ff9773"
25
+ "@dxos/async": "0.3.11-main.80db41b",
26
+ "@dxos/client-protocol": "0.3.11-main.80db41b",
27
+ "@dxos/config": "0.3.11-main.80db41b",
28
+ "@dxos/codec-protobuf": "0.3.11-main.80db41b",
29
+ "@dxos/context": "0.3.11-main.80db41b",
30
+ "@dxos/crypto": "0.3.11-main.80db41b",
31
+ "@dxos/debug": "0.3.11-main.80db41b",
32
+ "@dxos/credentials": "0.3.11-main.80db41b",
33
+ "@dxos/echo-db": "0.3.11-main.80db41b",
34
+ "@dxos/document-model": "0.3.11-main.80db41b",
35
+ "@dxos/echo-schema": "0.3.11-main.80db41b",
36
+ "@dxos/echo-pipeline": "0.3.11-main.80db41b",
37
+ "@dxos/invariant": "0.3.11-main.80db41b",
38
+ "@dxos/feed-store": "0.3.11-main.80db41b",
39
+ "@dxos/keys": "0.3.11-main.80db41b",
40
+ "@dxos/keyring": "0.3.11-main.80db41b",
41
+ "@dxos/log": "0.3.11-main.80db41b",
42
+ "@dxos/lock-file": "0.3.11-main.80db41b",
43
+ "@dxos/messaging": "0.3.11-main.80db41b",
44
+ "@dxos/model-factory": "0.3.11-main.80db41b",
45
+ "@dxos/network-manager": "0.3.11-main.80db41b",
46
+ "@dxos/protocols": "0.3.11-main.80db41b",
47
+ "@dxos/node-std": "0.3.11-main.80db41b",
48
+ "@dxos/random-access-storage": "0.3.11-main.80db41b",
49
+ "@dxos/teleport": "0.3.11-main.80db41b",
50
+ "@dxos/teleport-extension-gossip": "0.3.11-main.80db41b",
51
+ "@dxos/rpc": "0.3.11-main.80db41b",
52
+ "@dxos/teleport-extension-object-sync": "0.3.11-main.80db41b",
53
+ "@dxos/timeframe": "0.3.11-main.80db41b",
54
+ "@dxos/text-model": "0.3.11-main.80db41b",
55
+ "@dxos/tracing": "0.3.11-main.80db41b",
56
+ "@dxos/util": "0.3.11-main.80db41b",
57
+ "@dxos/websocket-rpc": "0.3.11-main.80db41b"
58
58
  },
59
59
  "devDependencies": {
60
60
  "@types/platform": "^1.3.4",
61
61
  "@types/readable-stream": "^2.3.9",
62
- "@dxos/signal": "0.3.11-main.7ff9773"
62
+ "@dxos/signal": "0.3.11-main.80db41b"
63
63
  },
64
64
  "publishConfig": {
65
65
  "access": "public"
@@ -274,7 +274,7 @@ export class IdentityManager {
274
274
  genesisFeedKey: spaceRecord.genesisFeedKey,
275
275
  },
276
276
  swarmIdentity,
277
- onNetworkConnection: () => {},
277
+ onAuthorizedConnection: () => {},
278
278
  onAuthFailure: () => {
279
279
  log.warn('auth failure');
280
280
  },
@@ -6,6 +6,7 @@ import { expect } from 'chai';
6
6
 
7
7
  import { asyncChain } from '@dxos/async';
8
8
  import { Context } from '@dxos/context';
9
+ import { AlreadyJoinedError } from '@dxos/protocols';
9
10
  import { Invitation } from '@dxos/protocols/proto/dxos/client/services';
10
11
  import { describe, test, afterTest } from '@dxos/test';
11
12
 
@@ -36,4 +37,17 @@ describe('services/device', () => {
36
37
  await Promise.all(performInvitation({ host, guest, options: { kind: Invitation.Kind.DEVICE } }));
37
38
  expect(guest.identityManager.identity?.identityKey).to.deep.eq(identity1.identityKey);
38
39
  });
40
+
41
+ test('invitation when already joined', async () => {
42
+ const [host, guest] = await asyncChain<ServiceContext>([closeAfterTest])(createPeers(2));
43
+
44
+ const identity1 = await host.createIdentity();
45
+ expect(host.identityManager.identity).to.eq(identity1);
46
+
47
+ await Promise.all(performInvitation({ host, guest, options: { kind: Invitation.Kind.DEVICE } }));
48
+ expect(guest.identityManager.identity?.identityKey).to.deep.eq(identity1.identityKey);
49
+
50
+ const [_, result] = performInvitation({ host, guest, options: { kind: Invitation.Kind.DEVICE } });
51
+ expect((await result).error).to.be.instanceOf(AlreadyJoinedError);
52
+ });
39
53
  });
@@ -4,6 +4,7 @@
4
4
 
5
5
  import { invariant } from '@dxos/invariant';
6
6
  import { type Keyring } from '@dxos/keyring';
7
+ import { AlreadyJoinedError } from '@dxos/protocols';
7
8
  import { Invitation } from '@dxos/protocols/proto/dxos/client/services';
8
9
  import {
9
10
  type AdmissionRequest,
@@ -46,6 +47,17 @@ export class DeviceInvitationProtocol implements InvitationProtocol {
46
47
  };
47
48
  }
48
49
 
50
+ checkInvitation(invitation: Partial<Invitation>) {
51
+ try {
52
+ const identity = this._getIdentity();
53
+ if (identity) {
54
+ return new AlreadyJoinedError('Currently only one identity per client is supported.');
55
+ }
56
+ } catch {
57
+ // No identity.
58
+ }
59
+ }
60
+
49
61
  createIntroduction(): IntroductionRequest {
50
62
  return {};
51
63
  }
@@ -71,6 +83,8 @@ export class DeviceInvitationProtocol implements InvitationProtocol {
71
83
  invariant(request.device);
72
84
  const { deviceKey, controlFeedKey, dataFeedKey } = request.device;
73
85
 
86
+ // TODO(wittjosiah): When multiple identities are supported, verify identity doesn't already exist before accepting.
87
+
74
88
  await this._acceptIdentity({
75
89
  identityKey,
76
90
  deviceKey,
@@ -2,24 +2,62 @@
2
2
  // Copyright 2023 DXOS.org
3
3
  //
4
4
 
5
- import { type Invitation } from '@dxos/protocols/proto/dxos/client/services';
6
- import { type ProfileDocument } from '@dxos/protocols/proto/dxos/halo/credentials';
7
- import {
8
- type AdmissionRequest,
9
- type AdmissionResponse,
10
- type IntroductionRequest,
5
+ import type { ApiError } from '@dxos/protocols';
6
+ import type { Invitation } from '@dxos/protocols/proto/dxos/client/services';
7
+ import type { ProfileDocument } from '@dxos/protocols/proto/dxos/halo/credentials';
8
+ import type {
9
+ AdmissionRequest,
10
+ AdmissionResponse,
11
+ IntroductionRequest,
11
12
  } from '@dxos/protocols/proto/dxos/halo/invitations';
12
13
 
13
14
  export interface InvitationProtocol {
15
+ //
14
16
  // Debugging
17
+ //
18
+
19
+ /**
20
+ * Protocol-specific debug info to include in logs.
21
+ */
15
22
  toJSON(): object;
16
23
 
24
+ //
17
25
  // Host
26
+ //
27
+
28
+ /**
29
+ * Protocol-specific information to include in the invitation.
30
+ */
18
31
  getInvitationContext(): Partial<Invitation> & Pick<Invitation, 'kind'>;
32
+
33
+ /**
34
+ * Once authentication is successful, the host can admit the guest to the requested resource.
35
+ */
19
36
  admit(request: AdmissionRequest, guestProfile?: ProfileDocument): Promise<AdmissionResponse>;
20
37
 
38
+ //
21
39
  // Guest
40
+ //
41
+
42
+ /**
43
+ * Check if the invitation is valid.
44
+ *
45
+ * For example, the guest may already be a member of the space.
46
+ */
47
+ checkInvitation(invitation: Partial<Invitation>): ApiError | undefined;
48
+
49
+ /**
50
+ * Get profile information to send to the host to identify the guest.
51
+ */
22
52
  createIntroduction(): IntroductionRequest;
53
+
54
+ /**
55
+ * Get key information to send to the host in order to create an admission credential for the guest.
56
+ */
23
57
  createAdmissionRequest(): Promise<AdmissionRequest>;
58
+
59
+ /**
60
+ * Redeem the admission credential.
61
+ */
24
62
  accept(response: AdmissionResponse, request: AdmissionRequest): Promise<Partial<Invitation>>;
25
63
  }
@@ -303,9 +303,6 @@ export class InvitationsHandler {
303
303
  }
304
304
  }
305
305
  }
306
- } else {
307
- // Notify that introduction is complete even if auth is not required.
308
- setState({ state: Invitation.State.READY_FOR_AUTHENTICATION });
309
306
  }
310
307
 
311
308
  // 3. Send admission credentials to host (with local space keys).
@@ -355,20 +352,25 @@ export class InvitationsHandler {
355
352
  };
356
353
 
357
354
  scheduleTask(ctx, async () => {
358
- invariant(invitation.swarmKey);
359
- const topic = invitation.swarmKey;
360
- const swarmConnection = await this._networkManager.joinSwarm({
361
- topic,
362
- peerId: PublicKey.random(),
363
- protocolProvider: createTeleportProtocolFactory(async (teleport) => {
364
- teleport.addExtension('dxos.halo.invitations', createExtension());
365
- }),
366
- topology: new StarTopology(topic),
367
- label: 'invitation guest',
368
- });
369
- ctx.onDispose(() => swarmConnection.close());
370
-
371
- setState({ state: Invitation.State.CONNECTING });
355
+ const error = protocol.checkInvitation(invitation);
356
+ if (error) {
357
+ stream.error(error);
358
+ } else {
359
+ invariant(invitation.swarmKey);
360
+ const topic = invitation.swarmKey;
361
+ const swarmConnection = await this._networkManager.joinSwarm({
362
+ topic,
363
+ peerId: PublicKey.random(),
364
+ protocolProvider: createTeleportProtocolFactory(async (teleport) => {
365
+ teleport.addExtension('dxos.halo.invitations', createExtension());
366
+ }),
367
+ topology: new StarTopology(topic),
368
+ label: 'invitation guest',
369
+ });
370
+ ctx.onDispose(() => swarmConnection.close());
371
+
372
+ setState({ state: Invitation.State.CONNECTING });
373
+ }
372
374
  });
373
375
 
374
376
  const observable = new AuthenticatingInvitation({
@@ -7,6 +7,7 @@ import { expect } from 'chai';
7
7
  import { asyncChain, Trigger } from '@dxos/async';
8
8
  import { raise } from '@dxos/debug';
9
9
  import { testLocalDatabase } from '@dxos/echo-pipeline/testing';
10
+ import { AlreadyJoinedError } from '@dxos/protocols';
10
11
  import { Invitation } from '@dxos/protocols/proto/dxos/client/services';
11
12
  import { afterTest, describe, test } from '@dxos/test';
12
13
 
@@ -71,6 +72,33 @@ describe('services/space-invitations-protocol', () => {
71
72
  }
72
73
  });
73
74
 
75
+ test('invitation when already joined', async () => {
76
+ const [host, guest] = await asyncChain<ServiceContext>([createIdentity, closeAfterTest])(createPeers(2));
77
+
78
+ const space1 = await host.dataSpaceManager!.createSpace();
79
+ const spaceKey = space1.key;
80
+
81
+ await Promise.all(performInvitation({ host, guest, options: { kind: Invitation.Kind.SPACE, spaceKey } }));
82
+
83
+ {
84
+ const space1 = host.dataSpaceManager!.spaces.get(spaceKey)!;
85
+ const space2 = guest.dataSpaceManager!.spaces.get(spaceKey)!;
86
+ expect(space1).not.to.be.undefined;
87
+ expect(space2).not.to.be.undefined;
88
+
89
+ await host.dataSpaceManager?.waitUntilSpaceReady(space1.key);
90
+ await guest.dataSpaceManager?.waitUntilSpaceReady(space2.key);
91
+ }
92
+
93
+ const [_, guestResult] = performInvitation({
94
+ host,
95
+ guest,
96
+ options: { kind: Invitation.Kind.SPACE, spaceKey },
97
+ });
98
+
99
+ expect((await guestResult).error).to.be.instanceOf(AlreadyJoinedError);
100
+ });
101
+
74
102
  test('creates and accepts invitation with retry', async () => {
75
103
  const [host, guest] = await asyncChain<ServiceContext>([createIdentity, closeAfterTest])(createPeers(2));
76
104
 
@@ -8,6 +8,7 @@ import { invariant } from '@dxos/invariant';
8
8
  import { type Keyring } from '@dxos/keyring';
9
9
  import { type PublicKey } from '@dxos/keys';
10
10
  import { log } from '@dxos/log';
11
+ import { AlreadyJoinedError } from '@dxos/protocols';
11
12
  import { Invitation } from '@dxos/protocols/proto/dxos/client/services';
12
13
  import { type FeedMessage } from '@dxos/protocols/proto/dxos/echo/feed';
13
14
  import { type ProfileDocument } from '@dxos/protocols/proto/dxos/halo/credentials';
@@ -76,6 +77,12 @@ export class SpaceInvitationProtocol implements InvitationProtocol {
76
77
  };
77
78
  }
78
79
 
80
+ checkInvitation(invitation: Partial<Invitation>) {
81
+ if (invitation.spaceKey && this._spaceManager.spaces.has(invitation.spaceKey)) {
82
+ return new AlreadyJoinedError('Already joined space.');
83
+ }
84
+ }
85
+
79
86
  createIntroduction(): IntroductionRequest {
80
87
  return {
81
88
  profile: this._signingContext.getProfile(),
@@ -104,6 +111,10 @@ export class SpaceInvitationProtocol implements InvitationProtocol {
104
111
  invariant(assertion['@type'] === 'dxos.halo.credentials.SpaceMember', 'Invalid credential');
105
112
  invariant(credential.subject.id.equals(this._signingContext.identityKey));
106
113
 
114
+ if (this._spaceManager.spaces.has(assertion.spaceKey)) {
115
+ throw new AlreadyJoinedError('Already joined space.');
116
+ }
117
+
107
118
  // Create local space.
108
119
  await this._spaceManager.acceptSpace({
109
120
  spaceKey: assertion.spaceKey,
@@ -12,13 +12,17 @@ import {
12
12
  } from '@dxos/protocols/proto/dxos/client/services';
13
13
 
14
14
  export class NetworkServiceImpl implements NetworkService {
15
- constructor(private readonly networkManager: NetworkManager, private readonly signalManager: SignalManager) {}
15
+ constructor(
16
+ private readonly networkManager: NetworkManager,
17
+ private readonly signalManager: SignalManager,
18
+ ) {}
16
19
 
17
20
  queryStatus() {
18
21
  return new Stream<NetworkStatus>(({ next }) => {
19
22
  const update = () => {
20
23
  next({
21
24
  swarm: this.networkManager.connectionState,
25
+ connectionInfo: this.networkManager.connectionLog?.swarms,
22
26
  signaling: this.signalManager.getStatus().map(({ host, state }) => ({ server: host, state })),
23
27
  });
24
28
  };
@@ -18,13 +18,17 @@ import {
18
18
  type Metrics,
19
19
  type NetworkStatus,
20
20
  type Space as SpaceProto,
21
+ type Platform,
21
22
  SpaceMember,
23
+ type LogEntry,
22
24
  } from '@dxos/protocols/proto/dxos/client/services';
23
25
  import { type SubscribeToFeedsResponse } from '@dxos/protocols/proto/dxos/devtools/host';
24
26
  import { type SwarmInfo } from '@dxos/protocols/proto/dxos/devtools/swarm';
25
27
  import { type Epoch } from '@dxos/protocols/proto/dxos/halo/credentials';
28
+ import { type Resource, type Span } from '@dxos/protocols/proto/dxos/tracing';
29
+ import { TRACE_PROCESSOR } from '@dxos/tracing';
26
30
 
27
- import { getPlatform, type Platform } from './platform';
31
+ import { getPlatform } from './platform';
28
32
  import { type ServiceContext } from './service-context';
29
33
  import { DXOS_VERSION } from '../../version';
30
34
  import { type DataSpace } from '../spaces';
@@ -32,22 +36,36 @@ import { type DataSpace } from '../spaces';
32
36
  const DEFAULT_TIMEOUT = 1_000;
33
37
 
34
38
  export type Diagnostics = {
35
- created: string;
36
- platform: Platform;
37
- config?: ConfigProto;
38
39
  client: {
39
- version: string;
40
- storage: {
41
- version: number;
40
+ config: ConfigProto;
41
+ trace: TraceDiagnostic;
42
+ };
43
+ services: {
44
+ trace: TraceDiagnostic;
45
+ created: string;
46
+ platform: Platform;
47
+ config?: ConfigProto;
48
+ client: {
49
+ version: string;
50
+ storage: {
51
+ version: number;
52
+ };
42
53
  };
54
+ identity?: Identity;
55
+ devices?: Device[];
56
+ spaces?: SpaceStats[];
57
+ networkStatus?: NetworkStatus;
58
+ swarms?: SwarmInfo[];
59
+ feeds?: Partial<SubscribeToFeedsResponse.Feed>[];
60
+ metrics?: Metrics;
61
+ storage?: { file: string; count: number }[];
43
62
  };
44
- identity?: Identity;
45
- devices?: Device[];
46
- spaces?: SpaceStats[];
47
- networkStatus?: NetworkStatus;
48
- swarms?: SwarmInfo[];
49
- feeds?: Partial<SubscribeToFeedsResponse.Feed>[];
50
- metrics?: Metrics;
63
+ };
64
+
65
+ export type TraceDiagnostic = {
66
+ resources: Record<string, Resource>;
67
+ spans: Span[];
68
+ logs: LogEntry[];
51
69
  };
52
70
 
53
71
  // TODO(burdon): Normalize for ECHO/HALO.
@@ -74,8 +92,8 @@ export const createDiagnostics = async (
74
92
  clientServices: Partial<ClientServices>,
75
93
  serviceContext: ServiceContext,
76
94
  config: Config,
77
- ): Promise<Diagnostics> => {
78
- const diagnostics: Diagnostics = {
95
+ ): Promise<Diagnostics['services']> => {
96
+ const diagnostics: Diagnostics['services'] = {
79
97
  created: new Date().toISOString(),
80
98
  platform: getPlatform(),
81
99
  client: {
@@ -84,6 +102,7 @@ export const createDiagnostics = async (
84
102
  version: STORAGE_VERSION,
85
103
  },
86
104
  },
105
+ trace: TRACE_PROCESSOR.getDiagnostics(),
87
106
  };
88
107
 
89
108
  // Trace metrics.
@@ -95,6 +114,23 @@ export const createDiagnostics = async (
95
114
  }).catch(() => undefined);
96
115
  }
97
116
 
117
+ if (typeof navigator !== 'undefined' && navigator.storage) {
118
+ const map = new Map();
119
+ const dir = await navigator.storage.getDirectory();
120
+ for await (const filename of (dir as any)?.keys()) {
121
+ const idx = filename.indexOf('-', filename.indexOf('-') + 1);
122
+ if (idx === -1) {
123
+ continue;
124
+ }
125
+
126
+ map.set(filename.slice(0, idx), (map.get(filename.slice(0, idx)) ?? 0) + 1);
127
+ }
128
+
129
+ diagnostics.storage = Array.from(map.entries())
130
+ .sort((a, b) => b[1] - a[1])
131
+ .map(([file, count]) => ({ file, count }));
132
+ }
133
+
98
134
  const identity = serviceContext.identityManager.identity;
99
135
  if (identity) {
100
136
  // Identity.
@@ -2,20 +2,7 @@
2
2
  // Copyright 2022 DXOS.org
3
3
  //
4
4
 
5
- export type Platform = {
6
- type: 'browser' | 'shared-worker' | 'node';
7
- userAgent?: string;
8
- platform?: string;
9
- runtime?: string;
10
- uptime?: number;
11
- memory?: {
12
- rss: number;
13
- heapTotal: number;
14
- heapUsed: number;
15
- external: number;
16
- arrayBuffers: number;
17
- };
18
- };
5
+ import { Platform } from '@dxos/protocols/proto/dxos/client/services';
19
6
 
20
7
  export const getPlatform = (): Platform => {
21
8
  if ((process as any).browser) {
@@ -23,14 +10,14 @@ export const getPlatform = (): Platform => {
23
10
  // Browser.
24
11
  const { userAgent } = window.navigator;
25
12
  return {
26
- type: 'browser',
13
+ type: Platform.PLATFORM_TYPE.BROWSER,
27
14
  userAgent,
28
15
  uptime: Math.floor((Date.now() - window.performance.timeOrigin) / 1_000),
29
16
  };
30
17
  } else {
31
18
  // Shared worker.
32
19
  return {
33
- type: 'shared-worker',
20
+ type: Platform.PLATFORM_TYPE.SHARED_WORKER,
34
21
  uptime: Math.floor((Date.now() - performance.timeOrigin) / 1_000),
35
22
  };
36
23
  }
@@ -38,9 +25,10 @@ export const getPlatform = (): Platform => {
38
25
  // Node.
39
26
  const { platform, version, arch } = process;
40
27
  return {
41
- type: 'node',
42
- platform: `${platform} ${version} ${arch}`,
43
- runtime: process.version,
28
+ type: Platform.PLATFORM_TYPE.NODE,
29
+ platform,
30
+ arch,
31
+ runtime: version,
44
32
  uptime: Math.floor(process.uptime()),
45
33
  memory: process.memoryUsage(),
46
34
  };
@@ -9,6 +9,8 @@ export class AutomergeSpaceState implements CredentialProcessor {
9
9
  public rootUrl: string | undefined = undefined;
10
10
  public lastEpoch: SpecificCredential<Epoch> | undefined = undefined;
11
11
 
12
+ constructor(private readonly _onNewRoot: (rootUrl: string) => void) {}
13
+
12
14
  async processCredential(credential: Credential) {
13
15
  if (!checkCredentialType(credential, 'dxos.halo.credentials.Epoch')) {
14
16
  return;
@@ -17,6 +19,8 @@ export class AutomergeSpaceState implements CredentialProcessor {
17
19
  this.lastEpoch = credential;
18
20
  if (credential.subject.assertion.automergeRoot) {
19
21
  this.rootUrl = credential.subject.assertion.automergeRoot;
22
+
23
+ this._onNewRoot(this.rootUrl);
20
24
  }
21
25
  }
22
26
  }
@@ -143,6 +143,9 @@ export class DataSpaceManager {
143
143
  const space = await this._constructSpace(metadata);
144
144
 
145
145
  const automergeRoot = this._automergeHost.repo.create();
146
+ automergeRoot.change((doc: any) => {
147
+ doc.experimental_spaceKey = spaceKey.toHex();
148
+ });
146
149
 
147
150
  const credentials = await spaceGenesis(this._keyring, this._signingContext, space.inner, automergeRoot.url);
148
151
  await this._metadataStore.addSpace(metadata);
@@ -222,12 +225,13 @@ export class DataSpaceManager {
222
225
  credentialProvider: createAuthProvider(this._signingContext.credentialSigner),
223
226
  credentialAuthenticator: deferFunction(() => dataSpace.authVerifier.verifier),
224
227
  },
225
- onNetworkConnection: (session) => {
228
+ onAuthorizedConnection: (session) => {
226
229
  session.addExtension(
227
230
  'dxos.mesh.teleport.gossip',
228
231
  gossip.createExtension({ remotePeerId: session.remotePeerId }),
229
232
  );
230
233
  session.addExtension('dxos.mesh.teleport.notarization', dataSpace.notarizationPlugin.createExtension());
234
+ this._automergeHost.authorizeDevice(space.key, session.remotePeerId);
231
235
  session.addExtension('dxos.mesh.teleport.automerge', this._automergeHost.createExtension());
232
236
  },
233
237
  onAuthFailure: () => {