@dxos/echo-pipeline 0.4.9 → 0.4.10-main.068c3d8

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 (69) hide show
  1. package/dist/lib/browser/{chunk-RTEEJ723.mjs → chunk-SYE4EK33.mjs} +30 -35
  2. package/dist/lib/browser/chunk-SYE4EK33.mjs.map +7 -0
  3. package/dist/lib/browser/index.mjs +593 -217
  4. package/dist/lib/browser/index.mjs.map +4 -4
  5. package/dist/lib/browser/meta.json +1 -1
  6. package/dist/lib/browser/testing/index.mjs +8 -2
  7. package/dist/lib/browser/testing/index.mjs.map +4 -4
  8. package/dist/lib/node/{chunk-7VZVCCNF.cjs → chunk-WCTX6RNS.cjs} +35 -40
  9. package/dist/lib/node/chunk-WCTX6RNS.cjs.map +7 -0
  10. package/dist/lib/node/index.cjs +611 -237
  11. package/dist/lib/node/index.cjs.map +4 -4
  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 +4 -4
  15. package/dist/types/src/automerge/automerge-doc-loader.d.ts +66 -0
  16. package/dist/types/src/automerge/automerge-doc-loader.d.ts.map +1 -0
  17. package/dist/types/src/automerge/automerge-doc-loader.test.d.ts +2 -0
  18. package/dist/types/src/automerge/automerge-doc-loader.test.d.ts.map +1 -0
  19. package/dist/types/src/automerge/automerge-host.d.ts +21 -18
  20. package/dist/types/src/automerge/automerge-host.d.ts.map +1 -1
  21. package/dist/types/src/automerge/automerge-repo.test.d.ts +2 -0
  22. package/dist/types/src/automerge/automerge-repo.test.d.ts.map +1 -0
  23. package/dist/types/src/automerge/index.d.ts +4 -0
  24. package/dist/types/src/automerge/index.d.ts.map +1 -1
  25. package/dist/types/src/automerge/level.test.d.ts +2 -0
  26. package/dist/types/src/automerge/level.test.d.ts.map +1 -0
  27. package/dist/types/src/automerge/leveldb-storage-adapter.d.ts +30 -0
  28. package/dist/types/src/automerge/leveldb-storage-adapter.d.ts.map +1 -0
  29. package/dist/types/src/automerge/local-host-network-adapter.d.ts +8 -1
  30. package/dist/types/src/automerge/local-host-network-adapter.d.ts.map +1 -1
  31. package/dist/types/src/automerge/migrations.d.ts +7 -0
  32. package/dist/types/src/automerge/migrations.d.ts.map +1 -0
  33. package/dist/types/src/automerge/reference.d.ts +15 -0
  34. package/dist/types/src/automerge/reference.d.ts.map +1 -0
  35. package/dist/types/src/automerge/storage-adapter.test.d.ts +2 -0
  36. package/dist/types/src/automerge/storage-adapter.test.d.ts.map +1 -0
  37. package/dist/types/src/automerge/types.d.ts +73 -0
  38. package/dist/types/src/automerge/types.d.ts.map +1 -0
  39. package/dist/types/src/metadata/metadata-store.d.ts +2 -1
  40. package/dist/types/src/metadata/metadata-store.d.ts.map +1 -1
  41. package/dist/types/src/space/space.d.ts +4 -8
  42. package/dist/types/src/space/space.d.ts.map +1 -1
  43. package/dist/types/src/testing/index.d.ts +1 -0
  44. package/dist/types/src/testing/index.d.ts.map +1 -1
  45. package/dist/types/src/testing/level.d.ts +3 -0
  46. package/dist/types/src/testing/level.d.ts.map +1 -0
  47. package/dist/types/src/testing/test-agent-builder.d.ts +2 -2
  48. package/package.json +33 -30
  49. package/src/automerge/automerge-doc-loader.test.ts +97 -0
  50. package/src/automerge/automerge-doc-loader.ts +241 -0
  51. package/src/automerge/automerge-host.test.ts +22 -8
  52. package/src/automerge/automerge-host.ts +65 -118
  53. package/src/automerge/automerge-repo.test.ts +29 -0
  54. package/src/automerge/index.ts +4 -0
  55. package/src/automerge/level.test.ts +64 -0
  56. package/src/automerge/leveldb-storage-adapter.ts +117 -0
  57. package/src/automerge/local-host-network-adapter.ts +19 -13
  58. package/src/automerge/migrations.ts +41 -0
  59. package/src/automerge/reference.ts +31 -0
  60. package/src/automerge/storage-adapter.test.ts +90 -0
  61. package/src/automerge/types.ts +86 -0
  62. package/src/db-host/data-service.ts +1 -1
  63. package/src/metadata/metadata-store.ts +17 -8
  64. package/src/space/space.test.ts +7 -7
  65. package/src/space/space.ts +6 -21
  66. package/src/testing/index.ts +1 -0
  67. package/src/testing/level.ts +11 -0
  68. package/dist/lib/browser/chunk-RTEEJ723.mjs.map +0 -7
  69. package/dist/lib/node/chunk-7VZVCCNF.cjs.map +0 -7
