@dxos/client-services 0.8.4-main.fffef41 → 0.8.4-staging.60fe92afc8

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 (253) hide show
  1. package/LICENSE +102 -5
  2. package/README.md +1 -1
  3. package/dist/lib/browser/{chunk-I2RGLVJF.mjs → chunk-HPR4MJ4W.mjs} +2870 -3767
  4. package/dist/lib/browser/chunk-HPR4MJ4W.mjs.map +7 -0
  5. package/dist/lib/browser/chunk-QCWEHHJW.mjs +24 -0
  6. package/dist/lib/browser/chunk-QCWEHHJW.mjs.map +7 -0
  7. package/dist/lib/browser/chunk-XJRPB3GA.mjs +22 -0
  8. package/dist/lib/browser/chunk-XJRPB3GA.mjs.map +7 -0
  9. package/dist/lib/browser/index.mjs +576 -139
  10. package/dist/lib/browser/index.mjs.map +4 -4
  11. package/dist/lib/browser/meta.json +1 -1
  12. package/dist/lib/browser/packlets/diagnostics/browser-diagnostics-broadcast.mjs +88 -0
  13. package/dist/lib/browser/packlets/diagnostics/browser-diagnostics-broadcast.mjs.map +7 -0
  14. package/dist/lib/browser/packlets/diagnostics/diagnostics-broadcast.mjs +11 -0
  15. package/dist/lib/browser/packlets/diagnostics/diagnostics-broadcast.mjs.map +7 -0
  16. package/dist/lib/browser/packlets/locks/browser.mjs +86 -0
  17. package/dist/lib/browser/packlets/locks/browser.mjs.map +7 -0
  18. package/dist/lib/browser/packlets/locks/node.mjs +48 -0
  19. package/dist/lib/browser/packlets/locks/node.mjs.map +7 -0
  20. package/dist/lib/browser/testing/index.mjs +58 -53
  21. package/dist/lib/browser/testing/index.mjs.map +3 -3
  22. package/dist/lib/node-esm/chunk-2DT3MZRL.mjs +22 -0
  23. package/dist/lib/node-esm/chunk-2DT3MZRL.mjs.map +7 -0
  24. package/dist/lib/node-esm/chunk-2SZHAWBN.mjs +24 -0
  25. package/dist/lib/node-esm/chunk-2SZHAWBN.mjs.map +7 -0
  26. package/dist/lib/node-esm/{chunk-QTUURCR4.mjs → chunk-JW6QHPRJ.mjs} +2810 -3576
  27. package/dist/lib/node-esm/chunk-JW6QHPRJ.mjs.map +7 -0
  28. package/dist/lib/node-esm/index.mjs +576 -139
  29. package/dist/lib/node-esm/index.mjs.map +4 -4
  30. package/dist/lib/node-esm/meta.json +1 -1
  31. package/dist/lib/node-esm/packlets/diagnostics/browser-diagnostics-broadcast.mjs +88 -0
  32. package/dist/lib/node-esm/packlets/diagnostics/browser-diagnostics-broadcast.mjs.map +7 -0
  33. package/dist/lib/node-esm/packlets/diagnostics/diagnostics-broadcast.mjs +11 -0
  34. package/dist/lib/node-esm/packlets/diagnostics/diagnostics-broadcast.mjs.map +7 -0
  35. package/dist/lib/node-esm/packlets/locks/browser.mjs +86 -0
  36. package/dist/lib/node-esm/packlets/locks/browser.mjs.map +7 -0
  37. package/dist/lib/node-esm/packlets/locks/node.mjs +48 -0
  38. package/dist/lib/node-esm/packlets/locks/node.mjs.map +7 -0
  39. package/dist/lib/node-esm/testing/index.mjs +58 -53
  40. package/dist/lib/node-esm/testing/index.mjs.map +3 -3
  41. package/dist/types/src/index.d.ts +1 -0
  42. package/dist/types/src/index.d.ts.map +1 -1
  43. package/dist/types/src/packlets/agents/edge-agent-manager.d.ts +3 -2
  44. package/dist/types/src/packlets/agents/edge-agent-manager.d.ts.map +1 -1
  45. package/dist/types/src/packlets/agents/edge-agent-service.d.ts +2 -1
  46. package/dist/types/src/packlets/agents/edge-agent-service.d.ts.map +1 -1
  47. package/dist/types/src/packlets/devices/devices-service.d.ts.map +1 -1
  48. package/dist/types/src/packlets/devtools/devtools.d.ts +7 -3
  49. package/dist/types/src/packlets/devtools/devtools.d.ts.map +1 -1
  50. package/dist/types/src/packlets/devtools/feeds.d.ts +1 -1
  51. package/dist/types/src/packlets/devtools/feeds.d.ts.map +1 -1
  52. package/dist/types/src/packlets/devtools/keys.d.ts +2 -2
  53. package/dist/types/src/packlets/devtools/keys.d.ts.map +1 -1
  54. package/dist/types/src/packlets/devtools/metadata.d.ts.map +1 -1
  55. package/dist/types/src/packlets/devtools/network.d.ts.map +1 -1
  56. package/dist/types/src/packlets/devtools/spaces.d.ts.map +1 -1
  57. package/dist/types/src/packlets/diagnostics/browser-diagnostics-broadcast.d.ts.map +1 -1
  58. package/dist/types/src/packlets/diagnostics/diagnostics-broadcast.d.ts.map +1 -1
  59. package/dist/types/src/packlets/diagnostics/diagnostics-collector.d.ts.map +1 -1
  60. package/dist/types/src/packlets/diagnostics/diagnostics.d.ts +2 -3
  61. package/dist/types/src/packlets/diagnostics/diagnostics.d.ts.map +1 -1
  62. package/dist/types/src/packlets/diagnostics/index.d.ts +1 -1
  63. package/dist/types/src/packlets/diagnostics/index.d.ts.map +1 -1
  64. package/dist/types/src/packlets/identity/authenticator.d.ts +3 -3
  65. package/dist/types/src/packlets/identity/authenticator.d.ts.map +1 -1
  66. package/dist/types/src/packlets/identity/contacts-service.d.ts +1 -1
  67. package/dist/types/src/packlets/identity/contacts-service.d.ts.map +1 -1
  68. package/dist/types/src/packlets/identity/identity-manager.d.ts +10 -10
  69. package/dist/types/src/packlets/identity/identity-manager.d.ts.map +1 -1
  70. package/dist/types/src/packlets/identity/identity-recovery-manager.d.ts +14 -9
  71. package/dist/types/src/packlets/identity/identity-recovery-manager.d.ts.map +1 -1
  72. package/dist/types/src/packlets/identity/identity-service.d.ts +7 -11
  73. package/dist/types/src/packlets/identity/identity-service.d.ts.map +1 -1
  74. package/dist/types/src/packlets/identity/identity.d.ts +10 -13
  75. package/dist/types/src/packlets/identity/identity.d.ts.map +1 -1
  76. package/dist/types/src/packlets/invitations/device-invitation-protocol.d.ts +7 -6
  77. package/dist/types/src/packlets/invitations/device-invitation-protocol.d.ts.map +1 -1
  78. package/dist/types/src/packlets/invitations/edge-invitation-handler.d.ts +1 -1
  79. package/dist/types/src/packlets/invitations/edge-invitation-handler.d.ts.map +1 -1
  80. package/dist/types/src/packlets/invitations/invitation-guest-extenstion.d.ts.map +1 -1
  81. package/dist/types/src/packlets/invitations/invitation-host-extension.d.ts.map +1 -1
  82. package/dist/types/src/packlets/invitations/invitation-protocol.d.ts +7 -4
  83. package/dist/types/src/packlets/invitations/invitation-protocol.d.ts.map +1 -1
  84. package/dist/types/src/packlets/invitations/invitation-state.d.ts.map +1 -1
  85. package/dist/types/src/packlets/invitations/invitation-topology.d.ts.map +1 -1
  86. package/dist/types/src/packlets/invitations/invitations-handler.d.ts +4 -4
  87. package/dist/types/src/packlets/invitations/invitations-handler.d.ts.map +1 -1
  88. package/dist/types/src/packlets/invitations/invitations-manager.d.ts +5 -5
  89. package/dist/types/src/packlets/invitations/invitations-manager.d.ts.map +1 -1
  90. package/dist/types/src/packlets/invitations/invitations-service.d.ts +3 -3
  91. package/dist/types/src/packlets/invitations/invitations-service.d.ts.map +1 -1
  92. package/dist/types/src/packlets/invitations/space-invitation-protocol.d.ts +6 -5
  93. package/dist/types/src/packlets/invitations/space-invitation-protocol.d.ts.map +1 -1
  94. package/dist/types/src/packlets/invitations/utils.d.ts.map +1 -1
  95. package/dist/types/src/packlets/locks/browser.d.ts.map +1 -1
  96. package/dist/types/src/packlets/locks/index.d.ts +1 -1
  97. package/dist/types/src/packlets/locks/index.d.ts.map +1 -1
  98. package/dist/types/src/packlets/locks/node.d.ts.map +1 -1
  99. package/dist/types/src/packlets/logging/logging-service.d.ts +4 -0
  100. package/dist/types/src/packlets/logging/logging-service.d.ts.map +1 -1
  101. package/dist/types/src/packlets/network/network-service.d.ts +5 -4
  102. package/dist/types/src/packlets/network/network-service.d.ts.map +1 -1
  103. package/dist/types/src/packlets/services/client-rpc-server.d.ts +5 -5
  104. package/dist/types/src/packlets/services/client-rpc-server.d.ts.map +1 -1
  105. package/dist/types/src/packlets/services/feed-syncer.d.ts +75 -0
  106. package/dist/types/src/packlets/services/feed-syncer.d.ts.map +1 -0
  107. package/dist/types/src/packlets/services/feed-syncer.test.d.ts +2 -0
  108. package/dist/types/src/packlets/services/feed-syncer.test.d.ts.map +1 -0
  109. package/dist/types/src/packlets/services/index.d.ts +1 -0
  110. package/dist/types/src/packlets/services/index.d.ts.map +1 -1
  111. package/dist/types/src/packlets/services/platform.d.ts.map +1 -1
  112. package/dist/types/src/packlets/services/service-context.d.ts +22 -19
  113. package/dist/types/src/packlets/services/service-context.d.ts.map +1 -1
  114. package/dist/types/src/packlets/services/service-host.d.ts +20 -13
  115. package/dist/types/src/packlets/services/service-host.d.ts.map +1 -1
  116. package/dist/types/src/packlets/services/service-registry.d.ts.map +1 -1
  117. package/dist/types/src/packlets/services/sqlite-storage.d.ts +27 -0
  118. package/dist/types/src/packlets/services/sqlite-storage.d.ts.map +1 -0
  119. package/dist/types/src/packlets/services/util.d.ts.map +1 -1
  120. package/dist/types/src/packlets/space-export/archive-format.d.ts +9 -0
  121. package/dist/types/src/packlets/space-export/archive-format.d.ts.map +1 -0
  122. package/dist/types/src/packlets/space-export/index.d.ts +4 -1
  123. package/dist/types/src/packlets/space-export/index.d.ts.map +1 -1
  124. package/dist/types/src/packlets/space-export/serialized-space-reader.d.ts +23 -0
  125. package/dist/types/src/packlets/space-export/serialized-space-reader.d.ts.map +1 -0
  126. package/dist/types/src/packlets/space-export/serialized-space-writer.d.ts +36 -0
  127. package/dist/types/src/packlets/space-export/serialized-space-writer.d.ts.map +1 -0
  128. package/dist/types/src/packlets/space-export/space-archive-reader.d.ts +9 -1
  129. package/dist/types/src/packlets/space-export/space-archive-reader.d.ts.map +1 -1
  130. package/dist/types/src/packlets/space-export/space-archive-writer.d.ts +7 -1
  131. package/dist/types/src/packlets/space-export/space-archive-writer.d.ts.map +1 -1
  132. package/dist/types/src/packlets/space-export/space-archive.test.d.ts +2 -0
  133. package/dist/types/src/packlets/space-export/space-archive.test.d.ts.map +1 -0
  134. package/dist/types/src/packlets/spaces/automerge-space-state.d.ts.map +1 -1
  135. package/dist/types/src/packlets/spaces/data-space-manager.d.ts +49 -22
  136. package/dist/types/src/packlets/spaces/data-space-manager.d.ts.map +1 -1
  137. package/dist/types/src/packlets/spaces/data-space.d.ts +38 -13
  138. package/dist/types/src/packlets/spaces/data-space.d.ts.map +1 -1
  139. package/dist/types/src/packlets/spaces/edge-feed-replicator.d.ts +2 -2
  140. package/dist/types/src/packlets/spaces/edge-feed-replicator.d.ts.map +1 -1
  141. package/dist/types/src/packlets/spaces/epoch-migrations.d.ts +1 -1
  142. package/dist/types/src/packlets/spaces/epoch-migrations.d.ts.map +1 -1
  143. package/dist/types/src/packlets/spaces/genesis.d.ts +4 -3
  144. package/dist/types/src/packlets/spaces/genesis.d.ts.map +1 -1
  145. package/dist/types/src/packlets/spaces/notarization-plugin.d.ts +6 -9
  146. package/dist/types/src/packlets/spaces/notarization-plugin.d.ts.map +1 -1
  147. package/dist/types/src/packlets/spaces/spaces-service.d.ts +10 -7
  148. package/dist/types/src/packlets/spaces/spaces-service.d.ts.map +1 -1
  149. package/dist/types/src/packlets/storage/index.d.ts +1 -0
  150. package/dist/types/src/packlets/storage/index.d.ts.map +1 -1
  151. package/dist/types/src/packlets/storage/level.d.ts.map +1 -1
  152. package/dist/types/src/packlets/storage/profile-archive-sqlite.d.ts +24 -0
  153. package/dist/types/src/packlets/storage/profile-archive-sqlite.d.ts.map +1 -0
  154. package/dist/types/src/packlets/storage/profile-archive-sqlite.test.d.ts +2 -0
  155. package/dist/types/src/packlets/storage/profile-archive-sqlite.test.d.ts.map +1 -0
  156. package/dist/types/src/packlets/storage/profile-archive.d.ts.map +1 -1
  157. package/dist/types/src/packlets/storage/storage.d.ts.map +1 -1
  158. package/dist/types/src/packlets/storage/util.d.ts.map +1 -1
  159. package/dist/types/src/packlets/system/system-service.d.ts +1 -1
  160. package/dist/types/src/packlets/system/system-service.d.ts.map +1 -1
  161. package/dist/types/src/packlets/testing/credential-utils.d.ts.map +1 -1
  162. package/dist/types/src/packlets/testing/invitation-utils.d.ts +6 -3
  163. package/dist/types/src/packlets/testing/invitation-utils.d.ts.map +1 -1
  164. package/dist/types/src/packlets/testing/test-builder.d.ts +20 -22
  165. package/dist/types/src/packlets/testing/test-builder.d.ts.map +1 -1
  166. package/dist/types/src/packlets/worker/worker-runtime.d.ts +41 -4
  167. package/dist/types/src/packlets/worker/worker-runtime.d.ts.map +1 -1
  168. package/dist/types/src/packlets/worker/worker-session.d.ts +2 -4
  169. package/dist/types/src/packlets/worker/worker-session.d.ts.map +1 -1
  170. package/dist/types/src/testing/setup.d.ts.map +1 -1
  171. package/dist/types/src/version.d.ts +1 -1
  172. package/dist/types/src/version.d.ts.map +1 -1
  173. package/dist/types/tsconfig.tsbuildinfo +1 -1
  174. package/package.json +71 -57
  175. package/src/index.ts +1 -0
  176. package/src/packlets/agents/edge-agent-manager.ts +8 -5
  177. package/src/packlets/agents/edge-agent-service.ts +4 -2
  178. package/src/packlets/devices/devices-service.test.ts +0 -1
  179. package/src/packlets/devtools/devtools.ts +28 -7
  180. package/src/packlets/devtools/feeds.ts +1 -1
  181. package/src/packlets/devtools/keys.ts +2 -2
  182. package/src/packlets/devtools/spaces.ts +1 -1
  183. package/src/packlets/diagnostics/diagnostics.ts +1 -2
  184. package/src/packlets/diagnostics/index.ts +1 -1
  185. package/src/packlets/identity/authenticator.ts +3 -3
  186. package/src/packlets/identity/contacts-service.ts +1 -2
  187. package/src/packlets/identity/identity-manager.test.ts +6 -6
  188. package/src/packlets/identity/identity-manager.ts +29 -28
  189. package/src/packlets/identity/identity-recovery-manager.ts +31 -22
  190. package/src/packlets/identity/identity-service.test.ts +6 -27
  191. package/src/packlets/identity/identity-service.ts +17 -83
  192. package/src/packlets/identity/identity.test.ts +3 -3
  193. package/src/packlets/identity/identity.ts +12 -35
  194. package/src/packlets/invitations/device-invitation-protocol.ts +10 -9
  195. package/src/packlets/invitations/edge-invitation-handler.ts +9 -5
  196. package/src/packlets/invitations/invitation-guest-extenstion.ts +6 -4
  197. package/src/packlets/invitations/invitation-host-extension.ts +13 -14
  198. package/src/packlets/invitations/invitation-protocol.ts +7 -4
  199. package/src/packlets/invitations/invitation-state.ts +1 -15
  200. package/src/packlets/invitations/invitations-handler.test.ts +4 -5
  201. package/src/packlets/invitations/invitations-handler.ts +74 -22
  202. package/src/packlets/invitations/invitations-manager.ts +42 -17
  203. package/src/packlets/invitations/invitations-service.ts +9 -9
  204. package/src/packlets/invitations/space-invitation-protocol.test.ts +17 -16
  205. package/src/packlets/invitations/space-invitation-protocol.ts +13 -18
  206. package/src/packlets/locks/index.ts +1 -1
  207. package/src/packlets/logging/logging-service.ts +19 -15
  208. package/src/packlets/network/network-service.test.ts +0 -1
  209. package/src/packlets/network/network-service.ts +10 -8
  210. package/src/packlets/services/client-rpc-server.ts +19 -16
  211. package/src/packlets/services/feed-syncer.test.ts +376 -0
  212. package/src/packlets/services/feed-syncer.ts +536 -0
  213. package/src/packlets/services/index.ts +1 -0
  214. package/src/packlets/services/platform.ts +7 -1
  215. package/src/packlets/services/service-context.test.ts +3 -2
  216. package/src/packlets/services/service-context.ts +215 -78
  217. package/src/packlets/services/service-host.test.ts +8 -10
  218. package/src/packlets/services/service-host.ts +102 -70
  219. package/src/packlets/services/service-registry.test.ts +0 -1
  220. package/src/packlets/services/sqlite-storage.ts +390 -0
  221. package/src/packlets/space-export/archive-format.ts +42 -0
  222. package/src/packlets/space-export/index.ts +4 -1
  223. package/src/packlets/space-export/serialized-space-reader.ts +129 -0
  224. package/src/packlets/space-export/serialized-space-writer.ts +260 -0
  225. package/src/packlets/space-export/space-archive-reader.ts +64 -3
  226. package/src/packlets/space-export/space-archive-writer.ts +41 -4
  227. package/src/packlets/space-export/space-archive.test.ts +482 -0
  228. package/src/packlets/spaces/data-space-manager.test.ts +169 -14
  229. package/src/packlets/spaces/data-space-manager.ts +192 -127
  230. package/src/packlets/spaces/data-space.ts +89 -43
  231. package/src/packlets/spaces/edge-feed-replicator.test.ts +2 -2
  232. package/src/packlets/spaces/edge-feed-replicator.ts +11 -9
  233. package/src/packlets/spaces/epoch-migrations.ts +7 -6
  234. package/src/packlets/spaces/genesis.ts +9 -4
  235. package/src/packlets/spaces/notarization-plugin.test.ts +2 -2
  236. package/src/packlets/spaces/notarization-plugin.ts +10 -9
  237. package/src/packlets/spaces/spaces-service.test.ts +18 -11
  238. package/src/packlets/spaces/spaces-service.ts +130 -24
  239. package/src/packlets/storage/index.ts +1 -0
  240. package/src/packlets/storage/profile-archive-sqlite.test.ts +79 -0
  241. package/src/packlets/storage/profile-archive-sqlite.ts +100 -0
  242. package/src/packlets/storage/profile-archive.ts +3 -0
  243. package/src/packlets/storage/storage.ts +4 -4
  244. package/src/packlets/testing/invitation-utils.ts +10 -6
  245. package/src/packlets/testing/test-builder.ts +59 -40
  246. package/src/packlets/worker/worker-runtime.ts +173 -17
  247. package/src/packlets/worker/worker-session.ts +12 -18
  248. package/src/version.ts +1 -1
  249. package/dist/lib/browser/chunk-I2RGLVJF.mjs.map +0 -7
  250. package/dist/lib/node-esm/chunk-QTUURCR4.mjs.map +0 -7
  251. package/dist/types/src/packlets/identity/default-space-state-machine.d.ts +0 -19
  252. package/dist/types/src/packlets/identity/default-space-state-machine.d.ts.map +0 -1
  253. package/src/packlets/identity/default-space-state-machine.ts +0 -44
