@dxos/echo-pipeline 0.7.2 → 0.7.3-staging.971cd8d

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 (40) hide show
  1. package/dist/lib/browser/{chunk-RRKGEIVZ.mjs → chunk-LZK5YFYE.mjs} +13 -13
  2. package/dist/lib/browser/{chunk-RRKGEIVZ.mjs.map → chunk-LZK5YFYE.mjs.map} +2 -2
  3. package/dist/lib/browser/index.mjs +42 -61
  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 +1 -1
  7. package/dist/lib/node/{chunk-6KAVD3GU.cjs → chunk-MACQJ2EP.cjs} +16 -16
  8. package/dist/lib/node/{chunk-6KAVD3GU.cjs.map → chunk-MACQJ2EP.cjs.map} +2 -2
  9. package/dist/lib/node/index.cjs +66 -87
  10. package/dist/lib/node/index.cjs.map +4 -4
  11. package/dist/lib/node/meta.json +1 -1
  12. package/dist/lib/node/testing/index.cjs +10 -10
  13. package/dist/lib/node-esm/{chunk-4QES5F4H.mjs → chunk-JIZPSASG.mjs} +13 -13
  14. package/dist/lib/node-esm/{chunk-4QES5F4H.mjs.map → chunk-JIZPSASG.mjs.map} +2 -2
  15. package/dist/lib/node-esm/index.mjs +42 -61
  16. package/dist/lib/node-esm/index.mjs.map +4 -4
  17. package/dist/lib/node-esm/meta.json +1 -1
  18. package/dist/lib/node-esm/testing/index.mjs +1 -1
  19. package/dist/types/src/automerge/automerge-host.d.ts +7 -1
  20. package/dist/types/src/automerge/automerge-host.d.ts.map +1 -1
  21. package/dist/types/src/db-host/echo-host.d.ts +3 -2
  22. package/dist/types/src/db-host/echo-host.d.ts.map +1 -1
  23. package/dist/types/src/db-host/index.d.ts +0 -1
  24. package/dist/types/src/db-host/index.d.ts.map +1 -1
  25. package/dist/types/src/edge/echo-edge-replicator.d.ts.map +1 -1
  26. package/dist/types/src/index.d.ts +1 -0
  27. package/dist/types/src/index.d.ts.map +1 -1
  28. package/dist/types/src/space/space.d.ts.map +1 -1
  29. package/dist/types/src/{db-host/migration.d.ts → util.d.ts} +1 -3
  30. package/dist/types/src/util.d.ts.map +1 -0
  31. package/package.json +34 -34
  32. package/src/automerge/automerge-host.ts +12 -3
  33. package/src/db-host/echo-host.ts +4 -1
  34. package/src/db-host/index.ts +0 -1
  35. package/src/edge/echo-edge-replicator.ts +3 -0
  36. package/src/index.ts +1 -0
  37. package/src/space/space.ts +1 -0
  38. package/src/util.ts +19 -0
  39. package/dist/types/src/db-host/migration.d.ts.map +0 -1
  40. package/src/db-host/migration.ts +0 -57
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
- "sources": ["../../../src/db-host/data-service.ts", "../../../src/db-host/documents-synchronizer.ts", "../../../src/automerge/automerge-host.ts", "../../../src/automerge/collection-synchronizer.ts", "../../../src/automerge/echo-network-adapter.ts", "../../../src/automerge/network-protocol.ts", "../../../src/automerge/heads-store.ts", "../../../src/automerge/leveldb-storage-adapter.ts", "../../../src/automerge/mesh-echo-replicator.ts", "../../../src/automerge/mesh-echo-replicator-connection.ts", "../../../src/automerge/space-collection.ts", "../../../src/automerge/echo-data-monitor.ts", "../../../src/db-host/echo-host.ts", "../../../src/db-host/documents-iterator.ts", "../../../src/db-host/query-service.ts", "../../../src/db-host/query-state.ts", "../../../src/db-host/space-state-manager.ts", "../../../src/db-host/database-root.ts", "../../../src/db-host/automerge-metrics.ts", "../../../src/db-host/migration.ts", "../../../src/edge/echo-edge-replicator.ts"],
4
- "sourcesContent": ["//\n// Copyright 2021 DXOS.org\n//\n\nimport { UpdateScheduler } from '@dxos/async';\nimport { type DocumentId } from '@dxos/automerge/automerge-repo';\nimport { type RequestOptions } from '@dxos/codec-protobuf';\nimport { Stream } from '@dxos/codec-protobuf/stream';\nimport { invariant } from '@dxos/invariant';\nimport { SpaceId } from '@dxos/keys';\nimport { log } from '@dxos/log';\nimport {\n type DataService,\n type FlushRequest,\n type SubscribeRequest,\n type BatchedDocumentUpdates,\n type UpdateSubscriptionRequest,\n type GetDocumentHeadsRequest,\n type GetDocumentHeadsResponse,\n type ReIndexHeadsRequest,\n type WaitUntilHeadsReplicatedRequest,\n type UpdateRequest,\n type GetSpaceSyncStateRequest,\n type SpaceSyncState,\n} from '@dxos/protocols/proto/dxos/echo/service';\n\nimport { DocumentsSynchronizer } from './documents-synchronizer';\nimport { deriveCollectionIdFromSpaceId, type AutomergeHost } from '../automerge';\n\nexport type DataServiceParams = {\n automergeHost: AutomergeHost;\n updateIndexes: () => Promise<void>;\n};\n\n/**\n * Data sync between client and services.\n */\n// TODO(burdon): Move to client-services.\nexport class DataServiceImpl implements DataService {\n /**\n * Map of subscriptions.\n * subscriptionId -> DocumentsSynchronizer\n */\n private readonly _subscriptions = new Map<string, DocumentsSynchronizer>();\n\n private readonly _automergeHost: AutomergeHost;\n private readonly _updateIndexes: () => Promise<void>;\n\n constructor(params: DataServiceParams) {\n this._automergeHost = params.automergeHost;\n this._updateIndexes = params.updateIndexes;\n }\n\n subscribe(request: SubscribeRequest): Stream<BatchedDocumentUpdates> {\n return new Stream<BatchedDocumentUpdates>(({ next, ready }) => {\n const synchronizer = new DocumentsSynchronizer({\n repo: this._automergeHost.repo,\n sendUpdates: (updates) => next(updates),\n });\n synchronizer\n .open()\n .then(() => {\n this._subscriptions.set(request.subscriptionId, synchronizer);\n ready();\n })\n .catch((err) => log.catch(err));\n return () => synchronizer.close();\n });\n }\n\n async updateSubscription(request: UpdateSubscriptionRequest) {\n const synchronizer = this._subscriptions.get(request.subscriptionId);\n invariant(synchronizer, 'Subscription not found');\n\n if (request.addIds?.length) {\n await synchronizer.addDocuments(request.addIds as DocumentId[]);\n }\n if (request.removeIds?.length) {\n await synchronizer.removeDocuments(request.removeIds as DocumentId[]);\n }\n }\n\n async update(request: UpdateRequest): Promise<void> {\n if (!request.updates) {\n return;\n }\n const synchronizer = this._subscriptions.get(request.subscriptionId);\n invariant(synchronizer, 'Subscription not found');\n\n synchronizer.update(request.updates);\n }\n\n async flush(request: FlushRequest): Promise<void> {\n await this._automergeHost.flush(request);\n }\n\n async getDocumentHeads(request: GetDocumentHeadsRequest): Promise<GetDocumentHeadsResponse> {\n const documentIds = request.documentIds;\n if (!documentIds) {\n return { heads: { entries: [] } };\n }\n const heads = await this._automergeHost.getHeads(documentIds as DocumentId[]);\n return {\n heads: {\n entries: heads.map((heads, idx) => ({ documentId: documentIds[idx], heads })),\n },\n };\n }\n\n async waitUntilHeadsReplicated(\n request: WaitUntilHeadsReplicatedRequest,\n options?: RequestOptions | undefined,\n ): Promise<void> {\n await this._automergeHost.waitUntilHeadsReplicated(request.heads);\n }\n\n async reIndexHeads(request: ReIndexHeadsRequest, options?: RequestOptions): Promise<void> {\n await this._automergeHost.reIndexHeads((request.documentIds ?? []) as DocumentId[]);\n }\n\n async updateIndexes() {\n await this._updateIndexes();\n }\n\n subscribeSpaceSyncState(request: GetSpaceSyncStateRequest): Stream<SpaceSyncState> {\n return new Stream<SpaceSyncState>(({ ctx, next, ready }) => {\n invariant(SpaceId.isValid(request.spaceId));\n const collectionId = deriveCollectionIdFromSpaceId(request.spaceId);\n\n const scheduler = new UpdateScheduler(ctx, async () => {\n const state = await this._automergeHost.getCollectionSyncState(collectionId);\n\n next({\n peers: state.peers.map((peer) => ({\n peerId: peer.peerId,\n missingOnRemote: peer.missingOnRemote,\n missingOnLocal: peer.missingOnLocal,\n differentDocuments: peer.differentDocuments,\n localDocumentCount: peer.localDocumentCount,\n remoteDocumentCount: peer.remoteDocumentCount,\n })),\n });\n });\n\n this._automergeHost.collectionStateUpdated.on(ctx, (e) => {\n if (e.collectionId === collectionId) {\n scheduler.trigger();\n }\n });\n scheduler.trigger();\n });\n }\n}\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport { UpdateScheduler } from '@dxos/async';\nimport { next as A, type Heads } from '@dxos/automerge/automerge';\nimport { type Repo, type DocHandle, type DocumentId } from '@dxos/automerge/automerge-repo';\nimport { Resource } from '@dxos/context';\nimport { type SpaceDoc } from '@dxos/echo-protocol';\nimport { invariant } from '@dxos/invariant';\nimport { log } from '@dxos/log';\nimport { type BatchedDocumentUpdates, type DocumentUpdate } from '@dxos/protocols/proto/dxos/echo/service';\n\nconst MAX_UPDATE_FREQ = 10; // [updates/sec]\n\nexport type DocumentsSynchronizerParams = {\n repo: Repo;\n sendUpdates: (updates: BatchedDocumentUpdates) => void;\n};\n\ninterface DocSyncState {\n handle: DocHandle<SpaceDoc>;\n lastSentHead?: Heads;\n clearSubscriptions?: () => void;\n}\n\n/**\n * Manages a connection and replication between worker's Automerge Repo and the client's Repo.\n */\nexport class DocumentsSynchronizer extends Resource {\n private readonly _syncStates = new Map<DocumentId, DocSyncState>();\n /**\n * Documents that have pending updates.\n * Used to batch updates.\n */\n private readonly _pendingUpdates = new Set<DocumentId>();\n\n /**\n * Job that schedules if there are pending updates.\n */\n private _sendUpdatesJob?: UpdateScheduler = undefined;\n\n constructor(private readonly _params: DocumentsSynchronizerParams) {\n super();\n }\n\n addDocuments(documentIds: DocumentId[], retryCounter = 0) {\n if (retryCounter > 3) {\n log.warn('Failed to load document, retry limit reached', { documentIds });\n return;\n }\n\n for (const documentId of documentIds) {\n const doc = this._params.repo.find(documentId as DocumentId);\n doc\n .whenReady()\n .then(() => {\n this._startSync(doc);\n this._pendingUpdates.add(doc.documentId);\n this._sendUpdatesJob!.trigger();\n })\n .catch((error) => {\n log.warn('Failed to load document, wraparound', { documentId, error });\n this.addDocuments([documentId], retryCounter + 1);\n });\n }\n }\n\n removeDocuments(documentIds: DocumentId[]) {\n for (const documentId of documentIds) {\n this._syncStates.get(documentId)?.clearSubscriptions?.();\n this._syncStates.delete(documentId);\n this._pendingUpdates.delete(documentId);\n }\n }\n\n protected override async _open(): Promise<void> {\n this._sendUpdatesJob = new UpdateScheduler(this._ctx, this._checkAndSendUpdates.bind(this), {\n maxFrequency: MAX_UPDATE_FREQ,\n });\n }\n\n protected override async _close(): Promise<void> {\n await this._sendUpdatesJob!.join();\n this._syncStates.clear();\n }\n\n update(updates: DocumentUpdate[]) {\n for (const { documentId, mutation, isNew } of updates) {\n if (isNew) {\n const doc = this._params.repo.find(documentId as DocumentId);\n doc.update((doc) => A.loadIncremental(doc, mutation));\n this._startSync(doc);\n } else {\n this._writeMutation(documentId as DocumentId, mutation);\n }\n }\n }\n\n private _startSync(doc: DocHandle<SpaceDoc>) {\n if (this._syncStates.has(doc.documentId)) {\n log.info('Document already being synced', { documentId: doc.documentId });\n return;\n }\n\n const syncState: DocSyncState = { handle: doc };\n this._subscribeForChanges(syncState);\n this._syncStates.set(doc.documentId, syncState);\n }\n\n _subscribeForChanges(syncState: DocSyncState) {\n const handler = () => {\n this._pendingUpdates.add(syncState.handle.documentId);\n this._sendUpdatesJob!.trigger();\n };\n syncState.handle.on('heads-changed', handler);\n syncState.clearSubscriptions = () => syncState.handle.off('heads-changed', handler);\n }\n\n private async _checkAndSendUpdates() {\n const updates: DocumentUpdate[] = [];\n\n const docsWithPendingUpdates = Array.from(this._pendingUpdates);\n this._pendingUpdates.clear();\n\n for (const documentId of docsWithPendingUpdates) {\n const update = this._getPendingChanges(documentId);\n if (update) {\n updates.push({\n documentId,\n mutation: update,\n });\n }\n }\n\n if (updates.length > 0) {\n this._params.sendUpdates({ updates });\n }\n }\n\n private _getPendingChanges(documentId: DocumentId): Uint8Array | void {\n const syncState = this._syncStates.get(documentId);\n invariant(syncState, 'Sync state for document not found');\n const doc = syncState.handle.docSync();\n if (!doc) {\n return;\n }\n const mutation = syncState.lastSentHead ? A.saveSince(doc, syncState.lastSentHead) : A.save(doc);\n if (mutation.length === 0) {\n return;\n }\n syncState.lastSentHead = A.getHeads(doc);\n return mutation;\n }\n\n private _writeMutation(documentId: DocumentId, mutation: Uint8Array) {\n const syncState = this._syncStates.get(documentId);\n invariant(syncState, 'Sync state for document not found');\n syncState.handle.update((doc) => {\n const headsBefore = A.getHeads(doc);\n const newDoc = A.loadIncremental(doc, mutation);\n if (A.equals(headsBefore, syncState.lastSentHead)) {\n syncState.lastSentHead = A.getHeads(newDoc);\n }\n return newDoc;\n });\n }\n}\n", "//\n// Copyright 2023 DXOS.org\n//\n\nimport { Event, asyncTimeout } from '@dxos/async';\nimport {\n getBackend,\n getHeads,\n isAutomerge,\n equals as headsEquals,\n save,\n type Doc,\n type Heads,\n} from '@dxos/automerge/automerge';\nimport {\n type DocHandleChangePayload,\n Repo,\n type AnyDocumentId,\n type DocHandle,\n type DocumentId,\n type PeerCandidatePayload,\n type PeerDisconnectedPayload,\n type PeerId,\n type StorageAdapterInterface,\n type StorageKey,\n} from '@dxos/automerge/automerge-repo';\nimport { Context, Resource, cancelWithContext, type Lifecycle } from '@dxos/context';\nimport { type CollectionId, type SpaceDoc } from '@dxos/echo-protocol';\nimport { type IndexMetadataStore } from '@dxos/indexing';\nimport { invariant } from '@dxos/invariant';\nimport { PublicKey } from '@dxos/keys';\nimport { type LevelDB } from '@dxos/kv-store';\nimport { log } from '@dxos/log';\nimport { objectPointerCodec } from '@dxos/protocols';\nimport { type DocHeadsList, type FlushRequest } from '@dxos/protocols/proto/dxos/echo/service';\nimport { trace } from '@dxos/tracing';\n\nimport { CollectionSynchronizer, diffCollectionState, type CollectionState } from './collection-synchronizer';\nimport { type EchoDataMonitor } from './echo-data-monitor';\nimport { EchoNetworkAdapter, isEchoPeerMetadata } from './echo-network-adapter';\nimport { type EchoReplicator, type RemoteDocumentExistenceCheckParams } from './echo-replicator';\nimport { HeadsStore } from './heads-store';\nimport { LevelDBStorageAdapter, type BeforeSaveParams } from './leveldb-storage-adapter';\n\nexport type AutomergeHostParams = {\n db: LevelDB;\n\n indexMetadataStore: IndexMetadataStore;\n dataMonitor?: EchoDataMonitor;\n};\n\nexport type LoadDocOptions = {\n timeout?: number;\n};\n\nexport type CreateDocOptions = {\n /**\n * Import the document together with its history.\n */\n preserveHistory?: boolean;\n};\n\n/**\n * Abstracts over the AutomergeRepo.\n */\n@trace.resource()\nexport class AutomergeHost extends Resource {\n private readonly _db: LevelDB;\n private readonly _indexMetadataStore: IndexMetadataStore;\n private readonly _echoNetworkAdapter: EchoNetworkAdapter;\n\n private readonly _collectionSynchronizer = new CollectionSynchronizer({\n queryCollectionState: this._queryCollectionState.bind(this),\n sendCollectionState: this._sendCollectionState.bind(this),\n shouldSyncCollection: this._shouldSyncCollection.bind(this),\n });\n\n private _repo!: Repo;\n private _storage!: StorageAdapterInterface & Lifecycle;\n private readonly _headsStore: HeadsStore;\n\n @trace.info()\n private _peerId!: PeerId;\n\n public readonly collectionStateUpdated = new Event<{ collectionId: CollectionId }>();\n\n constructor({ db, indexMetadataStore, dataMonitor }: AutomergeHostParams) {\n super();\n this._db = db;\n this._storage = new LevelDBStorageAdapter({\n db: db.sublevel('automerge'),\n callbacks: {\n beforeSave: async (params) => this._beforeSave(params),\n afterSave: async (key) => this._afterSave(key),\n },\n monitor: dataMonitor,\n });\n this._echoNetworkAdapter = new EchoNetworkAdapter({\n getContainingSpaceForDocument: this._getContainingSpaceForDocument.bind(this),\n isDocumentInRemoteCollection: this._isDocumentInRemoteCollection.bind(this),\n onCollectionStateQueried: this._onCollectionStateQueried.bind(this),\n onCollectionStateReceived: this._onCollectionStateReceived.bind(this),\n monitor: dataMonitor,\n });\n this._headsStore = new HeadsStore({ db: db.sublevel('heads') });\n this._indexMetadataStore = indexMetadataStore;\n }\n\n protected override async _open() {\n // TODO(burdon): Should this be stable?\n this._peerId = `host-${PublicKey.random().toHex()}` as PeerId;\n\n await this._storage.open?.();\n\n // Construct the automerge repo.\n this._repo = new Repo({\n peerId: this._peerId as PeerId,\n sharePolicy: this._sharePolicy.bind(this),\n storage: this._storage,\n network: [\n // Upstream swarm.\n this._echoNetworkAdapter,\n ],\n });\n\n Event.wrap(this._echoNetworkAdapter, 'peer-candidate').on(this._ctx, ((e: PeerCandidatePayload) =>\n this._onPeerConnected(e.peerId)) as any);\n Event.wrap(this._echoNetworkAdapter, 'peer-disconnected').on(this._ctx, ((e: PeerDisconnectedPayload) =>\n this._onPeerDisconnected(e.peerId)) as any);\n\n this._collectionSynchronizer.remoteStateUpdated.on(this._ctx, ({ collectionId, peerId }) => {\n this._onRemoteCollectionStateUpdated(collectionId, peerId);\n this.collectionStateUpdated.emit({ collectionId: collectionId as CollectionId });\n });\n\n await this._echoNetworkAdapter.open();\n await this._collectionSynchronizer.open();\n await this._echoNetworkAdapter.open();\n await this._echoNetworkAdapter.whenConnected();\n }\n\n protected override async _close() {\n await this._collectionSynchronizer.close();\n await this._storage.close?.();\n await this._echoNetworkAdapter.close();\n await this._ctx.dispose();\n }\n\n /**\n * @deprecated To be abstracted away.\n */\n get repo(): Repo {\n return this._repo;\n }\n\n get peerId(): PeerId {\n return this._peerId;\n }\n\n get loadedDocsCount(): number {\n return Object.keys(this._repo.handles).length;\n }\n\n async addReplicator(replicator: EchoReplicator) {\n await this._echoNetworkAdapter.addReplicator(replicator);\n }\n\n async removeReplicator(replicator: EchoReplicator) {\n await this._echoNetworkAdapter.removeReplicator(replicator);\n }\n\n /**\n * Loads the document handle from the repo and waits for it to be ready.\n */\n async loadDoc<T>(ctx: Context, documentId: AnyDocumentId, opts?: LoadDocOptions): Promise<DocHandle<T>> {\n let handle: DocHandle<T> | undefined;\n if (typeof documentId === 'string') {\n // NOTE: documentId might also be a URL, in which case this lookup will fail.\n handle = this._repo.handles[documentId as DocumentId];\n }\n if (!handle) {\n handle = this._repo.find(documentId as DocumentId);\n }\n\n // `whenReady` creates a timeout so we guard it with an if to skip it if the handle is already ready.\n if (!handle.isReady()) {\n if (!opts?.timeout) {\n await cancelWithContext(ctx, handle.whenReady());\n } else {\n await cancelWithContext(ctx, asyncTimeout(handle.whenReady(), opts.timeout));\n }\n }\n\n return handle;\n }\n\n /**\n * Create new persisted document.\n */\n createDoc<T>(initialValue?: T | Doc<T>, opts?: CreateDocOptions): DocHandle<T> {\n if (opts?.preserveHistory) {\n if (!isAutomerge(initialValue)) {\n throw new TypeError('Initial value must be an Automerge document');\n }\n // TODO(dmaretskyi): There's a more efficient way.\n return this._repo.import(save(initialValue as Doc<T>));\n } else {\n return this._repo.create(initialValue);\n }\n }\n\n async waitUntilHeadsReplicated(heads: DocHeadsList): Promise<void> {\n const entries = heads.entries;\n if (!entries?.length) {\n return;\n }\n const documentIds = entries.map((entry) => entry.documentId as DocumentId);\n const documentHeads = await this.getHeads(documentIds);\n const headsToWait = entries.filter((entry, index) => {\n const targetHeads = entry.heads;\n if (!targetHeads || targetHeads.length === 0) {\n return false;\n }\n const currentHeads = documentHeads[index];\n return !(currentHeads !== null && headsEquals(currentHeads, targetHeads));\n });\n if (headsToWait.length > 0) {\n await Promise.all(\n headsToWait.map(async (entry, index) => {\n const handle = await this.loadDoc(Context.default(), entry.documentId as DocumentId);\n await waitForHeads(handle, entry.heads!);\n }),\n );\n }\n\n // Flush to disk handles loaded to memory also so that the indexer can pick up the changes.\n await this._repo.flush(documentIds.filter((documentId) => !!this._repo.handles[documentId]));\n }\n\n async reIndexHeads(documentIds: DocumentId[]) {\n for (const documentId of documentIds) {\n log.info('re-indexing heads for document', { documentId });\n const handle = this._repo.find(documentId);\n await handle.whenReady(['ready', 'requesting']);\n if (handle.inState(['requesting'])) {\n log.warn('document is not available locally, skipping', { documentId });\n continue; // Handle not available locally.\n }\n\n const doc = handle.docSync();\n invariant(doc);\n\n const heads = getHeads(doc);\n const batch = this._db.batch();\n this._headsStore.setHeads(documentId, heads, batch);\n await batch.write();\n }\n log.info('done re-indexing heads');\n }\n\n // TODO(dmaretskyi): Share based on HALO permissions and space affinity.\n // Hosts, running in the worker, don't share documents unless requested by other peers.\n // NOTE: If both peers return sharePolicy=false the replication will not happen\n // https://github.com/automerge/automerge-repo/pull/292\n private async _sharePolicy(peerId: PeerId, documentId?: DocumentId): Promise<boolean> {\n if (peerId.startsWith('client-')) {\n return false; // Only send docs to clients if they are requested.\n }\n\n if (!documentId) {\n return false;\n }\n\n const peerMetadata = this.repo.peerMetadataByPeerId[peerId];\n if (isEchoPeerMetadata(peerMetadata)) {\n return this._echoNetworkAdapter.shouldAdvertise(peerId, { documentId });\n }\n\n return false;\n }\n\n private async _beforeSave({ path, batch }: BeforeSaveParams) {\n const handle = this._repo.handles[path[0] as DocumentId];\n if (!handle) {\n return;\n }\n const doc = handle.docSync();\n if (!doc) {\n return;\n }\n\n const heads = getHeads(doc);\n this._headsStore.setHeads(handle.documentId, heads, batch);\n\n const spaceKey = getSpaceKeyFromDoc(doc) ?? undefined;\n const objectIds = Object.keys(doc.objects ?? {});\n const encodedIds = objectIds.map((objectId) =>\n objectPointerCodec.encode({ documentId: handle.documentId, objectId, spaceKey }),\n );\n const idToLastHash = new Map(encodedIds.map((id) => [id, heads]));\n this._indexMetadataStore.markDirty(idToLastHash, batch);\n }\n\n private _shouldSyncCollection(collectionId: string, peerId: PeerId): boolean {\n const peerMetadata = this._repo.peerMetadataByPeerId[peerId];\n if (isEchoPeerMetadata(peerMetadata)) {\n return this._echoNetworkAdapter.shouldSyncCollection(peerId, { collectionId });\n }\n\n return false;\n }\n\n /**\n * Called by AutomergeStorageAdapter after levelDB batch commit.\n */\n private async _afterSave(path: StorageKey) {\n this._indexMetadataStore.notifyMarkedDirty();\n\n const documentId = path[0] as DocumentId;\n const document = this._repo.handles[documentId]?.docSync();\n if (document) {\n const heads = getHeads(document);\n this._onHeadsChanged(documentId, heads);\n }\n }\n\n @trace.info({ depth: null })\n private _automergePeers() {\n return this._repo.peers;\n }\n\n private async _isDocumentInRemoteCollection(params: RemoteDocumentExistenceCheckParams): Promise<boolean> {\n for (const collectionId of this._collectionSynchronizer.getRegisteredCollectionIds()) {\n const remoteCollections = this._collectionSynchronizer.getRemoteCollectionStates(collectionId);\n const remotePeerDocs = remoteCollections.get(params.peerId as PeerId)?.documents;\n if (remotePeerDocs && params.documentId in remotePeerDocs) {\n return true;\n }\n }\n return false;\n }\n\n private async _getContainingSpaceForDocument(documentId: string): Promise<PublicKey | null> {\n const doc = this._repo.handles[documentId as any]?.docSync();\n if (!doc) {\n return null;\n }\n\n const spaceKeyHex = getSpaceKeyFromDoc(doc);\n if (!spaceKeyHex) {\n return null;\n }\n\n return PublicKey.from(spaceKeyHex);\n }\n\n /**\n * Flush documents to disk.\n */\n @trace.span({ showInBrowserTimeline: true })\n async flush({ documentIds }: FlushRequest = {}): Promise<void> {\n // Note: Sync protocol for client and services ensures that all handles should have all changes.\n\n const loadedDocuments = documentIds?.filter(\n (documentId): documentId is DocumentId => !!this._repo.handles[documentId as DocumentId],\n );\n await this._repo.flush(loadedDocuments);\n }\n\n async getHeads(documentIds: DocumentId[]): Promise<(Heads | undefined)[]> {\n const result: (Heads | undefined)[] = [];\n const storeRequestIds: DocumentId[] = [];\n const storeResultIndices: number[] = [];\n for (const documentId of documentIds) {\n const doc = this._repo.handles[documentId]?.docSync();\n if (doc) {\n result.push(getHeads(doc));\n } else {\n storeRequestIds.push(documentId);\n storeResultIndices.push(result.length);\n result.push(undefined);\n }\n }\n if (storeRequestIds.length > 0) {\n const storedHeads = await this._headsStore.getHeads(storeRequestIds);\n for (let i = 0; i < storedHeads.length; i++) {\n result[storeResultIndices[i]] = storedHeads[i];\n }\n }\n return result;\n }\n\n //\n // Collection sync.\n //\n\n getLocalCollectionState(collectionId: string): CollectionState | undefined {\n return this._collectionSynchronizer.getLocalCollectionState(collectionId);\n }\n\n getRemoteCollectionStates(collectionId: string): ReadonlyMap<PeerId, CollectionState> {\n return this._collectionSynchronizer.getRemoteCollectionStates(collectionId);\n }\n\n refreshCollection(collectionId: string) {\n this._collectionSynchronizer.refreshCollection(collectionId);\n }\n\n async getCollectionSyncState(collectionId: string): Promise<CollectionSyncState> {\n const result: CollectionSyncState = {\n peers: [],\n };\n\n const localState = this.getLocalCollectionState(collectionId);\n const remoteState = this.getRemoteCollectionStates(collectionId);\n\n if (!localState) {\n return result;\n }\n\n for (const [peerId, state] of remoteState) {\n const diff = diffCollectionState(localState, state);\n result.peers.push({\n peerId,\n missingOnRemote: diff.missingOnRemote.length,\n missingOnLocal: diff.missingOnLocal.length,\n differentDocuments: diff.different.length,\n localDocumentCount: Object.keys(localState.documents).length,\n remoteDocumentCount: Object.keys(state.documents).length,\n });\n }\n\n return result;\n }\n\n /**\n * Update the local collection state based on the locally stored document heads.\n */\n async updateLocalCollectionState(collectionId: string, documentIds: DocumentId[]) {\n const heads = await this.getHeads(documentIds);\n const documents: Record<DocumentId, Heads> = Object.fromEntries(\n heads.map((heads, index) => [documentIds[index], heads ?? []]),\n );\n this._collectionSynchronizer.setLocalCollectionState(collectionId, { documents });\n }\n\n private _onCollectionStateQueried(collectionId: string, peerId: PeerId) {\n this._collectionSynchronizer.onCollectionStateQueried(collectionId, peerId);\n }\n\n private _onCollectionStateReceived(collectionId: string, peerId: PeerId, state: unknown) {\n this._collectionSynchronizer.onRemoteStateReceived(collectionId, peerId, decodeCollectionState(state));\n }\n\n private _queryCollectionState(collectionId: string, peerId: PeerId) {\n this._echoNetworkAdapter.queryCollectionState(collectionId, peerId);\n }\n\n private _sendCollectionState(collectionId: string, peerId: PeerId, state: CollectionState) {\n this._echoNetworkAdapter.sendCollectionState(collectionId, peerId, encodeCollectionState(state));\n }\n\n private _onPeerConnected(peerId: PeerId) {\n this._collectionSynchronizer.onConnectionOpen(peerId);\n }\n\n private _onPeerDisconnected(peerId: PeerId) {\n this._collectionSynchronizer.onConnectionClosed(peerId);\n }\n\n private _onRemoteCollectionStateUpdated(collectionId: string, peerId: PeerId) {\n const localState = this._collectionSynchronizer.getLocalCollectionState(collectionId);\n const remoteState = this._collectionSynchronizer.getRemoteCollectionStates(collectionId).get(peerId);\n\n if (!localState || !remoteState) {\n return;\n }\n\n const { different, missingOnLocal, missingOnRemote } = diffCollectionState(localState, remoteState);\n const toReplicate = [...missingOnLocal, ...missingOnRemote, ...different];\n\n if (toReplicate.length === 0) {\n return;\n }\n\n log.info('replication documents after collection sync', {\n count: toReplicate.length,\n });\n\n // Load the documents so they will start syncing.\n for (const documentId of toReplicate) {\n this._repo.find(documentId);\n }\n }\n\n private _onHeadsChanged(documentId: DocumentId, heads: Heads) {\n const collectionsChanged = new Set<CollectionId>();\n for (const collectionId of this._collectionSynchronizer.getRegisteredCollectionIds()) {\n const state = this._collectionSynchronizer.getLocalCollectionState(collectionId);\n if (state?.documents[documentId]) {\n const newState = structuredClone(state);\n newState.documents[documentId] = heads;\n this._collectionSynchronizer.setLocalCollectionState(collectionId, newState);\n collectionsChanged.add(collectionId as CollectionId);\n }\n }\n for (const collectionId of collectionsChanged) {\n this.collectionStateUpdated.emit({ collectionId });\n }\n }\n}\n\nexport const getSpaceKeyFromDoc = (doc: Doc<SpaceDoc>): string | null => {\n // experimental_spaceKey is set on old documents, new ones are created with doc.access.spaceKey\n const rawSpaceKey = doc.access?.spaceKey ?? (doc as any).experimental_spaceKey;\n if (rawSpaceKey == null) {\n return null;\n }\n\n return String(rawSpaceKey);\n};\n\nconst waitForHeads = async (handle: DocHandle<SpaceDoc>, heads: Heads) => {\n const unavailableHeads = new Set(heads);\n\n await handle.whenReady();\n await Event.wrap<DocHandleChangePayload<SpaceDoc>>(handle, 'change').waitForCondition(() => {\n // Check if unavailable heads became available.\n for (const changeHash of unavailableHeads.values()) {\n if (changeIsPresentInDoc(handle.docSync(), changeHash)) {\n unavailableHeads.delete(changeHash);\n }\n }\n\n return unavailableHeads.size === 0;\n });\n};\n\nconst changeIsPresentInDoc = (doc: Doc<any>, changeHash: string): boolean => {\n return !!getBackend(doc).getChangeByHash(changeHash);\n};\n\nconst decodeCollectionState = (state: unknown): CollectionState => {\n invariant(typeof state === 'object' && state !== null, 'Invalid state');\n\n return state as CollectionState;\n};\n\nconst encodeCollectionState = (state: CollectionState): unknown => {\n return state;\n};\n\nexport type CollectionSyncState = {\n peers: PeerSyncState[];\n};\n\nexport type PeerSyncState = {\n peerId: PeerId;\n /**\n * Documents that are present locally but not on the remote peer.\n */\n missingOnRemote: number;\n\n /**\n * Documents that are present on the remote peer but not locally.\n */\n missingOnLocal: number;\n\n /**\n * Documents that are present on both peers but have different heads.\n */\n differentDocuments: number;\n\n /**\n * Total number of documents locally.\n */\n localDocumentCount: number;\n\n /**\n * Total number of documents on the remote peer.\n */\n remoteDocumentCount: number;\n};\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport { asyncReturn, Event, scheduleTask, scheduleTaskInterval } from '@dxos/async';\nimport { next as am } from '@dxos/automerge/automerge';\nimport type { DocumentId, PeerId } from '@dxos/automerge/automerge-repo';\nimport { Resource, type Context } from '@dxos/context';\nimport { log } from '@dxos/log';\nimport { defaultMap } from '@dxos/util';\n\nconst MIN_QUERY_INTERVAL = 5_000;\n\nconst POLL_INTERVAL = 30_000;\n\nexport type CollectionSynchronizerParams = {\n sendCollectionState: (collectionId: string, peerId: PeerId, state: CollectionState) => void;\n queryCollectionState: (collectionId: string, peerId: PeerId) => void;\n shouldSyncCollection: (collectionId: string, peerId: PeerId) => boolean;\n};\n\n/**\n * Implements collection sync protocol.\n */\nexport class CollectionSynchronizer extends Resource {\n private readonly _sendCollectionState: CollectionSynchronizerParams['sendCollectionState'];\n private readonly _queryCollectionState: CollectionSynchronizerParams['queryCollectionState'];\n private readonly _shouldSyncCollection: CollectionSynchronizerParams['shouldSyncCollection'];\n\n /**\n * CollectionId -> State.\n */\n private readonly _perCollectionStates = new Map<string, PerCollectionState>();\n\n private readonly _connectedPeers = new Set<PeerId>();\n\n public readonly remoteStateUpdated = new Event<{ collectionId: string; peerId: PeerId }>();\n\n constructor(params: CollectionSynchronizerParams) {\n super();\n this._sendCollectionState = params.sendCollectionState;\n this._queryCollectionState = params.queryCollectionState;\n this._shouldSyncCollection = params.shouldSyncCollection;\n }\n\n protected override async _open(ctx: Context): Promise<void> {\n scheduleTaskInterval(\n this._ctx,\n async () => {\n for (const collectionId of this._perCollectionStates.keys()) {\n this.refreshCollection(collectionId);\n await asyncReturn();\n }\n },\n POLL_INTERVAL,\n );\n }\n\n getRegisteredCollectionIds(): string[] {\n return [...this._perCollectionStates.keys()];\n }\n\n getLocalCollectionState(collectionId: string): CollectionState | undefined {\n return this._getPerCollectionState(collectionId).localState;\n }\n\n setLocalCollectionState(collectionId: string, state: CollectionState) {\n log('setLocalCollectionState', { collectionId, state });\n this._getPerCollectionState(collectionId).localState = state;\n\n queueMicrotask(async () => {\n if (!this._ctx.disposed) {\n this._refreshInterestedPeers(collectionId);\n this.refreshCollection(collectionId);\n }\n });\n }\n\n getRemoteCollectionStates(collectionId: string): ReadonlyMap<PeerId, CollectionState> {\n return this._getPerCollectionState(collectionId).remoteStates;\n }\n\n refreshCollection(collectionId: string) {\n let scheduleAnotherRefresh = false;\n const state = this._getPerCollectionState(collectionId);\n for (const peerId of this._connectedPeers) {\n if (state.interestedPeers.has(peerId)) {\n const lastQueried = state.lastQueried.get(peerId) ?? 0;\n if (Date.now() - lastQueried > MIN_QUERY_INTERVAL) {\n state.lastQueried.set(peerId, Date.now());\n this._queryCollectionState(collectionId, peerId);\n } else {\n scheduleAnotherRefresh = true;\n }\n }\n }\n if (scheduleAnotherRefresh) {\n scheduleTask(this._ctx, () => this.refreshCollection(collectionId), MIN_QUERY_INTERVAL);\n }\n }\n\n /**\n * Callback when a connection to a peer is established.\n */\n onConnectionOpen(peerId: PeerId) {\n this._connectedPeers.add(peerId);\n\n queueMicrotask(async () => {\n if (this._ctx.disposed) {\n return;\n }\n for (const [collectionId, state] of this._perCollectionStates.entries()) {\n if (this._shouldSyncCollection(collectionId, peerId)) {\n state.interestedPeers.add(peerId);\n state.lastQueried.set(peerId, Date.now());\n this._queryCollectionState(collectionId, peerId);\n }\n }\n });\n }\n\n /**\n * Callback when a connection to a peer is closed.\n */\n onConnectionClosed(peerId: PeerId) {\n this._connectedPeers.delete(peerId);\n\n for (const perCollectionState of this._perCollectionStates.values()) {\n perCollectionState.remoteStates.delete(peerId);\n }\n }\n\n /**\n * Callback when a peer queries the state of a collection.\n */\n onCollectionStateQueried(collectionId: string, peerId: PeerId) {\n const perCollectionState = this._getPerCollectionState(collectionId);\n\n if (perCollectionState.localState) {\n this._sendCollectionState(collectionId, peerId, perCollectionState.localState);\n }\n }\n\n /**\n * Callback when a peer sends the state of a collection.\n */\n onRemoteStateReceived(collectionId: string, peerId: PeerId, state: CollectionState) {\n log('onRemoteStateReceived', { collectionId, peerId, state });\n validateCollectionState(state);\n const perCollectionState = this._getPerCollectionState(collectionId);\n perCollectionState.remoteStates.set(peerId, state);\n this.remoteStateUpdated.emit({ peerId, collectionId });\n }\n\n private _getPerCollectionState(collectionId: string): PerCollectionState {\n return defaultMap(this._perCollectionStates, collectionId, () => ({\n localState: undefined,\n remoteStates: new Map(),\n interestedPeers: new Set(),\n lastQueried: new Map(),\n }));\n }\n\n private _refreshInterestedPeers(collectionId: string) {\n for (const peerId of this._connectedPeers) {\n if (this._shouldSyncCollection(collectionId, peerId)) {\n this._getPerCollectionState(collectionId).interestedPeers.add(peerId);\n } else {\n this._getPerCollectionState(collectionId).interestedPeers.delete(peerId);\n }\n }\n }\n}\n\ntype PerCollectionState = {\n localState?: CollectionState;\n remoteStates: Map<PeerId, CollectionState>;\n interestedPeers: Set<PeerId>;\n lastQueried: Map<PeerId, number>;\n};\n\nexport type CollectionState = {\n /**\n * DocumentId -> Heads.\n */\n documents: Record<string, string[]>;\n};\n\nexport type CollectionStateDiff = {\n missingOnRemote: DocumentId[];\n missingOnLocal: DocumentId[];\n different: DocumentId[];\n};\n\nexport const diffCollectionState = (local: CollectionState, remote: CollectionState): CollectionStateDiff => {\n const allDocuments = new Set<DocumentId>([...Object.keys(local.documents), ...Object.keys(remote.documents)] as any);\n\n const missingOnRemote: DocumentId[] = [];\n const missingOnLocal: DocumentId[] = [];\n const different: DocumentId[] = [];\n for (const documentId of allDocuments) {\n if (!local.documents[documentId]) {\n missingOnLocal.push(documentId as DocumentId);\n } else if (!remote.documents[documentId]) {\n missingOnRemote.push(documentId as DocumentId);\n } else if (!am.equals(local.documents[documentId], remote.documents[documentId])) {\n different.push(documentId as DocumentId);\n }\n }\n\n return {\n missingOnRemote,\n missingOnLocal,\n different,\n };\n};\n\nconst validateCollectionState = (state: CollectionState) => {\n Object.entries(state.documents).forEach(([documentId, heads]) => {\n if (!isValidDocumentId(documentId as DocumentId)) {\n throw new Error(`Invalid documentId: ${documentId}`);\n }\n if (Array.isArray(heads) && heads.some((head) => typeof head !== 'string')) {\n throw new Error(`Invalid heads: ${heads}`);\n }\n });\n};\n\nconst isValidDocumentId = (documentId: DocumentId) => {\n return typeof documentId === 'string' && !documentId.includes(':');\n};\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport { synchronized, Trigger } from '@dxos/async';\nimport { NetworkAdapter, type Message, type PeerId, type PeerMetadata } from '@dxos/automerge/automerge-repo';\nimport { LifecycleState } from '@dxos/context';\nimport { invariant } from '@dxos/invariant';\nimport { type PublicKey } from '@dxos/keys';\nimport { log } from '@dxos/log';\nimport type { AutomergeProtocolMessage } from '@dxos/protocols';\nimport { nonNullable } from '@dxos/util';\n\nimport {\n type EchoReplicator,\n type RemoteDocumentExistenceCheckParams,\n type ReplicatorConnection,\n type ShouldAdvertiseParams,\n type ShouldSyncCollectionParams,\n} from './echo-replicator';\nimport {\n isCollectionQueryMessage,\n isCollectionStateMessage,\n type CollectionQueryMessage,\n type CollectionStateMessage,\n} from './network-protocol';\nimport { createIdFromSpaceKey } from '../common/space-id';\n\nexport interface NetworkDataMonitor {\n recordPeerConnected(peerId: string): void;\n recordPeerDisconnected(peerId: string): void;\n recordMessageSent(message: Message, duration: number): void;\n recordMessageReceived(message: Message): void;\n recordMessageSendingFailed(message: Message): void;\n}\n\nexport type EchoNetworkAdapterParams = {\n getContainingSpaceForDocument: (documentId: string) => Promise<PublicKey | null>;\n isDocumentInRemoteCollection: (params: RemoteDocumentExistenceCheckParams) => Promise<boolean>;\n onCollectionStateQueried: (collectionId: string, peerId: PeerId) => void;\n onCollectionStateReceived: (collectionId: string, peerId: PeerId, state: unknown) => void;\n monitor?: NetworkDataMonitor;\n};\n\n/**\n * Manages a set of {@link EchoReplicator} instances.\n */\nexport class EchoNetworkAdapter extends NetworkAdapter {\n private readonly _replicators = new Set<EchoReplicator>();\n /**\n * Remote peer id -> connection.\n */\n private readonly _connections = new Map<PeerId, ConnectionEntry>();\n private _lifecycleState: LifecycleState = LifecycleState.CLOSED;\n private readonly _connected = new Trigger();\n\n constructor(private readonly _params: EchoNetworkAdapterParams) {\n super();\n }\n\n override connect(peerId: PeerId, peerMetadata?: PeerMetadata | undefined): void {\n this.peerId = peerId;\n this.peerMetadata = peerMetadata;\n this._connected.wake();\n }\n\n override send(message: Message): void {\n this._send(message);\n }\n\n override disconnect(): void {\n // No-op\n }\n\n @synchronized\n async open() {\n if (this._lifecycleState === LifecycleState.OPEN) {\n return;\n }\n this._lifecycleState = LifecycleState.OPEN;\n\n log('emit ready');\n this.emit('ready', {\n network: this,\n });\n }\n\n @synchronized\n async close() {\n if (this._lifecycleState === LifecycleState.CLOSED) {\n return this;\n }\n\n for (const replicator of this._replicators) {\n await replicator.disconnect();\n }\n this._replicators.clear();\n\n this._lifecycleState = LifecycleState.CLOSED;\n }\n\n async whenConnected() {\n await this._connected.wait({ timeout: 10_000 });\n }\n\n @synchronized\n async addReplicator(replicator: EchoReplicator) {\n invariant(this._lifecycleState === LifecycleState.OPEN);\n invariant(this.peerId);\n invariant(!this._replicators.has(replicator));\n\n this._replicators.add(replicator);\n await replicator.connect({\n peerId: this.peerId,\n onConnectionOpen: this._onConnectionOpen.bind(this),\n onConnectionClosed: this._onConnectionClosed.bind(this),\n onConnectionAuthScopeChanged: this._onConnectionAuthScopeChanged.bind(this),\n isDocumentInRemoteCollection: this._params.isDocumentInRemoteCollection,\n getContainingSpaceForDocument: this._params.getContainingSpaceForDocument,\n getContainingSpaceIdForDocument: async (documentId) => {\n const key = await this._params.getContainingSpaceForDocument(documentId);\n return key ? createIdFromSpaceKey(key) : null;\n },\n });\n }\n\n @synchronized\n async removeReplicator(replicator: EchoReplicator) {\n invariant(this._lifecycleState === LifecycleState.OPEN);\n invariant(this._replicators.has(replicator));\n await replicator.disconnect();\n this._replicators.delete(replicator);\n }\n\n async shouldAdvertise(peerId: PeerId, params: ShouldAdvertiseParams): Promise<boolean> {\n const connection = this._connections.get(peerId);\n if (!connection) {\n return false;\n }\n\n return connection.connection.shouldAdvertise(params);\n }\n\n shouldSyncCollection(peerId: PeerId, params: ShouldSyncCollectionParams): boolean {\n const connection = this._connections.get(peerId);\n if (!connection) {\n return false;\n }\n\n return connection.connection.shouldSyncCollection(params);\n }\n\n queryCollectionState(collectionId: string, targetId: PeerId): void {\n const message: CollectionQueryMessage = {\n type: 'collection-query',\n senderId: this.peerId as PeerId,\n targetId,\n collectionId,\n };\n this._send(message);\n }\n\n sendCollectionState(collectionId: string, targetId: PeerId, state: unknown): void {\n const message: CollectionStateMessage = {\n type: 'collection-state',\n senderId: this.peerId as PeerId,\n targetId,\n collectionId,\n state,\n };\n this._send(message);\n }\n\n private _send(message: Message) {\n const connectionEntry = this._connections.get(message.targetId);\n if (!connectionEntry) {\n throw new Error('Connection not found.');\n }\n\n const writeStart = Date.now();\n // TODO(dmaretskyi): Find a way to enforce backpressure on AM-repo.\n connectionEntry.writer\n .write(message as AutomergeProtocolMessage)\n .then(() => {\n const durationMs = Date.now() - writeStart;\n this._params.monitor?.recordMessageSent(message, durationMs);\n })\n .catch((err) => {\n if (connectionEntry.isOpen) {\n log.catch(err);\n }\n this._params.monitor?.recordMessageSendingFailed(message);\n });\n }\n\n // TODO(dmaretskyi): Remove.\n getPeersInterestedInCollection(collectionId: string): PeerId[] {\n return Array.from(this._connections.values())\n .map((connection) => {\n return connection.connection.shouldSyncCollection({ collectionId })\n ? (connection.connection.peerId as PeerId)\n : null;\n })\n .filter(nonNullable);\n }\n\n private _onConnectionOpen(connection: ReplicatorConnection) {\n log('Connection opened', { peerId: connection.peerId });\n invariant(!this._connections.has(connection.peerId as PeerId));\n const reader = connection.readable.getReader();\n const writer = connection.writable.getWriter();\n const connectionEntry: ConnectionEntry = { connection, reader, writer, isOpen: true };\n this._connections.set(connection.peerId as PeerId, connectionEntry);\n\n queueMicrotask(async () => {\n try {\n while (true) {\n // TODO(dmaretskyi): Find a way to enforce backpressure on AM-repo.\n const { done, value } = await reader.read();\n if (done) {\n break;\n }\n\n this._onMessage(value as Message);\n }\n } catch (err) {\n if (connectionEntry.isOpen) {\n log.catch(err);\n }\n }\n });\n\n log('emit peer-candidate', { peerId: connection.peerId });\n this._emitPeerCandidate(connection);\n this._params.monitor?.recordPeerConnected(connection.peerId);\n }\n\n private _onMessage(message: Message) {\n if (isCollectionQueryMessage(message)) {\n this._params.onCollectionStateQueried(message.collectionId, message.senderId);\n } else if (isCollectionStateMessage(message)) {\n this._params.onCollectionStateReceived(message.collectionId, message.senderId, message.state);\n } else {\n this.emit('message', message);\n }\n this._params.monitor?.recordMessageReceived(message);\n }\n\n /**\n * Trigger doc-synchronizer shared documents set recalculation. Happens on peer-candidate.\n * TODO(y): replace with a proper API call when sharePolicy update becomes supported by automerge-repo\n */\n private _onConnectionAuthScopeChanged(connection: ReplicatorConnection) {\n log('Connection auth scope changed', { peerId: connection.peerId });\n const entry = this._connections.get(connection.peerId as PeerId);\n invariant(entry);\n this.emit('peer-disconnected', { peerId: connection.peerId as PeerId });\n this._emitPeerCandidate(connection);\n }\n\n private _onConnectionClosed(connection: ReplicatorConnection) {\n log('Connection closed', { peerId: connection.peerId });\n const entry = this._connections.get(connection.peerId as PeerId);\n invariant(entry);\n\n entry.isOpen = false;\n this.emit('peer-disconnected', { peerId: connection.peerId as PeerId });\n this._params.monitor?.recordPeerDisconnected(connection.peerId);\n\n void entry.reader.cancel().catch((err) => log.catch(err));\n void entry.writer.abort().catch((err) => log.catch(err));\n\n this._connections.delete(connection.peerId as PeerId);\n }\n\n private _emitPeerCandidate(connection: ReplicatorConnection) {\n this.emit('peer-candidate', {\n peerId: connection.peerId as PeerId,\n peerMetadata: createEchoPeerMetadata(),\n });\n }\n}\n\ntype ConnectionEntry = {\n connection: ReplicatorConnection;\n reader: ReadableStreamDefaultReader<AutomergeProtocolMessage>;\n writer: WritableStreamDefaultWriter<AutomergeProtocolMessage>;\n isOpen: boolean;\n};\n\nexport const createEchoPeerMetadata = (): PeerMetadata =>\n ({\n // TODO(dmaretskyi): Refactor this.\n dxos_peerSource: 'EchoNetworkAdapter',\n }) as any;\n\nexport const isEchoPeerMetadata = (metadata: PeerMetadata): boolean =>\n (metadata as any)?.dxos_peerSource === 'EchoNetworkAdapter';\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport type { Message } from '@dxos/automerge/automerge-repo';\nimport {\n type CollectionQueryMessage,\n type CollectionStateMessage,\n MESSAGE_TYPE_COLLECTION_QUERY,\n MESSAGE_TYPE_COLLECTION_STATE,\n} from '@dxos/protocols';\n\nexport { type CollectionStateMessage, type CollectionQueryMessage };\n\nexport const isCollectionQueryMessage = (message: Message): message is CollectionQueryMessage =>\n message.type === MESSAGE_TYPE_COLLECTION_QUERY;\n\nexport const isCollectionStateMessage = (message: Message): message is CollectionStateMessage =>\n message.type === MESSAGE_TYPE_COLLECTION_STATE;\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport type { Heads } from '@dxos/automerge/automerge';\nimport type { DocumentId } from '@dxos/automerge/automerge-repo';\nimport { headsEncoding } from '@dxos/indexing';\nimport type { BatchLevel, SublevelDB } from '@dxos/kv-store';\n\nexport type HeadsStoreParams = {\n db: SublevelDB;\n};\n\nexport class HeadsStore {\n private readonly _db: SublevelDB;\n\n constructor({ db }: HeadsStoreParams) {\n this._db = db;\n }\n\n setHeads(documentId: DocumentId, heads: Heads, batch: BatchLevel) {\n batch.put<DocumentId, Heads>(documentId, heads, {\n sublevel: this._db,\n keyEncoding: 'utf8',\n valueEncoding: headsEncoding,\n });\n }\n\n // TODO(dmaretskyi): Make batched.\n async getHeads(documentIds: DocumentId[]): Promise<Array<Heads | undefined>> {\n return this._db.getMany<DocumentId, Heads>(documentIds, {\n keyEncoding: 'utf8',\n valueEncoding: headsEncoding,\n });\n }\n}\n", "//\n// Copyright 2024 DXOS.org\n// s\n\nimport { type MixedEncoding } from 'level-transcoder';\n\nimport { type StorageAdapterInterface, type Chunk, type StorageKey } from '@dxos/automerge/automerge-repo';\nimport { LifecycleState, Resource } from '@dxos/context';\nimport { type BatchLevel, type SublevelDB } from '@dxos/kv-store';\nimport { type MaybePromise } from '@dxos/util';\n\nexport interface StorageAdapterDataMonitor {\n recordBytesStored(count: number): void;\n recordBytesLoaded(count: number): void;\n recordLoadDuration(durationMs: number): void;\n recordStoreDuration(durationMs: number): void;\n}\n\nexport type LevelDBStorageAdapterParams = {\n db: SublevelDB;\n callbacks?: StorageCallbacks;\n monitor?: StorageAdapterDataMonitor;\n};\n\nexport type BeforeSaveParams = { path: StorageKey; batch: BatchLevel };\n\nexport interface StorageCallbacks {\n beforeSave(params: BeforeSaveParams): MaybePromise<void>;\n afterSave(path: StorageKey): MaybePromise<void>;\n}\n\nexport class LevelDBStorageAdapter extends Resource implements StorageAdapterInterface {\n constructor(private readonly _params: LevelDBStorageAdapterParams) {\n super();\n }\n\n async load(keyArray: StorageKey): Promise<Uint8Array | undefined> {\n try {\n if (this._lifecycleState !== LifecycleState.OPEN) {\n // TODO(mykola): this should be an error.\n return undefined;\n }\n const startMs = Date.now();\n const chunk = await this._params.db.get<StorageKey, Uint8Array>(keyArray, { ...encodingOptions });\n this._params.monitor?.recordBytesLoaded(chunk.byteLength);\n this._params.monitor?.recordLoadDuration(Date.now() - startMs);\n return chunk;\n } catch (err: any) {\n if (isLevelDbNotFoundError(err)) {\n return undefined;\n }\n throw err;\n }\n }\n\n async save(keyArray: StorageKey, binary: Uint8Array): Promise<void> {\n if (this._lifecycleState !== LifecycleState.OPEN) {\n return undefined;\n }\n const startMs = Date.now();\n const batch = this._params.db.batch();\n\n await this._params.callbacks?.beforeSave?.({ path: keyArray, batch });\n batch.put<StorageKey, Uint8Array>(keyArray, Buffer.from(binary), {\n ...encodingOptions,\n });\n await batch.write();\n this._params.monitor?.recordBytesStored(binary.byteLength);\n\n await this._params.callbacks?.afterSave?.(keyArray);\n this._params.monitor?.recordStoreDuration(Date.now() - startMs);\n }\n\n async remove(keyArray: StorageKey): Promise<void> {\n if (this._lifecycleState !== LifecycleState.OPEN) {\n return undefined;\n }\n await this._params.db.del<StorageKey>(keyArray, { ...encodingOptions });\n }\n\n async loadRange(keyPrefix: StorageKey): Promise<Chunk[]> {\n if (this._lifecycleState !== LifecycleState.OPEN) {\n return [];\n }\n const startMs = Date.now();\n const result: Chunk[] = [];\n for await (const [key, value] of this._params.db.iterator<StorageKey, Uint8Array>({\n gte: keyPrefix,\n lte: [...keyPrefix, '\\uffff'],\n ...encodingOptions,\n })) {\n result.push({\n key,\n data: value,\n });\n this._params.monitor?.recordBytesLoaded(value.byteLength);\n }\n this._params.monitor?.recordLoadDuration(Date.now() - startMs);\n return result;\n }\n\n async removeRange(keyPrefix: StorageKey): Promise<void> {\n if (this._lifecycleState !== LifecycleState.OPEN) {\n return undefined;\n }\n const batch = this._params.db.batch();\n\n for await (const [key] of this._params.db.iterator<StorageKey, Uint8Array>({\n gte: keyPrefix,\n lte: [...keyPrefix, '\\uffff'],\n ...encodingOptions,\n })) {\n batch.del<StorageKey>(key, { ...encodingOptions });\n }\n await batch.write();\n }\n}\n\nconst keyEncoder: MixedEncoding<StorageKey, Uint8Array, StorageKey> = {\n encode: (key: StorageKey): Uint8Array =>\n Buffer.from(key.map((k) => k.replaceAll('%', '%25').replaceAll('-', '%2D')).join('-')),\n decode: (key: Uint8Array): StorageKey =>\n Buffer.from(key)\n .toString()\n .split('-')\n .map((k) => k.replaceAll('%2D', '-').replaceAll('%25', '%')),\n format: 'buffer',\n};\n\nexport const encodingOptions = {\n keyEncoding: keyEncoder,\n valueEncoding: 'buffer',\n};\n\nconst isLevelDbNotFoundError = (err: any): boolean => err.code === 'LEVEL_NOT_FOUND';\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport type { CollectionId } from '@dxos/echo-protocol';\nimport { invariant } from '@dxos/invariant';\nimport { PublicKey, type SpaceId } from '@dxos/keys';\nimport { log } from '@dxos/log';\nimport {\n type AutomergeReplicator,\n type AutomergeReplicatorFactory,\n} from '@dxos/teleport-extension-automerge-replicator';\nimport { ComplexSet, defaultMap } from '@dxos/util';\n\nimport { type EchoReplicator, type EchoReplicatorContext, type ShouldAdvertiseParams } from './echo-replicator';\nimport { MeshReplicatorConnection } from './mesh-echo-replicator-connection';\nimport { getSpaceIdFromCollectionId } from './space-collection';\nimport { createIdFromSpaceKey } from '../common/space-id';\n\n// TODO(dmaretskyi): Move out of @dxos/echo-pipeline.\n\n/**\n * Used to replicate with other peers over the network.\n */\nexport class MeshEchoReplicator implements EchoReplicator {\n private readonly _connections = new Set<MeshReplicatorConnection>();\n /**\n * Using automerge peerId as a key.\n */\n private readonly _connectionsPerPeer = new Map<string, MeshReplicatorConnection>();\n\n /**\n * spaceId -> deviceKey[]\n */\n private readonly _authorizedDevices = new Map<SpaceId, ComplexSet<PublicKey>>();\n\n private _context: EchoReplicatorContext | null = null;\n\n async connect(context: EchoReplicatorContext): Promise<void> {\n this._context = context;\n }\n\n async disconnect() {\n for (const connection of this._connectionsPerPeer.values()) {\n this._context?.onConnectionClosed(connection);\n }\n for (const connection of this._connections) {\n await connection.close();\n }\n this._connections.clear();\n this._connectionsPerPeer.clear();\n\n this._context = null;\n }\n\n createExtension(extensionFactory?: AutomergeReplicatorFactory): AutomergeReplicator {\n invariant(this._context);\n\n const connection: MeshReplicatorConnection = new MeshReplicatorConnection({\n ownPeerId: this._context.peerId,\n replicatorFactory: extensionFactory,\n onRemoteConnected: async () => {\n log('onRemoteConnected', { peerId: connection.peerId });\n invariant(this._context);\n\n if (this._connectionsPerPeer.has(connection.peerId)) {\n this._context.onConnectionAuthScopeChanged(connection);\n } else {\n this._connectionsPerPeer.set(connection.peerId, connection);\n this._context.onConnectionOpen(connection);\n connection.enable();\n }\n },\n onRemoteDisconnected: async () => {\n log('onRemoteDisconnected', { peerId: connection.peerId });\n this._context?.onConnectionClosed(connection);\n this._connectionsPerPeer.delete(connection.peerId);\n connection.disable();\n this._connections.delete(connection);\n },\n shouldAdvertise: async (params: ShouldAdvertiseParams) => {\n log('shouldAdvertise', { peerId: connection.peerId, documentId: params.documentId });\n invariant(this._context);\n try {\n const spaceKey = await this._context.getContainingSpaceForDocument(params.documentId);\n if (!spaceKey) {\n const remoteDocumentExists = await this._context.isDocumentInRemoteCollection({\n documentId: params.documentId,\n peerId: connection.peerId,\n });\n log('document not found locally for share policy check, accepting the remote document', {\n peerId: connection.peerId,\n documentId: params.documentId,\n remoteDocumentExists,\n });\n // If a document is not present locally return true if another peer claims to have it.\n // Simply returning true will add the peer to \"generous peers list\" for this document which will\n // start replication of the document after we receive, even if the peer is not in the corresponding space.\n return remoteDocumentExists;\n }\n\n const spaceId = await createIdFromSpaceKey(spaceKey);\n\n const authorizedDevices = this._authorizedDevices.get(spaceId);\n\n if (!connection.remoteDeviceKey) {\n log('device key not found for share policy check', {\n peerId: connection.peerId,\n documentId: params.documentId,\n });\n return false;\n }\n\n const isAuthorized = authorizedDevices?.has(connection.remoteDeviceKey) ?? false;\n log('share policy check', {\n localPeer: this._context.peerId,\n remotePeer: connection.peerId,\n documentId: params.documentId,\n deviceKey: connection.remoteDeviceKey,\n spaceKey,\n isAuthorized,\n });\n return isAuthorized;\n } catch (err) {\n log.catch(err);\n return false;\n }\n },\n shouldSyncCollection: ({ collectionId }) => {\n const spaceId = getSpaceIdFromCollectionId(collectionId as CollectionId);\n\n const authorizedDevices = this._authorizedDevices.get(spaceId);\n\n if (!connection.remoteDeviceKey) {\n log('device key not found for collection sync check', {\n peerId: connection.peerId,\n collectionId,\n });\n return false;\n }\n\n const isAuthorized = authorizedDevices?.has(connection.remoteDeviceKey) ?? false;\n return isAuthorized;\n },\n });\n this._connections.add(connection);\n\n return connection.replicatorExtension;\n }\n\n async authorizeDevice(spaceKey: PublicKey, deviceKey: PublicKey) {\n log('authorizeDevice', { spaceKey, deviceKey });\n const spaceId = await createIdFromSpaceKey(spaceKey);\n defaultMap(this._authorizedDevices, spaceId, () => new ComplexSet(PublicKey.hash)).add(deviceKey);\n for (const connection of this._connections) {\n if (connection.remoteDeviceKey && connection.remoteDeviceKey.equals(deviceKey)) {\n if (this._connectionsPerPeer.has(connection.peerId)) {\n this._context?.onConnectionAuthScopeChanged(connection);\n }\n }\n }\n }\n}\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport * as A from '@dxos/automerge/automerge';\nimport { cbor } from '@dxos/automerge/automerge-repo';\nimport { Resource } from '@dxos/context';\nimport { invariant } from '@dxos/invariant';\nimport { type PublicKey } from '@dxos/keys';\nimport { log } from '@dxos/log';\nimport type { AutomergeProtocolMessage } from '@dxos/protocols';\nimport { AutomergeReplicator, type AutomergeReplicatorFactory } from '@dxos/teleport-extension-automerge-replicator';\n\nimport type { ReplicatorConnection, ShouldAdvertiseParams, ShouldSyncCollectionParams } from './echo-replicator';\n\nconst DEFAULT_FACTORY: AutomergeReplicatorFactory = (params) => new AutomergeReplicator(...params);\n\nexport type MeshReplicatorConnectionParams = {\n ownPeerId: string;\n onRemoteConnected: () => void;\n onRemoteDisconnected: () => void;\n shouldAdvertise: (params: ShouldAdvertiseParams) => Promise<boolean>;\n shouldSyncCollection: (params: ShouldSyncCollectionParams) => boolean;\n replicatorFactory?: AutomergeReplicatorFactory;\n};\n\nexport class MeshReplicatorConnection extends Resource implements ReplicatorConnection {\n public readable: ReadableStream<AutomergeProtocolMessage>;\n public writable: WritableStream<AutomergeProtocolMessage>;\n public remoteDeviceKey: PublicKey | null = null;\n\n public readonly replicatorExtension: AutomergeReplicator;\n\n private _remotePeerId: string | null = null;\n private _isEnabled = false;\n\n constructor(private readonly _params: MeshReplicatorConnectionParams) {\n super();\n\n let readableStreamController!: ReadableStreamDefaultController<AutomergeProtocolMessage>;\n this.readable = new ReadableStream<AutomergeProtocolMessage>({\n start: (controller) => {\n readableStreamController = controller;\n this._ctx.onDispose(() => controller.close());\n },\n });\n\n this.writable = new WritableStream<AutomergeProtocolMessage>({\n write: async (message: AutomergeProtocolMessage, controller) => {\n invariant(this._isEnabled, 'Writing to a disabled connection');\n try {\n logSendSync(message);\n await this.replicatorExtension.sendSyncMessage({ payload: cbor.encode(message) });\n } catch (err) {\n controller.error(err);\n this._disconnectIfEnabled();\n }\n },\n });\n\n const createAutomergeReplicator = this._params.replicatorFactory ?? DEFAULT_FACTORY;\n this.replicatorExtension = createAutomergeReplicator([\n {\n peerId: this._params.ownPeerId,\n },\n {\n onStartReplication: async (info, remotePeerId /** Teleport ID */) => {\n // Note: We store only one extension per peer.\n // There can be a case where two connected peers have more than one teleport connection between them\n // and each of them uses different teleport connections to send messages.\n // It works because we receive messages from all teleport connections and Automerge Repo dedup them.\n // TODO(mykola): Use only one teleport connection per peer.\n\n // TODO(dmaretskyi): Critical bug.\n // - two peers get connected via swarm 1\n // - they get connected via swarm 2\n // - swarm 1 gets disconnected\n // - automerge repo thinks that peer 2 got disconnected even though swarm 2 is still active\n\n this.remoteDeviceKey = remotePeerId;\n\n // Set automerge id.\n this._remotePeerId = info.id;\n\n log('onStartReplication', { id: info.id, thisPeerId: this.peerId, remotePeerId: remotePeerId.toHex() });\n\n this._params.onRemoteConnected();\n },\n onSyncMessage: async ({ payload }) => {\n if (!this._isEnabled) {\n return;\n }\n const message = cbor.decode(payload) as AutomergeProtocolMessage;\n // Note: automerge Repo dedup messages.\n readableStreamController.enqueue(message);\n },\n onClose: async () => {\n this._disconnectIfEnabled();\n },\n },\n ]);\n }\n\n private _disconnectIfEnabled() {\n if (this._isEnabled) {\n this._params.onRemoteDisconnected();\n }\n }\n\n get peerId(): string {\n invariant(this._remotePeerId != null, 'Remote peer has not connected yet.');\n return this._remotePeerId;\n }\n\n async shouldAdvertise(params: ShouldAdvertiseParams): Promise<boolean> {\n return this._params.shouldAdvertise(params);\n }\n\n shouldSyncCollection(params: ShouldSyncCollectionParams): boolean {\n return this._params.shouldSyncCollection(params);\n }\n\n /**\n * Start exchanging messages with the remote peer.\n * Call after the remote peer has connected.\n */\n enable() {\n invariant(this._remotePeerId != null, 'Remote peer has not connected yet.');\n this._isEnabled = true;\n }\n\n /**\n * Stop exchanging messages with the remote peer.\n */\n disable() {\n this._isEnabled = false;\n }\n}\n\nconst logSendSync = (message: AutomergeProtocolMessage) => {\n log('sendSyncMessage', () => {\n const decodedSyncMessage = message.type === 'sync' && message.data ? A.decodeSyncMessage(message.data) : undefined;\n return {\n sync: decodedSyncMessage && {\n headsLength: decodedSyncMessage.heads.length,\n requesting: decodedSyncMessage.need.length > 0,\n sendingChanges: decodedSyncMessage.changes.length > 0,\n },\n type: message.type,\n from: message.senderId,\n to: message.targetId,\n };\n });\n};\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport type { CollectionId } from '@dxos/echo-protocol';\nimport { invariant } from '@dxos/invariant';\nimport { SpaceId } from '@dxos/keys';\n\nexport const deriveCollectionIdFromSpaceId = (spaceId: SpaceId): CollectionId => `space:${spaceId}` as CollectionId;\n\nexport const getSpaceIdFromCollectionId = (collectionId: CollectionId): SpaceId => {\n const spaceId = collectionId.replace(/^space:/, '');\n invariant(SpaceId.isValid(spaceId));\n return spaceId;\n};\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport { type Message } from '@dxos/automerge/automerge-repo';\nimport { type TimeAware, trace } from '@dxos/tracing';\nimport { CircularBuffer, mapValues, SlidingWindowSummary, type SlidingWindowSummaryConfig } from '@dxos/util';\n\nimport { type NetworkDataMonitor } from './echo-network-adapter';\nimport { type StorageAdapterDataMonitor } from './leveldb-storage-adapter';\nimport { isCollectionQueryMessage, isCollectionStateMessage } from './network-protocol';\n\nconst PER_SECOND_RATE_AVG_WINDOW_SIZE = 5;\nconst DEFAULT_AVG_WINDOW_SIZE = 25;\n\nexport type EchoDataMonitorOptions = {\n timeSeriesLength: number;\n};\n\n@trace.resource()\nexport class EchoDataMonitor implements StorageAdapterDataMonitor, NetworkDataMonitor, TimeAware {\n private _lastTick = 0;\n\n private _activeCounters = createLocalCounters();\n private _lastCompleteCounters: LocalCounters | undefined;\n private readonly _localTimeSeries = createLocalTimeSeries();\n private readonly _storageAverages = createStorageAverages();\n private readonly _replicationAverages = createNetworkAverages();\n private readonly _sizeByMessageType: { [type: string]: SlidingWindowSummary } = {};\n private readonly _lastReceivedMessages = new CircularBuffer<StoredMessage>(100);\n private readonly _lastSentMessages = new CircularBuffer<StoredMessage>(100);\n\n private _connectionsCount = 0;\n\n constructor(private readonly _params: EchoDataMonitorOptions = { timeSeriesLength: 30 }) {}\n\n public tick(timeMs: number) {\n this._advanceTimeWindow(timeMs - this._lastTick);\n this._lastTick = timeMs;\n }\n\n public computeStats(): EchoDataStats {\n return {\n meta: {\n rateAverageOverSeconds: PER_SECOND_RATE_AVG_WINDOW_SIZE,\n },\n storage: {\n reads: {\n payloadSize: this._storageAverages.loadedChunkSize.average(),\n opDuration: this._storageAverages.loadDuration.average(),\n countPerSecond: this._storageAverages.loadsPerSecond.average(),\n },\n writes: {\n payloadSize: this._storageAverages.storedChunkSize.average(),\n opDuration: this._storageAverages.storeDuration.average(),\n countPerSecond: this._storageAverages.storesPerSecond.average(),\n },\n },\n replicator: {\n connections: this._connectionsCount,\n receivedMessages: {\n payloadSize: this._replicationAverages.receivedMessageSize.average(),\n countPerSecond: this._replicationAverages.receivedPerSecond.average(),\n },\n sentMessages: {\n payloadSize: this._replicationAverages.sentMessageSize.average(),\n opDuration: this._replicationAverages.sendDuration.average(),\n countPerSecond: this._replicationAverages.sentPerSecond.average(),\n failedPerSecond: this._replicationAverages.sendsFailedPerSecond.average(),\n },\n countByMessageType: this._computeMessageHistogram('type'),\n avgSizeByMessageType: mapValues(this._sizeByMessageType, (summary) => summary.average()),\n },\n };\n }\n\n public get connectionsCount() {\n return this._connectionsCount;\n }\n\n /**\n * @internal\n */\n get lastPerSecondStats() {\n return this._lastCompleteCounters;\n }\n\n /**\n * @internal\n */\n get timeSeries() {\n return { ...this._localTimeSeries.storage, ...this._localTimeSeries.replication };\n }\n\n /**\n * @internal\n */\n get messagesByPeerId() {\n return this._computeMessageHistogram('peerId');\n }\n\n private _advanceTimeWindow(millisPassed: number) {\n const oldMetrics = Object.freeze(this._activeCounters);\n this._activeCounters = createLocalCounters();\n this._lastCompleteCounters = oldMetrics;\n for (const peerId of Object.keys(oldMetrics.byPeerId)) {\n this._activeCounters.byPeerId[peerId] = createMessageCounter();\n }\n this._addToTimeSeries(oldMetrics.replication, this._localTimeSeries.replication);\n this._addToTimeSeries(oldMetrics.storage, this._localTimeSeries.storage);\n // Prevent skewed measurements of incomplete buckets / after CPU freezes.\n if (Math.abs(millisPassed - 1000) < 100) {\n this._reportPerSecondRate(oldMetrics);\n }\n }\n\n private _addToTimeSeries<T extends object>(values: T, timeSeries: TimeSeries<T>) {\n for (const [key, value] of Object.entries(values)) {\n const values: (typeof value)[] = (timeSeries as any)[key];\n values.push(value);\n if (values.length > this._params.timeSeriesLength) {\n values.shift();\n }\n }\n }\n\n private _reportPerSecondRate(metrics: LocalCounters) {\n const toReport: [string, number, SlidingWindowSummary][] = [\n ['storage.load', metrics.storage.loadedChunks, this._storageAverages.loadsPerSecond],\n ['storage.store', metrics.storage.storedChunks, this._storageAverages.storesPerSecond],\n ['network.receive', metrics.replication.received, this._replicationAverages.receivedPerSecond],\n ['network.send', metrics.replication.sent, this._replicationAverages.sentPerSecond],\n ];\n for (const [metricName, metric, summary] of toReport) {\n summary.record(metric);\n if (metric > 0) {\n trace.metrics.distribution(`dxos.echo.${metricName}-rate`, metric);\n trace.metrics.increment(`dxos.echo.${metricName}`, 1, { tags: { status: 'busy' } });\n } else {\n trace.metrics.increment(`dxos.echo.${metricName}`, 1, { tags: { status: 'idle' } });\n }\n }\n this._replicationAverages.sendsFailedPerSecond.record(metrics.replication.failed);\n }\n\n public recordPeerConnected(peerId: string) {\n this._activeCounters.byPeerId[peerId] = createMessageCounter();\n this._connectionsCount++;\n }\n\n public recordPeerDisconnected(peerId: string) {\n this._connectionsCount--;\n delete this._activeCounters.byPeerId[peerId];\n }\n\n public recordBytesStored(count: number) {\n this._activeCounters.storage.storedChunks++;\n this._activeCounters.storage.storedBytes += count;\n this._storageAverages.storedChunkSize.record(count);\n trace.metrics.distribution('dxos.echo.storage.bytes-stored', count, { unit: 'bytes' });\n }\n\n public recordLoadDuration(durationMs: number): void {\n this._storageAverages.loadDuration.record(durationMs);\n }\n\n public recordStoreDuration(durationMs: number): void {\n this._storageAverages.storeDuration.record(durationMs);\n }\n\n public recordBytesLoaded(count: number) {\n this._activeCounters.storage.loadedChunks++;\n this._activeCounters.storage.loadedBytes += count;\n this._storageAverages.loadedChunkSize.record(count);\n trace.metrics.distribution('dxos.echo.storage.bytes-loaded', count, { unit: 'bytes' });\n }\n\n public recordMessageSent(message: Message, duration: number) {\n let metricsGroupName;\n const bytes = getByteCount(message);\n const tags = { type: message.type };\n if (isAutomergeProtocolMessage(message)) {\n this._activeCounters.replication.sent++;\n this._replicationAverages.sendDuration.record(duration);\n this._replicationAverages.sentMessageSize.record(bytes);\n metricsGroupName = 'replication';\n } else {\n metricsGroupName = 'collection-sync';\n }\n trace.metrics.distribution(`dxos.echo.${metricsGroupName}.bytes-sent`, bytes, { unit: 'bytes', tags });\n trace.metrics.distribution(`dxos.echo.${metricsGroupName}.send-duration`, duration, { unit: 'millisecond', tags });\n trace.metrics.increment(`dxos.echo.${metricsGroupName}.send-status`, 1, { tags: { ...tags, success: true } });\n const { messageSize, messageCounts } = this._getStatsForType(message);\n messageSize.record(bytes);\n messageCounts.sent++;\n this._lastSentMessages.push({ type: message.type, peerId: message.targetId });\n }\n\n public recordMessageReceived(message: Message) {\n const bytes = getByteCount(message);\n const tags = { type: message.type };\n if (isAutomergeProtocolMessage(message)) {\n this._activeCounters.replication.received++;\n this._replicationAverages.receivedMessageSize.record(bytes);\n trace.metrics.distribution('dxos.echo.replication.bytes-received', bytes, { unit: 'bytes', tags });\n } else {\n trace.metrics.distribution('dxos.echo.collection-sync.bytes-received', bytes, { unit: 'bytes', tags });\n }\n const { messageSize, messageCounts } = this._getStatsForType(message);\n messageSize.record(bytes);\n messageCounts.received++;\n this._lastReceivedMessages.push({ type: message.type, peerId: message.senderId });\n }\n\n public recordMessageSendingFailed(message: Message) {\n const tags = { type: message.type, success: false };\n if (isAutomergeProtocolMessage(message)) {\n this._activeCounters.replication.failed++;\n trace.metrics.increment('dxos.echo.replication.send-status', 1, { unit: 'bytes', tags });\n } else {\n trace.metrics.increment('dxos.echo.collection-sync.send-status', 1, { unit: 'bytes', tags });\n }\n const { messageCounts } = this._getStatsForType(message);\n messageCounts.failed++;\n }\n\n private _getStatsForType(message: Message) {\n const messageSize = (this._sizeByMessageType[message.type] ??= createSlidingWindow());\n const messageCounts = (this._activeCounters.byType[message.type] ??= createMessageCounter());\n return { messageCounts, messageSize };\n }\n\n private _computeMessageHistogram(groupKey: keyof StoredMessage): MessageAttributeHistogram {\n const result: MessageAttributeHistogram = {};\n for (const receivedMessage of this._lastReceivedMessages) {\n const counters = (result[receivedMessage[groupKey]] ??= { received: 0, sent: 0 });\n counters.received++;\n }\n for (const receivedMessage of this._lastSentMessages) {\n const counters = (result[receivedMessage[groupKey]] ??= { received: 0, sent: 0 });\n counters.sent++;\n }\n return result;\n }\n}\n\ntype BaseDataOpStats = {\n payloadSize: number;\n countPerSecond: number;\n};\n\nexport type TimedDataOpStats = BaseDataOpStats & { opDuration: number };\n\ntype TimeSeries<T extends object> = { [key in keyof T]: T[key][] };\n\ntype StorageCounts = {\n storedChunks: number;\n storedBytes: number;\n loadedChunks: number;\n loadedBytes: number;\n};\ntype StorageCountTimeSeries = TimeSeries<StorageCounts>;\n\ntype MessageCounts = {\n sent: number;\n received: number;\n failed: number;\n};\ntype MessageCountTimeSeries = TimeSeries<MessageCounts>;\n\ntype MessageAttributeHistogram = {\n [messageType: string]: {\n received: number;\n sent: number;\n };\n};\n\nexport type EchoDataStats = {\n meta: {\n rateAverageOverSeconds: number;\n };\n storage: {\n reads: TimedDataOpStats;\n writes: TimedDataOpStats;\n };\n replicator: {\n connections: number;\n receivedMessages: BaseDataOpStats;\n sentMessages: TimedDataOpStats & { failedPerSecond: number };\n avgSizeByMessageType: { [messageType: string]: number };\n countByMessageType: MessageAttributeHistogram;\n };\n};\n\ntype StoredMessage = { type: string; peerId: string };\n\ntype StorageAverages = {\n storedChunkSize: SlidingWindowSummary;\n storesPerSecond: SlidingWindowSummary;\n loadedChunkSize: SlidingWindowSummary;\n loadsPerSecond: SlidingWindowSummary;\n loadDuration: SlidingWindowSummary;\n storeDuration: SlidingWindowSummary;\n};\n\ntype NetworkAverages = {\n receivedMessageSize: SlidingWindowSummary;\n receivedPerSecond: SlidingWindowSummary;\n sentMessageSize: SlidingWindowSummary;\n sentPerSecond: SlidingWindowSummary;\n sendDuration: SlidingWindowSummary;\n sendsFailedPerSecond: SlidingWindowSummary;\n};\n\ntype LocalCounters = {\n storage: StorageCounts;\n replication: MessageCounts;\n byPeerId: { [peerId: string]: MessageCounts };\n byType: { [type: string]: MessageCounts };\n};\n\ntype LocalTimeSeries = {\n storage: StorageCountTimeSeries;\n replication: MessageCountTimeSeries;\n};\n\nconst isAutomergeProtocolMessage = (message: Message) => {\n return !(isCollectionQueryMessage(message) || isCollectionStateMessage(message));\n};\n\nconst createSlidingWindow = (overrides?: SlidingWindowSummaryConfig) =>\n new SlidingWindowSummary({ dataPoints: DEFAULT_AVG_WINDOW_SIZE, precision: 2, ...overrides });\n\nconst createLocalCounters = (): LocalCounters => ({\n storage: { loadedBytes: 0, storedBytes: 0, storedChunks: 0, loadedChunks: 0 },\n replication: createMessageCounter(),\n byPeerId: {},\n byType: {},\n});\n\nconst createLocalTimeSeries = (): LocalTimeSeries => ({\n storage: { loadedBytes: [], storedBytes: [], storedChunks: [], loadedChunks: [] },\n replication: { sent: [], failed: [], received: [] },\n});\n\nconst createMessageCounter = (): MessageCounts => ({ sent: 0, received: 0, failed: 0 });\n\nconst createNetworkAverages = (): NetworkAverages => ({\n receivedMessageSize: createSlidingWindow(),\n sentMessageSize: createSlidingWindow(),\n sendDuration: createSlidingWindow(),\n receivedPerSecond: createSlidingWindow({ dataPoints: PER_SECOND_RATE_AVG_WINDOW_SIZE }),\n sentPerSecond: createSlidingWindow({ dataPoints: PER_SECOND_RATE_AVG_WINDOW_SIZE }),\n sendsFailedPerSecond: createSlidingWindow({ dataPoints: PER_SECOND_RATE_AVG_WINDOW_SIZE }),\n});\n\nconst createStorageAverages = (): StorageAverages => ({\n storedChunkSize: createSlidingWindow(),\n loadedChunkSize: createSlidingWindow(),\n loadDuration: createSlidingWindow(),\n storeDuration: createSlidingWindow(),\n loadsPerSecond: createSlidingWindow({ dataPoints: PER_SECOND_RATE_AVG_WINDOW_SIZE }),\n storesPerSecond: createSlidingWindow({ dataPoints: PER_SECOND_RATE_AVG_WINDOW_SIZE }),\n});\n\nconst getByteCount = (message: Message): number => {\n return (\n message.type.length +\n message.senderId.length +\n message.targetId.length +\n (message.data?.byteLength ?? 0) +\n (message.documentId?.length ?? 0)\n );\n};\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport {\n type AnyDocumentId,\n type AutomergeUrl,\n type DocHandle,\n type DocumentId,\n type Repo,\n} from '@dxos/automerge/automerge-repo';\nimport { LifecycleState, Resource, type Context } from '@dxos/context';\nimport { todo } from '@dxos/debug';\nimport { createIdFromSpaceKey, SpaceDocVersion, type SpaceDoc } from '@dxos/echo-protocol';\nimport { IndexMetadataStore, IndexStore, Indexer } from '@dxos/indexing';\nimport { invariant } from '@dxos/invariant';\nimport { type PublicKey, type SpaceId } from '@dxos/keys';\nimport { type LevelDB } from '@dxos/kv-store';\nimport { IndexKind, type IndexConfig } from '@dxos/protocols/proto/dxos/echo/indexing';\nimport { trace } from '@dxos/tracing';\n\nimport { DataServiceImpl } from './data-service';\nimport { type DatabaseRoot } from './database-root';\nimport { createSelectedDocumentsIterator } from './documents-iterator';\nimport { QueryServiceImpl } from './query-service';\nimport { SpaceStateManager } from './space-state-manager';\nimport {\n AutomergeHost,\n EchoDataMonitor,\n deriveCollectionIdFromSpaceId,\n type LoadDocOptions,\n type CreateDocOptions,\n type EchoReplicator,\n type CollectionSyncState,\n type EchoDataStats,\n} from '../automerge';\n\nconst INDEXER_CONFIG: IndexConfig = {\n enabled: true,\n indexes: [{ kind: IndexKind.Kind.SCHEMA_MATCH }],\n};\n\nexport type EchoHostParams = {\n kv: LevelDB;\n};\n\n/**\n * Host for the Echo database.\n * Manages multiple spaces.\n * Stores data to disk.\n * Can sync with pluggable data replicators.\n */\nexport class EchoHost extends Resource {\n private readonly _indexMetadataStore: IndexMetadataStore;\n private readonly _indexer: Indexer;\n private readonly _automergeHost: AutomergeHost;\n private readonly _queryService: QueryServiceImpl;\n private readonly _dataService: DataServiceImpl;\n private readonly _spaceStateManager = new SpaceStateManager();\n private readonly _echoDataMonitor: EchoDataMonitor;\n\n constructor({ kv }: EchoHostParams) {\n super();\n\n this._indexMetadataStore = new IndexMetadataStore({ db: kv.sublevel('index-metadata') });\n\n this._echoDataMonitor = new EchoDataMonitor();\n\n this._automergeHost = new AutomergeHost({\n db: kv,\n dataMonitor: this._echoDataMonitor,\n indexMetadataStore: this._indexMetadataStore,\n });\n\n this._indexer = new Indexer({\n db: kv,\n indexStore: new IndexStore({ db: kv.sublevel('index-storage') }),\n metadataStore: this._indexMetadataStore,\n loadDocuments: createSelectedDocumentsIterator(this._automergeHost),\n indexCooldownTime: process.env.NODE_ENV === 'test' ? 0 : undefined,\n });\n this._indexer.setConfig(INDEXER_CONFIG);\n\n this._queryService = new QueryServiceImpl({\n automergeHost: this._automergeHost,\n indexer: this._indexer,\n });\n\n this._dataService = new DataServiceImpl({\n automergeHost: this._automergeHost,\n updateIndexes: async () => {\n await this._indexer.updateIndexes();\n },\n });\n\n trace.diagnostic<EchoStatsDiagnostic>({\n id: 'echo-stats',\n name: 'Echo Stats',\n fetch: async () => {\n return {\n dataStats: this._echoDataMonitor.computeStats(),\n loadedDocsCount: this._automergeHost.loadedDocsCount,\n };\n },\n });\n\n trace.diagnostic({\n id: 'database-roots',\n name: 'Database Roots',\n fetch: async () => {\n return Array.from(this._spaceStateManager.roots.values()).map((root) => ({\n url: root.url,\n isLoaded: root.isLoaded,\n spaceKey: root.getSpaceKey(),\n inlineObjects: root.getInlineObjectCount(),\n linkedObjects: root.getLinkedObjectCount(),\n }));\n },\n });\n\n trace.diagnostic({\n id: 'database-root-metrics',\n name: 'Database Roots (with metrics)',\n fetch: async () => {\n return Array.from(this._spaceStateManager.roots.values()).map((root) => ({\n url: root.url,\n isLoaded: root.isLoaded,\n spaceKey: root.getSpaceKey(),\n inlineObjects: root.getInlineObjectCount(),\n linkedObjects: root.getLinkedObjectCount(),\n ...(root.measureMetrics() ?? {}),\n }));\n },\n });\n }\n\n get queryService(): QueryServiceImpl {\n return this._queryService;\n }\n\n get dataService(): DataServiceImpl {\n return this._dataService;\n }\n\n /**\n * @deprecated To be abstracted away.\n */\n get automergeRepo(): Repo {\n return this._automergeHost.repo;\n }\n\n get roots(): ReadonlyMap<DocumentId, DatabaseRoot> {\n return this._spaceStateManager.roots;\n }\n\n protected override async _open(ctx: Context): Promise<void> {\n await this._automergeHost.open();\n await this._indexer.open(ctx);\n await this._queryService.open(ctx);\n await this._spaceStateManager.open(ctx);\n\n this._spaceStateManager.spaceDocumentListUpdated.on(this._ctx, (e) => {\n void this._automergeHost.updateLocalCollectionState(deriveCollectionIdFromSpaceId(e.spaceId), e.documentIds);\n });\n }\n\n protected override async _close(ctx: Context): Promise<void> {\n await this._spaceStateManager.close();\n await this._queryService.close(ctx);\n await this._indexer.close(ctx);\n await this._automergeHost.close();\n }\n\n /**\n * Flush all pending writes to the underlying storage.\n */\n async flush() {\n await this._automergeHost.repo.flush();\n }\n\n /**\n * Perform any pending index updates.\n */\n async updateIndexes() {\n await this._indexer.updateIndexes();\n }\n\n /**\n * Loads the document handle from the repo and waits for it to be ready.\n */\n async loadDoc<T>(ctx: Context, documentId: AnyDocumentId, opts?: LoadDocOptions): Promise<DocHandle<T>> {\n return await this._automergeHost.loadDoc(ctx, documentId, opts);\n }\n\n /**\n * Create new persisted document.\n */\n createDoc<T>(initialValue?: T, opts?: CreateDocOptions): DocHandle<T> {\n return this._automergeHost.createDoc(initialValue, opts);\n }\n\n /**\n * Create new space root.\n */\n async createSpaceRoot(spaceKey: PublicKey): Promise<DatabaseRoot> {\n invariant(this._lifecycleState === LifecycleState.OPEN);\n const spaceId = await createIdFromSpaceKey(spaceKey);\n\n const automergeRoot = this._automergeHost.createDoc<SpaceDoc>({\n version: SpaceDocVersion.CURRENT,\n access: { spaceKey: spaceKey.toHex() },\n\n // Better to initialize them right away to avoid merge conflicts and data loss that can occur if those maps get created on the fly.\n objects: {},\n links: {},\n });\n\n await this._automergeHost.flush({ documentIds: [automergeRoot.documentId] });\n\n return await this.openSpaceRoot(spaceId, automergeRoot.url);\n }\n\n // TODO(dmaretskyi): Change to document id.\n async openSpaceRoot(spaceId: SpaceId, automergeUrl: AutomergeUrl): Promise<DatabaseRoot> {\n invariant(this._lifecycleState === LifecycleState.OPEN);\n const handle = this._automergeHost.repo.find(automergeUrl);\n\n return this._spaceStateManager.assignRootToSpace(spaceId, handle);\n }\n\n // TODO(dmaretskyi): Change to document id.\n async closeSpaceRoot(automergeUrl: AutomergeUrl): Promise<void> {\n todo();\n }\n\n /**\n * Install data replicator.\n */\n async addReplicator(replicator: EchoReplicator): Promise<void> {\n await this._automergeHost.addReplicator(replicator);\n }\n\n /**\n * Remove data replicator.\n */\n async removeReplicator(replicator: EchoReplicator): Promise<void> {\n await this._automergeHost.removeReplicator(replicator);\n }\n\n async getSpaceSyncState(spaceId: SpaceId): Promise<CollectionSyncState> {\n const collectionId = deriveCollectionIdFromSpaceId(spaceId);\n return this._automergeHost.getCollectionSyncState(collectionId);\n }\n}\n\nexport type { EchoDataStats };\n\nexport type EchoStatsDiagnostic = {\n loadedDocsCount: number;\n dataStats: EchoDataStats;\n};\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport * as A from '@dxos/automerge/automerge';\nimport { type DocumentId } from '@dxos/automerge/automerge-repo';\nimport { Context } from '@dxos/context';\nimport { SpaceDocVersion, type SpaceDoc } from '@dxos/echo-protocol';\nimport { type ObjectSnapshot, type IdToHeads } from '@dxos/indexing';\nimport { invariant } from '@dxos/invariant';\nimport { log } from '@dxos/log';\nimport { ObjectPointerVersion, objectPointerCodec } from '@dxos/protocols';\n\nimport { type AutomergeHost, getSpaceKeyFromDoc } from '../automerge';\n\nconst LOG_VIEW_OPERATION_THRESHOLD = 300;\n\n/**\n * Factory for `loadDocuments` iterator.\n */\nexport const createSelectedDocumentsIterator = (automergeHost: AutomergeHost) =>\n /**\n * Get object data blobs from Automerge Repo by ids.\n */\n // TODO(mykola): Unload automerge handles after usage.\n async function* loadDocuments(objects: IdToHeads): AsyncGenerator<ObjectSnapshot[], void, void> {\n for (const [id, heads] of objects.entries()) {\n try {\n const { documentId, objectId } = objectPointerCodec.decode(id);\n const handle = await automergeHost.loadDoc(Context.default(), documentId as DocumentId);\n\n let doc: A.Doc<SpaceDoc> = handle.docSync();\n invariant(doc);\n\n const currentHeads = A.getHeads(doc);\n\n // Checkout the requested version of the document.\n if (!A.equals(currentHeads, heads)) {\n const begin = Date.now();\n // `view` can take a long time even if the document is already at the right version.\n doc = A.view(doc, heads);\n const end = Date.now();\n if (end - begin > LOG_VIEW_OPERATION_THRESHOLD) {\n log.info('Checking out document version is taking too long', {\n duration: end - begin,\n requestedHeads: heads,\n originalHeads: currentHeads,\n });\n }\n }\n\n // Skip outdated docs.\n if (doc.version !== SpaceDocVersion.CURRENT) {\n continue;\n }\n\n if (!doc.objects?.[objectId]) {\n continue;\n }\n\n // Upgrade V0 object pointers to V1.\n let newId = id;\n if (objectPointerCodec.getVersion(id) === ObjectPointerVersion.V0) {\n const spaceKey = getSpaceKeyFromDoc(doc) ?? undefined;\n newId = objectPointerCodec.encode({ documentId, objectId, spaceKey });\n }\n\n yield [{ id: newId, object: doc.objects![objectId], heads }];\n } catch (error) {\n log.error('Error loading document', { heads, id, error });\n }\n }\n };\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport { DeferredTask } from '@dxos/async';\nimport { getHeads, type Doc } from '@dxos/automerge/automerge';\nimport { type DocHandle, type DocumentId } from '@dxos/automerge/automerge-repo';\nimport { Stream } from '@dxos/codec-protobuf';\nimport { Context, Resource } from '@dxos/context';\nimport { type SpaceDoc } from '@dxos/echo-protocol';\nimport { type ObjectSnapshot, type Indexer, type IdToHeads } from '@dxos/indexing';\nimport { log } from '@dxos/log';\nimport { objectPointerCodec } from '@dxos/protocols';\nimport { type IndexConfig } from '@dxos/protocols/proto/dxos/echo/indexing';\nimport {\n type QueryRequest,\n type QueryResponse,\n type QueryService,\n type QueryResult,\n} from '@dxos/protocols/proto/dxos/echo/query';\nimport { trace } from '@dxos/tracing';\n\nimport { QueryState } from './query-state';\nimport { type AutomergeHost, getSpaceKeyFromDoc } from '../automerge';\n\nexport type QueryServiceParams = {\n indexer: Indexer;\n automergeHost: AutomergeHost;\n};\n\n/**\n * Represents an active query (stream and query state connected to that stream).\n */\ntype ActiveQuery = {\n state: QueryState;\n sendResults: (results: QueryResult[]) => void;\n close: () => Promise<void>;\n};\n\nexport class QueryServiceImpl extends Resource implements QueryService {\n private readonly _queries = new Set<ActiveQuery>();\n\n private readonly _updateQueries = new DeferredTask(this._ctx, async () => {\n await Promise.all(\n Array.from(this._queries).map(async (query) => {\n try {\n const { changed } = await query.state.execQuery();\n if (changed) {\n query.sendResults(query.state.getResults());\n }\n } catch (err) {\n log.catch(err);\n }\n }),\n );\n });\n\n // TODO(burdon): OK for options, but not params. Pass separately and type readonly here.\n constructor(private readonly _params: QueryServiceParams) {\n super();\n\n trace.diagnostic({\n id: 'active-queries',\n name: 'Active Queries',\n fetch: () => {\n return Array.from(this._queries).map((query) => {\n return {\n filter: JSON.stringify(query.state.filter),\n metrics: query.state.metrics,\n };\n });\n },\n });\n }\n\n override async _open() {\n this._params.indexer.updated.on(this._ctx, () => this._updateQueries.schedule());\n }\n\n override async _close() {\n await Promise.all(Array.from(this._queries).map((query) => query.close()));\n }\n\n async setConfig(config: IndexConfig): Promise<void> {\n if (this._params.indexer.initialized) {\n log.warn('Indexer already initialized.');\n return;\n }\n this._params.indexer.setConfig(config);\n }\n\n execQuery(request: QueryRequest): Stream<QueryResponse> {\n return new Stream<QueryResponse>(({ next, close, ctx }) => {\n const query: ActiveQuery = {\n state: new QueryState({\n indexer: this._params.indexer,\n automergeHost: this._params.automergeHost,\n request,\n }),\n sendResults: (results) => {\n if (ctx.disposed) {\n return;\n }\n next({ queryId: request.queryId, results });\n },\n close: async () => {\n close();\n await query.state.close();\n this._queries.delete(query);\n },\n };\n this._queries.add(query);\n\n queueMicrotask(async () => {\n await query.state.open();\n\n try {\n const { changed } = await query.state.execQuery();\n if (changed) {\n query.sendResults(query.state.getResults());\n }\n } catch (error) {\n log.catch(error);\n }\n });\n\n return query.close;\n });\n }\n\n /**\n * Re-index all loaded documents.\n */\n async reindex() {\n log.info('Reindexing all documents...');\n const iterator = createDocumentsIterator(this._params.automergeHost);\n const ids: IdToHeads = new Map();\n for await (const documents of iterator()) {\n for (const { id, heads } of documents) {\n ids.set(id, heads);\n }\n if (ids.size % 100 === 0) {\n log.info('Collected documents...', { count: ids.size });\n }\n }\n\n log.info('Marking all documents as dirty...', { count: ids.size });\n await this._params.indexer.reindex(ids);\n }\n}\n\n/**\n * Factory for `getAllDocuments` iterator.\n */\n// TODO(dmaretskyi): Get roots from echo-host.\nconst createDocumentsIterator = (automergeHost: AutomergeHost) =>\n /**\n * Recursively get all object data blobs from loaded documents from Automerge Repo.\n */\n // TODO(mykola): Unload automerge handles after usage.\n async function* getAllDocuments(): AsyncGenerator<ObjectSnapshot[], void, void> {\n /** visited automerge handles */\n const visited = new Set<string>();\n\n async function* getObjectsFromHandle(handle: DocHandle<any>): AsyncGenerator<ObjectSnapshot[]> {\n if (visited.has(handle.documentId)) {\n return;\n }\n const doc: Doc<SpaceDoc> = handle.docSync();\n\n const spaceKey = getSpaceKeyFromDoc(doc) ?? undefined;\n\n if (doc.objects) {\n yield Object.entries(doc.objects as { [key: string]: any }).map(([objectId, object]) => {\n return {\n id: objectPointerCodec.encode({ documentId: handle.documentId, objectId, spaceKey }),\n object,\n heads: getHeads(doc),\n };\n });\n }\n\n if (doc.links) {\n for (const id of Object.values(doc.links as { [echoId: string]: string })) {\n const urlString = id.toString();\n if (visited.has(urlString)) {\n continue;\n }\n const linkHandle = await automergeHost.loadDoc(Context.default(), urlString as DocumentId);\n for await (const result of getObjectsFromHandle(linkHandle)) {\n yield result;\n }\n }\n }\n\n visited.add(handle.documentId);\n }\n\n // TODO(mykola): Use list of roots instead of iterating over all handles.\n for (const handle of Object.values(automergeHost.repo.handles)) {\n if (visited.has(handle.documentId)) {\n continue;\n }\n for await (const result of getObjectsFromHandle(handle)) {\n yield result;\n }\n visited.add(handle.documentId);\n }\n };\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport { type DocumentId } from '@dxos/automerge/automerge-repo';\nimport { Context, LifecycleState, Resource } from '@dxos/context';\nimport { createIdFromSpaceKey } from '@dxos/echo-protocol';\nimport { type Indexer, type IndexQuery } from '@dxos/indexing';\nimport { invariant } from '@dxos/invariant';\nimport { DXN, PublicKey } from '@dxos/keys';\nimport { objectPointerCodec } from '@dxos/protocols';\nimport { type Filter as FilterProto } from '@dxos/protocols/proto/dxos/echo/filter';\nimport { type QueryRequest, type QueryResult } from '@dxos/protocols/proto/dxos/echo/query';\nimport { trace } from '@dxos/tracing';\nimport { nonNullable } from '@dxos/util';\n\nimport { type AutomergeHost, getSpaceKeyFromDoc } from '../automerge';\n\ntype QueryStateParams = {\n indexer: Indexer;\n automergeHost: AutomergeHost;\n request: QueryRequest;\n};\n\ntype QueryRunResult = {\n changed: boolean;\n};\n\nexport type QueryMetrics = {\n objectsReturned: number;\n objectsReturnedFromIndex: number;\n documentsLoaded: number;\n executionTime: number;\n indexQueryTime: number;\n documentLoadTime: number;\n};\n\n/**\n * Manages querying logic on service side.\n */\n@trace.resource()\nexport class QueryState extends Resource {\n private _results: QueryResult[] = [];\n\n /**\n * Metrics are only captured for the first run of the query since that is the most representative.\n * We plan to change the query logic so that reactive updates do not require a full re-run of the query.\n */\n private _firstRun = true;\n\n @trace.info({ depth: null })\n public readonly filter: FilterProto;\n\n @trace.info()\n public metrics: QueryMetrics = {\n objectsReturned: 0,\n objectsReturnedFromIndex: 0,\n documentsLoaded: 0,\n executionTime: 0,\n indexQueryTime: 0,\n documentLoadTime: 0,\n };\n\n @trace.info()\n get active() {\n return this._lifecycleState === LifecycleState.OPEN;\n }\n\n constructor(private readonly _params: QueryStateParams) {\n super();\n this.filter = _params.request.filter;\n }\n\n getResults() {\n return this._results;\n }\n\n // https://github.com/open-telemetry/semantic-conventions/blob/main/docs/attributes-registry/db.md#generic-database-attributes\n @trace.span({ showInBrowserTimeline: true, op: 'db.query', attributes: { 'db.system': 'echo' } })\n async execQuery(): Promise<QueryRunResult> {\n const filter = this._params.request.filter;\n\n const beginQuery = performance.now();\n\n // For object id filters, we return no results as those are handled by the SpaceQuerySource.\n const hits =\n filter.objectIds && filter.objectIds?.length > 0\n ? []\n : await this._params.indexer.execQuery(filterToIndexQuery(filter));\n if (this._firstRun) {\n this.metrics.indexQueryTime = performance.now() - beginQuery;\n }\n\n const beginFilter = performance.now();\n\n const results: QueryResult[] = (\n await Promise.all(\n hits.map(async (result) => {\n if (this._firstRun) {\n this.metrics.objectsReturnedFromIndex++;\n }\n\n const { objectId, documentId, spaceKey: spaceKeyInIndex } = objectPointerCodec.decode(result.id);\n\n let spaceKey: string | null;\n if (spaceKeyInIndex !== undefined) {\n spaceKey = spaceKeyInIndex;\n } else {\n // Indexes created by older versions of the indexer do not have the spaceKey in the index.\n // If the spaceKey is not in the index, we need to load the document to get it.\n\n if (this._firstRun) {\n this.metrics.documentsLoaded++;\n }\n\n const handle = await this._params.automergeHost.loadDoc(Context.default(), documentId as DocumentId);\n\n // `whenReady` creates a timeout so we guard it with an if to skip it if the handle is already ready.\n if (this._ctx.disposed) {\n return;\n }\n spaceKey = getSpaceKeyFromDoc(handle.docSync());\n }\n\n if (!spaceKey) {\n return;\n }\n // TODO(mykola): Remove business logic from here.\n if (\n this._params.request.filter.options?.spaces?.length &&\n !this._params.request.filter.options.spaces.some((key) => key.equals(spaceKey!))\n ) {\n return;\n }\n\n if (this._firstRun) {\n this.metrics.objectsReturned++;\n }\n\n return {\n id: objectId,\n documentId,\n spaceId: await createIdFromSpaceKey(PublicKey.from(spaceKey)),\n spaceKey: PublicKey.from(spaceKey),\n rank: result.rank,\n } satisfies QueryResult;\n }),\n )\n ).filter(nonNullable);\n\n if (this._firstRun) {\n this.metrics.documentLoadTime = performance.now() - beginFilter;\n }\n\n if (this._ctx.disposed) {\n return { changed: false };\n }\n\n const areResultsUnchanged =\n !this._firstRun &&\n this._results.length === results.length &&\n this._results.every((oldResult) => results.some((result) => result.id === oldResult.id)) &&\n results.every((result) => this._results.some((oldResult) => oldResult.id === result.id));\n\n if (this._firstRun) {\n this.metrics.executionTime = performance.now() - beginQuery;\n }\n\n this._firstRun = false;\n if (areResultsUnchanged) {\n return { changed: false };\n }\n\n this._results = results;\n return { changed: true };\n }\n}\n\n// TODO(burdon): Process Filter DSL.\nconst filterToIndexQuery = (filter: FilterProto): IndexQuery => {\n invariant(!(filter.type && (filter.or ?? []).length > 0), 'Cannot mix type and or filters.');\n invariant(\n (filter.or ?? []).every((subFilter) => !(subFilter.type && (subFilter.or ?? []).length > 0)),\n 'Cannot mix type and or filters.',\n );\n if (\n filter.type ||\n ((filter.or ?? []).length > 0 && (filter.or ?? []).every((subFilter) => !subFilter.not && subFilter.type))\n ) {\n return {\n typenames:\n filter.type && filter.type.length > 0\n ? filter.type.map((type) => dxnToIndexerTypename(DXN.parse(type)))\n : (filter.or ?? [])\n .flatMap((f) => f.type?.map((type) => dxnToIndexerTypename(DXN.parse(type))) ?? [])\n .filter(nonNullable),\n inverted: filter.not,\n };\n } else {\n // Query all objects.\n return { typenames: [] };\n }\n};\n\nconst dxnToIndexerTypename = (dxn: DXN) => {\n switch (dxn.kind) {\n case DXN.kind.TYPE:\n return dxn.parts[0];\n case DXN.kind.ECHO:\n return dxn.parts[1];\n default:\n throw new Error(`Invalid DXN kind: ${dxn.kind}`);\n }\n};\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport isEqual from 'lodash.isequal';\n\nimport { Event, UpdateScheduler } from '@dxos/async';\nimport { interpretAsDocumentId, type DocHandle, type DocumentId } from '@dxos/automerge/automerge-repo';\nimport { Resource, Context } from '@dxos/context';\nimport { type SpaceDoc } from '@dxos/echo-protocol';\nimport { type SpaceId } from '@dxos/keys';\n\nimport { DatabaseRoot } from './database-root';\n\nexport class SpaceStateManager extends Resource {\n private readonly _roots = new Map<DocumentId, DatabaseRoot>();\n private readonly _rootBySpace = new Map<SpaceId, DocumentId>();\n private readonly _perRootContext = new Map<DocumentId, Context>();\n private readonly _lastSpaceDocumentList = new Map<SpaceId, DocumentId[]>();\n\n public readonly spaceDocumentListUpdated = new Event<SpaceDocumentListUpdatedEvent>();\n\n protected override async _close(ctx: Context): Promise<void> {\n for (const [_, rootCtx] of this._perRootContext) {\n await rootCtx.dispose();\n }\n this._roots.clear();\n }\n\n get roots(): ReadonlyMap<DocumentId, DatabaseRoot> {\n return this._roots;\n }\n\n getRootByDocumentId(documentId: DocumentId): DatabaseRoot | undefined {\n return this._roots.get(documentId);\n }\n\n async assignRootToSpace(spaceId: SpaceId, handle: DocHandle<SpaceDoc>): Promise<DatabaseRoot> {\n let root: DatabaseRoot;\n if (this._roots.has(handle.documentId)) {\n root = this._roots.get(handle.documentId)!;\n } else {\n root = new DatabaseRoot(handle);\n this._roots.set(handle.documentId, root);\n }\n\n if (this._rootBySpace.get(spaceId) === root.handle.documentId) {\n return root;\n }\n\n const prevRootId = this._rootBySpace.get(spaceId);\n if (prevRootId) {\n void this._perRootContext.get(prevRootId)?.dispose();\n this._perRootContext.delete(prevRootId);\n }\n\n this._rootBySpace.set(spaceId, root.handle.documentId);\n const ctx = new Context();\n\n this._perRootContext.set(root.handle.documentId, ctx);\n\n await root.handle.whenReady();\n\n const documentListCheckScheduler = new UpdateScheduler(\n ctx,\n async () => {\n const documentIds = [root.documentId, ...root.getAllLinkedDocuments().map((url) => interpretAsDocumentId(url))];\n if (!isEqual(documentIds, this._lastSpaceDocumentList.get(spaceId))) {\n this._lastSpaceDocumentList.set(spaceId, documentIds);\n this.spaceDocumentListUpdated.emit(new SpaceDocumentListUpdatedEvent(spaceId, documentIds));\n }\n },\n { maxFrequency: 50 },\n );\n const triggerCheckOnChange = () => documentListCheckScheduler.trigger();\n root.handle.addListener('change', triggerCheckOnChange);\n ctx.onDispose(() => root.handle.removeListener('change', triggerCheckOnChange));\n\n documentListCheckScheduler.trigger();\n\n return root;\n }\n}\n\nexport class SpaceDocumentListUpdatedEvent {\n constructor(\n public readonly spaceId: SpaceId,\n public readonly documentIds: DocumentId[],\n ) {}\n}\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport type * as A from '@dxos/automerge/automerge';\nimport type { AutomergeUrl, DocHandle, DocumentId } from '@dxos/automerge/automerge-repo';\nimport { type SpaceDoc, SpaceDocVersion } from '@dxos/echo-protocol';\nimport { invariant } from '@dxos/invariant';\n\nimport { measureDocMetrics, type DocMetrics } from './automerge-metrics';\nimport { getSpaceKeyFromDoc } from '../automerge';\n\nexport class DatabaseRoot {\n constructor(private readonly _rootHandle: DocHandle<SpaceDoc>) {}\n\n get documentId(): DocumentId {\n return this._rootHandle.documentId;\n }\n\n get url() {\n return this._rootHandle.url;\n }\n\n get isLoaded(): boolean {\n return !!this._rootHandle.docSync();\n }\n\n get handle(): DocHandle<SpaceDoc> {\n return this._rootHandle;\n }\n\n docSync(): A.Doc<SpaceDoc> | null {\n return this._rootHandle.docSync();\n }\n\n getVersion(): SpaceDocVersion | null {\n const doc = this.docSync();\n if (!doc) {\n return null;\n }\n\n return doc.version ?? SpaceDocVersion.LEGACY;\n }\n\n getSpaceKey(): string | null {\n const doc = this.docSync();\n if (!doc) {\n return null;\n }\n\n return getSpaceKeyFromDoc(doc);\n }\n\n getInlineObjectCount(): number | null {\n const doc = this.docSync();\n if (!doc) {\n return null;\n }\n\n return Object.keys(doc.objects ?? {}).length;\n }\n\n getLinkedObjectCount(): number | null {\n const doc = this.docSync();\n if (!doc) {\n return null;\n }\n\n return Object.keys(doc.links ?? {}).length;\n }\n\n getAllLinkedDocuments(): AutomergeUrl[] {\n const doc = this.docSync();\n invariant(doc);\n\n // .toString() to handle RawString.\n return Object.values(doc.links ?? {}).map((s) => s.toString()) as AutomergeUrl[];\n }\n\n measureMetrics(): DocMetrics | null {\n const doc = this.docSync();\n if (!doc) {\n return null;\n }\n return measureDocMetrics(doc);\n }\n}\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport * as A from '@dxos/automerge/automerge';\nimport { log } from '@dxos/log';\n\nexport type DocMetrics = {\n compressedByteSize: number;\n loadTime: number;\n mutationCount: number;\n};\n\n/**\n * WARN: Slow to run on large docs.\n */\nexport const measureDocMetrics = (doc: A.Doc<any>): DocMetrics => {\n const snapshot = A.save(doc);\n\n const start = Date.now();\n const temp = A.load(snapshot);\n const end = Date.now();\n A.free(temp);\n\n const getAllChangesStart = Date.now();\n const mutationCount = A.getAllChanges(doc).length;\n const getAllChangesEnd = Date.now();\n\n if (getAllChangesEnd - getAllChangesStart > 300) {\n log.warn('getAllChanges took too long', { elapsed: getAllChangesEnd - getAllChangesStart });\n }\n\n return {\n compressedByteSize: snapshot.byteLength,\n loadTime: end - start,\n mutationCount,\n };\n};\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport { convertLegacyReference } from '@dxos/echo-protocol';\nimport {\n decodeReference,\n encodeReference,\n isLegacyReference,\n LEGACY_TYPE_PROPERTIES,\n type ObjectStructure,\n Reference,\n type SpaceDoc,\n SpaceDocVersion,\n} from '@dxos/echo-protocol';\nimport { TYPE_PROPERTIES } from '@dxos/echo-schema';\nimport { deepMapValuesAsync } from '@dxos/util';\n\nexport const convertLegacyReferences = async (doc: SpaceDoc): Promise<SpaceDoc> => {\n const newDoc = await deepMapValuesAsync(doc, async (value, recurse) => {\n if (isLegacyReference(value)) {\n return convertLegacyReference(value);\n }\n return recurse(value);\n });\n\n newDoc.version = SpaceDocVersion.CURRENT;\n return newDoc;\n};\n\nexport const convertLegacySpaceRootDoc = async (root: SpaceDoc): Promise<SpaceDoc> => {\n // Convert references.\n const newDoc: SpaceDoc = await convertLegacyReferences(root);\n\n // Update properties type.\n const properties = findInlineObjectOfType(newDoc, LEGACY_TYPE_PROPERTIES);\n if (properties) {\n const [_, obj] = properties;\n obj.system.type = encodeReference(Reference.fromLegacyTypename(TYPE_PROPERTIES));\n }\n\n return newDoc;\n};\n\n/**\n * Assumes properties are at root.\n */\nexport const findInlineObjectOfType = (spaceDoc: SpaceDoc, typename: string): [string, ObjectStructure] | undefined => {\n for (const id in spaceDoc.objects ?? {}) {\n const obj = spaceDoc.objects![id];\n if (obj.system.type && decodeReference(obj.system.type).objectId === typename) {\n return [id, obj];\n }\n }\n\n return undefined;\n};\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport { Mutex, scheduleTask, scheduleMicroTask } from '@dxos/async';\nimport * as A from '@dxos/automerge/automerge';\nimport { cbor } from '@dxos/automerge/automerge-repo';\nimport { Context, Resource } from '@dxos/context';\nimport { randomUUID } from '@dxos/crypto';\nimport type { CollectionId } from '@dxos/echo-protocol';\nimport { type EdgeConnection } from '@dxos/edge-client';\nimport { invariant } from '@dxos/invariant';\nimport type { SpaceId } from '@dxos/keys';\nimport { log } from '@dxos/log';\nimport { EdgeService, type AutomergeProtocolMessage, type PeerId } from '@dxos/protocols';\nimport { buf } from '@dxos/protocols/buf';\nimport {\n type Message as RouterMessage,\n MessageSchema as RouterMessageSchema,\n} from '@dxos/protocols/buf/dxos/edge/messenger_pb';\nimport { bufferToArray } from '@dxos/util';\n\nimport {\n getSpaceIdFromCollectionId,\n type EchoReplicator,\n type EchoReplicatorContext,\n type ReplicatorConnection,\n type ShouldAdvertiseParams,\n type ShouldSyncCollectionParams,\n} from '../automerge';\n\n/**\n * Delay before restarting the connection after receiving a forbidden error.\n */\nconst INITIAL_RESTART_DELAY = 500;\nconst RESTART_DELAY_JITTER = 250;\nconst MAX_RESTART_DELAY = 5000;\n\nexport type EchoEdgeReplicatorParams = {\n edgeConnection: EdgeConnection;\n disableSharePolicy?: boolean;\n};\n\nexport class EchoEdgeReplicator implements EchoReplicator {\n private readonly _edgeConnection: EdgeConnection;\n private readonly _mutex = new Mutex();\n\n private _ctx?: Context = undefined;\n private _context: EchoReplicatorContext | null = null;\n private _connectedSpaces = new Set<SpaceId>();\n private _connections = new Map<SpaceId, EdgeReplicatorConnection>();\n private _sharePolicyEnabled = true;\n\n constructor({ edgeConnection, disableSharePolicy }: EchoEdgeReplicatorParams) {\n this._edgeConnection = edgeConnection;\n this._sharePolicyEnabled = !disableSharePolicy;\n }\n\n async connect(context: EchoReplicatorContext): Promise<void> {\n log.info('connect', { peerId: context.peerId, connectedSpaces: this._connectedSpaces.size });\n this._context = context;\n\n this._ctx = Context.default();\n this._ctx.onDispose(\n this._edgeConnection.onReconnected(() => {\n this._ctx && scheduleMicroTask(this._ctx, () => this._handleReconnect());\n }),\n );\n }\n\n private async _handleReconnect() {\n using _guard = await this._mutex.acquire();\n\n const spaces = [...this._connectedSpaces];\n for (const connection of this._connections.values()) {\n await connection.close();\n }\n this._connections.clear();\n\n if (this._context !== null) {\n for (const spaceId of spaces) {\n await this._openConnection(spaceId);\n }\n }\n }\n\n async disconnect(): Promise<void> {\n using _guard = await this._mutex.acquire();\n await this._ctx?.dispose();\n\n for (const connection of this._connections.values()) {\n await connection.close();\n }\n this._connections.clear();\n }\n\n async connectToSpace(spaceId: SpaceId) {\n using _guard = await this._mutex.acquire();\n\n this._connectedSpaces.add(spaceId);\n\n // Check if AM-repo requested that we connect to remote peers.\n if (this._context !== null) {\n await this._openConnection(spaceId);\n }\n }\n\n async disconnectFromSpace(spaceId: SpaceId) {\n using _guard = await this._mutex.acquire();\n\n this._connectedSpaces.delete(spaceId);\n\n const connection = this._connections.get(spaceId);\n if (connection) {\n await connection.close();\n this._connections.delete(spaceId);\n }\n }\n\n private async _openConnection(spaceId: SpaceId, reconnects: number = 0) {\n invariant(this._context);\n invariant(!this._connections.has(spaceId));\n\n let restartScheduled = false;\n\n const connection = new EdgeReplicatorConnection({\n edgeConnection: this._edgeConnection,\n spaceId,\n context: this._context,\n sharedPolicyEnabled: this._sharePolicyEnabled,\n onRemoteConnected: async () => {\n this._context?.onConnectionOpen(connection);\n },\n onRemoteDisconnected: async () => {\n this._context?.onConnectionClosed(connection);\n },\n onRestartRequested: async () => {\n if (!this._ctx || restartScheduled) {\n return;\n }\n\n const restartDelay =\n Math.min(MAX_RESTART_DELAY, INITIAL_RESTART_DELAY * reconnects) + Math.random() * RESTART_DELAY_JITTER;\n\n log.info('connection restart scheduled', { spaceId, reconnects, restartDelay });\n\n restartScheduled = true;\n scheduleTask(\n this._ctx,\n async () => {\n using _guard = await this._mutex.acquire();\n if (this._connections.get(spaceId) !== connection) {\n return;\n }\n\n const ctx = this._ctx;\n await connection.close(); // Will call onRemoteDisconnected\n this._connections.delete(spaceId);\n if (ctx?.disposed) {\n return;\n }\n await this._openConnection(spaceId, reconnects + 1);\n },\n restartDelay,\n );\n },\n });\n this._connections.set(spaceId, connection);\n\n await connection.open();\n }\n}\n\ntype EdgeReplicatorConnectionsParams = {\n edgeConnection: EdgeConnection;\n spaceId: SpaceId;\n context: EchoReplicatorContext;\n sharedPolicyEnabled: boolean;\n onRemoteConnected: () => Promise<void>;\n onRemoteDisconnected: () => Promise<void>;\n onRestartRequested: () => Promise<void>;\n};\n\nclass EdgeReplicatorConnection extends Resource implements ReplicatorConnection {\n private readonly _edgeConnection: EdgeConnection;\n private _remotePeerId: string | null = null;\n private readonly _targetServiceId: string;\n private readonly _spaceId: SpaceId;\n private readonly _context: EchoReplicatorContext;\n private readonly _sharedPolicyEnabled: boolean;\n private readonly _onRemoteConnected: () => Promise<void>;\n private readonly _onRemoteDisconnected: () => Promise<void>;\n private readonly _onRestartRequested: () => void;\n\n private _readableStreamController!: ReadableStreamDefaultController<AutomergeProtocolMessage>;\n\n public readable: ReadableStream<AutomergeProtocolMessage>;\n public writable: WritableStream<AutomergeProtocolMessage>;\n\n constructor({\n edgeConnection,\n spaceId,\n context,\n sharedPolicyEnabled,\n onRemoteConnected,\n onRemoteDisconnected,\n onRestartRequested,\n }: EdgeReplicatorConnectionsParams) {\n super();\n this._edgeConnection = edgeConnection;\n this._spaceId = spaceId;\n this._context = context;\n\n // Generate a unique peer id for every connection.\n // This way automerge-repo will have separate sync states for every connection.\n // This is important because the previous connection might have had some messages that failed to deliver\n // abd if we don't clear the sync-state, automerge will not attempt to deliver them again.\n this._remotePeerId = `${EdgeService.AUTOMERGE_REPLICATOR}:${spaceId}-${randomUUID()}`;\n this._targetServiceId = `${EdgeService.AUTOMERGE_REPLICATOR}:${spaceId}`;\n this._sharedPolicyEnabled = sharedPolicyEnabled;\n this._onRemoteConnected = onRemoteConnected;\n this._onRemoteDisconnected = onRemoteDisconnected;\n this._onRestartRequested = onRestartRequested;\n\n this.readable = new ReadableStream<AutomergeProtocolMessage>({\n start: (controller) => {\n this._readableStreamController = controller;\n },\n });\n\n this.writable = new WritableStream<AutomergeProtocolMessage>({\n write: async (message: AutomergeProtocolMessage, controller) => {\n await this._sendMessage(message);\n },\n });\n }\n\n protected override async _open(ctx: Context): Promise<void> {\n log('open');\n // TODO: handle reconnects\n this._ctx.onDispose(\n this._edgeConnection.onMessage((msg: RouterMessage) => {\n this._onMessage(msg);\n }),\n );\n\n await this._onRemoteConnected();\n }\n\n protected override async _close(): Promise<void> {\n log('close');\n this._readableStreamController.close();\n await this._onRemoteDisconnected();\n }\n\n get peerId(): string {\n invariant(this._remotePeerId, 'Not connected');\n return this._remotePeerId;\n }\n\n async shouldAdvertise(params: ShouldAdvertiseParams): Promise<boolean> {\n if (!this._sharedPolicyEnabled) {\n return true;\n }\n const spaceId = await this._context.getContainingSpaceIdForDocument(params.documentId);\n if (!spaceId) {\n // There's no spaceId if the document is not present locally. This means the sharePolicy check is being\n // performed on message reception, so spaceId check was already performed in _onMessage.\n return true;\n }\n return spaceId === this._spaceId;\n }\n\n shouldSyncCollection(params: ShouldSyncCollectionParams): boolean {\n if (!this._sharedPolicyEnabled) {\n return true;\n }\n const spaceId = getSpaceIdFromCollectionId(params.collectionId as CollectionId);\n return spaceId === this._spaceId;\n }\n\n private _onMessage(message: RouterMessage) {\n if (message.serviceId !== this._targetServiceId) {\n return;\n }\n\n const payload = cbor.decode(message.payload!.value) as AutomergeProtocolMessage;\n log('recv', () => {\n const decodedData =\n payload.type === 'sync' && payload.data\n ? A.decodeSyncMessage(payload.data)\n : payload.type === 'collection-state'\n ? (payload as any).state\n : payload;\n return { from: message.serviceId, type: payload.type, decodedData };\n });\n // Fix the peer id.\n payload.senderId = this._remotePeerId! as PeerId;\n this._processMessage(payload);\n }\n\n private _processMessage(message: AutomergeProtocolMessage) {\n // There's a race between the credentials being replicated that are needed for access control and the data replication.\n // AutomergeReplicator might return a Forbidden error if the credentials are not yet replicated.\n // We restart the connection with some delay to account for that.\n if (isForbiddenErrorMessage(message)) {\n this._onRestartRequested();\n return;\n }\n\n this._readableStreamController.enqueue(message);\n }\n\n private async _sendMessage(message: AutomergeProtocolMessage) {\n // Fix the peer id.\n (message as any).targetId = this._targetServiceId as PeerId;\n\n log('send', {\n type: message.type,\n senderId: message.senderId,\n targetId: (message as any).targetId,\n documentId: (message as any).documentId,\n });\n const encoded = cbor.encode(message);\n\n await this._edgeConnection.send(\n buf.create(RouterMessageSchema, {\n serviceId: this._targetServiceId,\n source: {\n identityKey: this._edgeConnection.identityKey,\n peerKey: this._edgeConnection.peerKey,\n },\n payload: { value: bufferToArray(encoded) },\n }),\n );\n }\n}\n\n/**\n * This message is sent by EDGE AutomergeReplicator when the authorization is denied.\n */\nconst isForbiddenErrorMessage = (message: AutomergeProtocolMessage) =>\n message.type === 'error' && message.message === 'Forbidden';\n"],
5
- "mappings": "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",
6
- "names": ["UpdateScheduler", "Stream", "invariant", "SpaceId", "log", "UpdateScheduler", "next", "A", "Resource", "invariant", "log", "MAX_UPDATE_FREQ", "DocumentsSynchronizer", "constructor", "_params", "_syncStates", "Map", "_pendingUpdates", "Set", "_sendUpdatesJob", "undefined", "addDocuments", "documentIds", "retryCounter", "warn", "documentId", "doc", "repo", "find", "whenReady", "then", "_startSync", "add", "trigger", "catch", "error", "removeDocuments", "get", "clearSubscriptions", "delete", "_open", "_ctx", "_checkAndSendUpdates", "bind", "maxFrequency", "_close", "join", "clear", "update", "updates", "mutation", "isNew", "loadIncremental", "_writeMutation", "has", "info", "syncState", "handle", "_subscribeForChanges", "set", "handler", "on", "off", "docsWithPendingUpdates", "Array", "from", "_getPendingChanges", "push", "length", "sendUpdates", "docSync", "lastSentHead", "saveSince", "save", "getHeads", "headsBefore", "newDoc", "equals", "Event", "asyncTimeout", "getBackend", "getHeads", "isAutomerge", "equals", "headsEquals", "save", "Repo", "Context", "Resource", "cancelWithContext", "invariant", "PublicKey", "log", "objectPointerCodec", "trace", "asyncReturn", "Event", "scheduleTask", "scheduleTaskInterval", "next", "am", "Resource", "log", "defaultMap", "MIN_QUERY_INTERVAL", "POLL_INTERVAL", "CollectionSynchronizer", "constructor", "params", "_perCollectionStates", "Map", "_connectedPeers", "Set", "remoteStateUpdated", "_sendCollectionState", "sendCollectionState", "_queryCollectionState", "queryCollectionState", "_shouldSyncCollection", "shouldSyncCollection", "_open", "ctx", "_ctx", "collectionId", "keys", "refreshCollection", "getRegisteredCollectionIds", "getLocalCollectionState", "_getPerCollectionState", "localState", "setLocalCollectionState", "state", "queueMicrotask", "disposed", "_refreshInterestedPeers", "getRemoteCollectionStates", "remoteStates", "scheduleAnotherRefresh", "peerId", "interestedPeers", "has", "lastQueried", "get", "Date", "now", "set", "onConnectionOpen", "add", "entries", "onConnectionClosed", "delete", "perCollectionState", "values", "onCollectionStateQueried", "onRemoteStateReceived", "validateCollectionState", "emit", "undefined", "diffCollectionState", "local", "remote", "allDocuments", "Object", "documents", "missingOnRemote", "missingOnLocal", "different", "documentId", "push", "equals", "forEach", "heads", "isValidDocumentId", "Error", "Array", "isArray", "some", "head", "includes", "synchronized", "Trigger", "NetworkAdapter", "LifecycleState", "invariant", "log", "nonNullable", "MESSAGE_TYPE_COLLECTION_QUERY", "MESSAGE_TYPE_COLLECTION_STATE", "isCollectionQueryMessage", "message", "type", "MESSAGE_TYPE_COLLECTION_QUERY", "isCollectionStateMessage", "MESSAGE_TYPE_COLLECTION_STATE", "EchoNetworkAdapter", "NetworkAdapter", "constructor", "_params", "_replicators", "Set", "_connections", "Map", "_lifecycleState", "LifecycleState", "CLOSED", "_connected", "Trigger", "connect", "peerId", "peerMetadata", "wake", "send", "message", "_send", "disconnect", "open", "OPEN", "log", "emit", "network", "close", "replicator", "clear", "whenConnected", "wait", "timeout", "addReplicator", "invariant", "has", "add", "onConnectionOpen", "_onConnectionOpen", "bind", "onConnectionClosed", "_onConnectionClosed", "onConnectionAuthScopeChanged", "_onConnectionAuthScopeChanged", "isDocumentInRemoteCollection", "getContainingSpaceForDocument", "getContainingSpaceIdForDocument", "documentId", "key", "createIdFromSpaceKey", "removeReplicator", "delete", "shouldAdvertise", "params", "connection", "get", "shouldSyncCollection", "queryCollectionState", "collectionId", "targetId", "type", "senderId", "sendCollectionState", "state", "connectionEntry", "Error", "writeStart", "Date", "now", "writer", "write", "then", "durationMs", "monitor", "recordMessageSent", "catch", "err", "isOpen", "recordMessageSendingFailed", "getPeersInterestedInCollection", "Array", "from", "values", "map", "filter", "nonNullable", "reader", "readable", "getReader", "writable", "getWriter", "set", "queueMicrotask", "done", "value", "read", "_onMessage", "_emitPeerCandidate", "recordPeerConnected", "isCollectionQueryMessage", "onCollectionStateQueried", "isCollectionStateMessage", "onCollectionStateReceived", "recordMessageReceived", "entry", "recordPeerDisconnected", "cancel", "abort", "createEchoPeerMetadata", "synchronized", "dxos_peerSource", "isEchoPeerMetadata", "metadata", "headsEncoding", "HeadsStore", "constructor", "db", "_db", "setHeads", "documentId", "heads", "batch", "put", "sublevel", "keyEncoding", "valueEncoding", "headsEncoding", "getHeads", "documentIds", "getMany", "LifecycleState", "Resource", "LevelDBStorageAdapter", "Resource", "constructor", "_params", "load", "keyArray", "_lifecycleState", "LifecycleState", "OPEN", "undefined", "startMs", "Date", "now", "chunk", "db", "get", "encodingOptions", "monitor", "recordBytesLoaded", "byteLength", "recordLoadDuration", "err", "isLevelDbNotFoundError", "save", "binary", "batch", "callbacks", "beforeSave", "path", "put", "Buffer", "from", "write", "recordBytesStored", "afterSave", "recordStoreDuration", "remove", "del", "loadRange", "keyPrefix", "result", "key", "value", "iterator", "gte", "lte", "push", "data", "removeRange", "keyEncoder", "encode", "map", "k", "replaceAll", "join", "decode", "toString", "split", "format", "keyEncoding", "valueEncoding", "code", "AutomergeHost", "Resource", "constructor", "db", "indexMetadataStore", "dataMonitor", "_collectionSynchronizer", "CollectionSynchronizer", "queryCollectionState", "_queryCollectionState", "bind", "sendCollectionState", "_sendCollectionState", "shouldSyncCollection", "_shouldSyncCollection", "collectionStateUpdated", "Event", "_db", "_storage", "LevelDBStorageAdapter", "sublevel", "callbacks", "beforeSave", "params", "_beforeSave", "afterSave", "key", "_afterSave", "monitor", "_echoNetworkAdapter", "EchoNetworkAdapter", "getContainingSpaceForDocument", "_getContainingSpaceForDocument", "isDocumentInRemoteCollection", "_isDocumentInRemoteCollection", "onCollectionStateQueried", "_onCollectionStateQueried", "onCollectionStateReceived", "_onCollectionStateReceived", "_headsStore", "HeadsStore", "_indexMetadataStore", "_open", "_peerId", "PublicKey", "random", "toHex", "open", "_repo", "Repo", "peerId", "sharePolicy", "_sharePolicy", "storage", "network", "wrap", "on", "_ctx", "e", "_onPeerConnected", "_onPeerDisconnected", "remoteStateUpdated", "collectionId", "_onRemoteCollectionStateUpdated", "emit", "whenConnected", "_close", "close", "dispose", "repo", "loadedDocsCount", "Object", "keys", "handles", "length", "addReplicator", "replicator", "removeReplicator", "loadDoc", "ctx", "documentId", "opts", "handle", "find", "isReady", "timeout", "cancelWithContext", "whenReady", "asyncTimeout", "createDoc", "initialValue", "preserveHistory", "isAutomerge", "TypeError", "import", "save", "create", "waitUntilHeadsReplicated", "heads", "entries", "documentIds", "map", "entry", "documentHeads", "getHeads", "headsToWait", "filter", "index", "targetHeads", "currentHeads", "headsEquals", "Promise", "all", "Context", "default", "waitForHeads", "flush", "reIndexHeads", "log", "info", "inState", "warn", "doc", "docSync", "invariant", "batch", "setHeads", "write", "startsWith", "peerMetadata", "peerMetadataByPeerId", "isEchoPeerMetadata", "shouldAdvertise", "path", "spaceKey", "getSpaceKeyFromDoc", "undefined", "objectIds", "objects", "encodedIds", "objectId", "objectPointerCodec", "encode", "idToLastHash", "Map", "id", "markDirty", "notifyMarkedDirty", "document", "_onHeadsChanged", "_automergePeers", "peers", "getRegisteredCollectionIds", "remoteCollections", "getRemoteCollectionStates", "remotePeerDocs", "get", "documents", "spaceKeyHex", "from", "loadedDocuments", "result", "storeRequestIds", "storeResultIndices", "push", "storedHeads", "i", "getLocalCollectionState", "refreshCollection", "getCollectionSyncState", "localState", "remoteState", "state", "diff", "diffCollectionState", "missingOnRemote", "missingOnLocal", "differentDocuments", "different", "localDocumentCount", "remoteDocumentCount", "updateLocalCollectionState", "fromEntries", "setLocalCollectionState", "onRemoteStateReceived", "decodeCollectionState", "encodeCollectionState", "onConnectionOpen", "onConnectionClosed", "toReplicate", "count", "collectionsChanged", "Set", "newState", "structuredClone", "add", "trace", "depth", "span", "showInBrowserTimeline", "resource", "rawSpaceKey", "access", "experimental_spaceKey", "String", "unavailableHeads", "waitForCondition", "changeHash", "values", "changeIsPresentInDoc", "delete", "size", "getBackend", "getChangeByHash", "invariant", "PublicKey", "log", "ComplexSet", "defaultMap", "A", "cbor", "Resource", "invariant", "log", "AutomergeReplicator", "DEFAULT_FACTORY", "params", "MeshReplicatorConnection", "constructor", "_params", "remoteDeviceKey", "_remotePeerId", "_isEnabled", "readableStreamController", "readable", "ReadableStream", "start", "controller", "_ctx", "onDispose", "close", "writable", "WritableStream", "write", "message", "logSendSync", "replicatorExtension", "sendSyncMessage", "payload", "encode", "err", "error", "_disconnectIfEnabled", "createAutomergeReplicator", "replicatorFactory", "peerId", "ownPeerId", "onStartReplication", "info", "remotePeerId", "id", "thisPeerId", "toHex", "onRemoteConnected", "onSyncMessage", "decode", "enqueue", "onClose", "onRemoteDisconnected", "shouldAdvertise", "shouldSyncCollection", "enable", "disable", "decodedSyncMessage", "type", "data", "decodeSyncMessage", "undefined", "sync", "headsLength", "heads", "length", "requesting", "need", "sendingChanges", "changes", "from", "senderId", "to", "targetId", "invariant", "SpaceId", "deriveCollectionIdFromSpaceId", "spaceId", "getSpaceIdFromCollectionId", "collectionId", "replace", "isValid", "MeshEchoReplicator", "_connections", "Set", "_connectionsPerPeer", "Map", "_authorizedDevices", "_context", "connect", "context", "disconnect", "connection", "values", "onConnectionClosed", "close", "clear", "createExtension", "extensionFactory", "invariant", "MeshReplicatorConnection", "ownPeerId", "peerId", "replicatorFactory", "onRemoteConnected", "log", "has", "onConnectionAuthScopeChanged", "set", "onConnectionOpen", "enable", "onRemoteDisconnected", "delete", "disable", "shouldAdvertise", "params", "documentId", "spaceKey", "getContainingSpaceForDocument", "remoteDocumentExists", "isDocumentInRemoteCollection", "spaceId", "createIdFromSpaceKey", "authorizedDevices", "get", "remoteDeviceKey", "isAuthorized", "localPeer", "remotePeer", "deviceKey", "err", "catch", "shouldSyncCollection", "collectionId", "getSpaceIdFromCollectionId", "add", "replicatorExtension", "authorizeDevice", "defaultMap", "ComplexSet", "PublicKey", "hash", "equals", "trace", "CircularBuffer", "mapValues", "SlidingWindowSummary", "PER_SECOND_RATE_AVG_WINDOW_SIZE", "DEFAULT_AVG_WINDOW_SIZE", "EchoDataMonitor", "constructor", "_params", "timeSeriesLength", "_lastTick", "_activeCounters", "createLocalCounters", "_localTimeSeries", "createLocalTimeSeries", "_storageAverages", "createStorageAverages", "_replicationAverages", "createNetworkAverages", "_sizeByMessageType", "_lastReceivedMessages", "CircularBuffer", "_lastSentMessages", "_connectionsCount", "tick", "timeMs", "_advanceTimeWindow", "computeStats", "meta", "rateAverageOverSeconds", "storage", "reads", "payloadSize", "loadedChunkSize", "average", "opDuration", "loadDuration", "countPerSecond", "loadsPerSecond", "writes", "storedChunkSize", "storeDuration", "storesPerSecond", "replicator", "connections", "receivedMessages", "receivedMessageSize", "receivedPerSecond", "sentMessages", "sentMessageSize", "sendDuration", "sentPerSecond", "failedPerSecond", "sendsFailedPerSecond", "countByMessageType", "_computeMessageHistogram", "avgSizeByMessageType", "mapValues", "summary", "connectionsCount", "lastPerSecondStats", "_lastCompleteCounters", "timeSeries", "replication", "messagesByPeerId", "millisPassed", "oldMetrics", "Object", "freeze", "peerId", "keys", "byPeerId", "createMessageCounter", "_addToTimeSeries", "Math", "abs", "_reportPerSecondRate", "values", "key", "value", "entries", "push", "length", "shift", "metrics", "toReport", "loadedChunks", "storedChunks", "received", "sent", "metricName", "metric", "record", "trace", "distribution", "increment", "tags", "status", "failed", "recordPeerConnected", "recordPeerDisconnected", "recordBytesStored", "count", "storedBytes", "unit", "recordLoadDuration", "durationMs", "recordStoreDuration", "recordBytesLoaded", "loadedBytes", "recordMessageSent", "message", "duration", "metricsGroupName", "bytes", "getByteCount", "type", "isAutomergeProtocolMessage", "success", "messageSize", "messageCounts", "_getStatsForType", "targetId", "recordMessageReceived", "senderId", "recordMessageSendingFailed", "createSlidingWindow", "byType", "groupKey", "result", "receivedMessage", "counters", "resource", "isCollectionQueryMessage", "isCollectionStateMessage", "overrides", "SlidingWindowSummary", "dataPoints", "precision", "data", "byteLength", "documentId", "DataServiceImpl", "constructor", "params", "_subscriptions", "Map", "_automergeHost", "automergeHost", "_updateIndexes", "updateIndexes", "subscribe", "request", "Stream", "next", "ready", "synchronizer", "DocumentsSynchronizer", "repo", "sendUpdates", "updates", "open", "then", "set", "subscriptionId", "catch", "err", "log", "close", "updateSubscription", "get", "invariant", "addIds", "length", "addDocuments", "removeIds", "removeDocuments", "update", "flush", "getDocumentHeads", "documentIds", "heads", "entries", "getHeads", "map", "idx", "documentId", "waitUntilHeadsReplicated", "options", "reIndexHeads", "subscribeSpaceSyncState", "ctx", "SpaceId", "isValid", "spaceId", "collectionId", "deriveCollectionIdFromSpaceId", "scheduler", "UpdateScheduler", "state", "getCollectionSyncState", "peers", "peer", "peerId", "missingOnRemote", "missingOnLocal", "differentDocuments", "localDocumentCount", "remoteDocumentCount", "collectionStateUpdated", "on", "e", "trigger", "LifecycleState", "Resource", "todo", "createIdFromSpaceKey", "SpaceDocVersion", "IndexMetadataStore", "IndexStore", "Indexer", "invariant", "IndexKind", "trace", "A", "Context", "SpaceDocVersion", "invariant", "log", "ObjectPointerVersion", "objectPointerCodec", "LOG_VIEW_OPERATION_THRESHOLD", "createSelectedDocumentsIterator", "automergeHost", "loadDocuments", "objects", "id", "heads", "entries", "documentId", "objectId", "objectPointerCodec", "decode", "handle", "loadDoc", "Context", "default", "doc", "docSync", "invariant", "currentHeads", "getHeads", "equals", "begin", "Date", "now", "view", "end", "log", "info", "duration", "requestedHeads", "originalHeads", "version", "SpaceDocVersion", "CURRENT", "newId", "getVersion", "ObjectPointerVersion", "V0", "spaceKey", "getSpaceKeyFromDoc", "undefined", "encode", "object", "error", "DeferredTask", "getHeads", "Stream", "Context", "Resource", "log", "objectPointerCodec", "trace", "Context", "LifecycleState", "Resource", "createIdFromSpaceKey", "invariant", "DXN", "PublicKey", "objectPointerCodec", "trace", "nonNullable", "QueryState", "Resource", "active", "_lifecycleState", "LifecycleState", "OPEN", "constructor", "_params", "_results", "_firstRun", "metrics", "objectsReturned", "objectsReturnedFromIndex", "documentsLoaded", "executionTime", "indexQueryTime", "documentLoadTime", "filter", "request", "getResults", "execQuery", "beginQuery", "performance", "now", "hits", "objectIds", "length", "indexer", "filterToIndexQuery", "beginFilter", "results", "Promise", "all", "map", "result", "objectId", "documentId", "spaceKey", "spaceKeyInIndex", "objectPointerCodec", "decode", "id", "undefined", "handle", "automergeHost", "loadDoc", "Context", "default", "_ctx", "disposed", "getSpaceKeyFromDoc", "docSync", "options", "spaces", "some", "key", "equals", "spaceId", "createIdFromSpaceKey", "PublicKey", "from", "rank", "nonNullable", "changed", "areResultsUnchanged", "every", "oldResult", "trace", "info", "depth", "span", "showInBrowserTimeline", "op", "attributes", "resource", "invariant", "type", "or", "subFilter", "not", "typenames", "dxnToIndexerTypename", "DXN", "parse", "flatMap", "f", "inverted", "dxn", "kind", "TYPE", "parts", "ECHO", "Error", "QueryServiceImpl", "Resource", "constructor", "_params", "_queries", "Set", "_updateQueries", "DeferredTask", "_ctx", "Promise", "all", "Array", "from", "map", "query", "changed", "state", "execQuery", "sendResults", "getResults", "err", "log", "catch", "trace", "diagnostic", "id", "name", "fetch", "filter", "JSON", "stringify", "metrics", "_open", "indexer", "updated", "on", "schedule", "_close", "close", "setConfig", "config", "initialized", "warn", "request", "Stream", "next", "ctx", "QueryState", "automergeHost", "results", "disposed", "queryId", "delete", "add", "queueMicrotask", "open", "error", "reindex", "info", "iterator", "createDocumentsIterator", "ids", "Map", "documents", "heads", "set", "size", "count", "getAllDocuments", "visited", "getObjectsFromHandle", "handle", "has", "documentId", "doc", "docSync", "spaceKey", "getSpaceKeyFromDoc", "undefined", "objects", "Object", "entries", "objectId", "object", "objectPointerCodec", "encode", "getHeads", "links", "values", "urlString", "toString", "linkHandle", "loadDoc", "Context", "default", "result", "repo", "handles", "isEqual", "Event", "UpdateScheduler", "interpretAsDocumentId", "Resource", "Context", "SpaceDocVersion", "invariant", "A", "log", "measureDocMetrics", "doc", "snapshot", "save", "start", "Date", "now", "temp", "load", "end", "free", "getAllChangesStart", "mutationCount", "getAllChanges", "length", "getAllChangesEnd", "warn", "elapsed", "compressedByteSize", "byteLength", "loadTime", "DatabaseRoot", "constructor", "_rootHandle", "documentId", "url", "isLoaded", "docSync", "handle", "getVersion", "doc", "version", "SpaceDocVersion", "LEGACY", "getSpaceKey", "getSpaceKeyFromDoc", "getInlineObjectCount", "Object", "keys", "objects", "length", "getLinkedObjectCount", "links", "getAllLinkedDocuments", "invariant", "values", "map", "s", "toString", "measureMetrics", "measureDocMetrics", "SpaceStateManager", "Resource", "_roots", "Map", "_rootBySpace", "_perRootContext", "_lastSpaceDocumentList", "spaceDocumentListUpdated", "Event", "_close", "ctx", "_", "rootCtx", "dispose", "clear", "roots", "getRootByDocumentId", "documentId", "get", "assignRootToSpace", "spaceId", "handle", "root", "has", "DatabaseRoot", "set", "prevRootId", "delete", "Context", "whenReady", "documentListCheckScheduler", "UpdateScheduler", "documentIds", "getAllLinkedDocuments", "map", "url", "interpretAsDocumentId", "isEqual", "emit", "SpaceDocumentListUpdatedEvent", "maxFrequency", "triggerCheckOnChange", "trigger", "addListener", "onDispose", "removeListener", "constructor", "INDEXER_CONFIG", "enabled", "indexes", "kind", "IndexKind", "Kind", "SCHEMA_MATCH", "EchoHost", "Resource", "constructor", "kv", "_spaceStateManager", "SpaceStateManager", "_indexMetadataStore", "IndexMetadataStore", "db", "sublevel", "_echoDataMonitor", "EchoDataMonitor", "_automergeHost", "AutomergeHost", "dataMonitor", "indexMetadataStore", "_indexer", "Indexer", "indexStore", "IndexStore", "metadataStore", "loadDocuments", "createSelectedDocumentsIterator", "indexCooldownTime", "process", "undefined", "setConfig", "_queryService", "QueryServiceImpl", "automergeHost", "indexer", "_dataService", "DataServiceImpl", "updateIndexes", "trace", "diagnostic", "id", "name", "fetch", "dataStats", "computeStats", "loadedDocsCount", "Array", "from", "roots", "values", "map", "root", "url", "isLoaded", "spaceKey", "getSpaceKey", "inlineObjects", "getInlineObjectCount", "linkedObjects", "getLinkedObjectCount", "measureMetrics", "queryService", "dataService", "automergeRepo", "repo", "_open", "ctx", "open", "spaceDocumentListUpdated", "on", "_ctx", "e", "updateLocalCollectionState", "deriveCollectionIdFromSpaceId", "spaceId", "documentIds", "_close", "close", "flush", "loadDoc", "documentId", "opts", "createDoc", "initialValue", "createSpaceRoot", "invariant", "_lifecycleState", "LifecycleState", "OPEN", "createIdFromSpaceKey", "automergeRoot", "version", "SpaceDocVersion", "CURRENT", "access", "toHex", "objects", "links", "openSpaceRoot", "automergeUrl", "handle", "find", "assignRootToSpace", "closeSpaceRoot", "todo", "addReplicator", "replicator", "removeReplicator", "getSpaceSyncState", "collectionId", "getCollectionSyncState", "convertLegacyReference", "decodeReference", "encodeReference", "isLegacyReference", "LEGACY_TYPE_PROPERTIES", "Reference", "SpaceDocVersion", "TYPE_PROPERTIES", "deepMapValuesAsync", "convertLegacyReferences", "doc", "newDoc", "deepMapValuesAsync", "value", "recurse", "isLegacyReference", "convertLegacyReference", "version", "SpaceDocVersion", "CURRENT", "convertLegacySpaceRootDoc", "root", "properties", "findInlineObjectOfType", "LEGACY_TYPE_PROPERTIES", "_", "obj", "system", "type", "encodeReference", "Reference", "fromLegacyTypename", "TYPE_PROPERTIES", "spaceDoc", "typename", "id", "objects", "decodeReference", "objectId", "undefined", "Mutex", "scheduleTask", "scheduleMicroTask", "A", "cbor", "Context", "Resource", "randomUUID", "invariant", "log", "EdgeService", "buf", "MessageSchema", "RouterMessageSchema", "bufferToArray", "INITIAL_RESTART_DELAY", "RESTART_DELAY_JITTER", "MAX_RESTART_DELAY", "EchoEdgeReplicator", "constructor", "edgeConnection", "disableSharePolicy", "_mutex", "Mutex", "_ctx", "undefined", "_context", "_connectedSpaces", "Set", "_connections", "Map", "_sharePolicyEnabled", "_edgeConnection", "connect", "context", "log", "info", "peerId", "connectedSpaces", "size", "Context", "default", "onDispose", "onReconnected", "scheduleMicroTask", "_handleReconnect", "_guard", "acquire", "spaces", "connection", "values", "close", "clear", "spaceId", "_openConnection", "disconnect", "dispose", "connectToSpace", "add", "disconnectFromSpace", "delete", "get", "reconnects", "invariant", "has", "restartScheduled", "EdgeReplicatorConnection", "sharedPolicyEnabled", "onRemoteConnected", "onConnectionOpen", "onRemoteDisconnected", "onConnectionClosed", "onRestartRequested", "restartDelay", "Math", "min", "random", "scheduleTask", "ctx", "disposed", "set", "open", "Resource", "_remotePeerId", "_spaceId", "EdgeService", "AUTOMERGE_REPLICATOR", "randomUUID", "_targetServiceId", "_sharedPolicyEnabled", "_onRemoteConnected", "_onRemoteDisconnected", "_onRestartRequested", "readable", "ReadableStream", "start", "controller", "_readableStreamController", "writable", "WritableStream", "write", "message", "_sendMessage", "_open", "onMessage", "msg", "_onMessage", "_close", "shouldAdvertise", "params", "getContainingSpaceIdForDocument", "documentId", "shouldSyncCollection", "getSpaceIdFromCollectionId", "collectionId", "serviceId", "payload", "cbor", "decode", "value", "decodedData", "type", "data", "decodeSyncMessage", "state", "from", "senderId", "_processMessage", "isForbiddenErrorMessage", "enqueue", "targetId", "encoded", "encode", "send", "buf", "create", "RouterMessageSchema", "source", "identityKey", "peerKey", "bufferToArray"]
3
+ "sources": ["../../../src/db-host/data-service.ts", "../../../src/db-host/documents-synchronizer.ts", "../../../src/automerge/automerge-host.ts", "../../../src/automerge/collection-synchronizer.ts", "../../../src/automerge/echo-network-adapter.ts", "../../../src/automerge/network-protocol.ts", "../../../src/automerge/heads-store.ts", "../../../src/automerge/leveldb-storage-adapter.ts", "../../../src/automerge/mesh-echo-replicator.ts", "../../../src/automerge/mesh-echo-replicator-connection.ts", "../../../src/automerge/space-collection.ts", "../../../src/automerge/echo-data-monitor.ts", "../../../src/db-host/echo-host.ts", "../../../src/db-host/documents-iterator.ts", "../../../src/db-host/query-service.ts", "../../../src/db-host/query-state.ts", "../../../src/db-host/space-state-manager.ts", "../../../src/db-host/database-root.ts", "../../../src/db-host/automerge-metrics.ts", "../../../src/edge/echo-edge-replicator.ts", "../../../src/util.ts"],
4
+ "sourcesContent": ["//\n// Copyright 2021 DXOS.org\n//\n\nimport { UpdateScheduler } from '@dxos/async';\nimport { type DocumentId } from '@dxos/automerge/automerge-repo';\nimport { type RequestOptions } from '@dxos/codec-protobuf';\nimport { Stream } from '@dxos/codec-protobuf/stream';\nimport { invariant } from '@dxos/invariant';\nimport { SpaceId } from '@dxos/keys';\nimport { log } from '@dxos/log';\nimport {\n type DataService,\n type FlushRequest,\n type SubscribeRequest,\n type BatchedDocumentUpdates,\n type UpdateSubscriptionRequest,\n type GetDocumentHeadsRequest,\n type GetDocumentHeadsResponse,\n type ReIndexHeadsRequest,\n type WaitUntilHeadsReplicatedRequest,\n type UpdateRequest,\n type GetSpaceSyncStateRequest,\n type SpaceSyncState,\n} from '@dxos/protocols/proto/dxos/echo/service';\n\nimport { DocumentsSynchronizer } from './documents-synchronizer';\nimport { deriveCollectionIdFromSpaceId, type AutomergeHost } from '../automerge';\n\nexport type DataServiceParams = {\n automergeHost: AutomergeHost;\n updateIndexes: () => Promise<void>;\n};\n\n/**\n * Data sync between client and services.\n */\n// TODO(burdon): Move to client-services.\nexport class DataServiceImpl implements DataService {\n /**\n * Map of subscriptions.\n * subscriptionId -> DocumentsSynchronizer\n */\n private readonly _subscriptions = new Map<string, DocumentsSynchronizer>();\n\n private readonly _automergeHost: AutomergeHost;\n private readonly _updateIndexes: () => Promise<void>;\n\n constructor(params: DataServiceParams) {\n this._automergeHost = params.automergeHost;\n this._updateIndexes = params.updateIndexes;\n }\n\n subscribe(request: SubscribeRequest): Stream<BatchedDocumentUpdates> {\n return new Stream<BatchedDocumentUpdates>(({ next, ready }) => {\n const synchronizer = new DocumentsSynchronizer({\n repo: this._automergeHost.repo,\n sendUpdates: (updates) => next(updates),\n });\n synchronizer\n .open()\n .then(() => {\n this._subscriptions.set(request.subscriptionId, synchronizer);\n ready();\n })\n .catch((err) => log.catch(err));\n return () => synchronizer.close();\n });\n }\n\n async updateSubscription(request: UpdateSubscriptionRequest) {\n const synchronizer = this._subscriptions.get(request.subscriptionId);\n invariant(synchronizer, 'Subscription not found');\n\n if (request.addIds?.length) {\n await synchronizer.addDocuments(request.addIds as DocumentId[]);\n }\n if (request.removeIds?.length) {\n await synchronizer.removeDocuments(request.removeIds as DocumentId[]);\n }\n }\n\n async update(request: UpdateRequest): Promise<void> {\n if (!request.updates) {\n return;\n }\n const synchronizer = this._subscriptions.get(request.subscriptionId);\n invariant(synchronizer, 'Subscription not found');\n\n synchronizer.update(request.updates);\n }\n\n async flush(request: FlushRequest): Promise<void> {\n await this._automergeHost.flush(request);\n }\n\n async getDocumentHeads(request: GetDocumentHeadsRequest): Promise<GetDocumentHeadsResponse> {\n const documentIds = request.documentIds;\n if (!documentIds) {\n return { heads: { entries: [] } };\n }\n const heads = await this._automergeHost.getHeads(documentIds as DocumentId[]);\n return {\n heads: {\n entries: heads.map((heads, idx) => ({ documentId: documentIds[idx], heads })),\n },\n };\n }\n\n async waitUntilHeadsReplicated(\n request: WaitUntilHeadsReplicatedRequest,\n options?: RequestOptions | undefined,\n ): Promise<void> {\n await this._automergeHost.waitUntilHeadsReplicated(request.heads);\n }\n\n async reIndexHeads(request: ReIndexHeadsRequest, options?: RequestOptions): Promise<void> {\n await this._automergeHost.reIndexHeads((request.documentIds ?? []) as DocumentId[]);\n }\n\n async updateIndexes() {\n await this._updateIndexes();\n }\n\n subscribeSpaceSyncState(request: GetSpaceSyncStateRequest): Stream<SpaceSyncState> {\n return new Stream<SpaceSyncState>(({ ctx, next, ready }) => {\n invariant(SpaceId.isValid(request.spaceId));\n const collectionId = deriveCollectionIdFromSpaceId(request.spaceId);\n\n const scheduler = new UpdateScheduler(ctx, async () => {\n const state = await this._automergeHost.getCollectionSyncState(collectionId);\n\n next({\n peers: state.peers.map((peer) => ({\n peerId: peer.peerId,\n missingOnRemote: peer.missingOnRemote,\n missingOnLocal: peer.missingOnLocal,\n differentDocuments: peer.differentDocuments,\n localDocumentCount: peer.localDocumentCount,\n remoteDocumentCount: peer.remoteDocumentCount,\n })),\n });\n });\n\n this._automergeHost.collectionStateUpdated.on(ctx, (e) => {\n if (e.collectionId === collectionId) {\n scheduler.trigger();\n }\n });\n scheduler.trigger();\n });\n }\n}\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport { UpdateScheduler } from '@dxos/async';\nimport { next as A, type Heads } from '@dxos/automerge/automerge';\nimport { type Repo, type DocHandle, type DocumentId } from '@dxos/automerge/automerge-repo';\nimport { Resource } from '@dxos/context';\nimport { type SpaceDoc } from '@dxos/echo-protocol';\nimport { invariant } from '@dxos/invariant';\nimport { log } from '@dxos/log';\nimport { type BatchedDocumentUpdates, type DocumentUpdate } from '@dxos/protocols/proto/dxos/echo/service';\n\nconst MAX_UPDATE_FREQ = 10; // [updates/sec]\n\nexport type DocumentsSynchronizerParams = {\n repo: Repo;\n sendUpdates: (updates: BatchedDocumentUpdates) => void;\n};\n\ninterface DocSyncState {\n handle: DocHandle<SpaceDoc>;\n lastSentHead?: Heads;\n clearSubscriptions?: () => void;\n}\n\n/**\n * Manages a connection and replication between worker's Automerge Repo and the client's Repo.\n */\nexport class DocumentsSynchronizer extends Resource {\n private readonly _syncStates = new Map<DocumentId, DocSyncState>();\n /**\n * Documents that have pending updates.\n * Used to batch updates.\n */\n private readonly _pendingUpdates = new Set<DocumentId>();\n\n /**\n * Job that schedules if there are pending updates.\n */\n private _sendUpdatesJob?: UpdateScheduler = undefined;\n\n constructor(private readonly _params: DocumentsSynchronizerParams) {\n super();\n }\n\n addDocuments(documentIds: DocumentId[], retryCounter = 0) {\n if (retryCounter > 3) {\n log.warn('Failed to load document, retry limit reached', { documentIds });\n return;\n }\n\n for (const documentId of documentIds) {\n const doc = this._params.repo.find(documentId as DocumentId);\n doc\n .whenReady()\n .then(() => {\n this._startSync(doc);\n this._pendingUpdates.add(doc.documentId);\n this._sendUpdatesJob!.trigger();\n })\n .catch((error) => {\n log.warn('Failed to load document, wraparound', { documentId, error });\n this.addDocuments([documentId], retryCounter + 1);\n });\n }\n }\n\n removeDocuments(documentIds: DocumentId[]) {\n for (const documentId of documentIds) {\n this._syncStates.get(documentId)?.clearSubscriptions?.();\n this._syncStates.delete(documentId);\n this._pendingUpdates.delete(documentId);\n }\n }\n\n protected override async _open(): Promise<void> {\n this._sendUpdatesJob = new UpdateScheduler(this._ctx, this._checkAndSendUpdates.bind(this), {\n maxFrequency: MAX_UPDATE_FREQ,\n });\n }\n\n protected override async _close(): Promise<void> {\n await this._sendUpdatesJob!.join();\n this._syncStates.clear();\n }\n\n update(updates: DocumentUpdate[]) {\n for (const { documentId, mutation, isNew } of updates) {\n if (isNew) {\n const doc = this._params.repo.find(documentId as DocumentId);\n doc.update((doc) => A.loadIncremental(doc, mutation));\n this._startSync(doc);\n } else {\n this._writeMutation(documentId as DocumentId, mutation);\n }\n }\n }\n\n private _startSync(doc: DocHandle<SpaceDoc>) {\n if (this._syncStates.has(doc.documentId)) {\n log.info('Document already being synced', { documentId: doc.documentId });\n return;\n }\n\n const syncState: DocSyncState = { handle: doc };\n this._subscribeForChanges(syncState);\n this._syncStates.set(doc.documentId, syncState);\n }\n\n _subscribeForChanges(syncState: DocSyncState) {\n const handler = () => {\n this._pendingUpdates.add(syncState.handle.documentId);\n this._sendUpdatesJob!.trigger();\n };\n syncState.handle.on('heads-changed', handler);\n syncState.clearSubscriptions = () => syncState.handle.off('heads-changed', handler);\n }\n\n private async _checkAndSendUpdates() {\n const updates: DocumentUpdate[] = [];\n\n const docsWithPendingUpdates = Array.from(this._pendingUpdates);\n this._pendingUpdates.clear();\n\n for (const documentId of docsWithPendingUpdates) {\n const update = this._getPendingChanges(documentId);\n if (update) {\n updates.push({\n documentId,\n mutation: update,\n });\n }\n }\n\n if (updates.length > 0) {\n this._params.sendUpdates({ updates });\n }\n }\n\n private _getPendingChanges(documentId: DocumentId): Uint8Array | void {\n const syncState = this._syncStates.get(documentId);\n invariant(syncState, 'Sync state for document not found');\n const doc = syncState.handle.docSync();\n if (!doc) {\n return;\n }\n const mutation = syncState.lastSentHead ? A.saveSince(doc, syncState.lastSentHead) : A.save(doc);\n if (mutation.length === 0) {\n return;\n }\n syncState.lastSentHead = A.getHeads(doc);\n return mutation;\n }\n\n private _writeMutation(documentId: DocumentId, mutation: Uint8Array) {\n const syncState = this._syncStates.get(documentId);\n invariant(syncState, 'Sync state for document not found');\n syncState.handle.update((doc) => {\n const headsBefore = A.getHeads(doc);\n const newDoc = A.loadIncremental(doc, mutation);\n if (A.equals(headsBefore, syncState.lastSentHead)) {\n syncState.lastSentHead = A.getHeads(newDoc);\n }\n return newDoc;\n });\n }\n}\n", "//\n// Copyright 2023 DXOS.org\n//\n\nimport { Event, asyncTimeout } from '@dxos/async';\nimport {\n getBackend,\n getHeads,\n isAutomerge,\n equals as headsEquals,\n save,\n type Doc,\n type Heads,\n} from '@dxos/automerge/automerge';\nimport {\n type DocHandleChangePayload,\n Repo,\n type AnyDocumentId,\n type DocHandle,\n type DocumentId,\n type PeerCandidatePayload,\n type PeerDisconnectedPayload,\n type PeerId,\n type StorageAdapterInterface,\n type StorageKey,\n} from '@dxos/automerge/automerge-repo';\nimport { Context, Resource, cancelWithContext, type Lifecycle } from '@dxos/context';\nimport { type CollectionId, type SpaceDoc } from '@dxos/echo-protocol';\nimport { type IndexMetadataStore } from '@dxos/indexing';\nimport { invariant } from '@dxos/invariant';\nimport { PublicKey } from '@dxos/keys';\nimport { type LevelDB } from '@dxos/kv-store';\nimport { log } from '@dxos/log';\nimport { objectPointerCodec } from '@dxos/protocols';\nimport { type DocHeadsList, type FlushRequest } from '@dxos/protocols/proto/dxos/echo/service';\nimport { trace } from '@dxos/tracing';\n\nimport { CollectionSynchronizer, diffCollectionState, type CollectionState } from './collection-synchronizer';\nimport { type EchoDataMonitor } from './echo-data-monitor';\nimport { EchoNetworkAdapter, isEchoPeerMetadata } from './echo-network-adapter';\nimport { type EchoReplicator, type RemoteDocumentExistenceCheckParams } from './echo-replicator';\nimport { HeadsStore } from './heads-store';\nimport { LevelDBStorageAdapter, type BeforeSaveParams } from './leveldb-storage-adapter';\n\nexport type PeerIdProvider = () => string | undefined;\n\nexport type AutomergeHostParams = {\n db: LevelDB;\n\n indexMetadataStore: IndexMetadataStore;\n dataMonitor?: EchoDataMonitor;\n\n /**\n * Used for creating stable ids. A random key is generated on open, if no value is provided.\n */\n peerIdProvider?: PeerIdProvider;\n};\n\nexport type LoadDocOptions = {\n timeout?: number;\n};\n\nexport type CreateDocOptions = {\n /**\n * Import the document together with its history.\n */\n preserveHistory?: boolean;\n};\n\n/**\n * Abstracts over the AutomergeRepo.\n */\n@trace.resource()\nexport class AutomergeHost extends Resource {\n private readonly _db: LevelDB;\n private readonly _indexMetadataStore: IndexMetadataStore;\n private readonly _echoNetworkAdapter: EchoNetworkAdapter;\n\n private readonly _collectionSynchronizer = new CollectionSynchronizer({\n queryCollectionState: this._queryCollectionState.bind(this),\n sendCollectionState: this._sendCollectionState.bind(this),\n shouldSyncCollection: this._shouldSyncCollection.bind(this),\n });\n\n private _repo!: Repo;\n private _storage!: StorageAdapterInterface & Lifecycle;\n private readonly _headsStore: HeadsStore;\n\n @trace.info()\n private _peerId!: PeerId;\n\n private readonly _peerIdProvider?: PeerIdProvider;\n\n public readonly collectionStateUpdated = new Event<{ collectionId: CollectionId }>();\n\n constructor({ db, indexMetadataStore, dataMonitor, peerIdProvider }: AutomergeHostParams) {\n super();\n this._db = db;\n this._storage = new LevelDBStorageAdapter({\n db: db.sublevel('automerge'),\n callbacks: {\n beforeSave: async (params) => this._beforeSave(params),\n afterSave: async (key) => this._afterSave(key),\n },\n monitor: dataMonitor,\n });\n this._echoNetworkAdapter = new EchoNetworkAdapter({\n getContainingSpaceForDocument: this._getContainingSpaceForDocument.bind(this),\n isDocumentInRemoteCollection: this._isDocumentInRemoteCollection.bind(this),\n onCollectionStateQueried: this._onCollectionStateQueried.bind(this),\n onCollectionStateReceived: this._onCollectionStateReceived.bind(this),\n monitor: dataMonitor,\n });\n this._headsStore = new HeadsStore({ db: db.sublevel('heads') });\n this._indexMetadataStore = indexMetadataStore;\n this._peerIdProvider = peerIdProvider;\n }\n\n protected override async _open() {\n this._peerId = `host-${this._peerIdProvider?.() ?? PublicKey.random().toHex()}` as PeerId;\n\n await this._storage.open?.();\n\n // Construct the automerge repo.\n this._repo = new Repo({\n peerId: this._peerId as PeerId,\n sharePolicy: this._sharePolicy.bind(this),\n storage: this._storage,\n network: [\n // Upstream swarm.\n this._echoNetworkAdapter,\n ],\n });\n\n Event.wrap(this._echoNetworkAdapter, 'peer-candidate').on(this._ctx, ((e: PeerCandidatePayload) =>\n this._onPeerConnected(e.peerId)) as any);\n Event.wrap(this._echoNetworkAdapter, 'peer-disconnected').on(this._ctx, ((e: PeerDisconnectedPayload) =>\n this._onPeerDisconnected(e.peerId)) as any);\n\n this._collectionSynchronizer.remoteStateUpdated.on(this._ctx, ({ collectionId, peerId }) => {\n this._onRemoteCollectionStateUpdated(collectionId, peerId);\n this.collectionStateUpdated.emit({ collectionId: collectionId as CollectionId });\n });\n\n await this._echoNetworkAdapter.open();\n await this._collectionSynchronizer.open();\n await this._echoNetworkAdapter.open();\n await this._echoNetworkAdapter.whenConnected();\n }\n\n protected override async _close() {\n await this._collectionSynchronizer.close();\n await this._storage.close?.();\n await this._echoNetworkAdapter.close();\n await this._ctx.dispose();\n }\n\n /**\n * @deprecated To be abstracted away.\n */\n get repo(): Repo {\n return this._repo;\n }\n\n get peerId(): PeerId {\n return this._peerId;\n }\n\n get loadedDocsCount(): number {\n return Object.keys(this._repo.handles).length;\n }\n\n async addReplicator(replicator: EchoReplicator) {\n await this._echoNetworkAdapter.addReplicator(replicator);\n }\n\n async removeReplicator(replicator: EchoReplicator) {\n await this._echoNetworkAdapter.removeReplicator(replicator);\n }\n\n /**\n * Loads the document handle from the repo and waits for it to be ready.\n */\n async loadDoc<T>(ctx: Context, documentId: AnyDocumentId, opts?: LoadDocOptions): Promise<DocHandle<T>> {\n let handle: DocHandle<T> | undefined;\n if (typeof documentId === 'string') {\n // NOTE: documentId might also be a URL, in which case this lookup will fail.\n handle = this._repo.handles[documentId as DocumentId];\n }\n if (!handle) {\n handle = this._repo.find(documentId as DocumentId);\n }\n\n // `whenReady` creates a timeout so we guard it with an if to skip it if the handle is already ready.\n if (!handle.isReady()) {\n if (!opts?.timeout) {\n await cancelWithContext(ctx, handle.whenReady());\n } else {\n await cancelWithContext(ctx, asyncTimeout(handle.whenReady(), opts.timeout));\n }\n }\n\n return handle;\n }\n\n /**\n * Create new persisted document.\n */\n createDoc<T>(initialValue?: T | Doc<T>, opts?: CreateDocOptions): DocHandle<T> {\n if (opts?.preserveHistory) {\n if (!isAutomerge(initialValue)) {\n throw new TypeError('Initial value must be an Automerge document');\n }\n // TODO(dmaretskyi): There's a more efficient way.\n return this._repo.import(save(initialValue as Doc<T>));\n } else {\n return this._repo.create(initialValue);\n }\n }\n\n async waitUntilHeadsReplicated(heads: DocHeadsList): Promise<void> {\n const entries = heads.entries;\n if (!entries?.length) {\n return;\n }\n const documentIds = entries.map((entry) => entry.documentId as DocumentId);\n const documentHeads = await this.getHeads(documentIds);\n const headsToWait = entries.filter((entry, index) => {\n const targetHeads = entry.heads;\n if (!targetHeads || targetHeads.length === 0) {\n return false;\n }\n const currentHeads = documentHeads[index];\n return !(currentHeads !== null && headsEquals(currentHeads, targetHeads));\n });\n if (headsToWait.length > 0) {\n await Promise.all(\n headsToWait.map(async (entry, index) => {\n const handle = await this.loadDoc(Context.default(), entry.documentId as DocumentId);\n await waitForHeads(handle, entry.heads!);\n }),\n );\n }\n\n // Flush to disk handles loaded to memory also so that the indexer can pick up the changes.\n await this._repo.flush(documentIds.filter((documentId) => !!this._repo.handles[documentId]));\n }\n\n async reIndexHeads(documentIds: DocumentId[]) {\n for (const documentId of documentIds) {\n log.info('re-indexing heads for document', { documentId });\n const handle = this._repo.find(documentId);\n await handle.whenReady(['ready', 'requesting']);\n if (handle.inState(['requesting'])) {\n log.warn('document is not available locally, skipping', { documentId });\n continue; // Handle not available locally.\n }\n\n const doc = handle.docSync();\n invariant(doc);\n\n const heads = getHeads(doc);\n const batch = this._db.batch();\n this._headsStore.setHeads(documentId, heads, batch);\n await batch.write();\n }\n log.info('done re-indexing heads');\n }\n\n // TODO(dmaretskyi): Share based on HALO permissions and space affinity.\n // Hosts, running in the worker, don't share documents unless requested by other peers.\n // NOTE: If both peers return sharePolicy=false the replication will not happen\n // https://github.com/automerge/automerge-repo/pull/292\n private async _sharePolicy(peerId: PeerId, documentId?: DocumentId): Promise<boolean> {\n if (peerId.startsWith('client-')) {\n return false; // Only send docs to clients if they are requested.\n }\n\n if (!documentId) {\n return false;\n }\n\n const peerMetadata = this.repo.peerMetadataByPeerId[peerId];\n if (isEchoPeerMetadata(peerMetadata)) {\n return this._echoNetworkAdapter.shouldAdvertise(peerId, { documentId });\n }\n\n return false;\n }\n\n private async _beforeSave({ path, batch }: BeforeSaveParams) {\n const handle = this._repo.handles[path[0] as DocumentId];\n if (!handle) {\n return;\n }\n const doc = handle.docSync();\n if (!doc) {\n return;\n }\n\n const heads = getHeads(doc);\n this._headsStore.setHeads(handle.documentId, heads, batch);\n\n const spaceKey = getSpaceKeyFromDoc(doc) ?? undefined;\n const objectIds = Object.keys(doc.objects ?? {});\n const encodedIds = objectIds.map((objectId) =>\n objectPointerCodec.encode({ documentId: handle.documentId, objectId, spaceKey }),\n );\n const idToLastHash = new Map(encodedIds.map((id) => [id, heads]));\n this._indexMetadataStore.markDirty(idToLastHash, batch);\n }\n\n private _shouldSyncCollection(collectionId: string, peerId: PeerId): boolean {\n const peerMetadata = this._repo.peerMetadataByPeerId[peerId];\n if (isEchoPeerMetadata(peerMetadata)) {\n return this._echoNetworkAdapter.shouldSyncCollection(peerId, { collectionId });\n }\n\n return false;\n }\n\n /**\n * Called by AutomergeStorageAdapter after levelDB batch commit.\n */\n private async _afterSave(path: StorageKey) {\n this._indexMetadataStore.notifyMarkedDirty();\n\n const documentId = path[0] as DocumentId;\n const document = this._repo.handles[documentId]?.docSync();\n if (document) {\n const heads = getHeads(document);\n this._onHeadsChanged(documentId, heads);\n }\n }\n\n @trace.info({ depth: null })\n private _automergePeers() {\n return this._repo.peers;\n }\n\n private async _isDocumentInRemoteCollection(params: RemoteDocumentExistenceCheckParams): Promise<boolean> {\n for (const collectionId of this._collectionSynchronizer.getRegisteredCollectionIds()) {\n const remoteCollections = this._collectionSynchronizer.getRemoteCollectionStates(collectionId);\n const remotePeerDocs = remoteCollections.get(params.peerId as PeerId)?.documents;\n if (remotePeerDocs && params.documentId in remotePeerDocs) {\n return true;\n }\n }\n return false;\n }\n\n private async _getContainingSpaceForDocument(documentId: string): Promise<PublicKey | null> {\n const doc = this._repo.handles[documentId as any]?.docSync();\n if (!doc) {\n return null;\n }\n\n const spaceKeyHex = getSpaceKeyFromDoc(doc);\n if (!spaceKeyHex) {\n return null;\n }\n\n return PublicKey.from(spaceKeyHex);\n }\n\n /**\n * Flush documents to disk.\n */\n @trace.span({ showInBrowserTimeline: true })\n async flush({ documentIds }: FlushRequest = {}): Promise<void> {\n // Note: Sync protocol for client and services ensures that all handles should have all changes.\n\n const loadedDocuments = documentIds?.filter(\n (documentId): documentId is DocumentId => !!this._repo.handles[documentId as DocumentId],\n );\n await this._repo.flush(loadedDocuments);\n }\n\n async getHeads(documentIds: DocumentId[]): Promise<(Heads | undefined)[]> {\n const result: (Heads | undefined)[] = [];\n const storeRequestIds: DocumentId[] = [];\n const storeResultIndices: number[] = [];\n for (const documentId of documentIds) {\n const doc = this._repo.handles[documentId]?.docSync();\n if (doc) {\n result.push(getHeads(doc));\n } else {\n storeRequestIds.push(documentId);\n storeResultIndices.push(result.length);\n result.push(undefined);\n }\n }\n if (storeRequestIds.length > 0) {\n const storedHeads = await this._headsStore.getHeads(storeRequestIds);\n for (let i = 0; i < storedHeads.length; i++) {\n result[storeResultIndices[i]] = storedHeads[i];\n }\n }\n return result;\n }\n\n //\n // Collection sync.\n //\n\n getLocalCollectionState(collectionId: string): CollectionState | undefined {\n return this._collectionSynchronizer.getLocalCollectionState(collectionId);\n }\n\n getRemoteCollectionStates(collectionId: string): ReadonlyMap<PeerId, CollectionState> {\n return this._collectionSynchronizer.getRemoteCollectionStates(collectionId);\n }\n\n refreshCollection(collectionId: string) {\n this._collectionSynchronizer.refreshCollection(collectionId);\n }\n\n async getCollectionSyncState(collectionId: string): Promise<CollectionSyncState> {\n const result: CollectionSyncState = {\n peers: [],\n };\n\n const localState = this.getLocalCollectionState(collectionId);\n const remoteState = this.getRemoteCollectionStates(collectionId);\n\n if (!localState) {\n return result;\n }\n\n for (const [peerId, state] of remoteState) {\n const diff = diffCollectionState(localState, state);\n result.peers.push({\n peerId,\n missingOnRemote: diff.missingOnRemote.length,\n missingOnLocal: diff.missingOnLocal.length,\n differentDocuments: diff.different.length,\n localDocumentCount: Object.keys(localState.documents).length,\n remoteDocumentCount: Object.keys(state.documents).length,\n });\n }\n\n return result;\n }\n\n /**\n * Update the local collection state based on the locally stored document heads.\n */\n async updateLocalCollectionState(collectionId: string, documentIds: DocumentId[]) {\n const heads = await this.getHeads(documentIds);\n const documents: Record<DocumentId, Heads> = Object.fromEntries(\n heads.map((heads, index) => [documentIds[index], heads ?? []]),\n );\n this._collectionSynchronizer.setLocalCollectionState(collectionId, { documents });\n }\n\n private _onCollectionStateQueried(collectionId: string, peerId: PeerId) {\n this._collectionSynchronizer.onCollectionStateQueried(collectionId, peerId);\n }\n\n private _onCollectionStateReceived(collectionId: string, peerId: PeerId, state: unknown) {\n this._collectionSynchronizer.onRemoteStateReceived(collectionId, peerId, decodeCollectionState(state));\n }\n\n private _queryCollectionState(collectionId: string, peerId: PeerId) {\n this._echoNetworkAdapter.queryCollectionState(collectionId, peerId);\n }\n\n private _sendCollectionState(collectionId: string, peerId: PeerId, state: CollectionState) {\n this._echoNetworkAdapter.sendCollectionState(collectionId, peerId, encodeCollectionState(state));\n }\n\n private _onPeerConnected(peerId: PeerId) {\n this._collectionSynchronizer.onConnectionOpen(peerId);\n }\n\n private _onPeerDisconnected(peerId: PeerId) {\n this._collectionSynchronizer.onConnectionClosed(peerId);\n }\n\n private _onRemoteCollectionStateUpdated(collectionId: string, peerId: PeerId) {\n const localState = this._collectionSynchronizer.getLocalCollectionState(collectionId);\n const remoteState = this._collectionSynchronizer.getRemoteCollectionStates(collectionId).get(peerId);\n\n if (!localState || !remoteState) {\n return;\n }\n\n const { different, missingOnLocal, missingOnRemote } = diffCollectionState(localState, remoteState);\n const toReplicate = [...missingOnLocal, ...missingOnRemote, ...different];\n\n if (toReplicate.length === 0) {\n return;\n }\n\n log.info('replication documents after collection sync', {\n count: toReplicate.length,\n });\n\n // Load the documents so they will start syncing.\n for (const documentId of toReplicate) {\n this._repo.find(documentId);\n }\n }\n\n private _onHeadsChanged(documentId: DocumentId, heads: Heads) {\n const collectionsChanged = new Set<CollectionId>();\n for (const collectionId of this._collectionSynchronizer.getRegisteredCollectionIds()) {\n const state = this._collectionSynchronizer.getLocalCollectionState(collectionId);\n if (state?.documents[documentId]) {\n const newState = structuredClone(state);\n newState.documents[documentId] = heads;\n this._collectionSynchronizer.setLocalCollectionState(collectionId, newState);\n collectionsChanged.add(collectionId as CollectionId);\n }\n }\n for (const collectionId of collectionsChanged) {\n this.collectionStateUpdated.emit({ collectionId });\n }\n }\n}\n\nexport const getSpaceKeyFromDoc = (doc: Doc<SpaceDoc>): string | null => {\n // experimental_spaceKey is set on old documents, new ones are created with doc.access.spaceKey\n const rawSpaceKey = doc.access?.spaceKey ?? (doc as any).experimental_spaceKey;\n if (rawSpaceKey == null) {\n return null;\n }\n\n return String(rawSpaceKey);\n};\n\nconst waitForHeads = async (handle: DocHandle<SpaceDoc>, heads: Heads) => {\n const unavailableHeads = new Set(heads);\n\n await handle.whenReady();\n await Event.wrap<DocHandleChangePayload<SpaceDoc>>(handle, 'change').waitForCondition(() => {\n // Check if unavailable heads became available.\n for (const changeHash of unavailableHeads.values()) {\n if (changeIsPresentInDoc(handle.docSync(), changeHash)) {\n unavailableHeads.delete(changeHash);\n }\n }\n\n return unavailableHeads.size === 0;\n });\n};\n\nconst changeIsPresentInDoc = (doc: Doc<any>, changeHash: string): boolean => {\n return !!getBackend(doc).getChangeByHash(changeHash);\n};\n\nconst decodeCollectionState = (state: unknown): CollectionState => {\n invariant(typeof state === 'object' && state !== null, 'Invalid state');\n\n return state as CollectionState;\n};\n\nconst encodeCollectionState = (state: CollectionState): unknown => {\n return state;\n};\n\nexport type CollectionSyncState = {\n peers: PeerSyncState[];\n};\n\nexport type PeerSyncState = {\n peerId: PeerId;\n /**\n * Documents that are present locally but not on the remote peer.\n */\n missingOnRemote: number;\n\n /**\n * Documents that are present on the remote peer but not locally.\n */\n missingOnLocal: number;\n\n /**\n * Documents that are present on both peers but have different heads.\n */\n differentDocuments: number;\n\n /**\n * Total number of documents locally.\n */\n localDocumentCount: number;\n\n /**\n * Total number of documents on the remote peer.\n */\n remoteDocumentCount: number;\n};\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport { asyncReturn, Event, scheduleTask, scheduleTaskInterval } from '@dxos/async';\nimport { next as am } from '@dxos/automerge/automerge';\nimport type { DocumentId, PeerId } from '@dxos/automerge/automerge-repo';\nimport { Resource, type Context } from '@dxos/context';\nimport { log } from '@dxos/log';\nimport { defaultMap } from '@dxos/util';\n\nconst MIN_QUERY_INTERVAL = 5_000;\n\nconst POLL_INTERVAL = 30_000;\n\nexport type CollectionSynchronizerParams = {\n sendCollectionState: (collectionId: string, peerId: PeerId, state: CollectionState) => void;\n queryCollectionState: (collectionId: string, peerId: PeerId) => void;\n shouldSyncCollection: (collectionId: string, peerId: PeerId) => boolean;\n};\n\n/**\n * Implements collection sync protocol.\n */\nexport class CollectionSynchronizer extends Resource {\n private readonly _sendCollectionState: CollectionSynchronizerParams['sendCollectionState'];\n private readonly _queryCollectionState: CollectionSynchronizerParams['queryCollectionState'];\n private readonly _shouldSyncCollection: CollectionSynchronizerParams['shouldSyncCollection'];\n\n /**\n * CollectionId -> State.\n */\n private readonly _perCollectionStates = new Map<string, PerCollectionState>();\n\n private readonly _connectedPeers = new Set<PeerId>();\n\n public readonly remoteStateUpdated = new Event<{ collectionId: string; peerId: PeerId }>();\n\n constructor(params: CollectionSynchronizerParams) {\n super();\n this._sendCollectionState = params.sendCollectionState;\n this._queryCollectionState = params.queryCollectionState;\n this._shouldSyncCollection = params.shouldSyncCollection;\n }\n\n protected override async _open(ctx: Context): Promise<void> {\n scheduleTaskInterval(\n this._ctx,\n async () => {\n for (const collectionId of this._perCollectionStates.keys()) {\n this.refreshCollection(collectionId);\n await asyncReturn();\n }\n },\n POLL_INTERVAL,\n );\n }\n\n getRegisteredCollectionIds(): string[] {\n return [...this._perCollectionStates.keys()];\n }\n\n getLocalCollectionState(collectionId: string): CollectionState | undefined {\n return this._getPerCollectionState(collectionId).localState;\n }\n\n setLocalCollectionState(collectionId: string, state: CollectionState) {\n log('setLocalCollectionState', { collectionId, state });\n this._getPerCollectionState(collectionId).localState = state;\n\n queueMicrotask(async () => {\n if (!this._ctx.disposed) {\n this._refreshInterestedPeers(collectionId);\n this.refreshCollection(collectionId);\n }\n });\n }\n\n getRemoteCollectionStates(collectionId: string): ReadonlyMap<PeerId, CollectionState> {\n return this._getPerCollectionState(collectionId).remoteStates;\n }\n\n refreshCollection(collectionId: string) {\n let scheduleAnotherRefresh = false;\n const state = this._getPerCollectionState(collectionId);\n for (const peerId of this._connectedPeers) {\n if (state.interestedPeers.has(peerId)) {\n const lastQueried = state.lastQueried.get(peerId) ?? 0;\n if (Date.now() - lastQueried > MIN_QUERY_INTERVAL) {\n state.lastQueried.set(peerId, Date.now());\n this._queryCollectionState(collectionId, peerId);\n } else {\n scheduleAnotherRefresh = true;\n }\n }\n }\n if (scheduleAnotherRefresh) {\n scheduleTask(this._ctx, () => this.refreshCollection(collectionId), MIN_QUERY_INTERVAL);\n }\n }\n\n /**\n * Callback when a connection to a peer is established.\n */\n onConnectionOpen(peerId: PeerId) {\n this._connectedPeers.add(peerId);\n\n queueMicrotask(async () => {\n if (this._ctx.disposed) {\n return;\n }\n for (const [collectionId, state] of this._perCollectionStates.entries()) {\n if (this._shouldSyncCollection(collectionId, peerId)) {\n state.interestedPeers.add(peerId);\n state.lastQueried.set(peerId, Date.now());\n this._queryCollectionState(collectionId, peerId);\n }\n }\n });\n }\n\n /**\n * Callback when a connection to a peer is closed.\n */\n onConnectionClosed(peerId: PeerId) {\n this._connectedPeers.delete(peerId);\n\n for (const perCollectionState of this._perCollectionStates.values()) {\n perCollectionState.remoteStates.delete(peerId);\n }\n }\n\n /**\n * Callback when a peer queries the state of a collection.\n */\n onCollectionStateQueried(collectionId: string, peerId: PeerId) {\n const perCollectionState = this._getPerCollectionState(collectionId);\n\n if (perCollectionState.localState) {\n this._sendCollectionState(collectionId, peerId, perCollectionState.localState);\n }\n }\n\n /**\n * Callback when a peer sends the state of a collection.\n */\n onRemoteStateReceived(collectionId: string, peerId: PeerId, state: CollectionState) {\n log('onRemoteStateReceived', { collectionId, peerId, state });\n validateCollectionState(state);\n const perCollectionState = this._getPerCollectionState(collectionId);\n perCollectionState.remoteStates.set(peerId, state);\n this.remoteStateUpdated.emit({ peerId, collectionId });\n }\n\n private _getPerCollectionState(collectionId: string): PerCollectionState {\n return defaultMap(this._perCollectionStates, collectionId, () => ({\n localState: undefined,\n remoteStates: new Map(),\n interestedPeers: new Set(),\n lastQueried: new Map(),\n }));\n }\n\n private _refreshInterestedPeers(collectionId: string) {\n for (const peerId of this._connectedPeers) {\n if (this._shouldSyncCollection(collectionId, peerId)) {\n this._getPerCollectionState(collectionId).interestedPeers.add(peerId);\n } else {\n this._getPerCollectionState(collectionId).interestedPeers.delete(peerId);\n }\n }\n }\n}\n\ntype PerCollectionState = {\n localState?: CollectionState;\n remoteStates: Map<PeerId, CollectionState>;\n interestedPeers: Set<PeerId>;\n lastQueried: Map<PeerId, number>;\n};\n\nexport type CollectionState = {\n /**\n * DocumentId -> Heads.\n */\n documents: Record<string, string[]>;\n};\n\nexport type CollectionStateDiff = {\n missingOnRemote: DocumentId[];\n missingOnLocal: DocumentId[];\n different: DocumentId[];\n};\n\nexport const diffCollectionState = (local: CollectionState, remote: CollectionState): CollectionStateDiff => {\n const allDocuments = new Set<DocumentId>([...Object.keys(local.documents), ...Object.keys(remote.documents)] as any);\n\n const missingOnRemote: DocumentId[] = [];\n const missingOnLocal: DocumentId[] = [];\n const different: DocumentId[] = [];\n for (const documentId of allDocuments) {\n if (!local.documents[documentId]) {\n missingOnLocal.push(documentId as DocumentId);\n } else if (!remote.documents[documentId]) {\n missingOnRemote.push(documentId as DocumentId);\n } else if (!am.equals(local.documents[documentId], remote.documents[documentId])) {\n different.push(documentId as DocumentId);\n }\n }\n\n return {\n missingOnRemote,\n missingOnLocal,\n different,\n };\n};\n\nconst validateCollectionState = (state: CollectionState) => {\n Object.entries(state.documents).forEach(([documentId, heads]) => {\n if (!isValidDocumentId(documentId as DocumentId)) {\n throw new Error(`Invalid documentId: ${documentId}`);\n }\n if (Array.isArray(heads) && heads.some((head) => typeof head !== 'string')) {\n throw new Error(`Invalid heads: ${heads}`);\n }\n });\n};\n\nconst isValidDocumentId = (documentId: DocumentId) => {\n return typeof documentId === 'string' && !documentId.includes(':');\n};\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport { synchronized, Trigger } from '@dxos/async';\nimport { NetworkAdapter, type Message, type PeerId, type PeerMetadata } from '@dxos/automerge/automerge-repo';\nimport { LifecycleState } from '@dxos/context';\nimport { invariant } from '@dxos/invariant';\nimport { type PublicKey } from '@dxos/keys';\nimport { log } from '@dxos/log';\nimport type { AutomergeProtocolMessage } from '@dxos/protocols';\nimport { nonNullable } from '@dxos/util';\n\nimport {\n type EchoReplicator,\n type RemoteDocumentExistenceCheckParams,\n type ReplicatorConnection,\n type ShouldAdvertiseParams,\n type ShouldSyncCollectionParams,\n} from './echo-replicator';\nimport {\n isCollectionQueryMessage,\n isCollectionStateMessage,\n type CollectionQueryMessage,\n type CollectionStateMessage,\n} from './network-protocol';\nimport { createIdFromSpaceKey } from '../common/space-id';\n\nexport interface NetworkDataMonitor {\n recordPeerConnected(peerId: string): void;\n recordPeerDisconnected(peerId: string): void;\n recordMessageSent(message: Message, duration: number): void;\n recordMessageReceived(message: Message): void;\n recordMessageSendingFailed(message: Message): void;\n}\n\nexport type EchoNetworkAdapterParams = {\n getContainingSpaceForDocument: (documentId: string) => Promise<PublicKey | null>;\n isDocumentInRemoteCollection: (params: RemoteDocumentExistenceCheckParams) => Promise<boolean>;\n onCollectionStateQueried: (collectionId: string, peerId: PeerId) => void;\n onCollectionStateReceived: (collectionId: string, peerId: PeerId, state: unknown) => void;\n monitor?: NetworkDataMonitor;\n};\n\n/**\n * Manages a set of {@link EchoReplicator} instances.\n */\nexport class EchoNetworkAdapter extends NetworkAdapter {\n private readonly _replicators = new Set<EchoReplicator>();\n /**\n * Remote peer id -> connection.\n */\n private readonly _connections = new Map<PeerId, ConnectionEntry>();\n private _lifecycleState: LifecycleState = LifecycleState.CLOSED;\n private readonly _connected = new Trigger();\n\n constructor(private readonly _params: EchoNetworkAdapterParams) {\n super();\n }\n\n override connect(peerId: PeerId, peerMetadata?: PeerMetadata | undefined): void {\n this.peerId = peerId;\n this.peerMetadata = peerMetadata;\n this._connected.wake();\n }\n\n override send(message: Message): void {\n this._send(message);\n }\n\n override disconnect(): void {\n // No-op\n }\n\n @synchronized\n async open() {\n if (this._lifecycleState === LifecycleState.OPEN) {\n return;\n }\n this._lifecycleState = LifecycleState.OPEN;\n\n log('emit ready');\n this.emit('ready', {\n network: this,\n });\n }\n\n @synchronized\n async close() {\n if (this._lifecycleState === LifecycleState.CLOSED) {\n return this;\n }\n\n for (const replicator of this._replicators) {\n await replicator.disconnect();\n }\n this._replicators.clear();\n\n this._lifecycleState = LifecycleState.CLOSED;\n }\n\n async whenConnected() {\n await this._connected.wait({ timeout: 10_000 });\n }\n\n @synchronized\n async addReplicator(replicator: EchoReplicator) {\n invariant(this._lifecycleState === LifecycleState.OPEN);\n invariant(this.peerId);\n invariant(!this._replicators.has(replicator));\n\n this._replicators.add(replicator);\n await replicator.connect({\n peerId: this.peerId,\n onConnectionOpen: this._onConnectionOpen.bind(this),\n onConnectionClosed: this._onConnectionClosed.bind(this),\n onConnectionAuthScopeChanged: this._onConnectionAuthScopeChanged.bind(this),\n isDocumentInRemoteCollection: this._params.isDocumentInRemoteCollection,\n getContainingSpaceForDocument: this._params.getContainingSpaceForDocument,\n getContainingSpaceIdForDocument: async (documentId) => {\n const key = await this._params.getContainingSpaceForDocument(documentId);\n return key ? createIdFromSpaceKey(key) : null;\n },\n });\n }\n\n @synchronized\n async removeReplicator(replicator: EchoReplicator) {\n invariant(this._lifecycleState === LifecycleState.OPEN);\n invariant(this._replicators.has(replicator));\n await replicator.disconnect();\n this._replicators.delete(replicator);\n }\n\n async shouldAdvertise(peerId: PeerId, params: ShouldAdvertiseParams): Promise<boolean> {\n const connection = this._connections.get(peerId);\n if (!connection) {\n return false;\n }\n\n return connection.connection.shouldAdvertise(params);\n }\n\n shouldSyncCollection(peerId: PeerId, params: ShouldSyncCollectionParams): boolean {\n const connection = this._connections.get(peerId);\n if (!connection) {\n return false;\n }\n\n return connection.connection.shouldSyncCollection(params);\n }\n\n queryCollectionState(collectionId: string, targetId: PeerId): void {\n const message: CollectionQueryMessage = {\n type: 'collection-query',\n senderId: this.peerId as PeerId,\n targetId,\n collectionId,\n };\n this._send(message);\n }\n\n sendCollectionState(collectionId: string, targetId: PeerId, state: unknown): void {\n const message: CollectionStateMessage = {\n type: 'collection-state',\n senderId: this.peerId as PeerId,\n targetId,\n collectionId,\n state,\n };\n this._send(message);\n }\n\n private _send(message: Message) {\n const connectionEntry = this._connections.get(message.targetId);\n if (!connectionEntry) {\n throw new Error('Connection not found.');\n }\n\n const writeStart = Date.now();\n // TODO(dmaretskyi): Find a way to enforce backpressure on AM-repo.\n connectionEntry.writer\n .write(message as AutomergeProtocolMessage)\n .then(() => {\n const durationMs = Date.now() - writeStart;\n this._params.monitor?.recordMessageSent(message, durationMs);\n })\n .catch((err) => {\n if (connectionEntry.isOpen) {\n log.catch(err);\n }\n this._params.monitor?.recordMessageSendingFailed(message);\n });\n }\n\n // TODO(dmaretskyi): Remove.\n getPeersInterestedInCollection(collectionId: string): PeerId[] {\n return Array.from(this._connections.values())\n .map((connection) => {\n return connection.connection.shouldSyncCollection({ collectionId })\n ? (connection.connection.peerId as PeerId)\n : null;\n })\n .filter(nonNullable);\n }\n\n private _onConnectionOpen(connection: ReplicatorConnection) {\n log('Connection opened', { peerId: connection.peerId });\n invariant(!this._connections.has(connection.peerId as PeerId));\n const reader = connection.readable.getReader();\n const writer = connection.writable.getWriter();\n const connectionEntry: ConnectionEntry = { connection, reader, writer, isOpen: true };\n this._connections.set(connection.peerId as PeerId, connectionEntry);\n\n queueMicrotask(async () => {\n try {\n while (true) {\n // TODO(dmaretskyi): Find a way to enforce backpressure on AM-repo.\n const { done, value } = await reader.read();\n if (done) {\n break;\n }\n\n this._onMessage(value as Message);\n }\n } catch (err) {\n if (connectionEntry.isOpen) {\n log.catch(err);\n }\n }\n });\n\n log('emit peer-candidate', { peerId: connection.peerId });\n this._emitPeerCandidate(connection);\n this._params.monitor?.recordPeerConnected(connection.peerId);\n }\n\n private _onMessage(message: Message) {\n if (isCollectionQueryMessage(message)) {\n this._params.onCollectionStateQueried(message.collectionId, message.senderId);\n } else if (isCollectionStateMessage(message)) {\n this._params.onCollectionStateReceived(message.collectionId, message.senderId, message.state);\n } else {\n this.emit('message', message);\n }\n this._params.monitor?.recordMessageReceived(message);\n }\n\n /**\n * Trigger doc-synchronizer shared documents set recalculation. Happens on peer-candidate.\n * TODO(y): replace with a proper API call when sharePolicy update becomes supported by automerge-repo\n */\n private _onConnectionAuthScopeChanged(connection: ReplicatorConnection) {\n log('Connection auth scope changed', { peerId: connection.peerId });\n const entry = this._connections.get(connection.peerId as PeerId);\n invariant(entry);\n this.emit('peer-disconnected', { peerId: connection.peerId as PeerId });\n this._emitPeerCandidate(connection);\n }\n\n private _onConnectionClosed(connection: ReplicatorConnection) {\n log('Connection closed', { peerId: connection.peerId });\n const entry = this._connections.get(connection.peerId as PeerId);\n invariant(entry);\n\n entry.isOpen = false;\n this.emit('peer-disconnected', { peerId: connection.peerId as PeerId });\n this._params.monitor?.recordPeerDisconnected(connection.peerId);\n\n void entry.reader.cancel().catch((err) => log.catch(err));\n void entry.writer.abort().catch((err) => log.catch(err));\n\n this._connections.delete(connection.peerId as PeerId);\n }\n\n private _emitPeerCandidate(connection: ReplicatorConnection) {\n this.emit('peer-candidate', {\n peerId: connection.peerId as PeerId,\n peerMetadata: createEchoPeerMetadata(),\n });\n }\n}\n\ntype ConnectionEntry = {\n connection: ReplicatorConnection;\n reader: ReadableStreamDefaultReader<AutomergeProtocolMessage>;\n writer: WritableStreamDefaultWriter<AutomergeProtocolMessage>;\n isOpen: boolean;\n};\n\nexport const createEchoPeerMetadata = (): PeerMetadata =>\n ({\n // TODO(dmaretskyi): Refactor this.\n dxos_peerSource: 'EchoNetworkAdapter',\n }) as any;\n\nexport const isEchoPeerMetadata = (metadata: PeerMetadata): boolean =>\n (metadata as any)?.dxos_peerSource === 'EchoNetworkAdapter';\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport type { Message } from '@dxos/automerge/automerge-repo';\nimport {\n type CollectionQueryMessage,\n type CollectionStateMessage,\n MESSAGE_TYPE_COLLECTION_QUERY,\n MESSAGE_TYPE_COLLECTION_STATE,\n} from '@dxos/protocols';\n\nexport { type CollectionStateMessage, type CollectionQueryMessage };\n\nexport const isCollectionQueryMessage = (message: Message): message is CollectionQueryMessage =>\n message.type === MESSAGE_TYPE_COLLECTION_QUERY;\n\nexport const isCollectionStateMessage = (message: Message): message is CollectionStateMessage =>\n message.type === MESSAGE_TYPE_COLLECTION_STATE;\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport type { Heads } from '@dxos/automerge/automerge';\nimport type { DocumentId } from '@dxos/automerge/automerge-repo';\nimport { headsEncoding } from '@dxos/indexing';\nimport type { BatchLevel, SublevelDB } from '@dxos/kv-store';\n\nexport type HeadsStoreParams = {\n db: SublevelDB;\n};\n\nexport class HeadsStore {\n private readonly _db: SublevelDB;\n\n constructor({ db }: HeadsStoreParams) {\n this._db = db;\n }\n\n setHeads(documentId: DocumentId, heads: Heads, batch: BatchLevel) {\n batch.put<DocumentId, Heads>(documentId, heads, {\n sublevel: this._db,\n keyEncoding: 'utf8',\n valueEncoding: headsEncoding,\n });\n }\n\n // TODO(dmaretskyi): Make batched.\n async getHeads(documentIds: DocumentId[]): Promise<Array<Heads | undefined>> {\n return this._db.getMany<DocumentId, Heads>(documentIds, {\n keyEncoding: 'utf8',\n valueEncoding: headsEncoding,\n });\n }\n}\n", "//\n// Copyright 2024 DXOS.org\n// s\n\nimport { type MixedEncoding } from 'level-transcoder';\n\nimport { type StorageAdapterInterface, type Chunk, type StorageKey } from '@dxos/automerge/automerge-repo';\nimport { LifecycleState, Resource } from '@dxos/context';\nimport { type BatchLevel, type SublevelDB } from '@dxos/kv-store';\nimport { type MaybePromise } from '@dxos/util';\n\nexport interface StorageAdapterDataMonitor {\n recordBytesStored(count: number): void;\n recordBytesLoaded(count: number): void;\n recordLoadDuration(durationMs: number): void;\n recordStoreDuration(durationMs: number): void;\n}\n\nexport type LevelDBStorageAdapterParams = {\n db: SublevelDB;\n callbacks?: StorageCallbacks;\n monitor?: StorageAdapterDataMonitor;\n};\n\nexport type BeforeSaveParams = { path: StorageKey; batch: BatchLevel };\n\nexport interface StorageCallbacks {\n beforeSave(params: BeforeSaveParams): MaybePromise<void>;\n afterSave(path: StorageKey): MaybePromise<void>;\n}\n\nexport class LevelDBStorageAdapter extends Resource implements StorageAdapterInterface {\n constructor(private readonly _params: LevelDBStorageAdapterParams) {\n super();\n }\n\n async load(keyArray: StorageKey): Promise<Uint8Array | undefined> {\n try {\n if (this._lifecycleState !== LifecycleState.OPEN) {\n // TODO(mykola): this should be an error.\n return undefined;\n }\n const startMs = Date.now();\n const chunk = await this._params.db.get<StorageKey, Uint8Array>(keyArray, { ...encodingOptions });\n this._params.monitor?.recordBytesLoaded(chunk.byteLength);\n this._params.monitor?.recordLoadDuration(Date.now() - startMs);\n return chunk;\n } catch (err: any) {\n if (isLevelDbNotFoundError(err)) {\n return undefined;\n }\n throw err;\n }\n }\n\n async save(keyArray: StorageKey, binary: Uint8Array): Promise<void> {\n if (this._lifecycleState !== LifecycleState.OPEN) {\n return undefined;\n }\n const startMs = Date.now();\n const batch = this._params.db.batch();\n\n await this._params.callbacks?.beforeSave?.({ path: keyArray, batch });\n batch.put<StorageKey, Uint8Array>(keyArray, Buffer.from(binary), {\n ...encodingOptions,\n });\n await batch.write();\n this._params.monitor?.recordBytesStored(binary.byteLength);\n\n await this._params.callbacks?.afterSave?.(keyArray);\n this._params.monitor?.recordStoreDuration(Date.now() - startMs);\n }\n\n async remove(keyArray: StorageKey): Promise<void> {\n if (this._lifecycleState !== LifecycleState.OPEN) {\n return undefined;\n }\n await this._params.db.del<StorageKey>(keyArray, { ...encodingOptions });\n }\n\n async loadRange(keyPrefix: StorageKey): Promise<Chunk[]> {\n if (this._lifecycleState !== LifecycleState.OPEN) {\n return [];\n }\n const startMs = Date.now();\n const result: Chunk[] = [];\n for await (const [key, value] of this._params.db.iterator<StorageKey, Uint8Array>({\n gte: keyPrefix,\n lte: [...keyPrefix, '\\uffff'],\n ...encodingOptions,\n })) {\n result.push({\n key,\n data: value,\n });\n this._params.monitor?.recordBytesLoaded(value.byteLength);\n }\n this._params.monitor?.recordLoadDuration(Date.now() - startMs);\n return result;\n }\n\n async removeRange(keyPrefix: StorageKey): Promise<void> {\n if (this._lifecycleState !== LifecycleState.OPEN) {\n return undefined;\n }\n const batch = this._params.db.batch();\n\n for await (const [key] of this._params.db.iterator<StorageKey, Uint8Array>({\n gte: keyPrefix,\n lte: [...keyPrefix, '\\uffff'],\n ...encodingOptions,\n })) {\n batch.del<StorageKey>(key, { ...encodingOptions });\n }\n await batch.write();\n }\n}\n\nconst keyEncoder: MixedEncoding<StorageKey, Uint8Array, StorageKey> = {\n encode: (key: StorageKey): Uint8Array =>\n Buffer.from(key.map((k) => k.replaceAll('%', '%25').replaceAll('-', '%2D')).join('-')),\n decode: (key: Uint8Array): StorageKey =>\n Buffer.from(key)\n .toString()\n .split('-')\n .map((k) => k.replaceAll('%2D', '-').replaceAll('%25', '%')),\n format: 'buffer',\n};\n\nexport const encodingOptions = {\n keyEncoding: keyEncoder,\n valueEncoding: 'buffer',\n};\n\nconst isLevelDbNotFoundError = (err: any): boolean => err.code === 'LEVEL_NOT_FOUND';\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport type { CollectionId } from '@dxos/echo-protocol';\nimport { invariant } from '@dxos/invariant';\nimport { PublicKey, type SpaceId } from '@dxos/keys';\nimport { log } from '@dxos/log';\nimport {\n type AutomergeReplicator,\n type AutomergeReplicatorFactory,\n} from '@dxos/teleport-extension-automerge-replicator';\nimport { ComplexSet, defaultMap } from '@dxos/util';\n\nimport { type EchoReplicator, type EchoReplicatorContext, type ShouldAdvertiseParams } from './echo-replicator';\nimport { MeshReplicatorConnection } from './mesh-echo-replicator-connection';\nimport { getSpaceIdFromCollectionId } from './space-collection';\nimport { createIdFromSpaceKey } from '../common/space-id';\n\n// TODO(dmaretskyi): Move out of @dxos/echo-pipeline.\n\n/**\n * Used to replicate with other peers over the network.\n */\nexport class MeshEchoReplicator implements EchoReplicator {\n private readonly _connections = new Set<MeshReplicatorConnection>();\n /**\n * Using automerge peerId as a key.\n */\n private readonly _connectionsPerPeer = new Map<string, MeshReplicatorConnection>();\n\n /**\n * spaceId -> deviceKey[]\n */\n private readonly _authorizedDevices = new Map<SpaceId, ComplexSet<PublicKey>>();\n\n private _context: EchoReplicatorContext | null = null;\n\n async connect(context: EchoReplicatorContext): Promise<void> {\n this._context = context;\n }\n\n async disconnect() {\n for (const connection of this._connectionsPerPeer.values()) {\n this._context?.onConnectionClosed(connection);\n }\n for (const connection of this._connections) {\n await connection.close();\n }\n this._connections.clear();\n this._connectionsPerPeer.clear();\n\n this._context = null;\n }\n\n createExtension(extensionFactory?: AutomergeReplicatorFactory): AutomergeReplicator {\n invariant(this._context);\n\n const connection: MeshReplicatorConnection = new MeshReplicatorConnection({\n ownPeerId: this._context.peerId,\n replicatorFactory: extensionFactory,\n onRemoteConnected: async () => {\n log('onRemoteConnected', { peerId: connection.peerId });\n invariant(this._context);\n\n if (this._connectionsPerPeer.has(connection.peerId)) {\n this._context.onConnectionAuthScopeChanged(connection);\n } else {\n this._connectionsPerPeer.set(connection.peerId, connection);\n this._context.onConnectionOpen(connection);\n connection.enable();\n }\n },\n onRemoteDisconnected: async () => {\n log('onRemoteDisconnected', { peerId: connection.peerId });\n this._context?.onConnectionClosed(connection);\n this._connectionsPerPeer.delete(connection.peerId);\n connection.disable();\n this._connections.delete(connection);\n },\n shouldAdvertise: async (params: ShouldAdvertiseParams) => {\n log('shouldAdvertise', { peerId: connection.peerId, documentId: params.documentId });\n invariant(this._context);\n try {\n const spaceKey = await this._context.getContainingSpaceForDocument(params.documentId);\n if (!spaceKey) {\n const remoteDocumentExists = await this._context.isDocumentInRemoteCollection({\n documentId: params.documentId,\n peerId: connection.peerId,\n });\n log('document not found locally for share policy check, accepting the remote document', {\n peerId: connection.peerId,\n documentId: params.documentId,\n remoteDocumentExists,\n });\n // If a document is not present locally return true if another peer claims to have it.\n // Simply returning true will add the peer to \"generous peers list\" for this document which will\n // start replication of the document after we receive, even if the peer is not in the corresponding space.\n return remoteDocumentExists;\n }\n\n const spaceId = await createIdFromSpaceKey(spaceKey);\n\n const authorizedDevices = this._authorizedDevices.get(spaceId);\n\n if (!connection.remoteDeviceKey) {\n log('device key not found for share policy check', {\n peerId: connection.peerId,\n documentId: params.documentId,\n });\n return false;\n }\n\n const isAuthorized = authorizedDevices?.has(connection.remoteDeviceKey) ?? false;\n log('share policy check', {\n localPeer: this._context.peerId,\n remotePeer: connection.peerId,\n documentId: params.documentId,\n deviceKey: connection.remoteDeviceKey,\n spaceKey,\n isAuthorized,\n });\n return isAuthorized;\n } catch (err) {\n log.catch(err);\n return false;\n }\n },\n shouldSyncCollection: ({ collectionId }) => {\n const spaceId = getSpaceIdFromCollectionId(collectionId as CollectionId);\n\n const authorizedDevices = this._authorizedDevices.get(spaceId);\n\n if (!connection.remoteDeviceKey) {\n log('device key not found for collection sync check', {\n peerId: connection.peerId,\n collectionId,\n });\n return false;\n }\n\n const isAuthorized = authorizedDevices?.has(connection.remoteDeviceKey) ?? false;\n return isAuthorized;\n },\n });\n this._connections.add(connection);\n\n return connection.replicatorExtension;\n }\n\n async authorizeDevice(spaceKey: PublicKey, deviceKey: PublicKey) {\n log('authorizeDevice', { spaceKey, deviceKey });\n const spaceId = await createIdFromSpaceKey(spaceKey);\n defaultMap(this._authorizedDevices, spaceId, () => new ComplexSet(PublicKey.hash)).add(deviceKey);\n for (const connection of this._connections) {\n if (connection.remoteDeviceKey && connection.remoteDeviceKey.equals(deviceKey)) {\n if (this._connectionsPerPeer.has(connection.peerId)) {\n this._context?.onConnectionAuthScopeChanged(connection);\n }\n }\n }\n }\n}\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport * as A from '@dxos/automerge/automerge';\nimport { cbor } from '@dxos/automerge/automerge-repo';\nimport { Resource } from '@dxos/context';\nimport { invariant } from '@dxos/invariant';\nimport { type PublicKey } from '@dxos/keys';\nimport { log } from '@dxos/log';\nimport type { AutomergeProtocolMessage } from '@dxos/protocols';\nimport { AutomergeReplicator, type AutomergeReplicatorFactory } from '@dxos/teleport-extension-automerge-replicator';\n\nimport type { ReplicatorConnection, ShouldAdvertiseParams, ShouldSyncCollectionParams } from './echo-replicator';\n\nconst DEFAULT_FACTORY: AutomergeReplicatorFactory = (params) => new AutomergeReplicator(...params);\n\nexport type MeshReplicatorConnectionParams = {\n ownPeerId: string;\n onRemoteConnected: () => void;\n onRemoteDisconnected: () => void;\n shouldAdvertise: (params: ShouldAdvertiseParams) => Promise<boolean>;\n shouldSyncCollection: (params: ShouldSyncCollectionParams) => boolean;\n replicatorFactory?: AutomergeReplicatorFactory;\n};\n\nexport class MeshReplicatorConnection extends Resource implements ReplicatorConnection {\n public readable: ReadableStream<AutomergeProtocolMessage>;\n public writable: WritableStream<AutomergeProtocolMessage>;\n public remoteDeviceKey: PublicKey | null = null;\n\n public readonly replicatorExtension: AutomergeReplicator;\n\n private _remotePeerId: string | null = null;\n private _isEnabled = false;\n\n constructor(private readonly _params: MeshReplicatorConnectionParams) {\n super();\n\n let readableStreamController!: ReadableStreamDefaultController<AutomergeProtocolMessage>;\n this.readable = new ReadableStream<AutomergeProtocolMessage>({\n start: (controller) => {\n readableStreamController = controller;\n this._ctx.onDispose(() => controller.close());\n },\n });\n\n this.writable = new WritableStream<AutomergeProtocolMessage>({\n write: async (message: AutomergeProtocolMessage, controller) => {\n invariant(this._isEnabled, 'Writing to a disabled connection');\n try {\n logSendSync(message);\n await this.replicatorExtension.sendSyncMessage({ payload: cbor.encode(message) });\n } catch (err) {\n controller.error(err);\n this._disconnectIfEnabled();\n }\n },\n });\n\n const createAutomergeReplicator = this._params.replicatorFactory ?? DEFAULT_FACTORY;\n this.replicatorExtension = createAutomergeReplicator([\n {\n peerId: this._params.ownPeerId,\n },\n {\n onStartReplication: async (info, remotePeerId /** Teleport ID */) => {\n // Note: We store only one extension per peer.\n // There can be a case where two connected peers have more than one teleport connection between them\n // and each of them uses different teleport connections to send messages.\n // It works because we receive messages from all teleport connections and Automerge Repo dedup them.\n // TODO(mykola): Use only one teleport connection per peer.\n\n // TODO(dmaretskyi): Critical bug.\n // - two peers get connected via swarm 1\n // - they get connected via swarm 2\n // - swarm 1 gets disconnected\n // - automerge repo thinks that peer 2 got disconnected even though swarm 2 is still active\n\n this.remoteDeviceKey = remotePeerId;\n\n // Set automerge id.\n this._remotePeerId = info.id;\n\n log('onStartReplication', { id: info.id, thisPeerId: this.peerId, remotePeerId: remotePeerId.toHex() });\n\n this._params.onRemoteConnected();\n },\n onSyncMessage: async ({ payload }) => {\n if (!this._isEnabled) {\n return;\n }\n const message = cbor.decode(payload) as AutomergeProtocolMessage;\n // Note: automerge Repo dedup messages.\n readableStreamController.enqueue(message);\n },\n onClose: async () => {\n this._disconnectIfEnabled();\n },\n },\n ]);\n }\n\n private _disconnectIfEnabled() {\n if (this._isEnabled) {\n this._params.onRemoteDisconnected();\n }\n }\n\n get peerId(): string {\n invariant(this._remotePeerId != null, 'Remote peer has not connected yet.');\n return this._remotePeerId;\n }\n\n async shouldAdvertise(params: ShouldAdvertiseParams): Promise<boolean> {\n return this._params.shouldAdvertise(params);\n }\n\n shouldSyncCollection(params: ShouldSyncCollectionParams): boolean {\n return this._params.shouldSyncCollection(params);\n }\n\n /**\n * Start exchanging messages with the remote peer.\n * Call after the remote peer has connected.\n */\n enable() {\n invariant(this._remotePeerId != null, 'Remote peer has not connected yet.');\n this._isEnabled = true;\n }\n\n /**\n * Stop exchanging messages with the remote peer.\n */\n disable() {\n this._isEnabled = false;\n }\n}\n\nconst logSendSync = (message: AutomergeProtocolMessage) => {\n log('sendSyncMessage', () => {\n const decodedSyncMessage = message.type === 'sync' && message.data ? A.decodeSyncMessage(message.data) : undefined;\n return {\n sync: decodedSyncMessage && {\n headsLength: decodedSyncMessage.heads.length,\n requesting: decodedSyncMessage.need.length > 0,\n sendingChanges: decodedSyncMessage.changes.length > 0,\n },\n type: message.type,\n from: message.senderId,\n to: message.targetId,\n };\n });\n};\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport type { CollectionId } from '@dxos/echo-protocol';\nimport { invariant } from '@dxos/invariant';\nimport { SpaceId } from '@dxos/keys';\n\nexport const deriveCollectionIdFromSpaceId = (spaceId: SpaceId): CollectionId => `space:${spaceId}` as CollectionId;\n\nexport const getSpaceIdFromCollectionId = (collectionId: CollectionId): SpaceId => {\n const spaceId = collectionId.replace(/^space:/, '');\n invariant(SpaceId.isValid(spaceId));\n return spaceId;\n};\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport { type Message } from '@dxos/automerge/automerge-repo';\nimport { type TimeAware, trace } from '@dxos/tracing';\nimport { CircularBuffer, mapValues, SlidingWindowSummary, type SlidingWindowSummaryConfig } from '@dxos/util';\n\nimport { type NetworkDataMonitor } from './echo-network-adapter';\nimport { type StorageAdapterDataMonitor } from './leveldb-storage-adapter';\nimport { isCollectionQueryMessage, isCollectionStateMessage } from './network-protocol';\n\nconst PER_SECOND_RATE_AVG_WINDOW_SIZE = 5;\nconst DEFAULT_AVG_WINDOW_SIZE = 25;\n\nexport type EchoDataMonitorOptions = {\n timeSeriesLength: number;\n};\n\n@trace.resource()\nexport class EchoDataMonitor implements StorageAdapterDataMonitor, NetworkDataMonitor, TimeAware {\n private _lastTick = 0;\n\n private _activeCounters = createLocalCounters();\n private _lastCompleteCounters: LocalCounters | undefined;\n private readonly _localTimeSeries = createLocalTimeSeries();\n private readonly _storageAverages = createStorageAverages();\n private readonly _replicationAverages = createNetworkAverages();\n private readonly _sizeByMessageType: { [type: string]: SlidingWindowSummary } = {};\n private readonly _lastReceivedMessages = new CircularBuffer<StoredMessage>(100);\n private readonly _lastSentMessages = new CircularBuffer<StoredMessage>(100);\n\n private _connectionsCount = 0;\n\n constructor(private readonly _params: EchoDataMonitorOptions = { timeSeriesLength: 30 }) {}\n\n public tick(timeMs: number) {\n this._advanceTimeWindow(timeMs - this._lastTick);\n this._lastTick = timeMs;\n }\n\n public computeStats(): EchoDataStats {\n return {\n meta: {\n rateAverageOverSeconds: PER_SECOND_RATE_AVG_WINDOW_SIZE,\n },\n storage: {\n reads: {\n payloadSize: this._storageAverages.loadedChunkSize.average(),\n opDuration: this._storageAverages.loadDuration.average(),\n countPerSecond: this._storageAverages.loadsPerSecond.average(),\n },\n writes: {\n payloadSize: this._storageAverages.storedChunkSize.average(),\n opDuration: this._storageAverages.storeDuration.average(),\n countPerSecond: this._storageAverages.storesPerSecond.average(),\n },\n },\n replicator: {\n connections: this._connectionsCount,\n receivedMessages: {\n payloadSize: this._replicationAverages.receivedMessageSize.average(),\n countPerSecond: this._replicationAverages.receivedPerSecond.average(),\n },\n sentMessages: {\n payloadSize: this._replicationAverages.sentMessageSize.average(),\n opDuration: this._replicationAverages.sendDuration.average(),\n countPerSecond: this._replicationAverages.sentPerSecond.average(),\n failedPerSecond: this._replicationAverages.sendsFailedPerSecond.average(),\n },\n countByMessageType: this._computeMessageHistogram('type'),\n avgSizeByMessageType: mapValues(this._sizeByMessageType, (summary) => summary.average()),\n },\n };\n }\n\n public get connectionsCount() {\n return this._connectionsCount;\n }\n\n /**\n * @internal\n */\n get lastPerSecondStats() {\n return this._lastCompleteCounters;\n }\n\n /**\n * @internal\n */\n get timeSeries() {\n return { ...this._localTimeSeries.storage, ...this._localTimeSeries.replication };\n }\n\n /**\n * @internal\n */\n get messagesByPeerId() {\n return this._computeMessageHistogram('peerId');\n }\n\n private _advanceTimeWindow(millisPassed: number) {\n const oldMetrics = Object.freeze(this._activeCounters);\n this._activeCounters = createLocalCounters();\n this._lastCompleteCounters = oldMetrics;\n for (const peerId of Object.keys(oldMetrics.byPeerId)) {\n this._activeCounters.byPeerId[peerId] = createMessageCounter();\n }\n this._addToTimeSeries(oldMetrics.replication, this._localTimeSeries.replication);\n this._addToTimeSeries(oldMetrics.storage, this._localTimeSeries.storage);\n // Prevent skewed measurements of incomplete buckets / after CPU freezes.\n if (Math.abs(millisPassed - 1000) < 100) {\n this._reportPerSecondRate(oldMetrics);\n }\n }\n\n private _addToTimeSeries<T extends object>(values: T, timeSeries: TimeSeries<T>) {\n for (const [key, value] of Object.entries(values)) {\n const values: (typeof value)[] = (timeSeries as any)[key];\n values.push(value);\n if (values.length > this._params.timeSeriesLength) {\n values.shift();\n }\n }\n }\n\n private _reportPerSecondRate(metrics: LocalCounters) {\n const toReport: [string, number, SlidingWindowSummary][] = [\n ['storage.load', metrics.storage.loadedChunks, this._storageAverages.loadsPerSecond],\n ['storage.store', metrics.storage.storedChunks, this._storageAverages.storesPerSecond],\n ['network.receive', metrics.replication.received, this._replicationAverages.receivedPerSecond],\n ['network.send', metrics.replication.sent, this._replicationAverages.sentPerSecond],\n ];\n for (const [metricName, metric, summary] of toReport) {\n summary.record(metric);\n if (metric > 0) {\n trace.metrics.distribution(`dxos.echo.${metricName}-rate`, metric);\n trace.metrics.increment(`dxos.echo.${metricName}`, 1, { tags: { status: 'busy' } });\n } else {\n trace.metrics.increment(`dxos.echo.${metricName}`, 1, { tags: { status: 'idle' } });\n }\n }\n this._replicationAverages.sendsFailedPerSecond.record(metrics.replication.failed);\n }\n\n public recordPeerConnected(peerId: string) {\n this._activeCounters.byPeerId[peerId] = createMessageCounter();\n this._connectionsCount++;\n }\n\n public recordPeerDisconnected(peerId: string) {\n this._connectionsCount--;\n delete this._activeCounters.byPeerId[peerId];\n }\n\n public recordBytesStored(count: number) {\n this._activeCounters.storage.storedChunks++;\n this._activeCounters.storage.storedBytes += count;\n this._storageAverages.storedChunkSize.record(count);\n trace.metrics.distribution('dxos.echo.storage.bytes-stored', count, { unit: 'bytes' });\n }\n\n public recordLoadDuration(durationMs: number): void {\n this._storageAverages.loadDuration.record(durationMs);\n }\n\n public recordStoreDuration(durationMs: number): void {\n this._storageAverages.storeDuration.record(durationMs);\n }\n\n public recordBytesLoaded(count: number) {\n this._activeCounters.storage.loadedChunks++;\n this._activeCounters.storage.loadedBytes += count;\n this._storageAverages.loadedChunkSize.record(count);\n trace.metrics.distribution('dxos.echo.storage.bytes-loaded', count, { unit: 'bytes' });\n }\n\n public recordMessageSent(message: Message, duration: number) {\n let metricsGroupName;\n const bytes = getByteCount(message);\n const tags = { type: message.type };\n if (isAutomergeProtocolMessage(message)) {\n this._activeCounters.replication.sent++;\n this._replicationAverages.sendDuration.record(duration);\n this._replicationAverages.sentMessageSize.record(bytes);\n metricsGroupName = 'replication';\n } else {\n metricsGroupName = 'collection-sync';\n }\n trace.metrics.distribution(`dxos.echo.${metricsGroupName}.bytes-sent`, bytes, { unit: 'bytes', tags });\n trace.metrics.distribution(`dxos.echo.${metricsGroupName}.send-duration`, duration, { unit: 'millisecond', tags });\n trace.metrics.increment(`dxos.echo.${metricsGroupName}.send-status`, 1, { tags: { ...tags, success: true } });\n const { messageSize, messageCounts } = this._getStatsForType(message);\n messageSize.record(bytes);\n messageCounts.sent++;\n this._lastSentMessages.push({ type: message.type, peerId: message.targetId });\n }\n\n public recordMessageReceived(message: Message) {\n const bytes = getByteCount(message);\n const tags = { type: message.type };\n if (isAutomergeProtocolMessage(message)) {\n this._activeCounters.replication.received++;\n this._replicationAverages.receivedMessageSize.record(bytes);\n trace.metrics.distribution('dxos.echo.replication.bytes-received', bytes, { unit: 'bytes', tags });\n } else {\n trace.metrics.distribution('dxos.echo.collection-sync.bytes-received', bytes, { unit: 'bytes', tags });\n }\n const { messageSize, messageCounts } = this._getStatsForType(message);\n messageSize.record(bytes);\n messageCounts.received++;\n this._lastReceivedMessages.push({ type: message.type, peerId: message.senderId });\n }\n\n public recordMessageSendingFailed(message: Message) {\n const tags = { type: message.type, success: false };\n if (isAutomergeProtocolMessage(message)) {\n this._activeCounters.replication.failed++;\n trace.metrics.increment('dxos.echo.replication.send-status', 1, { unit: 'bytes', tags });\n } else {\n trace.metrics.increment('dxos.echo.collection-sync.send-status', 1, { unit: 'bytes', tags });\n }\n const { messageCounts } = this._getStatsForType(message);\n messageCounts.failed++;\n }\n\n private _getStatsForType(message: Message) {\n const messageSize = (this._sizeByMessageType[message.type] ??= createSlidingWindow());\n const messageCounts = (this._activeCounters.byType[message.type] ??= createMessageCounter());\n return { messageCounts, messageSize };\n }\n\n private _computeMessageHistogram(groupKey: keyof StoredMessage): MessageAttributeHistogram {\n const result: MessageAttributeHistogram = {};\n for (const receivedMessage of this._lastReceivedMessages) {\n const counters = (result[receivedMessage[groupKey]] ??= { received: 0, sent: 0 });\n counters.received++;\n }\n for (const receivedMessage of this._lastSentMessages) {\n const counters = (result[receivedMessage[groupKey]] ??= { received: 0, sent: 0 });\n counters.sent++;\n }\n return result;\n }\n}\n\ntype BaseDataOpStats = {\n payloadSize: number;\n countPerSecond: number;\n};\n\nexport type TimedDataOpStats = BaseDataOpStats & { opDuration: number };\n\ntype TimeSeries<T extends object> = { [key in keyof T]: T[key][] };\n\ntype StorageCounts = {\n storedChunks: number;\n storedBytes: number;\n loadedChunks: number;\n loadedBytes: number;\n};\ntype StorageCountTimeSeries = TimeSeries<StorageCounts>;\n\ntype MessageCounts = {\n sent: number;\n received: number;\n failed: number;\n};\ntype MessageCountTimeSeries = TimeSeries<MessageCounts>;\n\ntype MessageAttributeHistogram = {\n [messageType: string]: {\n received: number;\n sent: number;\n };\n};\n\nexport type EchoDataStats = {\n meta: {\n rateAverageOverSeconds: number;\n };\n storage: {\n reads: TimedDataOpStats;\n writes: TimedDataOpStats;\n };\n replicator: {\n connections: number;\n receivedMessages: BaseDataOpStats;\n sentMessages: TimedDataOpStats & { failedPerSecond: number };\n avgSizeByMessageType: { [messageType: string]: number };\n countByMessageType: MessageAttributeHistogram;\n };\n};\n\ntype StoredMessage = { type: string; peerId: string };\n\ntype StorageAverages = {\n storedChunkSize: SlidingWindowSummary;\n storesPerSecond: SlidingWindowSummary;\n loadedChunkSize: SlidingWindowSummary;\n loadsPerSecond: SlidingWindowSummary;\n loadDuration: SlidingWindowSummary;\n storeDuration: SlidingWindowSummary;\n};\n\ntype NetworkAverages = {\n receivedMessageSize: SlidingWindowSummary;\n receivedPerSecond: SlidingWindowSummary;\n sentMessageSize: SlidingWindowSummary;\n sentPerSecond: SlidingWindowSummary;\n sendDuration: SlidingWindowSummary;\n sendsFailedPerSecond: SlidingWindowSummary;\n};\n\ntype LocalCounters = {\n storage: StorageCounts;\n replication: MessageCounts;\n byPeerId: { [peerId: string]: MessageCounts };\n byType: { [type: string]: MessageCounts };\n};\n\ntype LocalTimeSeries = {\n storage: StorageCountTimeSeries;\n replication: MessageCountTimeSeries;\n};\n\nconst isAutomergeProtocolMessage = (message: Message) => {\n return !(isCollectionQueryMessage(message) || isCollectionStateMessage(message));\n};\n\nconst createSlidingWindow = (overrides?: SlidingWindowSummaryConfig) =>\n new SlidingWindowSummary({ dataPoints: DEFAULT_AVG_WINDOW_SIZE, precision: 2, ...overrides });\n\nconst createLocalCounters = (): LocalCounters => ({\n storage: { loadedBytes: 0, storedBytes: 0, storedChunks: 0, loadedChunks: 0 },\n replication: createMessageCounter(),\n byPeerId: {},\n byType: {},\n});\n\nconst createLocalTimeSeries = (): LocalTimeSeries => ({\n storage: { loadedBytes: [], storedBytes: [], storedChunks: [], loadedChunks: [] },\n replication: { sent: [], failed: [], received: [] },\n});\n\nconst createMessageCounter = (): MessageCounts => ({ sent: 0, received: 0, failed: 0 });\n\nconst createNetworkAverages = (): NetworkAverages => ({\n receivedMessageSize: createSlidingWindow(),\n sentMessageSize: createSlidingWindow(),\n sendDuration: createSlidingWindow(),\n receivedPerSecond: createSlidingWindow({ dataPoints: PER_SECOND_RATE_AVG_WINDOW_SIZE }),\n sentPerSecond: createSlidingWindow({ dataPoints: PER_SECOND_RATE_AVG_WINDOW_SIZE }),\n sendsFailedPerSecond: createSlidingWindow({ dataPoints: PER_SECOND_RATE_AVG_WINDOW_SIZE }),\n});\n\nconst createStorageAverages = (): StorageAverages => ({\n storedChunkSize: createSlidingWindow(),\n loadedChunkSize: createSlidingWindow(),\n loadDuration: createSlidingWindow(),\n storeDuration: createSlidingWindow(),\n loadsPerSecond: createSlidingWindow({ dataPoints: PER_SECOND_RATE_AVG_WINDOW_SIZE }),\n storesPerSecond: createSlidingWindow({ dataPoints: PER_SECOND_RATE_AVG_WINDOW_SIZE }),\n});\n\nconst getByteCount = (message: Message): number => {\n return (\n message.type.length +\n message.senderId.length +\n message.targetId.length +\n (message.data?.byteLength ?? 0) +\n (message.documentId?.length ?? 0)\n );\n};\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport {\n type AnyDocumentId,\n type AutomergeUrl,\n type DocHandle,\n type DocumentId,\n type Repo,\n} from '@dxos/automerge/automerge-repo';\nimport { LifecycleState, Resource, type Context } from '@dxos/context';\nimport { todo } from '@dxos/debug';\nimport { createIdFromSpaceKey, SpaceDocVersion, type SpaceDoc } from '@dxos/echo-protocol';\nimport { IndexMetadataStore, IndexStore, Indexer } from '@dxos/indexing';\nimport { invariant } from '@dxos/invariant';\nimport { type PublicKey, type SpaceId } from '@dxos/keys';\nimport { type LevelDB } from '@dxos/kv-store';\nimport { IndexKind, type IndexConfig } from '@dxos/protocols/proto/dxos/echo/indexing';\nimport { trace } from '@dxos/tracing';\n\nimport { DataServiceImpl } from './data-service';\nimport { type DatabaseRoot } from './database-root';\nimport { createSelectedDocumentsIterator } from './documents-iterator';\nimport { QueryServiceImpl } from './query-service';\nimport { SpaceStateManager } from './space-state-manager';\nimport {\n AutomergeHost,\n EchoDataMonitor,\n deriveCollectionIdFromSpaceId,\n type LoadDocOptions,\n type CreateDocOptions,\n type EchoReplicator,\n type CollectionSyncState,\n type EchoDataStats,\n type PeerIdProvider,\n} from '../automerge';\n\nconst INDEXER_CONFIG: IndexConfig = {\n enabled: true,\n indexes: [{ kind: IndexKind.Kind.SCHEMA_MATCH }],\n};\n\nexport type EchoHostParams = {\n kv: LevelDB;\n peerIdProvider?: PeerIdProvider;\n};\n\n/**\n * Host for the Echo database.\n * Manages multiple spaces.\n * Stores data to disk.\n * Can sync with pluggable data replicators.\n */\nexport class EchoHost extends Resource {\n private readonly _indexMetadataStore: IndexMetadataStore;\n private readonly _indexer: Indexer;\n private readonly _automergeHost: AutomergeHost;\n private readonly _queryService: QueryServiceImpl;\n private readonly _dataService: DataServiceImpl;\n private readonly _spaceStateManager = new SpaceStateManager();\n private readonly _echoDataMonitor: EchoDataMonitor;\n\n constructor({ kv, peerIdProvider }: EchoHostParams) {\n super();\n\n this._indexMetadataStore = new IndexMetadataStore({ db: kv.sublevel('index-metadata') });\n\n this._echoDataMonitor = new EchoDataMonitor();\n\n this._automergeHost = new AutomergeHost({\n db: kv,\n dataMonitor: this._echoDataMonitor,\n indexMetadataStore: this._indexMetadataStore,\n peerIdProvider,\n });\n\n this._indexer = new Indexer({\n db: kv,\n indexStore: new IndexStore({ db: kv.sublevel('index-storage') }),\n metadataStore: this._indexMetadataStore,\n loadDocuments: createSelectedDocumentsIterator(this._automergeHost),\n indexCooldownTime: process.env.NODE_ENV === 'test' ? 0 : undefined,\n });\n this._indexer.setConfig(INDEXER_CONFIG);\n\n this._queryService = new QueryServiceImpl({\n automergeHost: this._automergeHost,\n indexer: this._indexer,\n });\n\n this._dataService = new DataServiceImpl({\n automergeHost: this._automergeHost,\n updateIndexes: async () => {\n await this._indexer.updateIndexes();\n },\n });\n\n trace.diagnostic<EchoStatsDiagnostic>({\n id: 'echo-stats',\n name: 'Echo Stats',\n fetch: async () => {\n return {\n dataStats: this._echoDataMonitor.computeStats(),\n loadedDocsCount: this._automergeHost.loadedDocsCount,\n };\n },\n });\n\n trace.diagnostic({\n id: 'database-roots',\n name: 'Database Roots',\n fetch: async () => {\n return Array.from(this._spaceStateManager.roots.values()).map((root) => ({\n url: root.url,\n isLoaded: root.isLoaded,\n spaceKey: root.getSpaceKey(),\n inlineObjects: root.getInlineObjectCount(),\n linkedObjects: root.getLinkedObjectCount(),\n }));\n },\n });\n\n trace.diagnostic({\n id: 'database-root-metrics',\n name: 'Database Roots (with metrics)',\n fetch: async () => {\n return Array.from(this._spaceStateManager.roots.values()).map((root) => ({\n url: root.url,\n isLoaded: root.isLoaded,\n spaceKey: root.getSpaceKey(),\n inlineObjects: root.getInlineObjectCount(),\n linkedObjects: root.getLinkedObjectCount(),\n ...(root.measureMetrics() ?? {}),\n }));\n },\n });\n }\n\n get queryService(): QueryServiceImpl {\n return this._queryService;\n }\n\n get dataService(): DataServiceImpl {\n return this._dataService;\n }\n\n /**\n * @deprecated To be abstracted away.\n */\n get automergeRepo(): Repo {\n return this._automergeHost.repo;\n }\n\n get roots(): ReadonlyMap<DocumentId, DatabaseRoot> {\n return this._spaceStateManager.roots;\n }\n\n protected override async _open(ctx: Context): Promise<void> {\n await this._automergeHost.open();\n await this._indexer.open(ctx);\n await this._queryService.open(ctx);\n await this._spaceStateManager.open(ctx);\n\n this._spaceStateManager.spaceDocumentListUpdated.on(this._ctx, (e) => {\n void this._automergeHost.updateLocalCollectionState(deriveCollectionIdFromSpaceId(e.spaceId), e.documentIds);\n });\n }\n\n protected override async _close(ctx: Context): Promise<void> {\n await this._spaceStateManager.close();\n await this._queryService.close(ctx);\n await this._indexer.close(ctx);\n await this._automergeHost.close();\n }\n\n /**\n * Flush all pending writes to the underlying storage.\n */\n async flush() {\n await this._automergeHost.repo.flush();\n }\n\n /**\n * Perform any pending index updates.\n */\n async updateIndexes() {\n await this._indexer.updateIndexes();\n }\n\n /**\n * Loads the document handle from the repo and waits for it to be ready.\n */\n async loadDoc<T>(ctx: Context, documentId: AnyDocumentId, opts?: LoadDocOptions): Promise<DocHandle<T>> {\n return await this._automergeHost.loadDoc(ctx, documentId, opts);\n }\n\n /**\n * Create new persisted document.\n */\n createDoc<T>(initialValue?: T, opts?: CreateDocOptions): DocHandle<T> {\n return this._automergeHost.createDoc(initialValue, opts);\n }\n\n /**\n * Create new space root.\n */\n async createSpaceRoot(spaceKey: PublicKey): Promise<DatabaseRoot> {\n invariant(this._lifecycleState === LifecycleState.OPEN);\n const spaceId = await createIdFromSpaceKey(spaceKey);\n\n const automergeRoot = this._automergeHost.createDoc<SpaceDoc>({\n version: SpaceDocVersion.CURRENT,\n access: { spaceKey: spaceKey.toHex() },\n\n // Better to initialize them right away to avoid merge conflicts and data loss that can occur if those maps get created on the fly.\n objects: {},\n links: {},\n });\n\n await this._automergeHost.flush({ documentIds: [automergeRoot.documentId] });\n\n return await this.openSpaceRoot(spaceId, automergeRoot.url);\n }\n\n // TODO(dmaretskyi): Change to document id.\n async openSpaceRoot(spaceId: SpaceId, automergeUrl: AutomergeUrl): Promise<DatabaseRoot> {\n invariant(this._lifecycleState === LifecycleState.OPEN);\n const handle = this._automergeHost.repo.find(automergeUrl);\n\n return this._spaceStateManager.assignRootToSpace(spaceId, handle);\n }\n\n // TODO(dmaretskyi): Change to document id.\n async closeSpaceRoot(automergeUrl: AutomergeUrl): Promise<void> {\n todo();\n }\n\n /**\n * Install data replicator.\n */\n async addReplicator(replicator: EchoReplicator): Promise<void> {\n await this._automergeHost.addReplicator(replicator);\n }\n\n /**\n * Remove data replicator.\n */\n async removeReplicator(replicator: EchoReplicator): Promise<void> {\n await this._automergeHost.removeReplicator(replicator);\n }\n\n async getSpaceSyncState(spaceId: SpaceId): Promise<CollectionSyncState> {\n const collectionId = deriveCollectionIdFromSpaceId(spaceId);\n return this._automergeHost.getCollectionSyncState(collectionId);\n }\n}\n\nexport type { EchoDataStats };\n\nexport type EchoStatsDiagnostic = {\n loadedDocsCount: number;\n dataStats: EchoDataStats;\n};\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport * as A from '@dxos/automerge/automerge';\nimport { type DocumentId } from '@dxos/automerge/automerge-repo';\nimport { Context } from '@dxos/context';\nimport { SpaceDocVersion, type SpaceDoc } from '@dxos/echo-protocol';\nimport { type ObjectSnapshot, type IdToHeads } from '@dxos/indexing';\nimport { invariant } from '@dxos/invariant';\nimport { log } from '@dxos/log';\nimport { ObjectPointerVersion, objectPointerCodec } from '@dxos/protocols';\n\nimport { type AutomergeHost, getSpaceKeyFromDoc } from '../automerge';\n\nconst LOG_VIEW_OPERATION_THRESHOLD = 300;\n\n/**\n * Factory for `loadDocuments` iterator.\n */\nexport const createSelectedDocumentsIterator = (automergeHost: AutomergeHost) =>\n /**\n * Get object data blobs from Automerge Repo by ids.\n */\n // TODO(mykola): Unload automerge handles after usage.\n async function* loadDocuments(objects: IdToHeads): AsyncGenerator<ObjectSnapshot[], void, void> {\n for (const [id, heads] of objects.entries()) {\n try {\n const { documentId, objectId } = objectPointerCodec.decode(id);\n const handle = await automergeHost.loadDoc(Context.default(), documentId as DocumentId);\n\n let doc: A.Doc<SpaceDoc> = handle.docSync();\n invariant(doc);\n\n const currentHeads = A.getHeads(doc);\n\n // Checkout the requested version of the document.\n if (!A.equals(currentHeads, heads)) {\n const begin = Date.now();\n // `view` can take a long time even if the document is already at the right version.\n doc = A.view(doc, heads);\n const end = Date.now();\n if (end - begin > LOG_VIEW_OPERATION_THRESHOLD) {\n log.info('Checking out document version is taking too long', {\n duration: end - begin,\n requestedHeads: heads,\n originalHeads: currentHeads,\n });\n }\n }\n\n // Skip outdated docs.\n if (doc.version !== SpaceDocVersion.CURRENT) {\n continue;\n }\n\n if (!doc.objects?.[objectId]) {\n continue;\n }\n\n // Upgrade V0 object pointers to V1.\n let newId = id;\n if (objectPointerCodec.getVersion(id) === ObjectPointerVersion.V0) {\n const spaceKey = getSpaceKeyFromDoc(doc) ?? undefined;\n newId = objectPointerCodec.encode({ documentId, objectId, spaceKey });\n }\n\n yield [{ id: newId, object: doc.objects![objectId], heads }];\n } catch (error) {\n log.error('Error loading document', { heads, id, error });\n }\n }\n };\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport { DeferredTask } from '@dxos/async';\nimport { getHeads, type Doc } from '@dxos/automerge/automerge';\nimport { type DocHandle, type DocumentId } from '@dxos/automerge/automerge-repo';\nimport { Stream } from '@dxos/codec-protobuf';\nimport { Context, Resource } from '@dxos/context';\nimport { type SpaceDoc } from '@dxos/echo-protocol';\nimport { type ObjectSnapshot, type Indexer, type IdToHeads } from '@dxos/indexing';\nimport { log } from '@dxos/log';\nimport { objectPointerCodec } from '@dxos/protocols';\nimport { type IndexConfig } from '@dxos/protocols/proto/dxos/echo/indexing';\nimport {\n type QueryRequest,\n type QueryResponse,\n type QueryService,\n type QueryResult,\n} from '@dxos/protocols/proto/dxos/echo/query';\nimport { trace } from '@dxos/tracing';\n\nimport { QueryState } from './query-state';\nimport { type AutomergeHost, getSpaceKeyFromDoc } from '../automerge';\n\nexport type QueryServiceParams = {\n indexer: Indexer;\n automergeHost: AutomergeHost;\n};\n\n/**\n * Represents an active query (stream and query state connected to that stream).\n */\ntype ActiveQuery = {\n state: QueryState;\n sendResults: (results: QueryResult[]) => void;\n close: () => Promise<void>;\n};\n\nexport class QueryServiceImpl extends Resource implements QueryService {\n private readonly _queries = new Set<ActiveQuery>();\n\n private readonly _updateQueries = new DeferredTask(this._ctx, async () => {\n await Promise.all(\n Array.from(this._queries).map(async (query) => {\n try {\n const { changed } = await query.state.execQuery();\n if (changed) {\n query.sendResults(query.state.getResults());\n }\n } catch (err) {\n log.catch(err);\n }\n }),\n );\n });\n\n // TODO(burdon): OK for options, but not params. Pass separately and type readonly here.\n constructor(private readonly _params: QueryServiceParams) {\n super();\n\n trace.diagnostic({\n id: 'active-queries',\n name: 'Active Queries',\n fetch: () => {\n return Array.from(this._queries).map((query) => {\n return {\n filter: JSON.stringify(query.state.filter),\n metrics: query.state.metrics,\n };\n });\n },\n });\n }\n\n override async _open() {\n this._params.indexer.updated.on(this._ctx, () => this._updateQueries.schedule());\n }\n\n override async _close() {\n await Promise.all(Array.from(this._queries).map((query) => query.close()));\n }\n\n async setConfig(config: IndexConfig): Promise<void> {\n if (this._params.indexer.initialized) {\n log.warn('Indexer already initialized.');\n return;\n }\n this._params.indexer.setConfig(config);\n }\n\n execQuery(request: QueryRequest): Stream<QueryResponse> {\n return new Stream<QueryResponse>(({ next, close, ctx }) => {\n const query: ActiveQuery = {\n state: new QueryState({\n indexer: this._params.indexer,\n automergeHost: this._params.automergeHost,\n request,\n }),\n sendResults: (results) => {\n if (ctx.disposed) {\n return;\n }\n next({ queryId: request.queryId, results });\n },\n close: async () => {\n close();\n await query.state.close();\n this._queries.delete(query);\n },\n };\n this._queries.add(query);\n\n queueMicrotask(async () => {\n await query.state.open();\n\n try {\n const { changed } = await query.state.execQuery();\n if (changed) {\n query.sendResults(query.state.getResults());\n }\n } catch (error) {\n log.catch(error);\n }\n });\n\n return query.close;\n });\n }\n\n /**\n * Re-index all loaded documents.\n */\n async reindex() {\n log.info('Reindexing all documents...');\n const iterator = createDocumentsIterator(this._params.automergeHost);\n const ids: IdToHeads = new Map();\n for await (const documents of iterator()) {\n for (const { id, heads } of documents) {\n ids.set(id, heads);\n }\n if (ids.size % 100 === 0) {\n log.info('Collected documents...', { count: ids.size });\n }\n }\n\n log.info('Marking all documents as dirty...', { count: ids.size });\n await this._params.indexer.reindex(ids);\n }\n}\n\n/**\n * Factory for `getAllDocuments` iterator.\n */\n// TODO(dmaretskyi): Get roots from echo-host.\nconst createDocumentsIterator = (automergeHost: AutomergeHost) =>\n /**\n * Recursively get all object data blobs from loaded documents from Automerge Repo.\n */\n // TODO(mykola): Unload automerge handles after usage.\n async function* getAllDocuments(): AsyncGenerator<ObjectSnapshot[], void, void> {\n /** visited automerge handles */\n const visited = new Set<string>();\n\n async function* getObjectsFromHandle(handle: DocHandle<any>): AsyncGenerator<ObjectSnapshot[]> {\n if (visited.has(handle.documentId)) {\n return;\n }\n const doc: Doc<SpaceDoc> = handle.docSync();\n\n const spaceKey = getSpaceKeyFromDoc(doc) ?? undefined;\n\n if (doc.objects) {\n yield Object.entries(doc.objects as { [key: string]: any }).map(([objectId, object]) => {\n return {\n id: objectPointerCodec.encode({ documentId: handle.documentId, objectId, spaceKey }),\n object,\n heads: getHeads(doc),\n };\n });\n }\n\n if (doc.links) {\n for (const id of Object.values(doc.links as { [echoId: string]: string })) {\n const urlString = id.toString();\n if (visited.has(urlString)) {\n continue;\n }\n const linkHandle = await automergeHost.loadDoc(Context.default(), urlString as DocumentId);\n for await (const result of getObjectsFromHandle(linkHandle)) {\n yield result;\n }\n }\n }\n\n visited.add(handle.documentId);\n }\n\n // TODO(mykola): Use list of roots instead of iterating over all handles.\n for (const handle of Object.values(automergeHost.repo.handles)) {\n if (visited.has(handle.documentId)) {\n continue;\n }\n for await (const result of getObjectsFromHandle(handle)) {\n yield result;\n }\n visited.add(handle.documentId);\n }\n };\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport { type DocumentId } from '@dxos/automerge/automerge-repo';\nimport { Context, LifecycleState, Resource } from '@dxos/context';\nimport { createIdFromSpaceKey } from '@dxos/echo-protocol';\nimport { type Indexer, type IndexQuery } from '@dxos/indexing';\nimport { invariant } from '@dxos/invariant';\nimport { DXN, PublicKey } from '@dxos/keys';\nimport { objectPointerCodec } from '@dxos/protocols';\nimport { type Filter as FilterProto } from '@dxos/protocols/proto/dxos/echo/filter';\nimport { type QueryRequest, type QueryResult } from '@dxos/protocols/proto/dxos/echo/query';\nimport { trace } from '@dxos/tracing';\nimport { nonNullable } from '@dxos/util';\n\nimport { type AutomergeHost, getSpaceKeyFromDoc } from '../automerge';\n\ntype QueryStateParams = {\n indexer: Indexer;\n automergeHost: AutomergeHost;\n request: QueryRequest;\n};\n\ntype QueryRunResult = {\n changed: boolean;\n};\n\nexport type QueryMetrics = {\n objectsReturned: number;\n objectsReturnedFromIndex: number;\n documentsLoaded: number;\n executionTime: number;\n indexQueryTime: number;\n documentLoadTime: number;\n};\n\n/**\n * Manages querying logic on service side.\n */\n@trace.resource()\nexport class QueryState extends Resource {\n private _results: QueryResult[] = [];\n\n /**\n * Metrics are only captured for the first run of the query since that is the most representative.\n * We plan to change the query logic so that reactive updates do not require a full re-run of the query.\n */\n private _firstRun = true;\n\n @trace.info({ depth: null })\n public readonly filter: FilterProto;\n\n @trace.info()\n public metrics: QueryMetrics = {\n objectsReturned: 0,\n objectsReturnedFromIndex: 0,\n documentsLoaded: 0,\n executionTime: 0,\n indexQueryTime: 0,\n documentLoadTime: 0,\n };\n\n @trace.info()\n get active() {\n return this._lifecycleState === LifecycleState.OPEN;\n }\n\n constructor(private readonly _params: QueryStateParams) {\n super();\n this.filter = _params.request.filter;\n }\n\n getResults() {\n return this._results;\n }\n\n // https://github.com/open-telemetry/semantic-conventions/blob/main/docs/attributes-registry/db.md#generic-database-attributes\n @trace.span({ showInBrowserTimeline: true, op: 'db.query', attributes: { 'db.system': 'echo' } })\n async execQuery(): Promise<QueryRunResult> {\n const filter = this._params.request.filter;\n\n const beginQuery = performance.now();\n\n // For object id filters, we return no results as those are handled by the SpaceQuerySource.\n const hits =\n filter.objectIds && filter.objectIds?.length > 0\n ? []\n : await this._params.indexer.execQuery(filterToIndexQuery(filter));\n if (this._firstRun) {\n this.metrics.indexQueryTime = performance.now() - beginQuery;\n }\n\n const beginFilter = performance.now();\n\n const results: QueryResult[] = (\n await Promise.all(\n hits.map(async (result) => {\n if (this._firstRun) {\n this.metrics.objectsReturnedFromIndex++;\n }\n\n const { objectId, documentId, spaceKey: spaceKeyInIndex } = objectPointerCodec.decode(result.id);\n\n let spaceKey: string | null;\n if (spaceKeyInIndex !== undefined) {\n spaceKey = spaceKeyInIndex;\n } else {\n // Indexes created by older versions of the indexer do not have the spaceKey in the index.\n // If the spaceKey is not in the index, we need to load the document to get it.\n\n if (this._firstRun) {\n this.metrics.documentsLoaded++;\n }\n\n const handle = await this._params.automergeHost.loadDoc(Context.default(), documentId as DocumentId);\n\n // `whenReady` creates a timeout so we guard it with an if to skip it if the handle is already ready.\n if (this._ctx.disposed) {\n return;\n }\n spaceKey = getSpaceKeyFromDoc(handle.docSync());\n }\n\n if (!spaceKey) {\n return;\n }\n // TODO(mykola): Remove business logic from here.\n if (\n this._params.request.filter.options?.spaces?.length &&\n !this._params.request.filter.options.spaces.some((key) => key.equals(spaceKey!))\n ) {\n return;\n }\n\n if (this._firstRun) {\n this.metrics.objectsReturned++;\n }\n\n return {\n id: objectId,\n documentId,\n spaceId: await createIdFromSpaceKey(PublicKey.from(spaceKey)),\n spaceKey: PublicKey.from(spaceKey),\n rank: result.rank,\n } satisfies QueryResult;\n }),\n )\n ).filter(nonNullable);\n\n if (this._firstRun) {\n this.metrics.documentLoadTime = performance.now() - beginFilter;\n }\n\n if (this._ctx.disposed) {\n return { changed: false };\n }\n\n const areResultsUnchanged =\n !this._firstRun &&\n this._results.length === results.length &&\n this._results.every((oldResult) => results.some((result) => result.id === oldResult.id)) &&\n results.every((result) => this._results.some((oldResult) => oldResult.id === result.id));\n\n if (this._firstRun) {\n this.metrics.executionTime = performance.now() - beginQuery;\n }\n\n this._firstRun = false;\n if (areResultsUnchanged) {\n return { changed: false };\n }\n\n this._results = results;\n return { changed: true };\n }\n}\n\n// TODO(burdon): Process Filter DSL.\nconst filterToIndexQuery = (filter: FilterProto): IndexQuery => {\n invariant(!(filter.type && (filter.or ?? []).length > 0), 'Cannot mix type and or filters.');\n invariant(\n (filter.or ?? []).every((subFilter) => !(subFilter.type && (subFilter.or ?? []).length > 0)),\n 'Cannot mix type and or filters.',\n );\n if (\n filter.type ||\n ((filter.or ?? []).length > 0 && (filter.or ?? []).every((subFilter) => !subFilter.not && subFilter.type))\n ) {\n return {\n typenames:\n filter.type && filter.type.length > 0\n ? filter.type.map((type) => dxnToIndexerTypename(DXN.parse(type)))\n : (filter.or ?? [])\n .flatMap((f) => f.type?.map((type) => dxnToIndexerTypename(DXN.parse(type))) ?? [])\n .filter(nonNullable),\n inverted: filter.not,\n };\n } else {\n // Query all objects.\n return { typenames: [] };\n }\n};\n\nconst dxnToIndexerTypename = (dxn: DXN) => {\n switch (dxn.kind) {\n case DXN.kind.TYPE:\n return dxn.parts[0];\n case DXN.kind.ECHO:\n return dxn.parts[1];\n default:\n throw new Error(`Invalid DXN kind: ${dxn.kind}`);\n }\n};\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport isEqual from 'lodash.isequal';\n\nimport { Event, UpdateScheduler } from '@dxos/async';\nimport { interpretAsDocumentId, type DocHandle, type DocumentId } from '@dxos/automerge/automerge-repo';\nimport { Resource, Context } from '@dxos/context';\nimport { type SpaceDoc } from '@dxos/echo-protocol';\nimport { type SpaceId } from '@dxos/keys';\n\nimport { DatabaseRoot } from './database-root';\n\nexport class SpaceStateManager extends Resource {\n private readonly _roots = new Map<DocumentId, DatabaseRoot>();\n private readonly _rootBySpace = new Map<SpaceId, DocumentId>();\n private readonly _perRootContext = new Map<DocumentId, Context>();\n private readonly _lastSpaceDocumentList = new Map<SpaceId, DocumentId[]>();\n\n public readonly spaceDocumentListUpdated = new Event<SpaceDocumentListUpdatedEvent>();\n\n protected override async _close(ctx: Context): Promise<void> {\n for (const [_, rootCtx] of this._perRootContext) {\n await rootCtx.dispose();\n }\n this._roots.clear();\n }\n\n get roots(): ReadonlyMap<DocumentId, DatabaseRoot> {\n return this._roots;\n }\n\n getRootByDocumentId(documentId: DocumentId): DatabaseRoot | undefined {\n return this._roots.get(documentId);\n }\n\n async assignRootToSpace(spaceId: SpaceId, handle: DocHandle<SpaceDoc>): Promise<DatabaseRoot> {\n let root: DatabaseRoot;\n if (this._roots.has(handle.documentId)) {\n root = this._roots.get(handle.documentId)!;\n } else {\n root = new DatabaseRoot(handle);\n this._roots.set(handle.documentId, root);\n }\n\n if (this._rootBySpace.get(spaceId) === root.handle.documentId) {\n return root;\n }\n\n const prevRootId = this._rootBySpace.get(spaceId);\n if (prevRootId) {\n void this._perRootContext.get(prevRootId)?.dispose();\n this._perRootContext.delete(prevRootId);\n }\n\n this._rootBySpace.set(spaceId, root.handle.documentId);\n const ctx = new Context();\n\n this._perRootContext.set(root.handle.documentId, ctx);\n\n await root.handle.whenReady();\n\n const documentListCheckScheduler = new UpdateScheduler(\n ctx,\n async () => {\n const documentIds = [root.documentId, ...root.getAllLinkedDocuments().map((url) => interpretAsDocumentId(url))];\n if (!isEqual(documentIds, this._lastSpaceDocumentList.get(spaceId))) {\n this._lastSpaceDocumentList.set(spaceId, documentIds);\n this.spaceDocumentListUpdated.emit(new SpaceDocumentListUpdatedEvent(spaceId, documentIds));\n }\n },\n { maxFrequency: 50 },\n );\n const triggerCheckOnChange = () => documentListCheckScheduler.trigger();\n root.handle.addListener('change', triggerCheckOnChange);\n ctx.onDispose(() => root.handle.removeListener('change', triggerCheckOnChange));\n\n documentListCheckScheduler.trigger();\n\n return root;\n }\n}\n\nexport class SpaceDocumentListUpdatedEvent {\n constructor(\n public readonly spaceId: SpaceId,\n public readonly documentIds: DocumentId[],\n ) {}\n}\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport type * as A from '@dxos/automerge/automerge';\nimport type { AutomergeUrl, DocHandle, DocumentId } from '@dxos/automerge/automerge-repo';\nimport { type SpaceDoc, SpaceDocVersion } from '@dxos/echo-protocol';\nimport { invariant } from '@dxos/invariant';\n\nimport { measureDocMetrics, type DocMetrics } from './automerge-metrics';\nimport { getSpaceKeyFromDoc } from '../automerge';\n\nexport class DatabaseRoot {\n constructor(private readonly _rootHandle: DocHandle<SpaceDoc>) {}\n\n get documentId(): DocumentId {\n return this._rootHandle.documentId;\n }\n\n get url() {\n return this._rootHandle.url;\n }\n\n get isLoaded(): boolean {\n return !!this._rootHandle.docSync();\n }\n\n get handle(): DocHandle<SpaceDoc> {\n return this._rootHandle;\n }\n\n docSync(): A.Doc<SpaceDoc> | null {\n return this._rootHandle.docSync();\n }\n\n getVersion(): SpaceDocVersion | null {\n const doc = this.docSync();\n if (!doc) {\n return null;\n }\n\n return doc.version ?? SpaceDocVersion.LEGACY;\n }\n\n getSpaceKey(): string | null {\n const doc = this.docSync();\n if (!doc) {\n return null;\n }\n\n return getSpaceKeyFromDoc(doc);\n }\n\n getInlineObjectCount(): number | null {\n const doc = this.docSync();\n if (!doc) {\n return null;\n }\n\n return Object.keys(doc.objects ?? {}).length;\n }\n\n getLinkedObjectCount(): number | null {\n const doc = this.docSync();\n if (!doc) {\n return null;\n }\n\n return Object.keys(doc.links ?? {}).length;\n }\n\n getAllLinkedDocuments(): AutomergeUrl[] {\n const doc = this.docSync();\n invariant(doc);\n\n // .toString() to handle RawString.\n return Object.values(doc.links ?? {}).map((s) => s.toString()) as AutomergeUrl[];\n }\n\n measureMetrics(): DocMetrics | null {\n const doc = this.docSync();\n if (!doc) {\n return null;\n }\n return measureDocMetrics(doc);\n }\n}\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport * as A from '@dxos/automerge/automerge';\nimport { log } from '@dxos/log';\n\nexport type DocMetrics = {\n compressedByteSize: number;\n loadTime: number;\n mutationCount: number;\n};\n\n/**\n * WARN: Slow to run on large docs.\n */\nexport const measureDocMetrics = (doc: A.Doc<any>): DocMetrics => {\n const snapshot = A.save(doc);\n\n const start = Date.now();\n const temp = A.load(snapshot);\n const end = Date.now();\n A.free(temp);\n\n const getAllChangesStart = Date.now();\n const mutationCount = A.getAllChanges(doc).length;\n const getAllChangesEnd = Date.now();\n\n if (getAllChangesEnd - getAllChangesStart > 300) {\n log.warn('getAllChanges took too long', { elapsed: getAllChangesEnd - getAllChangesStart });\n }\n\n return {\n compressedByteSize: snapshot.byteLength,\n loadTime: end - start,\n mutationCount,\n };\n};\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport { Mutex, scheduleTask, scheduleMicroTask } from '@dxos/async';\nimport * as A from '@dxos/automerge/automerge';\nimport { cbor } from '@dxos/automerge/automerge-repo';\nimport { Context, Resource } from '@dxos/context';\nimport { randomUUID } from '@dxos/crypto';\nimport type { CollectionId } from '@dxos/echo-protocol';\nimport { type EdgeConnection } from '@dxos/edge-client';\nimport { invariant } from '@dxos/invariant';\nimport type { SpaceId } from '@dxos/keys';\nimport { log } from '@dxos/log';\nimport { EdgeService, type AutomergeProtocolMessage, type PeerId } from '@dxos/protocols';\nimport { buf } from '@dxos/protocols/buf';\nimport {\n type Message as RouterMessage,\n MessageSchema as RouterMessageSchema,\n} from '@dxos/protocols/buf/dxos/edge/messenger_pb';\nimport { bufferToArray } from '@dxos/util';\n\nimport {\n getSpaceIdFromCollectionId,\n type EchoReplicator,\n type EchoReplicatorContext,\n type ReplicatorConnection,\n type ShouldAdvertiseParams,\n type ShouldSyncCollectionParams,\n} from '../automerge';\n\n/**\n * Delay before restarting the connection after receiving a forbidden error.\n */\nconst INITIAL_RESTART_DELAY = 500;\nconst RESTART_DELAY_JITTER = 250;\nconst MAX_RESTART_DELAY = 5000;\n\nexport type EchoEdgeReplicatorParams = {\n edgeConnection: EdgeConnection;\n disableSharePolicy?: boolean;\n};\n\nexport class EchoEdgeReplicator implements EchoReplicator {\n private readonly _edgeConnection: EdgeConnection;\n private readonly _mutex = new Mutex();\n\n private _ctx?: Context = undefined;\n private _context: EchoReplicatorContext | null = null;\n private _connectedSpaces = new Set<SpaceId>();\n private _connections = new Map<SpaceId, EdgeReplicatorConnection>();\n private _sharePolicyEnabled = true;\n\n constructor({ edgeConnection, disableSharePolicy }: EchoEdgeReplicatorParams) {\n this._edgeConnection = edgeConnection;\n this._sharePolicyEnabled = !disableSharePolicy;\n }\n\n async connect(context: EchoReplicatorContext): Promise<void> {\n log.info('connect', { peerId: context.peerId, connectedSpaces: this._connectedSpaces.size });\n this._context = context;\n\n this._ctx = Context.default();\n this._ctx.onDispose(\n this._edgeConnection.onReconnected(() => {\n this._ctx && scheduleMicroTask(this._ctx, () => this._handleReconnect());\n }),\n );\n }\n\n private async _handleReconnect() {\n using _guard = await this._mutex.acquire();\n\n const spaces = [...this._connectedSpaces];\n for (const connection of this._connections.values()) {\n await connection.close();\n }\n this._connections.clear();\n\n if (this._context !== null) {\n for (const spaceId of spaces) {\n await this._openConnection(spaceId);\n }\n }\n }\n\n async disconnect(): Promise<void> {\n using _guard = await this._mutex.acquire();\n await this._ctx?.dispose();\n\n for (const connection of this._connections.values()) {\n await connection.close();\n }\n this._connections.clear();\n }\n\n async connectToSpace(spaceId: SpaceId) {\n using _guard = await this._mutex.acquire();\n\n if (this._connectedSpaces.has(spaceId)) {\n return;\n }\n this._connectedSpaces.add(spaceId);\n\n // Check if AM-repo requested that we connect to remote peers.\n if (this._context !== null) {\n await this._openConnection(spaceId);\n }\n }\n\n async disconnectFromSpace(spaceId: SpaceId) {\n using _guard = await this._mutex.acquire();\n\n this._connectedSpaces.delete(spaceId);\n\n const connection = this._connections.get(spaceId);\n if (connection) {\n await connection.close();\n this._connections.delete(spaceId);\n }\n }\n\n private async _openConnection(spaceId: SpaceId, reconnects: number = 0) {\n invariant(this._context);\n invariant(!this._connections.has(spaceId));\n\n let restartScheduled = false;\n\n const connection = new EdgeReplicatorConnection({\n edgeConnection: this._edgeConnection,\n spaceId,\n context: this._context,\n sharedPolicyEnabled: this._sharePolicyEnabled,\n onRemoteConnected: async () => {\n this._context?.onConnectionOpen(connection);\n },\n onRemoteDisconnected: async () => {\n this._context?.onConnectionClosed(connection);\n },\n onRestartRequested: async () => {\n if (!this._ctx || restartScheduled) {\n return;\n }\n\n const restartDelay =\n Math.min(MAX_RESTART_DELAY, INITIAL_RESTART_DELAY * reconnects) + Math.random() * RESTART_DELAY_JITTER;\n\n log.info('connection restart scheduled', { spaceId, reconnects, restartDelay });\n\n restartScheduled = true;\n scheduleTask(\n this._ctx,\n async () => {\n using _guard = await this._mutex.acquire();\n if (this._connections.get(spaceId) !== connection) {\n return;\n }\n\n const ctx = this._ctx;\n await connection.close(); // Will call onRemoteDisconnected\n this._connections.delete(spaceId);\n if (ctx?.disposed) {\n return;\n }\n await this._openConnection(spaceId, reconnects + 1);\n },\n restartDelay,\n );\n },\n });\n this._connections.set(spaceId, connection);\n\n await connection.open();\n }\n}\n\ntype EdgeReplicatorConnectionsParams = {\n edgeConnection: EdgeConnection;\n spaceId: SpaceId;\n context: EchoReplicatorContext;\n sharedPolicyEnabled: boolean;\n onRemoteConnected: () => Promise<void>;\n onRemoteDisconnected: () => Promise<void>;\n onRestartRequested: () => Promise<void>;\n};\n\nclass EdgeReplicatorConnection extends Resource implements ReplicatorConnection {\n private readonly _edgeConnection: EdgeConnection;\n private _remotePeerId: string | null = null;\n private readonly _targetServiceId: string;\n private readonly _spaceId: SpaceId;\n private readonly _context: EchoReplicatorContext;\n private readonly _sharedPolicyEnabled: boolean;\n private readonly _onRemoteConnected: () => Promise<void>;\n private readonly _onRemoteDisconnected: () => Promise<void>;\n private readonly _onRestartRequested: () => void;\n\n private _readableStreamController!: ReadableStreamDefaultController<AutomergeProtocolMessage>;\n\n public readable: ReadableStream<AutomergeProtocolMessage>;\n public writable: WritableStream<AutomergeProtocolMessage>;\n\n constructor({\n edgeConnection,\n spaceId,\n context,\n sharedPolicyEnabled,\n onRemoteConnected,\n onRemoteDisconnected,\n onRestartRequested,\n }: EdgeReplicatorConnectionsParams) {\n super();\n this._edgeConnection = edgeConnection;\n this._spaceId = spaceId;\n this._context = context;\n\n // Generate a unique peer id for every connection.\n // This way automerge-repo will have separate sync states for every connection.\n // This is important because the previous connection might have had some messages that failed to deliver\n // abd if we don't clear the sync-state, automerge will not attempt to deliver them again.\n this._remotePeerId = `${EdgeService.AUTOMERGE_REPLICATOR}:${spaceId}-${randomUUID()}`;\n this._targetServiceId = `${EdgeService.AUTOMERGE_REPLICATOR}:${spaceId}`;\n this._sharedPolicyEnabled = sharedPolicyEnabled;\n this._onRemoteConnected = onRemoteConnected;\n this._onRemoteDisconnected = onRemoteDisconnected;\n this._onRestartRequested = onRestartRequested;\n\n this.readable = new ReadableStream<AutomergeProtocolMessage>({\n start: (controller) => {\n this._readableStreamController = controller;\n },\n });\n\n this.writable = new WritableStream<AutomergeProtocolMessage>({\n write: async (message: AutomergeProtocolMessage, controller) => {\n await this._sendMessage(message);\n },\n });\n }\n\n protected override async _open(ctx: Context): Promise<void> {\n log('open');\n // TODO: handle reconnects\n this._ctx.onDispose(\n this._edgeConnection.onMessage((msg: RouterMessage) => {\n this._onMessage(msg);\n }),\n );\n\n await this._onRemoteConnected();\n }\n\n protected override async _close(): Promise<void> {\n log('close');\n this._readableStreamController.close();\n await this._onRemoteDisconnected();\n }\n\n get peerId(): string {\n invariant(this._remotePeerId, 'Not connected');\n return this._remotePeerId;\n }\n\n async shouldAdvertise(params: ShouldAdvertiseParams): Promise<boolean> {\n if (!this._sharedPolicyEnabled) {\n return true;\n }\n const spaceId = await this._context.getContainingSpaceIdForDocument(params.documentId);\n if (!spaceId) {\n // There's no spaceId if the document is not present locally. This means the sharePolicy check is being\n // performed on message reception, so spaceId check was already performed in _onMessage.\n return true;\n }\n return spaceId === this._spaceId;\n }\n\n shouldSyncCollection(params: ShouldSyncCollectionParams): boolean {\n if (!this._sharedPolicyEnabled) {\n return true;\n }\n const spaceId = getSpaceIdFromCollectionId(params.collectionId as CollectionId);\n return spaceId === this._spaceId;\n }\n\n private _onMessage(message: RouterMessage) {\n if (message.serviceId !== this._targetServiceId) {\n return;\n }\n\n const payload = cbor.decode(message.payload!.value) as AutomergeProtocolMessage;\n log('recv', () => {\n const decodedData =\n payload.type === 'sync' && payload.data\n ? A.decodeSyncMessage(payload.data)\n : payload.type === 'collection-state'\n ? (payload as any).state\n : payload;\n return { from: message.serviceId, type: payload.type, decodedData };\n });\n // Fix the peer id.\n payload.senderId = this._remotePeerId! as PeerId;\n this._processMessage(payload);\n }\n\n private _processMessage(message: AutomergeProtocolMessage) {\n // There's a race between the credentials being replicated that are needed for access control and the data replication.\n // AutomergeReplicator might return a Forbidden error if the credentials are not yet replicated.\n // We restart the connection with some delay to account for that.\n if (isForbiddenErrorMessage(message)) {\n this._onRestartRequested();\n return;\n }\n\n this._readableStreamController.enqueue(message);\n }\n\n private async _sendMessage(message: AutomergeProtocolMessage) {\n // Fix the peer id.\n (message as any).targetId = this._targetServiceId as PeerId;\n\n log('send', {\n type: message.type,\n senderId: message.senderId,\n targetId: (message as any).targetId,\n documentId: (message as any).documentId,\n });\n const encoded = cbor.encode(message);\n\n await this._edgeConnection.send(\n buf.create(RouterMessageSchema, {\n serviceId: this._targetServiceId,\n source: {\n identityKey: this._edgeConnection.identityKey,\n peerKey: this._edgeConnection.peerKey,\n },\n payload: { value: bufferToArray(encoded) },\n }),\n );\n }\n}\n\n/**\n * This message is sent by EDGE AutomergeReplicator when the authorization is denied.\n */\nconst isForbiddenErrorMessage = (message: AutomergeProtocolMessage) =>\n message.type === 'error' && message.message === 'Forbidden';\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport { decodeReference, type ObjectStructure, type SpaceDoc } from '@dxos/echo-protocol';\n\n/**\n * Assumes properties are at root.\n */\nexport const findInlineObjectOfType = (spaceDoc: SpaceDoc, typename: string): [string, ObjectStructure] | undefined => {\n for (const id in spaceDoc.objects ?? {}) {\n const obj = spaceDoc.objects![id];\n if (obj.system.type && decodeReference(obj.system.type).objectId === typename) {\n return [id, obj];\n }\n }\n\n return undefined;\n};\n"],
5
+ "mappings": "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",
6
+ "names": ["UpdateScheduler", "Stream", "invariant", "SpaceId", "log", "UpdateScheduler", "next", "A", "Resource", "invariant", "log", "MAX_UPDATE_FREQ", "DocumentsSynchronizer", "constructor", "_params", "_syncStates", "Map", "_pendingUpdates", "Set", "_sendUpdatesJob", "undefined", "addDocuments", "documentIds", "retryCounter", "warn", "documentId", "doc", "repo", "find", "whenReady", "then", "_startSync", "add", "trigger", "catch", "error", "removeDocuments", "get", "clearSubscriptions", "delete", "_open", "_ctx", "_checkAndSendUpdates", "bind", "maxFrequency", "_close", "join", "clear", "update", "updates", "mutation", "isNew", "loadIncremental", "_writeMutation", "has", "info", "syncState", "handle", "_subscribeForChanges", "set", "handler", "on", "off", "docsWithPendingUpdates", "Array", "from", "_getPendingChanges", "push", "length", "sendUpdates", "docSync", "lastSentHead", "saveSince", "save", "getHeads", "headsBefore", "newDoc", "equals", "Event", "asyncTimeout", "getBackend", "getHeads", "isAutomerge", "equals", "headsEquals", "save", "Repo", "Context", "Resource", "cancelWithContext", "invariant", "PublicKey", "log", "objectPointerCodec", "trace", "asyncReturn", "Event", "scheduleTask", "scheduleTaskInterval", "next", "am", "Resource", "log", "defaultMap", "MIN_QUERY_INTERVAL", "POLL_INTERVAL", "CollectionSynchronizer", "constructor", "params", "_perCollectionStates", "Map", "_connectedPeers", "Set", "remoteStateUpdated", "_sendCollectionState", "sendCollectionState", "_queryCollectionState", "queryCollectionState", "_shouldSyncCollection", "shouldSyncCollection", "_open", "ctx", "_ctx", "collectionId", "keys", "refreshCollection", "getRegisteredCollectionIds", "getLocalCollectionState", "_getPerCollectionState", "localState", "setLocalCollectionState", "state", "queueMicrotask", "disposed", "_refreshInterestedPeers", "getRemoteCollectionStates", "remoteStates", "scheduleAnotherRefresh", "peerId", "interestedPeers", "has", "lastQueried", "get", "Date", "now", "set", "onConnectionOpen", "add", "entries", "onConnectionClosed", "delete", "perCollectionState", "values", "onCollectionStateQueried", "onRemoteStateReceived", "validateCollectionState", "emit", "undefined", "diffCollectionState", "local", "remote", "allDocuments", "Object", "documents", "missingOnRemote", "missingOnLocal", "different", "documentId", "push", "equals", "forEach", "heads", "isValidDocumentId", "Error", "Array", "isArray", "some", "head", "includes", "synchronized", "Trigger", "NetworkAdapter", "LifecycleState", "invariant", "log", "nonNullable", "MESSAGE_TYPE_COLLECTION_QUERY", "MESSAGE_TYPE_COLLECTION_STATE", "isCollectionQueryMessage", "message", "type", "MESSAGE_TYPE_COLLECTION_QUERY", "isCollectionStateMessage", "MESSAGE_TYPE_COLLECTION_STATE", "EchoNetworkAdapter", "NetworkAdapter", "constructor", "_params", "_replicators", "Set", "_connections", "Map", "_lifecycleState", "LifecycleState", "CLOSED", "_connected", "Trigger", "connect", "peerId", "peerMetadata", "wake", "send", "message", "_send", "disconnect", "open", "OPEN", "log", "emit", "network", "close", "replicator", "clear", "whenConnected", "wait", "timeout", "addReplicator", "invariant", "has", "add", "onConnectionOpen", "_onConnectionOpen", "bind", "onConnectionClosed", "_onConnectionClosed", "onConnectionAuthScopeChanged", "_onConnectionAuthScopeChanged", "isDocumentInRemoteCollection", "getContainingSpaceForDocument", "getContainingSpaceIdForDocument", "documentId", "key", "createIdFromSpaceKey", "removeReplicator", "delete", "shouldAdvertise", "params", "connection", "get", "shouldSyncCollection", "queryCollectionState", "collectionId", "targetId", "type", "senderId", "sendCollectionState", "state", "connectionEntry", "Error", "writeStart", "Date", "now", "writer", "write", "then", "durationMs", "monitor", "recordMessageSent", "catch", "err", "isOpen", "recordMessageSendingFailed", "getPeersInterestedInCollection", "Array", "from", "values", "map", "filter", "nonNullable", "reader", "readable", "getReader", "writable", "getWriter", "set", "queueMicrotask", "done", "value", "read", "_onMessage", "_emitPeerCandidate", "recordPeerConnected", "isCollectionQueryMessage", "onCollectionStateQueried", "isCollectionStateMessage", "onCollectionStateReceived", "recordMessageReceived", "entry", "recordPeerDisconnected", "cancel", "abort", "createEchoPeerMetadata", "synchronized", "dxos_peerSource", "isEchoPeerMetadata", "metadata", "headsEncoding", "HeadsStore", "constructor", "db", "_db", "setHeads", "documentId", "heads", "batch", "put", "sublevel", "keyEncoding", "valueEncoding", "headsEncoding", "getHeads", "documentIds", "getMany", "LifecycleState", "Resource", "LevelDBStorageAdapter", "Resource", "constructor", "_params", "load", "keyArray", "_lifecycleState", "LifecycleState", "OPEN", "undefined", "startMs", "Date", "now", "chunk", "db", "get", "encodingOptions", "monitor", "recordBytesLoaded", "byteLength", "recordLoadDuration", "err", "isLevelDbNotFoundError", "save", "binary", "batch", "callbacks", "beforeSave", "path", "put", "Buffer", "from", "write", "recordBytesStored", "afterSave", "recordStoreDuration", "remove", "del", "loadRange", "keyPrefix", "result", "key", "value", "iterator", "gte", "lte", "push", "data", "removeRange", "keyEncoder", "encode", "map", "k", "replaceAll", "join", "decode", "toString", "split", "format", "keyEncoding", "valueEncoding", "code", "AutomergeHost", "Resource", "constructor", "db", "indexMetadataStore", "dataMonitor", "peerIdProvider", "_collectionSynchronizer", "CollectionSynchronizer", "queryCollectionState", "_queryCollectionState", "bind", "sendCollectionState", "_sendCollectionState", "shouldSyncCollection", "_shouldSyncCollection", "collectionStateUpdated", "Event", "_db", "_storage", "LevelDBStorageAdapter", "sublevel", "callbacks", "beforeSave", "params", "_beforeSave", "afterSave", "key", "_afterSave", "monitor", "_echoNetworkAdapter", "EchoNetworkAdapter", "getContainingSpaceForDocument", "_getContainingSpaceForDocument", "isDocumentInRemoteCollection", "_isDocumentInRemoteCollection", "onCollectionStateQueried", "_onCollectionStateQueried", "onCollectionStateReceived", "_onCollectionStateReceived", "_headsStore", "HeadsStore", "_indexMetadataStore", "_peerIdProvider", "_open", "_peerId", "PublicKey", "random", "toHex", "open", "_repo", "Repo", "peerId", "sharePolicy", "_sharePolicy", "storage", "network", "wrap", "on", "_ctx", "e", "_onPeerConnected", "_onPeerDisconnected", "remoteStateUpdated", "collectionId", "_onRemoteCollectionStateUpdated", "emit", "whenConnected", "_close", "close", "dispose", "repo", "loadedDocsCount", "Object", "keys", "handles", "length", "addReplicator", "replicator", "removeReplicator", "loadDoc", "ctx", "documentId", "opts", "handle", "find", "isReady", "timeout", "cancelWithContext", "whenReady", "asyncTimeout", "createDoc", "initialValue", "preserveHistory", "isAutomerge", "TypeError", "import", "save", "create", "waitUntilHeadsReplicated", "heads", "entries", "documentIds", "map", "entry", "documentHeads", "getHeads", "headsToWait", "filter", "index", "targetHeads", "currentHeads", "headsEquals", "Promise", "all", "Context", "default", "waitForHeads", "flush", "reIndexHeads", "log", "info", "inState", "warn", "doc", "docSync", "invariant", "batch", "setHeads", "write", "startsWith", "peerMetadata", "peerMetadataByPeerId", "isEchoPeerMetadata", "shouldAdvertise", "path", "spaceKey", "getSpaceKeyFromDoc", "undefined", "objectIds", "objects", "encodedIds", "objectId", "objectPointerCodec", "encode", "idToLastHash", "Map", "id", "markDirty", "notifyMarkedDirty", "document", "_onHeadsChanged", "_automergePeers", "peers", "getRegisteredCollectionIds", "remoteCollections", "getRemoteCollectionStates", "remotePeerDocs", "get", "documents", "spaceKeyHex", "from", "loadedDocuments", "result", "storeRequestIds", "storeResultIndices", "push", "storedHeads", "i", "getLocalCollectionState", "refreshCollection", "getCollectionSyncState", "localState", "remoteState", "state", "diff", "diffCollectionState", "missingOnRemote", "missingOnLocal", "differentDocuments", "different", "localDocumentCount", "remoteDocumentCount", "updateLocalCollectionState", "fromEntries", "setLocalCollectionState", "onRemoteStateReceived", "decodeCollectionState", "encodeCollectionState", "onConnectionOpen", "onConnectionClosed", "toReplicate", "count", "collectionsChanged", "Set", "newState", "structuredClone", "add", "trace", "depth", "span", "showInBrowserTimeline", "resource", "rawSpaceKey", "access", "experimental_spaceKey", "String", "unavailableHeads", "waitForCondition", "changeHash", "values", "changeIsPresentInDoc", "delete", "size", "getBackend", "getChangeByHash", "invariant", "PublicKey", "log", "ComplexSet", "defaultMap", "A", "cbor", "Resource", "invariant", "log", "AutomergeReplicator", "DEFAULT_FACTORY", "params", "MeshReplicatorConnection", "constructor", "_params", "remoteDeviceKey", "_remotePeerId", "_isEnabled", "readableStreamController", "readable", "ReadableStream", "start", "controller", "_ctx", "onDispose", "close", "writable", "WritableStream", "write", "message", "logSendSync", "replicatorExtension", "sendSyncMessage", "payload", "encode", "err", "error", "_disconnectIfEnabled", "createAutomergeReplicator", "replicatorFactory", "peerId", "ownPeerId", "onStartReplication", "info", "remotePeerId", "id", "thisPeerId", "toHex", "onRemoteConnected", "onSyncMessage", "decode", "enqueue", "onClose", "onRemoteDisconnected", "shouldAdvertise", "shouldSyncCollection", "enable", "disable", "decodedSyncMessage", "type", "data", "decodeSyncMessage", "undefined", "sync", "headsLength", "heads", "length", "requesting", "need", "sendingChanges", "changes", "from", "senderId", "to", "targetId", "invariant", "SpaceId", "deriveCollectionIdFromSpaceId", "spaceId", "getSpaceIdFromCollectionId", "collectionId", "replace", "isValid", "MeshEchoReplicator", "_connections", "Set", "_connectionsPerPeer", "Map", "_authorizedDevices", "_context", "connect", "context", "disconnect", "connection", "values", "onConnectionClosed", "close", "clear", "createExtension", "extensionFactory", "invariant", "MeshReplicatorConnection", "ownPeerId", "peerId", "replicatorFactory", "onRemoteConnected", "log", "has", "onConnectionAuthScopeChanged", "set", "onConnectionOpen", "enable", "onRemoteDisconnected", "delete", "disable", "shouldAdvertise", "params", "documentId", "spaceKey", "getContainingSpaceForDocument", "remoteDocumentExists", "isDocumentInRemoteCollection", "spaceId", "createIdFromSpaceKey", "authorizedDevices", "get", "remoteDeviceKey", "isAuthorized", "localPeer", "remotePeer", "deviceKey", "err", "catch", "shouldSyncCollection", "collectionId", "getSpaceIdFromCollectionId", "add", "replicatorExtension", "authorizeDevice", "defaultMap", "ComplexSet", "PublicKey", "hash", "equals", "trace", "CircularBuffer", "mapValues", "SlidingWindowSummary", "PER_SECOND_RATE_AVG_WINDOW_SIZE", "DEFAULT_AVG_WINDOW_SIZE", "EchoDataMonitor", "constructor", "_params", "timeSeriesLength", "_lastTick", "_activeCounters", "createLocalCounters", "_localTimeSeries", "createLocalTimeSeries", "_storageAverages", "createStorageAverages", "_replicationAverages", "createNetworkAverages", "_sizeByMessageType", "_lastReceivedMessages", "CircularBuffer", "_lastSentMessages", "_connectionsCount", "tick", "timeMs", "_advanceTimeWindow", "computeStats", "meta", "rateAverageOverSeconds", "storage", "reads", "payloadSize", "loadedChunkSize", "average", "opDuration", "loadDuration", "countPerSecond", "loadsPerSecond", "writes", "storedChunkSize", "storeDuration", "storesPerSecond", "replicator", "connections", "receivedMessages", "receivedMessageSize", "receivedPerSecond", "sentMessages", "sentMessageSize", "sendDuration", "sentPerSecond", "failedPerSecond", "sendsFailedPerSecond", "countByMessageType", "_computeMessageHistogram", "avgSizeByMessageType", "mapValues", "summary", "connectionsCount", "lastPerSecondStats", "_lastCompleteCounters", "timeSeries", "replication", "messagesByPeerId", "millisPassed", "oldMetrics", "Object", "freeze", "peerId", "keys", "byPeerId", "createMessageCounter", "_addToTimeSeries", "Math", "abs", "_reportPerSecondRate", "values", "key", "value", "entries", "push", "length", "shift", "metrics", "toReport", "loadedChunks", "storedChunks", "received", "sent", "metricName", "metric", "record", "trace", "distribution", "increment", "tags", "status", "failed", "recordPeerConnected", "recordPeerDisconnected", "recordBytesStored", "count", "storedBytes", "unit", "recordLoadDuration", "durationMs", "recordStoreDuration", "recordBytesLoaded", "loadedBytes", "recordMessageSent", "message", "duration", "metricsGroupName", "bytes", "getByteCount", "type", "isAutomergeProtocolMessage", "success", "messageSize", "messageCounts", "_getStatsForType", "targetId", "recordMessageReceived", "senderId", "recordMessageSendingFailed", "createSlidingWindow", "byType", "groupKey", "result", "receivedMessage", "counters", "resource", "isCollectionQueryMessage", "isCollectionStateMessage", "overrides", "SlidingWindowSummary", "dataPoints", "precision", "data", "byteLength", "documentId", "DataServiceImpl", "constructor", "params", "_subscriptions", "Map", "_automergeHost", "automergeHost", "_updateIndexes", "updateIndexes", "subscribe", "request", "Stream", "next", "ready", "synchronizer", "DocumentsSynchronizer", "repo", "sendUpdates", "updates", "open", "then", "set", "subscriptionId", "catch", "err", "log", "close", "updateSubscription", "get", "invariant", "addIds", "length", "addDocuments", "removeIds", "removeDocuments", "update", "flush", "getDocumentHeads", "documentIds", "heads", "entries", "getHeads", "map", "idx", "documentId", "waitUntilHeadsReplicated", "options", "reIndexHeads", "subscribeSpaceSyncState", "ctx", "SpaceId", "isValid", "spaceId", "collectionId", "deriveCollectionIdFromSpaceId", "scheduler", "UpdateScheduler", "state", "getCollectionSyncState", "peers", "peer", "peerId", "missingOnRemote", "missingOnLocal", "differentDocuments", "localDocumentCount", "remoteDocumentCount", "collectionStateUpdated", "on", "e", "trigger", "LifecycleState", "Resource", "todo", "createIdFromSpaceKey", "SpaceDocVersion", "IndexMetadataStore", "IndexStore", "Indexer", "invariant", "IndexKind", "trace", "A", "Context", "SpaceDocVersion", "invariant", "log", "ObjectPointerVersion", "objectPointerCodec", "LOG_VIEW_OPERATION_THRESHOLD", "createSelectedDocumentsIterator", "automergeHost", "loadDocuments", "objects", "id", "heads", "entries", "documentId", "objectId", "objectPointerCodec", "decode", "handle", "loadDoc", "Context", "default", "doc", "docSync", "invariant", "currentHeads", "getHeads", "equals", "begin", "Date", "now", "view", "end", "log", "info", "duration", "requestedHeads", "originalHeads", "version", "SpaceDocVersion", "CURRENT", "newId", "getVersion", "ObjectPointerVersion", "V0", "spaceKey", "getSpaceKeyFromDoc", "undefined", "encode", "object", "error", "DeferredTask", "getHeads", "Stream", "Context", "Resource", "log", "objectPointerCodec", "trace", "Context", "LifecycleState", "Resource", "createIdFromSpaceKey", "invariant", "DXN", "PublicKey", "objectPointerCodec", "trace", "nonNullable", "QueryState", "Resource", "active", "_lifecycleState", "LifecycleState", "OPEN", "constructor", "_params", "_results", "_firstRun", "metrics", "objectsReturned", "objectsReturnedFromIndex", "documentsLoaded", "executionTime", "indexQueryTime", "documentLoadTime", "filter", "request", "getResults", "execQuery", "beginQuery", "performance", "now", "hits", "objectIds", "length", "indexer", "filterToIndexQuery", "beginFilter", "results", "Promise", "all", "map", "result", "objectId", "documentId", "spaceKey", "spaceKeyInIndex", "objectPointerCodec", "decode", "id", "undefined", "handle", "automergeHost", "loadDoc", "Context", "default", "_ctx", "disposed", "getSpaceKeyFromDoc", "docSync", "options", "spaces", "some", "key", "equals", "spaceId", "createIdFromSpaceKey", "PublicKey", "from", "rank", "nonNullable", "changed", "areResultsUnchanged", "every", "oldResult", "trace", "info", "depth", "span", "showInBrowserTimeline", "op", "attributes", "resource", "invariant", "type", "or", "subFilter", "not", "typenames", "dxnToIndexerTypename", "DXN", "parse", "flatMap", "f", "inverted", "dxn", "kind", "TYPE", "parts", "ECHO", "Error", "QueryServiceImpl", "Resource", "constructor", "_params", "_queries", "Set", "_updateQueries", "DeferredTask", "_ctx", "Promise", "all", "Array", "from", "map", "query", "changed", "state", "execQuery", "sendResults", "getResults", "err", "log", "catch", "trace", "diagnostic", "id", "name", "fetch", "filter", "JSON", "stringify", "metrics", "_open", "indexer", "updated", "on", "schedule", "_close", "close", "setConfig", "config", "initialized", "warn", "request", "Stream", "next", "ctx", "QueryState", "automergeHost", "results", "disposed", "queryId", "delete", "add", "queueMicrotask", "open", "error", "reindex", "info", "iterator", "createDocumentsIterator", "ids", "Map", "documents", "heads", "set", "size", "count", "getAllDocuments", "visited", "getObjectsFromHandle", "handle", "has", "documentId", "doc", "docSync", "spaceKey", "getSpaceKeyFromDoc", "undefined", "objects", "Object", "entries", "objectId", "object", "objectPointerCodec", "encode", "getHeads", "links", "values", "urlString", "toString", "linkHandle", "loadDoc", "Context", "default", "result", "repo", "handles", "isEqual", "Event", "UpdateScheduler", "interpretAsDocumentId", "Resource", "Context", "SpaceDocVersion", "invariant", "A", "log", "measureDocMetrics", "doc", "snapshot", "save", "start", "Date", "now", "temp", "load", "end", "free", "getAllChangesStart", "mutationCount", "getAllChanges", "length", "getAllChangesEnd", "warn", "elapsed", "compressedByteSize", "byteLength", "loadTime", "DatabaseRoot", "constructor", "_rootHandle", "documentId", "url", "isLoaded", "docSync", "handle", "getVersion", "doc", "version", "SpaceDocVersion", "LEGACY", "getSpaceKey", "getSpaceKeyFromDoc", "getInlineObjectCount", "Object", "keys", "objects", "length", "getLinkedObjectCount", "links", "getAllLinkedDocuments", "invariant", "values", "map", "s", "toString", "measureMetrics", "measureDocMetrics", "SpaceStateManager", "Resource", "_roots", "Map", "_rootBySpace", "_perRootContext", "_lastSpaceDocumentList", "spaceDocumentListUpdated", "Event", "_close", "ctx", "_", "rootCtx", "dispose", "clear", "roots", "getRootByDocumentId", "documentId", "get", "assignRootToSpace", "spaceId", "handle", "root", "has", "DatabaseRoot", "set", "prevRootId", "delete", "Context", "whenReady", "documentListCheckScheduler", "UpdateScheduler", "documentIds", "getAllLinkedDocuments", "map", "url", "interpretAsDocumentId", "isEqual", "emit", "SpaceDocumentListUpdatedEvent", "maxFrequency", "triggerCheckOnChange", "trigger", "addListener", "onDispose", "removeListener", "constructor", "INDEXER_CONFIG", "enabled", "indexes", "kind", "IndexKind", "Kind", "SCHEMA_MATCH", "EchoHost", "Resource", "constructor", "kv", "peerIdProvider", "_spaceStateManager", "SpaceStateManager", "_indexMetadataStore", "IndexMetadataStore", "db", "sublevel", "_echoDataMonitor", "EchoDataMonitor", "_automergeHost", "AutomergeHost", "dataMonitor", "indexMetadataStore", "_indexer", "Indexer", "indexStore", "IndexStore", "metadataStore", "loadDocuments", "createSelectedDocumentsIterator", "indexCooldownTime", "process", "undefined", "setConfig", "_queryService", "QueryServiceImpl", "automergeHost", "indexer", "_dataService", "DataServiceImpl", "updateIndexes", "trace", "diagnostic", "id", "name", "fetch", "dataStats", "computeStats", "loadedDocsCount", "Array", "from", "roots", "values", "map", "root", "url", "isLoaded", "spaceKey", "getSpaceKey", "inlineObjects", "getInlineObjectCount", "linkedObjects", "getLinkedObjectCount", "measureMetrics", "queryService", "dataService", "automergeRepo", "repo", "_open", "ctx", "open", "spaceDocumentListUpdated", "on", "_ctx", "e", "updateLocalCollectionState", "deriveCollectionIdFromSpaceId", "spaceId", "documentIds", "_close", "close", "flush", "loadDoc", "documentId", "opts", "createDoc", "initialValue", "createSpaceRoot", "invariant", "_lifecycleState", "LifecycleState", "OPEN", "createIdFromSpaceKey", "automergeRoot", "version", "SpaceDocVersion", "CURRENT", "access", "toHex", "objects", "links", "openSpaceRoot", "automergeUrl", "handle", "find", "assignRootToSpace", "closeSpaceRoot", "todo", "addReplicator", "replicator", "removeReplicator", "getSpaceSyncState", "collectionId", "getCollectionSyncState", "Mutex", "scheduleTask", "scheduleMicroTask", "A", "cbor", "Context", "Resource", "randomUUID", "invariant", "log", "EdgeService", "buf", "MessageSchema", "RouterMessageSchema", "bufferToArray", "INITIAL_RESTART_DELAY", "RESTART_DELAY_JITTER", "MAX_RESTART_DELAY", "EchoEdgeReplicator", "constructor", "edgeConnection", "disableSharePolicy", "_mutex", "Mutex", "_ctx", "undefined", "_context", "_connectedSpaces", "Set", "_connections", "Map", "_sharePolicyEnabled", "_edgeConnection", "connect", "context", "log", "info", "peerId", "connectedSpaces", "size", "Context", "default", "onDispose", "onReconnected", "scheduleMicroTask", "_handleReconnect", "_guard", "acquire", "spaces", "connection", "values", "close", "clear", "spaceId", "_openConnection", "disconnect", "dispose", "connectToSpace", "has", "add", "disconnectFromSpace", "delete", "get", "reconnects", "invariant", "restartScheduled", "EdgeReplicatorConnection", "sharedPolicyEnabled", "onRemoteConnected", "onConnectionOpen", "onRemoteDisconnected", "onConnectionClosed", "onRestartRequested", "restartDelay", "Math", "min", "random", "scheduleTask", "ctx", "disposed", "set", "open", "Resource", "_remotePeerId", "_spaceId", "EdgeService", "AUTOMERGE_REPLICATOR", "randomUUID", "_targetServiceId", "_sharedPolicyEnabled", "_onRemoteConnected", "_onRemoteDisconnected", "_onRestartRequested", "readable", "ReadableStream", "start", "controller", "_readableStreamController", "writable", "WritableStream", "write", "message", "_sendMessage", "_open", "onMessage", "msg", "_onMessage", "_close", "shouldAdvertise", "params", "getContainingSpaceIdForDocument", "documentId", "shouldSyncCollection", "getSpaceIdFromCollectionId", "collectionId", "serviceId", "payload", "cbor", "decode", "value", "decodedData", "type", "data", "decodeSyncMessage", "state", "from", "senderId", "_processMessage", "isForbiddenErrorMessage", "enqueue", "targetId", "encoded", "encode", "send", "buf", "create", "RouterMessageSchema", "source", "identityKey", "peerKey", "bufferToArray", "decodeReference", "findInlineObjectOfType", "spaceDoc", "typename", "id", "objects", "obj", "system", "type", "decodeReference", "objectId", "undefined"]
7
7
  }