@@ -0,0 +1,90 @@
1
+ //
2
+ // Copyright 2024 DXOS.org
3
+ //
4
+
5
+ import { expect } from 'chai';
6
+
7
+ import { type StorageAdapterInterface } from '@dxos/automerge/automerge-repo';
8
+ import { PublicKey } from '@dxos/keys';
9
+ import { StorageType, createStorage } from '@dxos/random-access-storage';
10
+ import { afterTest, describe, openAndClose, test } from '@dxos/test';
11
+ import { type MaybePromise } from '@dxos/util';
12
+
13
+ import { AutomergeStorageAdapter } from './automerge-storage-adapter';
14
+ import { LevelDBStorageAdapter } from './leveldb-storage-adapter';
15
+ import { createTestLevel } from '../testing';
16
+
17
+ const runTests = (
18
+ testNamespace: string,
19
+ /** Run per test. Expects automatic clean-up with `afterTest`. */ createAdapter: () => MaybePromise<StorageAdapterInterface>,
20
+ ) => {
21
+ describe(testNamespace, () => {
22
+ const chunks = [
23
+ { key: ['a', 'b', 'c', '1'], data: PublicKey.random().asUint8Array() },
24
+ { key: ['a', 'b', 'c', '2'], data: PublicKey.random().asUint8Array() },
25
+ { key: ['a', 'b', 'd', '3'], data: PublicKey.random().asUint8Array() },
26
+ { key: ['a', 'b', 'd', '4'], data: PublicKey.random().asUint8Array() },
27
+ ];
28
+
29
+ test('should store and retrieve data', async () => {
30
+ const adapter = await createAdapter();
31
+
32
+ await adapter.save(chunks[0].key, chunks[0].data);
33
+ expect(await adapter.load(chunks[0].key)).to.deep.equal(chunks[0].data);
34
+ });
35
+
36
+ test('loadRange return inputs with correct prefixes', async () => {
37
+ const adapter = await createAdapter();
38
+
39
+ for (const chunk of chunks) {
40
+ await adapter.save(chunk.key, chunk.data);
41
+ }
42
+
43
+ expect((await adapter.loadRange(['a', 'b'])).length).to.equal(4);
44
+ expect((await adapter.loadRange(['a', 'b', 'c']))[0]).to.deep.equal(chunks[0]);
45
+ expect((await adapter.loadRange(['a', 'b', 'c'])).length).to.equal(2);
46
+ });
47
+
48
+ test('deletion works', async () => {
49
+ const adapter = await createAdapter();
50
+
51
+ for (const chunk of chunks) {
52
+ await adapter.save(chunk.key, chunk.data);
53
+ }
54
+
55
+ await adapter.remove(['a', 'b', 'c', '1']);
56
+
57
+ expect((await adapter.loadRange(['a', 'b'])).length).to.equal(3);
58
+ expect((await adapter.loadRange(['a', 'b', 'c'])).length).to.equal(1);
59
+
60
+ await adapter.removeRange(['a', 'b', 'd']);
61
+
62
+ expect((await adapter.loadRange(['a', 'b'])).length).to.equal(1);
63
+ expect((await adapter.loadRange(['a', 'b']))[0]).to.deep.equal(chunks[1]);
64
+ expect(await adapter.load(['a', 'b', 'c', '2'])).to.deep.equal(chunks[1].data);
65
+ expect(await adapter.load(['a', 'b', 'd', '3'])).to.be.undefined;
66
+ });
67
+ });
68
+ };
69
+
70
+ /**
71
+ * Run tests for AutomergeStorageAdapter.
72
+ */
73
+ runTests('AutomergeStorageAdapter', () => {
74
+ const storage = createStorage({ type: StorageType.RAM });
75
+ afterTest(() => storage.close());
76
+ const dir = storage.createDirectory('automerge');
77
+ const adapter = new AutomergeStorageAdapter(dir);
78
+ afterTest(() => adapter.close());
79
+ return adapter;
80
+ });
81
+
82
+ /**
83
+ * Run tests for LevelDBStorageAdapter.
84
+ */
85
+ runTests('LevelDBStorageAdapter', async () => {
86
+ const level = createTestLevel();
87
+ const adapter = new LevelDBStorageAdapter({ db: level.sublevel('automerge') });
88
+ await openAndClose(level, adapter as any);
89
+ return adapter;
90
+ });
@@ -0,0 +1,86 @@
1
+ //
2
+ // Copyright 2024 DXOS.org
3
+ //
4
+
5
+ import { type AbstractChainedBatch, type AbstractSublevel } from 'abstract-level';
6
+ import { type Level } from 'level';
7
+
8
+ import { type EncodedReferenceObject } from './reference';
9
+
10
+ export type SpaceState = {
11
+ // Url of the root automerge document.
12
+ rootUrl?: string;
13
+ };
14
+
15
+ export interface SpaceDoc {
16
+ access?: {
17
+ spaceKey: string;
18
+ };
19
+ /**
20
+ * Objects inlined in the current document.
21
+ */
22
+ objects?: {
23
+ [key: string]: ObjectStructure;
24
+ };
25
+ /**
26
+ * Object id points to an automerge doc url where the object is embedded.
27
+ */
28
+ links?: {
29
+ [echoId: string]: string;
30
+ };
31
+ }
32
+
33
+ /**
34
+ * Representation of an ECHO object in an AM document.
35
+ */
36
+ export type ObjectStructure = {
37
+ data: Record<string, any>;
38
+ meta: ObjectMeta;
39
+ system: ObjectSystem;
40
+ };
41
+
42
+ /**
43
+ * Echo object metadata.
44
+ */
45
+ export type ObjectMeta = {
46
+ /**
47
+ * Foreign keys.
48
+ */
49
+ keys: ForeignKey[];
50
+ };
51
+
52
+ /**
53
+ * Reference to an object in a foreign database.
54
+ */
55
+ export type ForeignKey = {
56
+ /**
57
+ * Name of the foreign database/system.
58
+ * E.g. `github.com`.
59
+ */
60
+ source?: string;
61
+
62
+ /**
63
+ * Id within the foreign database.
64
+ */
65
+ id?: string;
66
+ };
67
+
68
+ /**
69
+ * Automerge object system properties.
70
+ * (Is automerge specific.)
71
+ */
72
+ export type ObjectSystem = {
73
+ /**
74
+ * Deletion marker.
75
+ */
76
+ deleted?: boolean;
77
+
78
+ /**
79
+ * Object reference ('protobuf' protocol) type.
80
+ */
81
+ type?: EncodedReferenceObject;
82
+ };
83
+
84
+ export type LevelDB = Level<string, string>;
85
+ export type SubLevelDB = AbstractSublevel<any, string | Buffer | Uint8Array, string, string>;
86
+ export type BatchLevel = AbstractChainedBatch<any, string, string>;
@@ -33,7 +33,7 @@ export class DataServiceImpl implements DataService {
33
33
  }