@@ -0,0 +1,482 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ import type { DocumentId } from '@automerge/automerge-repo';
6
+ import { describe, expect, test } from 'vitest';
7
+
8
+ import { type SerializedSpace } from '@dxos/echo-client';
9
+ import { EntityId, SpaceId, URI } from '@dxos/keys';
10
+ import {
11
+ FEED_ARCHIVE_BLOCKS_PER_CHUNK,
12
+ type FeedArchiveBlock,
13
+ SpaceArchiveFileStructure,
14
+ SpaceArchiveVersion,
15
+ } from '@dxos/protocols';
16
+ import { SpaceArchive } from '@dxos/protocols/proto/dxos/client/services';
17
+
18
+ import { detectSpaceArchiveFormat } from './archive-format';
19
+ import { buildDatabaseDirectoryFromObjects, readSerializedSpaceArchive } from './serialized-space-reader';
20
+ import { objectStructureToObjJson, orderObjJsonFields } from './serialized-space-writer';
21
+ import { extractSpaceArchive } from './space-archive-reader';
22
+ import { SpaceArchiveWriter } from './space-archive-writer';
23
+
24
+ describe('SpaceArchive', () => {
25
+ describe('SpaceArchiveWriter', () => {
26
+ test('writes and reads documents', async () => {
27
+ const writer = new SpaceArchiveWriter();
28
+ await writer.open();
29
+ try {
30
+ const spaceId = SpaceId.random();
31
+ await writer.begin({ spaceId });
32
+ await writer.setCurrentRootUrl('automerge:test123');
33
+ await writer.writeDocument('doc1', new Uint8Array([1, 2, 3]));
34
+ await writer.writeDocument('doc2', new Uint8Array([4, 5, 6]));
35
+
36
+ const archive = await writer.finish();
37
+
38
+ expect(archive.filename).toContain(spaceId);
39
+ expect(archive.contents).toBeInstanceOf(Uint8Array);
40
+
41
+ const extracted = await extractSpaceArchive(archive);
42
+ expect(extracted.metadata.version).toBe(SpaceArchiveVersion.V1);
43
+ expect(extracted.metadata.originalSpaceId).toBe(spaceId);
44
+ expect(extracted.metadata.echo?.currentRootUrl).toBe('automerge:test123');
45
+ expect(extracted.documents['doc1' as DocumentId]).toEqual(new Uint8Array([1, 2, 3]));
46
+ expect(extracted.documents['doc2' as DocumentId]).toEqual(new Uint8Array([4, 5, 6]));
47
+ } finally {
48
+ await writer.close();
49
+ }
50
+ });
51
+ });
52
+
53
+ describe('Feed Archive', () => {
54
+ test('writes and reads a single feed with blocks', async () => {
55
+ const writer = new SpaceArchiveWriter();
56
+ await writer.open();
57
+ try {
58
+ const spaceId = SpaceId.random();
59
+ await writer.begin({ spaceId });
60
+ await writer.setCurrentRootUrl('automerge:root');
61
+
62
+ const blocks: FeedArchiveBlock[] = [
63
+ {
64
+ actorId: 'actor1',
65
+ sequence: 0,
66
+ prevActorId: null,
67
+ prevSequence: null,
68
+ position: 0,
69
+ timestamp: 1000,
70
+ data: btoa('block0'),
71
+ },
72
+ {
73
+ actorId: 'actor1',
74
+ sequence: 1,
75
+ prevActorId: 'actor1',
76
+ prevSequence: 0,
77
+ position: 1,
78
+ timestamp: 2000,
79
+ data: btoa('block1'),
80
+ },
81
+ ];
82
+
83
+ await writer.writeFeed('feed-123', 'data', blocks);
84
+
85
+ const archive = await writer.finish();
86
+ const extracted = await extractSpaceArchive(archive);
87
+
88
+ expect(Object.keys(extracted.feeds)).toHaveLength(1);
89
+ expect(extracted.feeds['feed-123']).toBeDefined();
90
+ expect(extracted.feeds['feed-123'].metadata.id).toBe('feed-123');
91
+ expect(extracted.feeds['feed-123'].metadata.namespace).toBe('data');
92
+ expect(extracted.feeds['feed-123'].blocks).toHaveLength(2);
93
+ expect(extracted.feeds['feed-123'].blocks[0].actorId).toBe('actor1');
94
+ expect(extracted.feeds['feed-123'].blocks[0].sequence).toBe(0);
95
+ expect(extracted.feeds['feed-123'].blocks[1].sequence).toBe(1);
96
+ } finally {
97
+ await writer.close();
98
+ }
99
+ });
100
+
101
+ test('writes and reads multiple feeds', async () => {
102
+ const writer = new SpaceArchiveWriter();
103
+ await writer.open();
104
+ try {
105
+ const spaceId = SpaceId.random();
106
+ await writer.begin({ spaceId });
107
+ await writer.setCurrentRootUrl('automerge:root');
108
+
109
+ const dataBlocks: FeedArchiveBlock[] = [
110
+ {
111
+ actorId: 'actor1',
112
+ sequence: 0,
113
+ prevActorId: null,
114
+ prevSequence: null,
115
+ position: 0,
116
+ timestamp: 1000,
117
+ data: btoa('data-block'),
118
+ },
119
+ ];
120
+
121
+ const traceBlocks: FeedArchiveBlock[] = [
122
+ {
123
+ actorId: 'actor2',
124
+ sequence: 0,
125
+ prevActorId: null,
126
+ prevSequence: null,
127
+ position: 0,
128
+ timestamp: 2000,
129
+ data: btoa('trace-block'),
130
+ },
131
+ ];
132
+
133
+ await writer.writeFeed('feed-data', 'data', dataBlocks);
134
+ await writer.writeFeed('feed-trace', 'trace', traceBlocks);
135
+
136
+ const archive = await writer.finish();
137
+ const extracted = await extractSpaceArchive(archive);
138
+
139
+ expect(Object.keys(extracted.feeds)).toHaveLength(2);
140
+ expect(extracted.feeds['feed-data'].metadata.namespace).toBe('data');
141
+ expect(extracted.feeds['feed-trace'].metadata.namespace).toBe('trace');
142
+ } finally {
143
+ await writer.close();
144
+ }
145
+ });
146
+
147
+ test('writes blocks in chunks when exceeding chunk size', async () => {
148
+ const writer = new SpaceArchiveWriter();
149
+ await writer.open();
150
+ try {
151
+ const spaceId = SpaceId.random();
152
+ await writer.begin({ spaceId });
153
+ await writer.setCurrentRootUrl('automerge:root');
154
+
155
+ const numBlocks = FEED_ARCHIVE_BLOCKS_PER_CHUNK + 50;
156
+ const blocks: FeedArchiveBlock[] = [];
157
+ for (let i = 0; i < numBlocks; i++) {
158
+ blocks.push({
159
+ actorId: 'actor1',
160
+ sequence: i,
161
+ prevActorId: i > 0 ? 'actor1' : null,
162
+ prevSequence: i > 0 ? i - 1 : null,
163
+ position: i,
164
+ timestamp: 1000 + i,
165
+ data: btoa(`block-${i}`),
166
+ });
167
+ }
168
+
169
+ await writer.writeFeed('large-feed', 'data', blocks);
170
+
171
+ const archive = await writer.finish();
172
+ const extracted = await extractSpaceArchive(archive);
173
+
174
+ expect(extracted.feeds['large-feed'].blocks).toHaveLength(numBlocks);
175
+ expect(extracted.feeds['large-feed'].blocks[0].sequence).toBe(0);
176
+ expect(extracted.feeds['large-feed'].blocks[numBlocks - 1].sequence).toBe(numBlocks - 1);
177
+ } finally {
178
+ await writer.close();
179
+ }
180
+ });
181
+
182
+ test('handles empty feeds', async () => {
183
+ const writer = new SpaceArchiveWriter();
184
+ await writer.open();
185
+ try {
186
+ const spaceId = SpaceId.random();
187
+ await writer.begin({ spaceId });
188
+ await writer.setCurrentRootUrl('automerge:root');
189
+
190
+ await writer.writeFeed('empty-feed', 'data', []);
191
+
192
+ const archive = await writer.finish();
193
+ const extracted = await extractSpaceArchive(archive);
194
+
195
+ expect(extracted.feeds['empty-feed']).toBeDefined();
196
+ expect(extracted.feeds['empty-feed'].metadata.id).toBe('empty-feed');
197
+ expect(extracted.feeds['empty-feed'].blocks).toHaveLength(0);
198
+ } finally {
199
+ await writer.close();
200
+ }
201
+ });
202
+
203
+ test('preserves block data encoding', async () => {
204
+ const writer = new SpaceArchiveWriter();
205
+ await writer.open();
206
+ try {
207
+ const spaceId = SpaceId.random();
208
+ await writer.begin({ spaceId });
209
+ await writer.setCurrentRootUrl('automerge:root');
210
+
211
+ const originalData = new Uint8Array([0, 1, 2, 255, 128, 64]);
212
+ const base64Data = btoa(String.fromCharCode(...originalData));
213
+
214
+ const blocks: FeedArchiveBlock[] = [
215
+ {
216
+ actorId: 'actor1',
217
+ sequence: 0,
218
+ prevActorId: null,
219
+ prevSequence: null,
220
+ position: null,
221
+ timestamp: 1000,
222
+ data: base64Data,
223
+ },
224
+ ];
225
+
226
+ await writer.writeFeed('binary-feed', 'data', blocks);
227
+
228
+ const archive = await writer.finish();
229
+ const extracted = await extractSpaceArchive(archive);
230
+
231
+ const extractedData = extracted.feeds['binary-feed'].blocks[0].data;
232
+ expect(extractedData).toBe(base64Data);
233
+
234
+ const decoded = new Uint8Array(
235
+ atob(extractedData)
236
+ .split('')
237
+ .map((char) => char.charCodeAt(0)),
238
+ );
239
+ expect(decoded).toEqual(originalData);
240
+ } finally {
241
+ await writer.close();
242
+ }
243
+ });
244
+
245
+ test('combines documents and feeds in archive', async () => {
246
+ const writer = new SpaceArchiveWriter();
247
+ await writer.open();
248
+ try {
249
+ const spaceId = SpaceId.random();
250
+ await writer.begin({ spaceId });
251
+ await writer.setCurrentRootUrl('automerge:root');
252
+
253
+ await writer.writeDocument('doc1', new Uint8Array([1, 2, 3]));
254
+ await writer.writeFeed('feed1', 'data', [
255
+ {
256
+ actorId: 'actor1',
257
+ sequence: 0,
258
+ prevActorId: null,
259
+ prevSequence: null,
260
+ position: 0,
261
+ timestamp: 1000,
262
+ data: btoa('test'),
263
+ },
264
+ ]);
265
+
266
+ const archive = await writer.finish();
267
+ const extracted = await extractSpaceArchive(archive);
268
+
269
+ expect(Object.keys(extracted.documents)).toHaveLength(1);
270
+ expect(Object.keys(extracted.feeds)).toHaveLength(1);
271
+ expect(extracted.documents['doc1' as DocumentId]).toEqual(new Uint8Array([1, 2, 3]));
272
+ expect(extracted.feeds['feed1'].blocks).toHaveLength(1);
273
+ } finally {
274
+ await writer.close();
275
+ }
276
+ });
277
+ });
278
+
279
+ describe('File Structure', () => {
280
+ test('file structure constants are correct', () => {
281
+ expect(SpaceArchiveFileStructure.metadata).toBe('metadata.json');
282
+ expect(SpaceArchiveFileStructure.documents).toBe('documents');
283
+ expect(SpaceArchiveFileStructure.feeds).toBe('feeds');
284
+ expect(SpaceArchiveFileStructure.feedMetadata).toBe('metadata.json');
285
+ expect(SpaceArchiveFileStructure.feedBlocksPrefix).toBe('blocks-');
286
+ });
287
+
288
+ test('blocks per chunk constant is set', () => {
289
+ expect(FEED_ARCHIVE_BLOCKS_PER_CHUNK).toBe(100);
290
+ });
291
+ });
292
+
293
+ describe('detectSpaceArchiveFormat', () => {
294
+ test('detects JSON via .dx.json extension', () => {
295
+ const format = detectSpaceArchiveFormat({ filename: 'space.dx.json', contents: new Uint8Array() });
296
+ expect(format).toBe(SpaceArchive.Format.JSON);
297
+ });
298
+
299
+ test('detects JSON via .json extension', () => {
300
+ const format = detectSpaceArchiveFormat({ filename: 'backup.json', contents: new Uint8Array() });
301
+ expect(format).toBe(SpaceArchive.Format.JSON);
302
+ });
303
+
304
+ test('detects BINARY via .tar extension', () => {
305
+ const format = detectSpaceArchiveFormat({ filename: 'space.tar', contents: new Uint8Array() });
306
+ expect(format).toBe(SpaceArchive.Format.BINARY);
307
+ });
308
+
309
+ test('detects BINARY via .tar.gz extension', () => {
310
+ const format = detectSpaceArchiveFormat({ filename: 'space.tar.gz', contents: new Uint8Array() });
311
+ expect(format).toBe(SpaceArchive.Format.BINARY);
312
+ });
313
+
314
+ test('falls back to JSON via leading { byte', () => {
315
+ const contents = new TextEncoder().encode('{"version":1}');
316
+ const format = detectSpaceArchiveFormat({ filename: 'unknown', contents });
317
+ expect(format).toBe(SpaceArchive.Format.JSON);
318
+ });
319
+
320
+ test('skips leading whitespace before sniffing', () => {
321
+ const contents = new TextEncoder().encode(' \n\t{"version":1}');
322
+ const format = detectSpaceArchiveFormat({ filename: 'unknown', contents });
323
+ expect(format).toBe(SpaceArchive.Format.JSON);
324
+ });
325
+
326
+ test('falls back to BINARY on non-JSON bytes', () => {
327
+ const format = detectSpaceArchiveFormat({
328
+ filename: 'unknown',
329
+ contents: new Uint8Array([0x00, 0x01, 0x02]),
330
+ });
331
+ expect(format).toBe(SpaceArchive.Format.BINARY);
332
+ });
333
+ });
334
+
335
+ describe('SerializedSpace reader', () => {
336
+ test('parses a minimal JSON archive', () => {
337
+ const serialized: SerializedSpace = {
338
+ version: 1,
339
+ objects: [],
340
+ };
341
+ const contents = new TextEncoder().encode(JSON.stringify(serialized));
342
+ const archive: SpaceArchive = {
343
+ filename: 'test.dx.json',
344
+ contents,
345
+ format: SpaceArchive.Format.JSON,
346
+ };
347
+ const result = readSerializedSpaceArchive(archive);
348
+ expect(result.version).toBe(1);
349
+ expect(result.objects).toEqual([]);
350
+ });
351
+
352
+ test('rejects archives missing required fields', () => {
353
+ const bogus = new TextEncoder().encode(JSON.stringify({ objects: [] }));
354
+ expect(() =>
355
+ readSerializedSpaceArchive({
356
+ filename: 'bad.json',
357
+ contents: bogus,
358
+ format: SpaceArchive.Format.JSON,
359
+ }),
360
+ ).toThrow();
361
+ });
362
+
363
+ test('buildDatabaseDirectoryFromObjects round-trips data and type info', () => {
364
+ const id = EntityId.random();
365
+ const objects = [
366
+ {
367
+ id,
368
+ '@type': 'dxn:example.Thing',
369
+ '@meta': { keys: [] },
370
+ title: 'hello',
371
+ },
372
+ ];
373
+ const directory = buildDatabaseDirectoryFromObjects(objects as any);
374
+ expect(directory.objects).toBeDefined();
375
+ const structure = directory.objects![id];
376
+ expect(structure).toBeDefined();
377
+ expect(structure.data).toEqual({ title: 'hello' });
378
+ expect(structure.system?.type).toEqual({ '/': 'dxn:example.Thing' });
379
+ expect(structure.system?.kind).toBe('object');
380
+ });
381
+
382
+ test('objectStructureToObjJson emits fields in canonical order', () => {
383
+ const id = EntityId.random();
384
+ const sourceId = EntityId.random();
385
+ const targetId = EntityId.random();
386
+ const parentId = EntityId.random();
387
+ const obj = objectStructureToObjJson(id, {
388
+ data: { title: 'hello', count: 42 },
389
+ meta: { keys: [] },
390
+ system: {
391
+ type: { '/': URI.make('dxn:example.Link') },
392
+ kind: 'relation',
393
+ source: { '/': URI.make(sourceId) },
394
+ target: { '/': URI.make(targetId) },
395
+ parent: { '/': URI.make(parentId) },
396
+ deleted: true,
397
+ },
398
+ });
399
+
400
+ expect(Object.keys(obj)).toEqual([
401
+ 'id',
402
+ '@type',
403
+ '@meta',
404
+ '@deleted',
405
+ '@parent',
406
+ '@relationSource',
407
+ '@relationTarget',
408
+ 'title',
409
+ 'count',
410
+ ]);
411
+ });
412
+
413
+ test('orderObjJsonFields reorders feed queue messages with id/@type/@meta first', () => {
414
+ const id = EntityId.random();
415
+ const message = {
416
+ payload: { value: 'x' },
417
+ timestamp: 1000,
418
+ id,
419
+ '@meta': { keys: [] },
420
+ '@type': 'dxn:example.Message',
421
+ } as any;
422
+
423
+ const ordered = orderObjJsonFields(message);
424
+ expect(Object.keys(ordered)).toEqual(['id', '@type', '@meta', 'payload', 'timestamp']);
425
+ expect(ordered).toEqual(message);
426
+ });
427
+
428
+ test('orderObjJsonFields preserves unknown @-prefixed fields between system and data', () => {
429
+ const id = EntityId.random();
430
+ const obj = {
431
+ data: 1,
432
+ '@custom': 'extension',
433
+ '@type': 'dxn:example.Thing',
434
+ id,
435
+ } as any;
436
+
437
+ const ordered = orderObjJsonFields(obj);
438
+ expect(Object.keys(ordered)).toEqual(['id', '@type', '@custom', 'data']);
439
+ });
440
+
441
+ test('buildDatabaseDirectoryFromObjects flags relations', () => {
442
+ const id = EntityId.random();
443
+ const sourceId = EntityId.random();
444
+ const targetId = EntityId.random();
445
+ const objects = [
446
+ {
447
+ id,
448
+ '@type': 'dxn:example.Link',
449
+ '@meta': { keys: [] },
450
+ '@relationSource': sourceId,
451
+ '@relationTarget': targetId,
452
+ },
453
+ ];
454
+ const directory = buildDatabaseDirectoryFromObjects(objects as any);
455
+ const structure = directory.objects![id];
456
+ expect(structure.system?.kind).toBe('relation');
457
+ expect(structure.system?.source).toEqual({ '/': sourceId });
458
+ expect(structure.system?.target).toEqual({ '/': targetId });
459
+ });
460
+
461
+ test('buildDatabaseDirectoryFromObjects flags persisted Type.Type entities as kind=type', () => {
462
+ const id = EntityId.random();
463
+ const objects = [
464
+ {
465
+ id,
466
+ '@type': 'dxn:org.dxos.type.schema:0.1.0',
467
+ '@meta': { keys: [] },
468
+ name: 'Custom Type',
469
+ typename: 'example.type.custom',
470
+ version: '0.1.0',
471
+ jsonSchema: { $id: `echo:/${id}`, type: 'object', properties: {} },
472
+ },
473
+ ];
474
+ const directory = buildDatabaseDirectoryFromObjects(objects as any);
475
+ const structure = directory.objects![id];
476
+ expect(structure.system?.kind).toBe('type');
477
+ // Type entities aren't relations — source/target stay unset.
478
+ expect(structure.system?.source).toBeUndefined();
479
+ expect(structure.system?.target).toBeUndefined();
480
+ });
481
+ });
482
+ });