@dxos/client-services 0.3.11-main.7ff9773 → 0.3.11-main.800163e
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.
- package/dist/lib/browser/{chunk-5EFJSN6A.mjs → chunk-GUFT4GX3.mjs} +164 -82
- package/dist/lib/browser/chunk-GUFT4GX3.mjs.map +7 -0
- package/dist/lib/browser/index.mjs +1 -1
- package/dist/lib/browser/meta.json +1 -1
- package/dist/lib/browser/packlets/testing/index.mjs +4 -11
- package/dist/lib/browser/packlets/testing/index.mjs.map +3 -3
- package/dist/lib/node/{chunk-34H4H2HZ.cjs → chunk-GNBZGQOY.cjs} +300 -218
- package/dist/lib/node/chunk-GNBZGQOY.cjs.map +7 -0
- package/dist/lib/node/index.cjs +37 -37
- package/dist/lib/node/meta.json +1 -1
- package/dist/lib/node/packlets/testing/index.cjs +11 -18
- package/dist/lib/node/packlets/testing/index.cjs.map +3 -3
- package/dist/types/src/packlets/invitations/device-invitation-protocol.d.ts +2 -0
- package/dist/types/src/packlets/invitations/device-invitation-protocol.d.ts.map +1 -1
- package/dist/types/src/packlets/invitations/invitation-protocol.d.ts +28 -3
- package/dist/types/src/packlets/invitations/invitation-protocol.d.ts.map +1 -1
- package/dist/types/src/packlets/invitations/invitations-handler.d.ts.map +1 -1
- package/dist/types/src/packlets/invitations/space-invitation-protocol.d.ts +2 -0
- package/dist/types/src/packlets/invitations/space-invitation-protocol.d.ts.map +1 -1
- package/dist/types/src/packlets/network/network-service.d.ts.map +1 -1
- package/dist/types/src/packlets/services/diagnostics.d.ts +5 -2
- package/dist/types/src/packlets/services/diagnostics.d.ts.map +1 -1
- package/dist/types/src/packlets/services/platform.d.ts +1 -14
- package/dist/types/src/packlets/services/platform.d.ts.map +1 -1
- package/dist/types/src/packlets/spaces/automerge-space-state.d.ts +2 -0
- package/dist/types/src/packlets/spaces/automerge-space-state.d.ts.map +1 -1
- package/dist/types/src/packlets/spaces/data-space-manager.d.ts.map +1 -1
- package/dist/types/src/packlets/spaces/data-space.d.ts +1 -0
- package/dist/types/src/packlets/spaces/data-space.d.ts.map +1 -1
- package/dist/types/src/packlets/system/system-service.d.ts +2 -1
- package/dist/types/src/packlets/system/system-service.d.ts.map +1 -1
- package/dist/types/src/packlets/testing/invitation-utils.d.ts.map +1 -1
- package/dist/types/src/version.d.ts +1 -1
- package/package.json +35 -35
- package/src/packlets/identity/identity-manager.ts +1 -1
- package/src/packlets/invitations/device-invitation-protocol.test.ts +14 -0
- package/src/packlets/invitations/device-invitation-protocol.ts +14 -0
- package/src/packlets/invitations/invitation-protocol.ts +44 -6
- package/src/packlets/invitations/invitations-handler.ts +19 -17
- package/src/packlets/invitations/space-invitation-protocol.test.ts +28 -0
- package/src/packlets/invitations/space-invitation-protocol.ts +11 -0
- package/src/packlets/network/network-service.ts +1 -0
- package/src/packlets/services/diagnostics.ts +20 -1
- package/src/packlets/services/platform.ts +7 -19
- package/src/packlets/spaces/automerge-space-state.ts +4 -0
- package/src/packlets/spaces/data-space-manager.ts +5 -1
- package/src/packlets/spaces/data-space.ts +21 -1
- package/src/packlets/spaces/spaces-service.ts +1 -1
- package/src/packlets/system/system-service.ts +6 -0
- package/src/packlets/testing/invitation-utils.ts +2 -10
- package/src/version.ts +1 -1
- package/dist/lib/browser/chunk-5EFJSN6A.mjs.map +0 -7
- package/dist/lib/node/chunk-34H4H2HZ.cjs.map +0 -7
|
@@ -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 {
|
|
6
|
-
import {
|
|
7
|
-
import {
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
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
|
-
|
|
359
|
-
|
|
360
|
-
|
|
361
|
-
|
|
362
|
-
|
|
363
|
-
|
|
364
|
-
|
|
365
|
-
|
|
366
|
-
|
|
367
|
-
|
|
368
|
-
|
|
369
|
-
|
|
370
|
-
|
|
371
|
-
|
|
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,
|
|
@@ -19,6 +19,7 @@ export class NetworkServiceImpl implements NetworkService {
|
|
|
19
19
|
const update = () => {
|
|
20
20
|
next({
|
|
21
21
|
swarm: this.networkManager.connectionState,
|
|
22
|
+
connectionInfo: this.networkManager.connectionLog?.swarms,
|
|
22
23
|
signaling: this.signalManager.getStatus().map(({ host, state }) => ({ server: host, state })),
|
|
23
24
|
});
|
|
24
25
|
};
|
|
@@ -18,13 +18,14 @@ import {
|
|
|
18
18
|
type Metrics,
|
|
19
19
|
type NetworkStatus,
|
|
20
20
|
type Space as SpaceProto,
|
|
21
|
+
type Platform,
|
|
21
22
|
SpaceMember,
|
|
22
23
|
} from '@dxos/protocols/proto/dxos/client/services';
|
|
23
24
|
import { type SubscribeToFeedsResponse } from '@dxos/protocols/proto/dxos/devtools/host';
|
|
24
25
|
import { type SwarmInfo } from '@dxos/protocols/proto/dxos/devtools/swarm';
|
|
25
26
|
import { type Epoch } from '@dxos/protocols/proto/dxos/halo/credentials';
|
|
26
27
|
|
|
27
|
-
import { getPlatform
|
|
28
|
+
import { getPlatform } from './platform';
|
|
28
29
|
import { type ServiceContext } from './service-context';
|
|
29
30
|
import { DXOS_VERSION } from '../../version';
|
|
30
31
|
import { type DataSpace } from '../spaces';
|
|
@@ -48,6 +49,7 @@ export type Diagnostics = {
|
|
|
48
49
|
swarms?: SwarmInfo[];
|
|
49
50
|
feeds?: Partial<SubscribeToFeedsResponse.Feed>[];
|
|
50
51
|
metrics?: Metrics;
|
|
52
|
+
storage?: { file: string; count: number }[];
|
|
51
53
|
};
|
|
52
54
|
|
|
53
55
|
// TODO(burdon): Normalize for ECHO/HALO.
|
|
@@ -95,6 +97,23 @@ export const createDiagnostics = async (
|
|
|
95
97
|
}).catch(() => undefined);
|
|
96
98
|
}
|
|
97
99
|
|
|
100
|
+
if (typeof navigator !== 'undefined' && navigator.storage) {
|
|
101
|
+
const map = new Map();
|
|
102
|
+
const dir = await navigator.storage.getDirectory();
|
|
103
|
+
for await (const filename of (dir as any)?.keys()) {
|
|
104
|
+
const idx = filename.indexOf('-', filename.indexOf('-') + 1);
|
|
105
|
+
if (idx === -1) {
|
|
106
|
+
continue;
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
map.set(filename.slice(0, idx), (map.get(filename.slice(0, idx)) ?? 0) + 1);
|
|
110
|
+
}
|
|
111
|
+
|
|
112
|
+
diagnostics.storage = Array.from(map.entries())
|
|
113
|
+
.sort((a, b) => b[1] - a[1])
|
|
114
|
+
.map(([file, count]) => ({ file, count }));
|
|
115
|
+
}
|
|
116
|
+
|
|
98
117
|
const identity = serviceContext.identityManager.identity;
|
|
99
118
|
if (identity) {
|
|
100
119
|
// Identity.
|
|
@@ -2,20 +2,7 @@
|
|
|
2
2
|
// Copyright 2022 DXOS.org
|
|
3
3
|
//
|
|
4
4
|
|
|
5
|
-
|
|
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:
|
|
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:
|
|
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:
|
|
42
|
-
platform
|
|
43
|
-
|
|
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
|
-
|
|
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: () => {
|
|
@@ -15,6 +15,7 @@ import {
|
|
|
15
15
|
type AutomergeHost,
|
|
16
16
|
} from '@dxos/echo-pipeline';
|
|
17
17
|
import { type FeedStore } from '@dxos/feed-store';
|
|
18
|
+
import { failedInvariant } from '@dxos/invariant';
|
|
18
19
|
import { type Keyring } from '@dxos/keyring';
|
|
19
20
|
import { PublicKey } from '@dxos/keys';
|
|
20
21
|
import { log } from '@dxos/log';
|
|
@@ -92,7 +93,7 @@ export class DataSpace {
|
|
|
92
93
|
private readonly _automergeHost: AutomergeHost;
|
|
93
94
|
|
|
94
95
|
// TODO(dmaretskyi): Move into Space?
|
|
95
|
-
private readonly _automergeSpaceState = new AutomergeSpaceState();
|
|
96
|
+
private readonly _automergeSpaceState = new AutomergeSpaceState((rootUrl) => this._onNewAutomergeRoot(rootUrl));
|
|
96
97
|
|
|
97
98
|
private _state = SpaceState.CLOSED;
|
|
98
99
|
|
|
@@ -363,6 +364,25 @@ export class DataSpace {
|
|
|
363
364
|
}
|
|
364
365
|
}
|
|
365
366
|
|
|
367
|
+
private _onNewAutomergeRoot(rootUrl: string) {
|
|
368
|
+
log.info('loading automerge root doc for space', { space: this.key, rootUrl });
|
|
369
|
+
const handle = this._automergeHost.repo.find(rootUrl as any);
|
|
370
|
+
|
|
371
|
+
queueMicrotask(async () => {
|
|
372
|
+
try {
|
|
373
|
+
await handle.whenReady();
|
|
374
|
+
const doc = handle.docSync() ?? failedInvariant();
|
|
375
|
+
if (!doc.experimental_spaceKey) {
|
|
376
|
+
handle.change((doc: any) => {
|
|
377
|
+
doc.experimental_spaceKey = this.key.toHex();
|
|
378
|
+
});
|
|
379
|
+
}
|
|
380
|
+
} catch (err) {
|
|
381
|
+
log.warn('error loading automerge root doc', { space: this.key, rootUrl, err });
|
|
382
|
+
}
|
|
383
|
+
});
|
|
384
|
+
}
|
|
385
|
+
|
|
366
386
|
// TODO(dmaretskyi): Use profile from signing context.
|
|
367
387
|
async updateOwnProfile(profile: ProfileDocument) {
|
|
368
388
|
const credential = await this._signingContext.credentialSigner.createCredential({
|
|
@@ -41,7 +41,7 @@ export class SpacesServiceImpl implements SpacesService {
|
|
|
41
41
|
|
|
42
42
|
async createSpace(): Promise<Space> {
|
|
43
43
|
if (!this._identityManager.identity) {
|
|
44
|
-
throw new Error('This device has no HALO identity available. See https://docs.dxos.org/guide/halo');
|
|
44
|
+
throw new Error('This device has no HALO identity available. See https://docs.dxos.org/guide/platform/halo');
|
|
45
45
|
}
|
|
46
46
|
|
|
47
47
|
const dataSpaceManager = await this._getDataSpaceManager();
|
|
@@ -12,10 +12,12 @@ import {
|
|
|
12
12
|
type UpdateStatusRequest,
|
|
13
13
|
type QueryStatusRequest,
|
|
14
14
|
type QueryStatusResponse,
|
|
15
|
+
type Platform,
|
|
15
16
|
} from '@dxos/protocols/proto/dxos/client/services';
|
|
16
17
|
import { jsonKeyReplacer, type MaybePromise } from '@dxos/util';
|
|
17
18
|
|
|
18
19
|
import { type Diagnostics } from '../services';
|
|
20
|
+
import { getPlatform } from '../services/platform';
|
|
19
21
|
|
|
20
22
|
export type SystemServiceOptions = {
|
|
21
23
|
config?: Config;
|
|
@@ -73,6 +75,10 @@ export class SystemServiceImpl implements SystemService {
|
|
|
73
75
|
};
|
|
74
76
|
}
|
|
75
77
|
|
|
78
|
+
async getPlatform(): Promise<Platform> {
|
|
79
|
+
return getPlatform();
|
|
80
|
+
}
|
|
81
|
+
|
|
76
82
|
async updateStatus({ status }: UpdateStatusRequest) {
|
|
77
83
|
await this._onUpdateStatus(status);
|
|
78
84
|
}
|
|
@@ -3,7 +3,7 @@
|
|
|
3
3
|
//
|
|
4
4
|
|
|
5
5
|
import { Trigger } from '@dxos/async';
|
|
6
|
-
import { type AuthenticatingInvitation, type CancellableInvitation } from '@dxos/client-protocol';
|
|
6
|
+
import { InvitationEncoder, type AuthenticatingInvitation, type CancellableInvitation } from '@dxos/client-protocol';
|
|
7
7
|
import { invariant } from '@dxos/invariant';
|
|
8
8
|
import { Invitation } from '@dxos/protocols/proto/dxos/client/services';
|
|
9
9
|
|
|
@@ -13,15 +13,7 @@ import { ServiceContext } from '../services';
|
|
|
13
13
|
* Strip secrets from invitation before giving it to the peer.
|
|
14
14
|
*/
|
|
15
15
|
export const sanitizeInvitation = (invitation: Invitation): Invitation => {
|
|
16
|
-
return
|
|
17
|
-
invitationId: invitation.invitationId,
|
|
18
|
-
type: invitation.type,
|
|
19
|
-
kind: invitation.kind,
|
|
20
|
-
authMethod: invitation.authMethod,
|
|
21
|
-
swarmKey: invitation.swarmKey,
|
|
22
|
-
state: invitation.state,
|
|
23
|
-
timeout: invitation.timeout,
|
|
24
|
-
};
|
|
16
|
+
return InvitationEncoder.decode(InvitationEncoder.encode(invitation));
|
|
25
17
|
};
|
|
26
18
|
|
|
27
19
|
export type InvitationHost = {
|
package/src/version.ts
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
export const DXOS_VERSION = "0.3.11-main.
|
|
1
|
+
export const DXOS_VERSION = "0.3.11-main.800163e";
|