34
34
 
35
35
  async flush(request: FlushRequest): Promise<void> {
36
- // TODO(dmaretskyi): Implement with automerge.
36
+ await this._automergeHost.flush(request);
37
37
  }
38
38
 
39
39
  // Automerge specific.
@@ -11,7 +11,7 @@ import { invariant } from '@dxos/invariant';
11
11
  import { PublicKey } from '@dxos/keys';
12
12
  import { log } from '@dxos/log';
13
13
  import { DataCorruptionError, STORAGE_VERSION, schema } from '@dxos/protocols';
14
- import { type Invitation, SpaceState } from '@dxos/protocols/proto/dxos/client/services';
14
+ import { Invitation, SpaceState } from '@dxos/protocols/proto/dxos/client/services';
15
15
  import {
16
16
  type ControlPipelineSnapshot,
17
17
  EchoMetadata,
@@ -174,13 +174,8 @@ export class MetadataStore {
174
174
  scheduleTaskInterval(
175
175
  this._invitationCleanupCtx,
176
176
  async () => {
177
- for (const invitation of this.getInvitations()) {
178
- if (
179
- invitation.created &&
180
- invitation.lifetime &&
181
- invitation.lifetime !== 0 &&
182
- invitation.created.getTime() + invitation.lifetime * 1000 < Date.now()
183
- ) {
177
+ for (const invitation of this._metadata.invitations ?? []) {
178
+ if (hasInvitationExpired(invitation) || isLegacyInvitationFormat(invitation)) {
184
179
  await this.removeInvitation(invitation.invitationId);
185
180
  }
186
181
  }
@@ -343,3 +338,17 @@ export class MetadataStore {
343
338
  }
344
339
 
345
340
  const fromBytesInt32 = (buf: Buffer) => buf.readInt32LE(0);
341
+
342
+ export const hasInvitationExpired = (invitation: Invitation): boolean => {
343
+ return Boolean(
344
+ invitation.created &&
345
+ invitation.lifetime &&
346
+ invitation.lifetime !== 0 &&
347
+ invitation.created.getTime() + invitation.lifetime * 1000 < Date.now(),
348
+ );
349
+ };
350
+
351
+ // TODO: remove once "multiuse" type invitations get removed from local metadata of existing profiles
352
+ const isLegacyInvitationFormat = (invitation: Invitation): boolean => {
353
+ return invitation.type === Invitation.Type.MULTIUSE;
354
+ };
@@ -20,7 +20,7 @@ describe('space/space', () => {
20
20
  const agent = await builder.createPeer();
21
21
  const space = await agent.createSpace();
22
22
 
23
- await space.open(new Context());
23
+ await space.open(Context.default());
24
24
  expect(space.isOpen).toBeTruthy();
25
25
  afterTest(() => space.close());
26
26
 
@@ -43,7 +43,7 @@ describe('space/space', () => {
43
43
  const agent = await builder.createPeer();
44
44
  const space = await agent.createSpace(agent.identityKey);
45
45
 
46
- await space.open(new Context());
46
+ await space.open(Context.default());
47
47
  expect(space.isOpen).toBeTruthy();
48
48
  afterTest(() => space.close());
49
49
 
@@ -62,7 +62,7 @@ describe('space/space', () => {
62
62
  const agent = await builder.createPeer();
63
63
  const space = await agent.createSpace(agent.identityKey, space1.key, space1.genesisFeedKey, undefined, true);
64
64
 
65
- await space.open(new Context());
65
+ await space.open(Context.default());
66
66
  expect(space.isOpen).toBeTruthy();
67
67
  afterTest(() => space.close());
68
68
 
@@ -114,7 +114,7 @@ describe('space/space', () => {
114
114
  const agent = await builder.createPeer();
115
115
  const space1 = await agent.createSpace();
116
116
 
117
- await space1.open(new Context());
117
+ await space1.open(Context.default());
118
118
  expect(space1.isOpen).toBeTruthy();
119
119
  afterTest(() => space1.close());
120
120
 
@@ -128,7 +128,7 @@ describe('space/space', () => {
128
128
  // Re-open.
129
129
  const space2 = await agent.createSpace(agent.identityKey, space1.key, space1.genesisFeedKey, space1.dataFeedKey);
130
130
 
131
- await space2.open(new Context());
131
+ await space2.open(Context.default());
132
132
  await space2.controlPipeline.state!.waitUntilTimeframe(space2.controlPipeline.state!.endTimeframe);
133
133
  });
134
134
 
@@ -139,7 +139,7 @@ describe('space/space', () => {
139
139
  const space = await agent.createSpace();
140
140
 
141
141
  {
142
- await space.open(new Context());
142
+ await space.open(Context.default());
143
143
  afterTest(() => space.close());
144
144
  expect(space.isOpen).toBeTruthy();
145
145
 
@@ -153,7 +153,7 @@ describe('space/space', () => {
153
153
 
154
154
  // Re-open.
155
155
  {
156
- await space.open(new Context());
156
+ await space.open(Context.default());
157
157
  expect(space.isOpen).toBeTruthy();
158
158
 
159
159
  await space.controlPipeline.state!.waitUntilTimeframe(space.controlPipeline.state!.endTimeframe);
@@ -3,7 +3,7 @@
3
3
  //
4
4
 
5
5
  import { Event, Mutex, synchronized, trackLeaks } from '@dxos/async';
6
- import { type Context } from '@dxos/context';
6
+ import { Resource, type Context, LifecycleState } from '@dxos/context';
7
7
  import { type FeedInfo } from '@dxos/credentials';
8
8
  import { type FeedOptions, type FeedWrapper } from '@dxos/feed-store';
9
9
  import { invariant } from '@dxos/invariant';
@@ -48,7 +48,7 @@ export type CreatePipelineParams = {
48
48
  // TODO(dmaretskyi): Extract database stuff.
49
49
  @trackLeaks('open', 'close')
50
50
  @trace.resource()
51
- export class Space {
51
+ export class Space extends Resource {
52
52
  private readonly _addFeedMutex = new Mutex();
53
53
 
54
54
  public readonly onCredentialProcessed = new Callback<AsyncCallback<Credential>>();
@@ -64,11 +64,11 @@ export class Space {
64
64
 
65
65
  private readonly _snapshotManager: SnapshotManager;
66
66
 
67
- private _isOpen = false;
68
67
  private _controlFeed?: FeedWrapper<FeedMessage>;
69
68
  private _dataFeed?: FeedWrapper<FeedMessage>;
70
69
 
71
70
  constructor(params: SpaceParams) {
71
+ super();
72
72
  invariant(params.spaceKey && params.feedProvider);
73
73
  this._key = params.spaceKey;
74
74
  this._genesisFeedKey = params.genesisFeed.key;
@@ -112,7 +112,7 @@ export class Space {
112
112
  }
113
113
 
114
114
  get isOpen() {
115
- return this._isOpen;
115
+ return this._lifecycleState === LifecycleState.OPEN;
116
116
  }
117
117
 
118
118
  get genesisFeedKey(): PublicKey {
@@ -162,40 +162,25 @@ export class Space {
162
162
  return Array.from(this._controlPipeline.spaceState.feeds.values());
163
163
  }
164
164
 
165
- /**
166
- * Use for diagnostics.
167
- */
168
- // getDataFeeds(): FeedInfo[] {
169
- // return this._dataPipeline?.getFeeds();
170
- // }
171
- @synchronized
172
165
  @trace.span()
173
- async open(ctx: Context) {
166
+ protected override async _open(ctx: Context) {
174
167
  log('opening...');
175
- if (this._isOpen) {
176
- return;
177
- }
178
168
 
179
169
  // Order is important.
180
170
  await this._controlPipeline.start();
181
171
  await this.protocol.start();
182
172
 
183
- this._isOpen = true;
184
173
  log('opened');
185
174
  }
186
175
 
187
176
  @synchronized
188
- async close() {
177
+ protected override async _close() {
189
178
  log('closing...', { key: this._key });
190
- if (!this._isOpen) {
191
- return;
192
- }
193
179
 
194
180
  // Closes in reverse order to open.
195
181
  await this.protocol.stop();
196
182
  await this._controlPipeline.stop();
197
183
 
198
- this._isOpen = false;
199
184
  log('closed');
200
185
  }
201
186
  }
@@ -5,3 +5,4 @@
5
5
  export * from './change-metadata';
6
6
  export * from './test-agent-builder';
7
7
  export * from './test-feed-builder';
8
+ export * from './level';
@@ -0,0 +1,11 @@
1
+ //
2
+ // Copyright 2024 DXOS.org
3
+ //
4
+
5
+ import { Level } from 'level';
6
+
7
+ import { PublicKey } from '@dxos/keys';
8
+
9
+ import { type LevelDB } from '../automerge/types';
10
+
11
+ export const createTestLevel = (): LevelDB => new Level<string, string>(`/tmp/dxos-${PublicKey.random().toHex()}`);