@dxos/echo-pipeline 0.4.10-main.c32f430 → 0.4.10-main.c42bfdb
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/lib/browser/{chunk-RTEEJ723.mjs → chunk-RA6MLCZM.mjs} +14 -27
- package/dist/lib/browser/chunk-RA6MLCZM.mjs.map +7 -0
- package/dist/lib/browser/index.mjs +50 -26
- package/dist/lib/browser/index.mjs.map +4 -4
- package/dist/lib/browser/meta.json +1 -1
- package/dist/lib/browser/testing/index.mjs +1 -1
- package/dist/lib/node/{chunk-7VZVCCNF.cjs → chunk-KGIYLJBT.cjs} +20 -33
- package/dist/lib/node/chunk-KGIYLJBT.cjs.map +7 -0
- package/dist/lib/node/index.cjs +72 -46
- package/dist/lib/node/index.cjs.map +4 -4
- package/dist/lib/node/meta.json +1 -1
- package/dist/lib/node/testing/index.cjs +11 -11
- package/dist/types/src/automerge/automerge-doc-loader.d.ts +2 -0
- package/dist/types/src/automerge/automerge-doc-loader.d.ts.map +1 -1
- package/dist/types/src/automerge/automerge-host.d.ts +2 -1
- package/dist/types/src/automerge/automerge-host.d.ts.map +1 -1
- package/dist/types/src/automerge/index.d.ts +1 -0
- package/dist/types/src/automerge/index.d.ts.map +1 -1
- package/dist/types/src/automerge/reference.d.ts +15 -0
- package/dist/types/src/automerge/reference.d.ts.map +1 -0
- package/dist/types/src/automerge/types.d.ts +2 -2
- package/dist/types/src/automerge/types.d.ts.map +1 -1
- package/dist/types/src/space/space.d.ts +4 -8
- package/dist/types/src/space/space.d.ts.map +1 -1
- package/dist/types/src/testing/test-agent-builder.d.ts +2 -2
- package/package.json +30 -30
- package/src/automerge/automerge-doc-loader.ts +6 -0
- package/src/automerge/automerge-host.ts +15 -6
- package/src/automerge/index.ts +1 -0
- package/src/automerge/reference.ts +31 -0
- package/src/automerge/types.ts +2 -2
- package/src/db-host/data-service.ts +1 -1
- package/src/space/space.test.ts +7 -7
- package/src/space/space.ts +6 -21
- package/dist/lib/browser/chunk-RTEEJ723.mjs.map +0 -7
- package/dist/lib/node/chunk-7VZVCCNF.cjs.map +0 -7
|
@@ -20,7 +20,7 @@ import {
|
|
|
20
20
|
mapTimeframeToFeedIndexes,
|
|
21
21
|
startAfter,
|
|
22
22
|
valueEncoding
|
|
23
|
-
} from "./chunk-
|
|
23
|
+
} from "./chunk-RA6MLCZM.mjs";
|
|
24
24
|
|
|
25
25
|
// packages/core/echo/echo-pipeline/src/automerge/automerge-host.ts
|
|
26
26
|
import { next as automerge, getHeads } from "@dxos/automerge/automerge";
|
|
@@ -431,7 +431,7 @@ var AutomergeHost = class {
|
|
|
431
431
|
isRequested
|
|
432
432
|
}, {
|
|
433
433
|
F: __dxlog_file3,
|
|
434
|
-
L:
|
|
434
|
+
L: 101,
|
|
435
435
|
S: this,
|
|
436
436
|
C: (f, a) => f(...a)
|
|
437
437
|
});
|
|
@@ -445,7 +445,7 @@ var AutomergeHost = class {
|
|
|
445
445
|
documentId
|
|
446
446
|
}, {
|
|
447
447
|
F: __dxlog_file3,
|
|
448
|
-
L:
|
|
448
|
+
L: 108,
|
|
449
449
|
S: this,
|
|
450
450
|
C: (f, a) => f(...a)
|
|
451
451
|
});
|
|
@@ -459,7 +459,7 @@ var AutomergeHost = class {
|
|
|
459
459
|
documentId
|
|
460
460
|
}, {
|
|
461
461
|
F: __dxlog_file3,
|
|
462
|
-
L:
|
|
462
|
+
L: 117,
|
|
463
463
|
S: this,
|
|
464
464
|
C: (f, a) => f(...a)
|
|
465
465
|
});
|
|
@@ -476,7 +476,7 @@ var AutomergeHost = class {
|
|
|
476
476
|
isAuthorized
|
|
477
477
|
}, {
|
|
478
478
|
F: __dxlog_file3,
|
|
479
|
-
L:
|
|
479
|
+
L: 123,
|
|
480
480
|
S: this,
|
|
481
481
|
C: (f, a) => f(...a)
|
|
482
482
|
});
|
|
@@ -484,7 +484,7 @@ var AutomergeHost = class {
|
|
|
484
484
|
} catch (err) {
|
|
485
485
|
log3.catch(err, void 0, {
|
|
486
486
|
F: __dxlog_file3,
|
|
487
|
-
L:
|
|
487
|
+
L: 133,
|
|
488
488
|
S: this,
|
|
489
489
|
C: (f, a) => f(...a)
|
|
490
490
|
});
|
|
@@ -499,6 +499,7 @@ var AutomergeHost = class {
|
|
|
499
499
|
this._repo.on("document", listener);
|
|
500
500
|
this._ctx.onDispose(() => {
|
|
501
501
|
this._repo.off("document", listener);
|
|
502
|
+
Object.values(this._repo.handles).forEach((handle) => handle.off("change"));
|
|
502
503
|
});
|
|
503
504
|
}
|
|
504
505
|
}
|
|
@@ -514,9 +515,6 @@ var AutomergeHost = class {
|
|
|
514
515
|
_onDocument(handle) {
|
|
515
516
|
const listener = (event) => this._onUpdate(event);
|
|
516
517
|
handle.on("change", listener);
|
|
517
|
-
this._ctx.onDispose(() => {
|
|
518
|
-
handle.off("change", listener);
|
|
519
|
-
});
|
|
520
518
|
}
|
|
521
519
|
_onUpdate(event) {
|
|
522
520
|
if (this._metadata == null) {
|
|
@@ -544,7 +542,7 @@ var AutomergeHost = class {
|
|
|
544
542
|
}).catch((err) => {
|
|
545
543
|
this._ctx.disposed && log3.catch(err, void 0, {
|
|
546
544
|
F: __dxlog_file3,
|
|
547
|
-
L:
|
|
545
|
+
L: 191,
|
|
548
546
|
S: this,
|
|
549
547
|
C: (f, a) => f(...a)
|
|
550
548
|
});
|
|
@@ -556,7 +554,7 @@ var AutomergeHost = class {
|
|
|
556
554
|
state: handle.state,
|
|
557
555
|
hasDoc: !!handle.docSync(),
|
|
558
556
|
heads: handle.docSync() ? automerge.getHeads(handle.docSync()) : null,
|
|
559
|
-
data: handle.docSync()
|
|
557
|
+
data: handle.docSync() && mapValues(handle.docSync(), (value, key) => {
|
|
560
558
|
try {
|
|
561
559
|
switch (key) {
|
|
562
560
|
case "access":
|
|
@@ -584,6 +582,10 @@ var AutomergeHost = class {
|
|
|
584
582
|
//
|
|
585
583
|
// Methods for client-services.
|
|
586
584
|
//
|
|
585
|
+
async flush({ documentIds }) {
|
|
586
|
+
await Promise.all(documentIds?.map((id) => this._repo.find(id).whenReady()) ?? []);
|
|
587
|
+
await this._repo.flush(documentIds);
|
|
588
|
+
}
|
|
587
589
|
syncRepo(request) {
|
|
588
590
|
return this._clientNetwork.syncRepo(request);
|
|
589
591
|
}
|
|
@@ -605,7 +607,7 @@ var AutomergeHost = class {
|
|
|
605
607
|
deviceKey
|
|
606
608
|
}, {
|
|
607
609
|
F: __dxlog_file3,
|
|
608
|
-
L:
|
|
610
|
+
L: 264,
|
|
609
611
|
S: this,
|
|
610
612
|
C: (f, a) => f(...a)
|
|
611
613
|
});
|
|
@@ -670,6 +672,11 @@ var AutomergeDocumentLoaderImpl = class {
|
|
|
670
672
|
this._objectsPendingDocumentLoad = /* @__PURE__ */ new Set();
|
|
671
673
|
this.onObjectDocumentLoaded = new Event();
|
|
672
674
|
}
|
|
675
|
+
getAllHandles() {
|
|
676
|
+
return [
|
|
677
|
+
...new Set(this._objectDocumentHandles.values())
|
|
678
|
+
];
|
|
679
|
+
}
|
|
673
680
|
async loadSpaceRootDocHandle(ctx, spaceState) {
|
|
674
681
|
if (this._spaceRootDocHandle != null) {
|
|
675
682
|
return;
|
|
@@ -679,7 +686,7 @@ var AutomergeDocumentLoaderImpl = class {
|
|
|
679
686
|
spaceKey: this._spaceKey
|
|
680
687
|
}, {
|
|
681
688
|
F: __dxlog_file4,
|
|
682
|
-
L:
|
|
689
|
+
L: 66,
|
|
683
690
|
S: this,
|
|
684
691
|
C: (f, a) => f(...a)
|
|
685
692
|
});
|
|
@@ -689,7 +696,7 @@ var AutomergeDocumentLoaderImpl = class {
|
|
|
689
696
|
const doc = existingDocHandle.docSync();
|
|
690
697
|
invariant3(doc, void 0, {
|
|
691
698
|
F: __dxlog_file4,
|
|
692
|
-
L:
|
|
699
|
+
L: 71,
|
|
693
700
|
S: this,
|
|
694
701
|
A: [
|
|
695
702
|
"doc",
|
|
@@ -705,7 +712,7 @@ var AutomergeDocumentLoaderImpl = class {
|
|
|
705
712
|
loadObjectDocument(objectId) {
|
|
706
713
|
invariant3(this._spaceRootDocHandle, void 0, {
|
|
707
714
|
F: __dxlog_file4,
|
|
708
|
-
L:
|
|
715
|
+
L: 80,
|
|
709
716
|
S: this,
|
|
710
717
|
A: [
|
|
711
718
|
"this._spaceRootDocHandle",
|
|
@@ -718,7 +725,7 @@ var AutomergeDocumentLoaderImpl = class {
|
|
|
718
725
|
const spaceRootDoc = this._spaceRootDocHandle.docSync();
|
|
719
726
|
invariant3(spaceRootDoc, void 0, {
|
|
720
727
|
F: __dxlog_file4,
|
|
721
|
-
L:
|
|
728
|
+
L: 85,
|
|
722
729
|
S: this,
|
|
723
730
|
A: [
|
|
724
731
|
"spaceRootDoc",
|
|
@@ -732,7 +739,7 @@ var AutomergeDocumentLoaderImpl = class {
|
|
|
732
739
|
objectId
|
|
733
740
|
}, {
|
|
734
741
|
F: __dxlog_file4,
|
|
735
|
-
L:
|
|
742
|
+
L: 89,
|
|
736
743
|
S: this,
|
|
737
744
|
C: (f, a) => f(...a)
|
|
738
745
|
});
|
|
@@ -753,7 +760,7 @@ var AutomergeDocumentLoaderImpl = class {
|
|
|
753
760
|
getSpaceRootDocHandle() {
|
|
754
761
|
invariant3(this._spaceRootDocHandle, void 0, {
|
|
755
762
|
F: __dxlog_file4,
|
|
756
|
-
L:
|
|
763
|
+
L: 107,
|
|
757
764
|
S: this,
|
|
758
765
|
A: [
|
|
759
766
|
"this._spaceRootDocHandle",
|
|
@@ -765,7 +772,7 @@ var AutomergeDocumentLoaderImpl = class {
|
|
|
765
772
|
createDocumentForObject(objectId) {
|
|
766
773
|
invariant3(this._spaceRootDocHandle, void 0, {
|
|
767
774
|
F: __dxlog_file4,
|
|
768
|
-
L:
|
|
775
|
+
L: 112,
|
|
769
776
|
S: this,
|
|
770
777
|
A: [
|
|
771
778
|
"this._spaceRootDocHandle",
|
|
@@ -808,7 +815,7 @@ var AutomergeDocumentLoaderImpl = class {
|
|
|
808
815
|
actualDocumentUrl: objectDocumentHandle.url
|
|
809
816
|
}, {
|
|
810
817
|
F: __dxlog_file4,
|
|
811
|
-
L:
|
|
818
|
+
L: 142,
|
|
812
819
|
S: this,
|
|
813
820
|
C: (f, a) => f(...a)
|
|
814
821
|
});
|
|
@@ -817,7 +824,7 @@ var AutomergeDocumentLoaderImpl = class {
|
|
|
817
824
|
if (objectDocumentHandle?.url === automergeUrl) {
|
|
818
825
|
log4.warn("object document was already loaded", logMeta, {
|
|
819
826
|
F: __dxlog_file4,
|
|
820
|
-
L:
|
|
827
|
+
L: 149,
|
|
821
828
|
S: this,
|
|
822
829
|
C: (f, a) => f(...a)
|
|
823
830
|
});
|
|
@@ -826,7 +833,7 @@ var AutomergeDocumentLoaderImpl = class {
|
|
|
826
833
|
const handle = this._repo.find(automergeUrl);
|
|
827
834
|
log4.debug("document loading triggered", logMeta, {
|
|
828
835
|
F: __dxlog_file4,
|
|
829
|
-
L:
|
|
836
|
+
L: 153,
|
|
830
837
|
S: this,
|
|
831
838
|
C: (f, a) => f(...a)
|
|
832
839
|
});
|
|
@@ -849,7 +856,7 @@ var AutomergeDocumentLoaderImpl = class {
|
|
|
849
856
|
state: docHandle.state
|
|
850
857
|
}, {
|
|
851
858
|
F: __dxlog_file4,
|
|
852
|
-
L:
|
|
859
|
+
L: 169,
|
|
853
860
|
S: this,
|
|
854
861
|
C: (f, a) => f(...a)
|
|
855
862
|
});
|
|
@@ -891,7 +898,7 @@ var AutomergeDocumentLoaderImpl = class {
|
|
|
891
898
|
if (this.onObjectDocumentLoaded.listenerCount() === 0) {
|
|
892
899
|
log4.info("document loaded after all listeners were removed", logMeta, {
|
|
893
900
|
F: __dxlog_file4,
|
|
894
|
-
L:
|
|
901
|
+
L: 205,
|
|
895
902
|
S: this,
|
|
896
903
|
C: (f, a) => f(...a)
|
|
897
904
|
});
|
|
@@ -901,7 +908,7 @@ var AutomergeDocumentLoaderImpl = class {
|
|
|
901
908
|
if (objectDocHandle?.url !== handle.url) {
|
|
902
909
|
log4.warn("object was rebound while a document was loading, discarding handle", logMeta, {
|
|
903
910
|
F: __dxlog_file4,
|
|
904
|
-
L:
|
|
911
|
+
L: 210,
|
|
905
912
|
S: this,
|
|
906
913
|
C: (f, a) => f(...a)
|
|
907
914
|
});
|
|
@@ -920,7 +927,7 @@ var AutomergeDocumentLoaderImpl = class {
|
|
|
920
927
|
err
|
|
921
928
|
}, {
|
|
922
929
|
F: __dxlog_file4,
|
|
923
|
-
L:
|
|
930
|
+
L: 216,
|
|
924
931
|
S: this,
|
|
925
932
|
C: (f, a) => f(...a)
|
|
926
933
|
});
|
|
@@ -930,6 +937,19 @@ var AutomergeDocumentLoaderImpl = class {
|
|
|
930
937
|
}
|
|
931
938
|
}
|
|
932
939
|
};
|
|
940
|
+
|
|
941
|
+
// packages/core/echo/echo-pipeline/src/automerge/reference.ts
|
|
942
|
+
import { Reference } from "@dxos/echo-db";
|
|
943
|
+
var REFERENCE_TYPE_TAG = "dxos.echo.model.document.Reference";
|
|
944
|
+
var encodeReference = (reference) => ({
|
|
945
|
+
"@type": REFERENCE_TYPE_TAG,
|
|
946
|
+
// NOTE: Automerge do not support undefined values, so we need to use null instead.
|
|
947
|
+
itemId: reference.itemId ?? null,
|
|
948
|
+
protocol: reference.protocol ?? null,
|
|
949
|
+
host: reference.host ?? null
|
|
950
|
+
});
|
|
951
|
+
var decodeReference = (value) => new Reference(value.itemId, value.protocol ?? void 0, value.host ?? void 0);
|
|
952
|
+
var isEncodedReferenceObject = (value) => typeof value === "object" && value !== null && value["@type"] === REFERENCE_TYPE_TAG;
|
|
933
953
|
export {
|
|
934
954
|
AuthExtension,
|
|
935
955
|
AuthStatus,
|
|
@@ -943,6 +963,7 @@ export {
|
|
|
943
963
|
MeshNetworkAdapter,
|
|
944
964
|
MetadataStore,
|
|
945
965
|
Pipeline,
|
|
966
|
+
REFERENCE_TYPE_TAG,
|
|
946
967
|
SnapshotManager,
|
|
947
968
|
SnapshotStore,
|
|
948
969
|
Space,
|
|
@@ -952,7 +973,10 @@ export {
|
|
|
952
973
|
TimeframeClock,
|
|
953
974
|
codec,
|
|
954
975
|
createMappedFeedWriter,
|
|
976
|
+
decodeReference,
|
|
977
|
+
encodeReference,
|
|
955
978
|
getSpaceKeyFromDoc,
|
|
979
|
+
isEncodedReferenceObject,
|
|
956
980
|
mapFeedIndexesToTimeframe,
|
|
957
981
|
mapTimeframeToFeedIndexes,
|
|
958
982
|
startAfter,
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
|
-
"sources": ["../../../src/automerge/automerge-host.ts", "../../../src/automerge/automerge-storage-adapter.ts", "../../../src/automerge/automerge-storage\u2013wrapper.ts", "../../../src/automerge/local-host-network-adapter.ts", "../../../src/automerge/mesh-network-adapter.ts", "../../../src/automerge/automerge-doc-loader.ts"],
|
|
4
|
-
"sourcesContent": ["//\n// Copyright 2023 DXOS.org\n//\n\nimport { next as automerge, getHeads } from '@dxos/automerge/automerge';\nimport {\n Repo,\n type PeerId,\n type DocumentId,\n type StorageKey,\n type DocHandle,\n type DocHandleChangePayload,\n} from '@dxos/automerge/automerge-repo';\nimport { IndexedDBStorageAdapter } from '@dxos/automerge/automerge-repo-storage-indexeddb';\nimport { type Stream } from '@dxos/codec-protobuf';\nimport { Context } from '@dxos/context';\nimport { PublicKey } from '@dxos/keys';\nimport { log } from '@dxos/log';\nimport { idCodec } from '@dxos/protocols';\nimport { type HostInfo, type SyncRepoRequest, type SyncRepoResponse } from '@dxos/protocols/proto/dxos/echo/service';\nimport { StorageType, type Directory } from '@dxos/random-access-storage';\nimport { type AutomergeReplicator } from '@dxos/teleport-extension-automerge-replicator';\nimport { trace } from '@dxos/tracing';\nimport { ComplexMap, ComplexSet, defaultMap, mapValues } from '@dxos/util';\n\nimport { AutomergeStorageAdapter } from './automerge-storage-adapter';\nimport { AutomergeStorageWrapper } from './automerge-storage–wrapper';\nimport { LocalHostNetworkAdapter } from './local-host-network-adapter';\nimport { MeshNetworkAdapter } from './mesh-network-adapter';\n\nexport type { DocumentId };\n\nexport interface MetadataMethods {\n markDirty(idToLastHash: Map<string, string>): Promise<void>;\n}\n\nexport type AutomergeHostParams = {\n directory: Directory;\n metadata?: MetadataMethods;\n};\n\n@trace.resource()\nexport class AutomergeHost {\n private readonly _ctx = new Context();\n private readonly _repo: Repo;\n private readonly _meshNetwork: MeshNetworkAdapter;\n private readonly _clientNetwork: LocalHostNetworkAdapter;\n private readonly _storage: AutomergeStorageWrapper;\n\n @trace.info()\n private readonly _peerId: string;\n\n /**\n * spaceKey -> deviceKey[]\n */\n private readonly _authorizedDevices = new ComplexMap<PublicKey, ComplexSet<PublicKey>>(PublicKey.hash);\n\n private readonly _updatingMetadata = new Map<string, Promise<void>>();\n private readonly _metadata?: MetadataMethods;\n\n public _requestedDocs = new Set<string>();\n\n constructor({ directory, metadata }: AutomergeHostParams) {\n this._metadata = metadata;\n this._meshNetwork = new MeshNetworkAdapter();\n this._clientNetwork = new LocalHostNetworkAdapter();\n\n this._storage = new AutomergeStorageWrapper({\n storage:\n // TODO(mykola): Delete specific handling of IDB storage.\n directory.type === StorageType.IDB\n ? new IndexedDBStorageAdapter(directory.path, 'data')\n : new AutomergeStorageAdapter(directory),\n callbacks: { beforeSave: (params) => this._beforeSave(params) },\n });\n this._peerId = `host-${PublicKey.random().toHex()}` as PeerId;\n this._repo = new Repo({\n peerId: this._peerId as PeerId,\n network: [this._clientNetwork, this._meshNetwork],\n storage: this._storage,\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 sharePolicy: async (peerId /* device key */, documentId /* space key */) => {\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 doc = this._repo.handles[documentId]?.docSync();\n if (!doc) {\n const isRequested = this._requestedDocs.has(`automerge:${documentId}`);\n log('doc share policy check', { peerId, documentId, isRequested });\n return isRequested;\n }\n\n try {\n const spaceKey = getSpaceKeyFromDoc(doc);\n if (!spaceKey) {\n log('space key not found for share policy check', { peerId, documentId });\n return false;\n }\n\n const authorizedDevices = this._authorizedDevices.get(PublicKey.from(spaceKey));\n\n // TODO(mykola): Hack, stop abusing `peerMetadata` field.\n const deviceKeyHex = (this.repo.peerMetadataByPeerId[peerId] as any)?.dxos_deviceKey;\n if (!deviceKeyHex) {\n log('device key not found for share policy check', { peerId, documentId });\n return false;\n }\n const deviceKey = PublicKey.from(deviceKeyHex);\n\n const isAuthorized = authorizedDevices?.has(deviceKey) ?? false;\n log('share policy check', {\n localPeer: this._peerId,\n remotePeer: peerId,\n documentId,\n deviceKey,\n spaceKey,\n isAuthorized,\n });\n return isAuthorized;\n } catch (err) {\n log.catch(err);\n return false;\n }\n },\n });\n this._clientNetwork.ready();\n this._meshNetwork.ready();\n\n {\n const listener = ({ handle }: { handle: DocHandle<any> }) => this._onDocument(handle);\n this._repo.on('document', listener);\n this._ctx.onDispose(() => {\n this._repo.off('document', listener);\n });\n }\n }\n\n get repo(): Repo {\n return this._repo;\n }\n\n private async _beforeSave(path: StorageKey) {\n const id = path[0];\n if (this._updatingMetadata.has(id)) {\n return this._updatingMetadata.get(id);\n }\n }\n\n private _onDocument(handle: DocHandle<any>) {\n const listener = (event: DocHandleChangePayload<any>) => this._onUpdate(event);\n handle.on('change', listener);\n this._ctx.onDispose(() => {\n handle.off('change', listener);\n });\n }\n\n private _onUpdate(event: DocHandleChangePayload<any>) {\n if (this._metadata == null) {\n return;\n }\n\n const objectIds = getInlineChanges(event);\n if (objectIds.length === 0) {\n return;\n }\n\n const heads = getHeads(event.doc);\n const lastAvailableHash = heads.join('');\n if (!lastAvailableHash) {\n return;\n }\n\n const encodedIds = objectIds.map((objectId) => idCodec.encode({ documentId: event.handle.documentId, objectId }));\n const idToLastHash = new Map(encodedIds.map((id) => [id, lastAvailableHash]));\n const markingDirtyPromise = this._metadata\n .markDirty(idToLastHash)\n .then(() => {\n this._updatingMetadata.delete(event.handle.documentId);\n })\n .catch((err: Error) => {\n this._ctx.disposed && log.catch(err);\n });\n this._updatingMetadata.set(event.handle.documentId, markingDirtyPromise);\n }\n\n @trace.info({ depth: null })\n private _automergeDocs() {\n return mapValues(this._repo.handles, (handle) => ({\n state: handle.state,\n hasDoc: !!handle.docSync(),\n heads: handle.docSync() ? automerge.getHeads(handle.docSync()) : null,\n data:\n handle.docSync()?.doc &&\n mapValues(handle.docSync()?.doc, (value, key) => {\n try {\n switch (key) {\n case 'access':\n case 'links':\n return value;\n case 'objects':\n return Object.keys(value as any);\n default:\n return `${value}`;\n }\n } catch (err) {\n return `${err}`;\n }\n }),\n }));\n }\n\n @trace.info({ depth: null })\n private _automergePeers() {\n return this._repo.peers;\n }\n\n async close() {\n await this._storage.close();\n await this._clientNetwork.close();\n await this._ctx.dispose();\n }\n\n //\n // Methods for client-services.\n //\n\n syncRepo(request: SyncRepoRequest): Stream<SyncRepoResponse> {\n return this._clientNetwork.syncRepo(request);\n }\n\n sendSyncMessage(request: SyncRepoRequest): Promise<void> {\n return this._clientNetwork.sendSyncMessage(request);\n }\n\n async getHostInfo(): Promise<HostInfo> {\n return this._clientNetwork.getHostInfo();\n }\n\n //\n // Mesh replication.\n //\n\n createExtension(): AutomergeReplicator {\n return this._meshNetwork.createExtension();\n }\n\n authorizeDevice(spaceKey: PublicKey, deviceKey: PublicKey) {\n log('authorizeDevice', { spaceKey, deviceKey });\n defaultMap(this._authorizedDevices, spaceKey, () => new ComplexSet(PublicKey.hash)).add(deviceKey);\n }\n}\n\n// TODO(mykola): Reconcile with `getInlineAndLinkChanges` in AutomergeDB.\nconst getInlineChanges = (event: DocHandleChangePayload<any>) => {\n const inlineChangedObjectIds = new Set<string>();\n for (const { path } of event.patches) {\n if (path.length < 2) {\n continue;\n }\n switch (path[0]) {\n case 'objects':\n if (path.length >= 2) {\n inlineChangedObjectIds.add(path[1]);\n }\n break;\n }\n }\n return [...inlineChangedObjectIds];\n};\n\nexport const getSpaceKeyFromDoc = (doc: any): 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.experimental_spaceKey;\n if (rawSpaceKey == null) {\n return null;\n }\n\n return String(rawSpaceKey);\n};\n", "//\n// Copyright 2024 DXOS.org\n//\n//\n// Copyright 2023 DXOS.org\n//\n\nimport { type Chunk, type StorageKey, type StorageAdapterInterface } from '@dxos/automerge/automerge-repo';\nimport { type Directory } from '@dxos/random-access-storage';\nimport { arrayToBuffer, bufferToArray } from '@dxos/util';\n\nexport class AutomergeStorageAdapter implements StorageAdapterInterface {\n // TODO(mykola): Hack for restricting automerge Repo to access storage if Host is `closed`.\n // Automerge Repo do not have any lifetime management.\n private _state: 'opened' | 'closed' = 'opened';\n\n constructor(private readonly _directory: Directory) {}\n\n async load(key: StorageKey): Promise<Uint8Array | undefined> {\n if (this._state !== 'opened') {\n return undefined;\n }\n const filename = this._getFilename(key);\n const file = this._directory.getOrCreateFile(filename);\n const { size } = await file.stat();\n if (!size || size === 0) {\n return undefined;\n }\n const buffer = await file.read(0, size);\n return bufferToArray(buffer);\n }\n\n async save(key: StorageKey, data: Uint8Array): Promise<void> {\n if (this._state !== 'opened') {\n return undefined;\n }\n const filename = this._getFilename(key);\n const file = this._directory.getOrCreateFile(filename);\n await file.write(0, arrayToBuffer(data));\n await file.truncate?.(data.length);\n\n await file.flush?.();\n }\n\n async remove(key: StorageKey): Promise<void> {\n if (this._state !== 'opened') {\n return undefined;\n }\n // TODO(dmaretskyi): Better deletion.\n const filename = this._getFilename(key);\n const file = this._directory.getOrCreateFile(filename);\n await file.destroy();\n }\n\n async loadRange(keyPrefix: StorageKey): Promise<Chunk[]> {\n if (this._state !== 'opened') {\n return [];\n }\n const filename = this._getFilename(keyPrefix);\n const entries = await this._directory.list();\n return Promise.all(\n entries\n .filter((entry) => entry.startsWith(filename))\n .map(async (entry): Promise<Chunk> => {\n const file = this._directory.getOrCreateFile(entry);\n const { size } = await file.stat();\n const buffer = await file.read(0, size);\n return {\n key: this._getKeyFromFilename(entry),\n data: bufferToArray(buffer),\n };\n }),\n );\n }\n\n async removeRange(keyPrefix: StorageKey): Promise<void> {\n if (this._state !== 'opened') {\n return undefined;\n }\n const filename = this._getFilename(keyPrefix);\n const entries = await this._directory.list();\n await Promise.all(\n entries\n .filter((entry) => entry.startsWith(filename))\n .map(async (entry): Promise<void> => {\n const file = this._directory.getOrCreateFile(entry);\n await file.destroy();\n }),\n );\n }\n\n async close(): Promise<void> {\n this._state = 'closed';\n }\n\n private _getFilename(key: StorageKey): string {\n return key.map((k) => k.replaceAll('%', '%25').replaceAll('-', '%2D')).join('-');\n }\n\n private _getKeyFromFilename(filename: string): StorageKey {\n return filename.split('-').map((k) => k.replaceAll('%2D', '-').replaceAll('%25', '%'));\n }\n}\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport { type StorageKey, type Chunk, type StorageAdapterInterface } from '@dxos/automerge/automerge-repo';\nimport { type MaybePromise } from '@dxos/util';\n\nimport { AutomergeStorageAdapter } from './automerge-storage-adapter';\n\nexport type StorageCallbacks = {\n beforeSave?: (path: string[]) => MaybePromise<void>;\n afterSave?: (path: string[]) => MaybePromise<void>;\n};\n\nexport type AutomergeStorageWrapperParams = {\n storage: StorageAdapterInterface;\n callbacks: StorageCallbacks;\n};\n\n/**\n * Wrapper for automerge storage that adds callback on save.\n */\nexport class AutomergeStorageWrapper implements StorageAdapterInterface {\n private readonly _storage: StorageAdapterInterface;\n private readonly _callbacks: StorageCallbacks;\n\n constructor({ storage, callbacks }: AutomergeStorageWrapperParams) {\n this._storage = storage;\n this._callbacks = callbacks;\n }\n\n async load(key: StorageKey): Promise<Uint8Array | undefined> {\n return this._storage.load(key);\n }\n\n async save(key: StorageKey, value: Uint8Array): Promise<void> {\n await this._callbacks.beforeSave?.(key);\n await this._storage.save(key, value);\n await this._callbacks.afterSave?.(key);\n }\n\n async remove(key: StorageKey): Promise<void> {\n return this._storage.remove(key);\n }\n\n async loadRange(keyPrefix: StorageKey): Promise<Chunk[]> {\n return this._storage.loadRange(keyPrefix);\n }\n\n async removeRange(keyPrefix: StorageKey): Promise<void> {\n return this._storage.removeRange(keyPrefix);\n }\n\n async close() {\n if (this._storage instanceof AutomergeStorageAdapter) {\n return this._storage.close();\n }\n }\n}\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport { Trigger } from '@dxos/async';\nimport { NetworkAdapter, type Message, type PeerId, cbor } from '@dxos/automerge/automerge-repo';\nimport { Stream } from '@dxos/codec-protobuf';\nimport { invariant } from '@dxos/invariant';\nimport { log } from '@dxos/log';\nimport { type HostInfo, type SyncRepoRequest, type SyncRepoResponse } from '@dxos/protocols/proto/dxos/echo/service';\n\ntype ClientSyncState = {\n connected: boolean;\n send: (message: Message) => void;\n disconnect: () => void;\n};\n\n/**\n * Used to replicate with apps running on the same device.\n */\nexport class LocalHostNetworkAdapter extends NetworkAdapter {\n private readonly _peers: Map<PeerId, ClientSyncState> = new Map();\n\n /**\n * Emits `ready` event. That signals to `Repo` that it can start using the adapter.\n */\n ready() {\n // NOTE: Emitting `ready` event in NetworkAdapter`s constructor causes a race condition\n // because `Repo` waits for `ready` event (which it never receives) before it starts using the adapter.\n this.emit('ready', {\n network: this,\n });\n }\n\n private _connected = new Trigger();\n\n override connect(peerId: PeerId): void {\n this.peerId = peerId;\n this._connected.wake();\n // No-op. Client always connects first\n }\n\n override send(message: Message): void {\n const peer = this._peers.get(message.targetId);\n invariant(peer, 'Peer not found.');\n peer.send(message);\n }\n\n async close() {\n this._peers.forEach((peer) => peer.disconnect());\n this.emit('close');\n }\n\n override disconnect(): void {\n // TODO(mykola): `disconnect` is not used anywhere in `Repo` from `@automerge/automerge-repo`. Should we remove it?\n // No-op\n }\n\n syncRepo({ id, syncMessage }: SyncRepoRequest): Stream<SyncRepoResponse> {\n const peerId = this._getPeerId(id);\n\n return new Stream(({ next, close }) => {\n invariant(!this._peers.has(peerId), 'Peer already connected.');\n this._peers.set(peerId, {\n connected: true,\n send: (message) => {\n next({\n syncMessage: cbor.encode(message),\n });\n },\n disconnect: () => {\n this._peers.delete(peerId);\n close();\n this.emit('peer-disconnected', {\n peerId,\n });\n },\n });\n\n this._connected\n .wait({ timeout: 1_000 })\n .then(() => {\n this.emit('peer-candidate', {\n peerMetadata: {},\n peerId,\n });\n })\n .catch((err) => log.catch(err));\n });\n }\n\n async sendSyncMessage({ id, syncMessage }: SyncRepoRequest): Promise<void> {\n await this._connected.wait({ timeout: 1_000 });\n const message = cbor.decode(syncMessage!) as Message;\n this.emit('message', message);\n }\n\n async getHostInfo(): Promise<HostInfo> {\n await this._connected.wait({ timeout: 1_000 });\n invariant(this.peerId, 'Peer id not set.');\n return {\n peerId: this.peerId,\n };\n }\n\n private _getPeerId(id: string): PeerId {\n return id as PeerId;\n }\n}\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport { Trigger } from '@dxos/async';\nimport { NetworkAdapter, type Message, type PeerId, cbor } from '@dxos/automerge/automerge-repo';\nimport { invariant } from '@dxos/invariant';\nimport { log } from '@dxos/log';\nimport { type PeerInfo } from '@dxos/protocols/proto/dxos/mesh/teleport/automerge';\nimport { AutomergeReplicator } from '@dxos/teleport-extension-automerge-replicator';\n\n/**\n * Used to replicate with other peers over the network.\n */\nexport class MeshNetworkAdapter extends NetworkAdapter {\n private readonly _extensions: Map<string, AutomergeReplicator> = new Map();\n private _connected = new Trigger();\n\n /**\n * Emits `ready` event. That signals to `Repo` that it can start using the adapter.\n */\n ready() {\n // NOTE: Emitting `ready` event in NetworkAdapter`s constructor causes a race condition\n // because `Repo` waits for `ready` event (which it never receives) before it starts using the adapter.\n this.emit('ready', {\n network: this,\n });\n }\n\n override connect(peerId: PeerId): void {\n this.peerId = peerId;\n this._connected.wake();\n }\n\n override send(message: Message): void {\n const receiverId = message.targetId;\n const extension = this._extensions.get(receiverId);\n invariant(extension, 'Extension not found.');\n extension.sendSyncMessage({ payload: cbor.encode(message) }).catch((err) => log.catch(err));\n }\n\n override disconnect(): void {\n // No-op\n }\n\n createExtension(): AutomergeReplicator {\n invariant(this.peerId, 'Peer id not set.');\n\n let peerInfo: PeerInfo;\n const extension = new AutomergeReplicator(\n {\n peerId: this.peerId,\n },\n {\n onStartReplication: async (info, remotePeerId /** Teleport ID */) => {\n await this._connected.wait();\n\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 log('onStartReplication', { id: info.id, thisPeerId: this.peerId, remotePeerId: remotePeerId.toHex() });\n if (!this._extensions.has(info.id)) {\n peerInfo = info;\n // TODO(mykola): Fix race condition?\n this._extensions.set(info.id, extension);\n\n log('peer-candidate', { id: info.id, thisPeerId: this.peerId, remotePeerId: remotePeerId.toHex() });\n this.emit('peer-candidate', {\n // TODO(mykola): Hack, stop abusing `peerMetadata` field.\n peerMetadata: {\n dxos_deviceKey: remotePeerId.toHex(),\n } as any,\n peerId: info.id as PeerId,\n });\n }\n },\n onSyncMessage: async ({ payload }) => {\n if (!peerInfo) {\n return;\n }\n const message = cbor.decode(payload) as Message;\n // Note: automerge Repo dedup messages.\n this.emit('message', message);\n },\n onClose: async () => {\n if (!peerInfo) {\n return;\n }\n this.emit('peer-disconnected', {\n peerId: peerInfo.id as PeerId,\n });\n this._extensions.delete(peerInfo.id);\n },\n },\n );\n return extension;\n }\n}\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport { Event } from '@dxos/async';\nimport { type DocHandle, type AutomergeUrl, type DocumentId, type Repo } from '@dxos/automerge/automerge-repo';\nimport { cancelWithContext, type Context } from '@dxos/context';\nimport { warnAfterTimeout } from '@dxos/debug';\nimport { invariant } from '@dxos/invariant';\nimport { type PublicKey } from '@dxos/keys';\nimport { log } from '@dxos/log';\n\nimport { type SpaceState, type SpaceDoc } from './types';\n\ntype SpaceDocumentLinks = SpaceDoc['links'];\n\nexport interface AutomergeDocumentLoader {\n onObjectDocumentLoaded: Event<ObjectDocumentLoaded>;\n\n loadSpaceRootDocHandle(ctx: Context, spaceState: SpaceState): Promise<void>;\n loadObjectDocument(objectId: string): void;\n getSpaceRootDocHandle(): DocHandle<SpaceDoc>;\n createDocumentForObject(objectId: string): DocHandle<SpaceDoc>;\n onObjectLinksUpdated(links: SpaceDocumentLinks): void;\n onObjectBoundToDocument(handle: DocHandle<SpaceDoc>, objectId: string): void;\n\n /**\n * @returns objectIds for which we had document handles or were loading one.\n */\n clearHandleReferences(): string[];\n}\n\n/**\n * Manages object <-> docHandle binding and automerge document loading.\n */\nexport class AutomergeDocumentLoaderImpl implements AutomergeDocumentLoader {\n private _spaceRootDocHandle: DocHandle<SpaceDoc> | null = null;\n /**\n * An object id pointer to a handle of the document where the object is stored inline.\n */\n private readonly _objectDocumentHandles = new Map<string, DocHandle<SpaceDoc>>();\n /**\n * If object was requested via loadObjectDocument but root document links weren't updated yet\n * loading will be triggered in onObjectLinksUpdated callback.\n */\n private readonly _objectsPendingDocumentLoad = new Set<string>();\n\n public readonly onObjectDocumentLoaded = new Event<ObjectDocumentLoaded>();\n\n constructor(\n private readonly _spaceKey: PublicKey,\n private readonly _repo: Repo,\n ) {}\n\n public async loadSpaceRootDocHandle(ctx: Context, spaceState: SpaceState): Promise<void> {\n if (this._spaceRootDocHandle != null) {\n return;\n }\n if (!spaceState.rootUrl) {\n log.error('Database opened with no rootUrl', { spaceKey: this._spaceKey });\n this._createContextBoundSpaceRootDocument(ctx);\n } else {\n const existingDocHandle = await this._initDocHandle(ctx, spaceState.rootUrl);\n const doc = existingDocHandle.docSync();\n invariant(doc);\n if (doc.access == null) {\n this._initDocAccess(existingDocHandle);\n }\n this._spaceRootDocHandle = existingDocHandle;\n }\n }\n\n public loadObjectDocument(objectId: string) {\n invariant(this._spaceRootDocHandle);\n if (this._objectDocumentHandles.has(objectId) || this._objectsPendingDocumentLoad.has(objectId)) {\n return;\n }\n const spaceRootDoc = this._spaceRootDocHandle.docSync();\n invariant(spaceRootDoc);\n const documentUrl = (spaceRootDoc.links ?? {})[objectId];\n if (documentUrl == null) {\n this._objectsPendingDocumentLoad.add(objectId);\n log.info('loading delayed until object links are initialized', { objectId });\n return;\n }\n this._loadLinkedObjects({ [objectId]: documentUrl });\n }\n\n public onObjectLinksUpdated(links: SpaceDocumentLinks) {\n if (!links) {\n return;\n }\n const linksAwaitingLoad = Object.entries(links).filter(([objectId]) =>\n this._objectsPendingDocumentLoad.has(objectId),\n );\n this._loadLinkedObjects(Object.fromEntries(linksAwaitingLoad));\n linksAwaitingLoad.forEach(([objectId]) => this._objectsPendingDocumentLoad.delete(objectId));\n }\n\n public getSpaceRootDocHandle(): DocHandle<SpaceDoc> {\n invariant(this._spaceRootDocHandle);\n return this._spaceRootDocHandle;\n }\n\n public createDocumentForObject(objectId: string): DocHandle<SpaceDoc> {\n invariant(this._spaceRootDocHandle);\n const spaceDocHandle = this._repo.create<SpaceDoc>();\n this._initDocAccess(spaceDocHandle);\n this.onObjectBoundToDocument(spaceDocHandle, objectId);\n this._spaceRootDocHandle.change((newDoc: SpaceDoc) => {\n newDoc.links ??= {};\n newDoc.links[objectId] = spaceDocHandle.url;\n });\n return spaceDocHandle;\n }\n\n public onObjectBoundToDocument(handle: DocHandle<SpaceDoc>, objectId: string) {\n this._objectDocumentHandles.set(objectId, handle);\n }\n\n public clearHandleReferences(): string[] {\n const objectsWithHandles = [...this._objectDocumentHandles.keys()];\n this._objectDocumentHandles.clear();\n this._spaceRootDocHandle = null;\n return objectsWithHandles;\n }\n\n private _loadLinkedObjects(links: SpaceDocumentLinks) {\n if (!links) {\n return;\n }\n for (const [objectId, automergeUrl] of Object.entries(links)) {\n const logMeta = { objectId, automergeUrl };\n const objectDocumentHandle = this._objectDocumentHandles.get(objectId);\n if (objectDocumentHandle != null && objectDocumentHandle.url !== automergeUrl) {\n log.warn('object already inlined in a different document, ignoring the link', {\n ...logMeta,\n actualDocumentUrl: objectDocumentHandle.url,\n });\n continue;\n }\n if (objectDocumentHandle?.url === automergeUrl) {\n log.warn('object document was already loaded', logMeta);\n continue;\n }\n const handle = this._repo.find<SpaceDoc>(automergeUrl as DocumentId);\n log.debug('document loading triggered', logMeta);\n this._objectDocumentHandles.set(objectId, handle);\n void this._createObjectOnDocumentLoad(handle, objectId);\n }\n }\n\n private async _initDocHandle(ctx: Context, url: string) {\n const docHandle = this._repo.find<SpaceDoc>(url as DocumentId);\n while (true) {\n try {\n await warnAfterTimeout(5_000, 'Automerge root doc load timeout (AutomergeDb)', async () => {\n await cancelWithContext(ctx, docHandle.whenReady()); // TODO(dmaretskyi): Temporary 5s timeout for debugging.\n });\n break;\n } catch (err) {\n if (`${err}`.includes('Timeout')) {\n log.info('wraparound', { id: docHandle.documentId, state: docHandle.state });\n continue;\n }\n\n throw err;\n }\n }\n\n if (docHandle.state === 'unavailable') {\n throw new Error('Automerge document is unavailable');\n }\n\n return docHandle;\n }\n\n private _createContextBoundSpaceRootDocument(ctx: Context) {\n const docHandle = this._repo.create<SpaceDoc>();\n this._spaceRootDocHandle = docHandle;\n ctx.onDispose(() => {\n docHandle.delete();\n this._spaceRootDocHandle = null;\n });\n }\n\n private _initDocAccess(handle: DocHandle<SpaceDoc>) {\n handle.change((newDoc: SpaceDoc) => {\n newDoc.access ??= { spaceKey: this._spaceKey.toHex() };\n newDoc.access.spaceKey = this._spaceKey.toHex();\n });\n }\n\n private async _createObjectOnDocumentLoad(handle: DocHandle<SpaceDoc>, objectId: string) {\n try {\n await handle.doc(['ready']);\n const logMeta = { objectId, docUrl: handle.url };\n if (this.onObjectDocumentLoaded.listenerCount() === 0) {\n log.info('document loaded after all listeners were removed', logMeta);\n return;\n }\n const objectDocHandle = this._objectDocumentHandles.get(objectId);\n if (objectDocHandle?.url !== handle.url) {\n log.warn('object was rebound while a document was loading, discarding handle', logMeta);\n return;\n }\n this.onObjectDocumentLoaded.emit({ handle, objectId });\n } catch (err) {\n const shouldRetryLoading = this.onObjectDocumentLoaded.listenerCount() > 0;\n log.warn('failed to load a document', {\n objectId,\n automergeUrl: handle.url,\n retryLoading: shouldRetryLoading,\n err,\n });\n if (shouldRetryLoading) {\n await this._createObjectOnDocumentLoad(handle, objectId);\n }\n }\n }\n}\n\nexport interface ObjectDocumentLoaded {\n handle: DocHandle<SpaceDoc>;\n objectId: string;\n}\n\nexport interface DocumentChanges {\n createdObjectIds: string[];\n updatedObjectIds: string[];\n objectsToRebind: string[];\n linkedDocuments: {\n [echoId: string]: AutomergeUrl;\n };\n}\n"],
|
|
5
|
-
"mappings": ";;;;;;;;;;;;;;;;;;;;;;;;;AAIA,SAASA,QAAQC,WAAWC,gBAAgB;AAC5C,SACEC,YAMK;AACP,SAASC,+BAA+B;AAExC,SAASC,eAAe;AACxB,SAASC,iBAAiB;AAC1B,SAASC,OAAAA,YAAW;AACpB,SAASC,eAAe;AAExB,SAASC,mBAAmC;AAE5C,SAASC,aAAa;AACtB,SAASC,YAAYC,YAAYC,YAAYC,iBAAiB;;;ACd9D,SAASC,eAAeC,qBAAqB;AAEtC,IAAMC,0BAAN,MAAMA;EAKXC,YAA6BC,YAAuB;SAAvBA,aAAAA;SAFrBC,SAA8B;EAEe;EAErD,MAAMC,KAAKC,KAAkD;AAC3D,QAAI,KAAKF,WAAW,UAAU;AAC5B,aAAOG;IACT;AACA,UAAMC,WAAW,KAAKC,aAAaH,GAAAA;AACnC,UAAMI,OAAO,KAAKP,WAAWQ,gBAAgBH,QAAAA;AAC7C,UAAM,EAAEI,KAAI,IAAK,MAAMF,KAAKG,KAAI;AAChC,QAAI,CAACD,QAAQA,SAAS,GAAG;AACvB,aAAOL;IACT;AACA,UAAMO,SAAS,MAAMJ,KAAKK,KAAK,GAAGH,IAAAA;AAClC,WAAOI,cAAcF,MAAAA;EACvB;EAEA,MAAMG,KAAKX,KAAiBY,MAAiC;AAC3D,QAAI,KAAKd,WAAW,UAAU;AAC5B,aAAOG;IACT;AACA,UAAMC,WAAW,KAAKC,aAAaH,GAAAA;AACnC,UAAMI,OAAO,KAAKP,WAAWQ,gBAAgBH,QAAAA;AAC7C,UAAME,KAAKS,MAAM,GAAGC,cAAcF,IAAAA,CAAAA;AAClC,UAAMR,KAAKW,WAAWH,KAAKI,MAAM;AAEjC,UAAMZ,KAAKa,QAAK;EAClB;EAEA,MAAMC,OAAOlB,KAAgC;AAC3C,QAAI,KAAKF,WAAW,UAAU;AAC5B,aAAOG;IACT;AAEA,UAAMC,WAAW,KAAKC,aAAaH,GAAAA;AACnC,UAAMI,OAAO,KAAKP,WAAWQ,gBAAgBH,QAAAA;AAC7C,UAAME,KAAKe,QAAO;EACpB;EAEA,MAAMC,UAAUC,WAAyC;AACvD,QAAI,KAAKvB,WAAW,UAAU;AAC5B,aAAO,CAAA;IACT;AACA,UAAMI,WAAW,KAAKC,aAAakB,SAAAA;AACnC,UAAMC,UAAU,MAAM,KAAKzB,WAAW0B,KAAI;AAC1C,WAAOC,QAAQC,IACbH,QACGI,OAAO,CAACC,UAAUA,MAAMC,WAAW1B,QAAAA,CAAAA,EACnC2B,IAAI,OAAOF,UAAAA;AACV,YAAMvB,OAAO,KAAKP,WAAWQ,gBAAgBsB,KAAAA;AAC7C,YAAM,EAAErB,KAAI,IAAK,MAAMF,KAAKG,KAAI;AAChC,YAAMC,SAAS,MAAMJ,KAAKK,KAAK,GAAGH,IAAAA;AAClC,aAAO;QACLN,KAAK,KAAK8B,oBAAoBH,KAAAA;QAC9Bf,MAAMF,cAAcF,MAAAA;MACtB;IACF,CAAA,CAAA;EAEN;EAEA,MAAMuB,YAAYV,WAAsC;AACtD,QAAI,KAAKvB,WAAW,UAAU;AAC5B,aAAOG;IACT;AACA,UAAMC,WAAW,KAAKC,aAAakB,SAAAA;AACnC,UAAMC,UAAU,MAAM,KAAKzB,WAAW0B,KAAI;AAC1C,UAAMC,QAAQC,IACZH,QACGI,OAAO,CAACC,UAAUA,MAAMC,WAAW1B,QAAAA,CAAAA,EACnC2B,IAAI,OAAOF,UAAAA;AACV,YAAMvB,OAAO,KAAKP,WAAWQ,gBAAgBsB,KAAAA;AAC7C,YAAMvB,KAAKe,QAAO;IACpB,CAAA,CAAA;EAEN;EAEA,MAAMa,QAAuB;AAC3B,SAAKlC,SAAS;EAChB;EAEQK,aAAaH,KAAyB;AAC5C,WAAOA,IAAI6B,IAAI,CAACI,MAAMA,EAAEC,WAAW,KAAK,KAAA,EAAOA,WAAW,KAAK,KAAA,CAAA,EAAQC,KAAK,GAAA;EAC9E;EAEQL,oBAAoB5B,UAA8B;AACxD,WAAOA,SAASkC,MAAM,GAAA,EAAKP,IAAI,CAACI,MAAMA,EAAEC,WAAW,OAAO,GAAA,EAAKA,WAAW,OAAO,GAAA,CAAA;EACnF;AACF;;;AChFO,IAAMG,0BAAN,MAAMA;EAIXC,YAAY,EAAEC,SAASC,UAAS,GAAmC;AACjE,SAAKC,WAAWF;AAChB,SAAKG,aAAaF;EACpB;EAEA,MAAMG,KAAKC,KAAkD;AAC3D,WAAO,KAAKH,SAASE,KAAKC,GAAAA;EAC5B;EAEA,MAAMC,KAAKD,KAAiBE,OAAkC;AAC5D,UAAM,KAAKJ,WAAWK,aAAaH,GAAAA;AACnC,UAAM,KAAKH,SAASI,KAAKD,KAAKE,KAAAA;AAC9B,UAAM,KAAKJ,WAAWM,YAAYJ,GAAAA;EACpC;EAEA,MAAMK,OAAOL,KAAgC;AAC3C,WAAO,KAAKH,SAASQ,OAAOL,GAAAA;EAC9B;EAEA,MAAMM,UAAUC,WAAyC;AACvD,WAAO,KAAKV,SAASS,UAAUC,SAAAA;EACjC;EAEA,MAAMC,YAAYD,WAAsC;AACtD,WAAO,KAAKV,SAASW,YAAYD,SAAAA;EACnC;EAEA,MAAME,QAAQ;AACZ,QAAI,KAAKZ,oBAAoBa,yBAAyB;AACpD,aAAO,KAAKb,SAASY,MAAK;IAC5B;EACF;AACF;;;ACtDA,SAASE,eAAe;AACxB,SAASC,gBAA2CC,YAAY;AAChE,SAASC,cAAc;AACvB,SAASC,iBAAiB;AAC1B,SAASC,WAAW;;AAYb,IAAMC,0BAAN,cAAsCL,eAAAA;EAAtC;;AACYM,kBAAuC,oBAAIC,IAAAA;AAapDC,sBAAa,IAAIT,QAAAA;;;;;EARzBU,QAAQ;AAGN,SAAKC,KAAK,SAAS;MACjBC,SAAS;IACX,CAAA;EACF;EAISC,QAAQC,QAAsB;AACrC,SAAKA,SAASA;AACd,SAAKL,WAAWM,KAAI;EAEtB;EAESC,KAAKC,SAAwB;AACpC,UAAMC,OAAO,KAAKX,OAAOY,IAAIF,QAAQG,QAAQ;AAC7ChB,cAAUc,MAAM,mBAAA;;;;;;;;;AAChBA,SAAKF,KAAKC,OAAAA;EACZ;EAEA,MAAMI,QAAQ;AACZ,SAAKd,OAAOe,QAAQ,CAACJ,SAASA,KAAKK,WAAU,CAAA;AAC7C,SAAKZ,KAAK,OAAA;EACZ;EAESY,aAAmB;EAG5B;EAEAC,SAAS,EAAEC,IAAIC,YAAW,GAA+C;AACvE,UAAMZ,SAAS,KAAKa,WAAWF,EAAAA;AAE/B,WAAO,IAAItB,OAAO,CAAC,EAAEyB,MAAMP,MAAK,MAAE;AAChCjB,gBAAU,CAAC,KAAKG,OAAOsB,IAAIf,MAAAA,GAAS,2BAAA;;;;;;;;;AACpC,WAAKP,OAAOuB,IAAIhB,QAAQ;QACtBiB,WAAW;QACXf,MAAM,CAACC,YAAAA;AACLW,eAAK;YACHF,aAAaxB,KAAK8B,OAAOf,OAAAA;UAC3B,CAAA;QACF;QACAM,YAAY,MAAA;AACV,eAAKhB,OAAO0B,OAAOnB,MAAAA;AACnBO,gBAAAA;AACA,eAAKV,KAAK,qBAAqB;YAC7BG;UACF,CAAA;QACF;MACF,CAAA;AAEA,WAAKL,WACFyB,KAAK;QAAEC,SAAS;MAAM,CAAA,EACtBC,KAAK,MAAA;AACJ,aAAKzB,KAAK,kBAAkB;UAC1B0B,cAAc,CAAC;UACfvB;QACF,CAAA;MACF,CAAA,EACCwB,MAAM,CAACC,QAAQlC,IAAIiC,MAAMC,KAAAA,QAAAA;;;;;;IAC9B,CAAA;EACF;EAEA,MAAMC,gBAAgB,EAAEf,IAAIC,YAAW,GAAoC;AACzE,UAAM,KAAKjB,WAAWyB,KAAK;MAAEC,SAAS;IAAM,CAAA;AAC5C,UAAMlB,UAAUf,KAAKuC,OAAOf,WAAAA;AAC5B,SAAKf,KAAK,WAAWM,OAAAA;EACvB;EAEA,MAAMyB,cAAiC;AACrC,UAAM,KAAKjC,WAAWyB,KAAK;MAAEC,SAAS;IAAM,CAAA;AAC5C/B,cAAU,KAAKU,QAAQ,oBAAA;;;;;;;;;AACvB,WAAO;MACLA,QAAQ,KAAKA;IACf;EACF;EAEQa,WAAWF,IAAoB;AACrC,WAAOA;EACT;AACF;;;ACxGA,SAASkB,WAAAA,gBAAe;AACxB,SAASC,kBAAAA,iBAA2CC,QAAAA,aAAY;AAChE,SAASC,aAAAA,kBAAiB;AAC1B,SAASC,OAAAA,YAAW;AAEpB,SAASC,2BAA2B;;AAK7B,IAAMC,qBAAN,cAAiCL,gBAAAA;EAAjC;;AACYM,uBAAgD,oBAAIC,IAAAA;AAC7DC,sBAAa,IAAIT,SAAAA;;;;;EAKzBU,QAAQ;AAGN,SAAKC,KAAK,SAAS;MACjBC,SAAS;IACX,CAAA;EACF;EAESC,QAAQC,QAAsB;AACrC,SAAKA,SAASA;AACd,SAAKL,WAAWM,KAAI;EACtB;EAESC,KAAKC,SAAwB;AACpC,UAAMC,aAAaD,QAAQE;AAC3B,UAAMC,YAAY,KAAKb,YAAYc,IAAIH,UAAAA;AACvCf,IAAAA,WAAUiB,WAAW,wBAAA;;;;;;;;;AACrBA,cAAUE,gBAAgB;MAAEC,SAASrB,MAAKsB,OAAOP,OAAAA;IAAS,CAAA,EAAGQ,MAAM,CAACC,QAAQtB,KAAIqB,MAAMC,KAAAA,QAAAA;;;;;;EACxF;EAESC,aAAmB;EAE5B;EAEAC,kBAAuC;AACrCzB,IAAAA,WAAU,KAAKW,QAAQ,oBAAA;;;;;;;;;AAEvB,QAAIe;AACJ,UAAMT,YAAY,IAAIf,oBACpB;MACES,QAAQ,KAAKA;IACf,GACA;MACEgB,oBAAoB,OAAOC,MAAMC,iBAA6B;AAC5D,cAAM,KAAKvB,WAAWwB,KAAI;AAc1B7B,QAAAA,KAAI,sBAAsB;UAAE8B,IAAIH,KAAKG;UAAIC,YAAY,KAAKrB;UAAQkB,cAAcA,aAAaI,MAAK;QAAG,GAAA;;;;;;AACrG,YAAI,CAAC,KAAK7B,YAAY8B,IAAIN,KAAKG,EAAE,GAAG;AAClCL,qBAAWE;AAEX,eAAKxB,YAAY+B,IAAIP,KAAKG,IAAId,SAAAA;AAE9BhB,UAAAA,KAAI,kBAAkB;YAAE8B,IAAIH,KAAKG;YAAIC,YAAY,KAAKrB;YAAQkB,cAAcA,aAAaI,MAAK;UAAG,GAAA;;;;;;AACjG,eAAKzB,KAAK,kBAAkB;;YAE1B4B,cAAc;cACZC,gBAAgBR,aAAaI,MAAK;YACpC;YACAtB,QAAQiB,KAAKG;UACf,CAAA;QACF;MACF;MACAO,eAAe,OAAO,EAAElB,QAAO,MAAE;AAC/B,YAAI,CAACM,UAAU;AACb;QACF;AACA,cAAMZ,UAAUf,MAAKwC,OAAOnB,OAAAA;AAE5B,aAAKZ,KAAK,WAAWM,OAAAA;MACvB;MACA0B,SAAS,YAAA;AACP,YAAI,CAACd,UAAU;AACb;QACF;AACA,aAAKlB,KAAK,qBAAqB;UAC7BG,QAAQe,SAASK;QACnB,CAAA;AACA,aAAK3B,YAAYqC,OAAOf,SAASK,EAAE;MACrC;IACF,CAAA;AAEF,WAAOd;EACT;AACF;;;;;;;;;;;;;;AJhEO,IAAMyB,gBAAN,MAAMA;EAoBXC,YAAY,EAAEC,WAAWC,SAAQ,GAAyB;AAnBzCC,gBAAO,IAAIC,QAAAA;AAYXC;;;8BAAqB,IAAIC,WAA6CC,UAAUC,IAAI;AAEpFC,6BAAoB,oBAAIC,IAAAA;AAGlCC,0BAAiB,oBAAIC,IAAAA;AAG1B,SAAKC,YAAYX;AACjB,SAAKY,eAAe,IAAIC,mBAAAA;AACxB,SAAKC,iBAAiB,IAAIC,wBAAAA;AAE1B,SAAKC,WAAW,IAAIC,wBAAwB;MAC1CC;;QAEEnB,UAAUoB,SAASC,YAAYC,MAC3B,IAAIC,wBAAwBvB,UAAUwB,MAAM,MAAA,IAC5C,IAAIC,wBAAwBzB,SAAAA;;MAClC0B,WAAW;QAAEC,YAAY,CAACC,WAAW,KAAKC,YAAYD,MAAAA;MAAQ;IAChE,CAAA;AACA,SAAKE,UAAU,QAAQxB,UAAUyB,OAAM,EAAGC,MAAK,CAAA;AAC/C,SAAKC,QAAQ,IAAIC,KAAK;MACpBC,QAAQ,KAAKL;MACbM,SAAS;QAAC,KAAKrB;QAAgB,KAAKF;;MACpCM,SAAS,KAAKF;;;MAIdoB,aAAa,OAAOF,QAAyBG,eAAwB;AACnE,YAAIH,OAAOI,WAAW,SAAA,GAAY;AAChC,iBAAO;QACT;AAEA,YAAI,CAACD,YAAY;AACf,iBAAO;QACT;AAEA,cAAME,MAAM,KAAKP,MAAMQ,QAAQH,UAAAA,GAAaI,QAAAA;AAC5C,YAAI,CAACF,KAAK;AACR,gBAAMG,cAAc,KAAKjC,eAAekC,IAAI,aAAaN,UAAAA,EAAY;AACrEO,UAAAA,KAAI,0BAA0B;YAAEV;YAAQG;YAAYK;UAAY,GAAA;;;;;;AAChE,iBAAOA;QACT;AAEA,YAAI;AACF,gBAAMG,WAAWC,mBAAmBP,GAAAA;AACpC,cAAI,CAACM,UAAU;AACbD,YAAAA,KAAI,8CAA8C;cAAEV;cAAQG;YAAW,GAAA;;;;;;AACvE,mBAAO;UACT;AAEA,gBAAMU,oBAAoB,KAAK5C,mBAAmB6C,IAAI3C,UAAU4C,KAAKJ,QAAAA,CAAAA;AAGrE,gBAAMK,eAAgB,KAAKC,KAAKC,qBAAqBlB,MAAAA,GAAiBmB;AACtE,cAAI,CAACH,cAAc;AACjBN,YAAAA,KAAI,+CAA+C;cAAEV;cAAQG;YAAW,GAAA;;;;;;AACxE,mBAAO;UACT;AACA,gBAAMiB,YAAYjD,UAAU4C,KAAKC,YAAAA;AAEjC,gBAAMK,eAAeR,mBAAmBJ,IAAIW,SAAAA,KAAc;AAC1DV,UAAAA,KAAI,sBAAsB;YACxBY,WAAW,KAAK3B;YAChB4B,YAAYvB;YACZG;YACAiB;YACAT;YACAU;UACF,GAAA;;;;;;AACA,iBAAOA;QACT,SAASG,KAAK;AACZd,UAAAA,KAAIe,MAAMD,KAAAA,QAAAA;;;;;;AACV,iBAAO;QACT;MACF;IACF,CAAA;AACA,SAAK5C,eAAe8C,MAAK;AACzB,SAAKhD,aAAagD,MAAK;AAEvB;AACE,YAAMC,WAAW,CAAC,EAAEC,OAAM,MAAmC,KAAKC,YAAYD,MAAAA;AAC9E,WAAK9B,MAAMgC,GAAG,YAAYH,QAAAA;AAC1B,WAAK5D,KAAKgE,UAAU,MAAA;AAClB,aAAKjC,MAAMkC,IAAI,YAAYL,QAAAA;MAC7B,CAAA;IACF;EACF;EAEA,IAAIV,OAAa;AACf,WAAO,KAAKnB;EACd;EAEA,MAAcJ,YAAYL,MAAkB;AAC1C,UAAM4C,KAAK5C,KAAK,CAAA;AAChB,QAAI,KAAKhB,kBAAkBoC,IAAIwB,EAAAA,GAAK;AAClC,aAAO,KAAK5D,kBAAkByC,IAAImB,EAAAA;IACpC;EACF;EAEQJ,YAAYD,QAAwB;AAC1C,UAAMD,WAAW,CAACO,UAAuC,KAAKC,UAAUD,KAAAA;AACxEN,WAAOE,GAAG,UAAUH,QAAAA;AACpB,SAAK5D,KAAKgE,UAAU,MAAA;AAClBH,aAAOI,IAAI,UAAUL,QAAAA;IACvB,CAAA;EACF;EAEQQ,UAAUD,OAAoC;AACpD,QAAI,KAAKzD,aAAa,MAAM;AAC1B;IACF;AAEA,UAAM2D,YAAYC,iBAAiBH,KAAAA;AACnC,QAAIE,UAAUE,WAAW,GAAG;AAC1B;IACF;AAEA,UAAMC,QAAQC,SAASN,MAAM7B,GAAG;AAChC,UAAMoC,oBAAoBF,MAAMG,KAAK,EAAA;AACrC,QAAI,CAACD,mBAAmB;AACtB;IACF;AAEA,UAAME,aAAaP,UAAUQ,IAAI,CAACC,aAAaC,QAAQC,OAAO;MAAE5C,YAAY+B,MAAMN,OAAOzB;MAAY0C;IAAS,CAAA,CAAA;AAC9G,UAAMG,eAAe,IAAI1E,IAAIqE,WAAWC,IAAI,CAACX,OAAO;MAACA;MAAIQ;KAAkB,CAAA;AAC3E,UAAMQ,sBAAsB,KAAKxE,UAC9ByE,UAAUF,YAAAA,EACVG,KAAK,MAAA;AACJ,WAAK9E,kBAAkB+E,OAAOlB,MAAMN,OAAOzB,UAAU;IACvD,CAAA,EACCsB,MAAM,CAACD,QAAAA;AACN,WAAKzD,KAAKsF,YAAY3C,KAAIe,MAAMD,KAAAA,QAAAA;;;;;;IAClC,CAAA;AACF,SAAKnD,kBAAkBiF,IAAIpB,MAAMN,OAAOzB,YAAY8C,mBAAAA;EACtD;EAGQM,iBAAiB;AACvB,WAAOC,UAAU,KAAK1D,MAAMQ,SAAS,CAACsB,YAAY;MAChD6B,OAAO7B,OAAO6B;MACdC,QAAQ,CAAC,CAAC9B,OAAOrB,QAAO;MACxBgC,OAAOX,OAAOrB,QAAO,IAAKoD,UAAUnB,SAASZ,OAAOrB,QAAO,CAAA,IAAM;MACjEqD,MACEhC,OAAOrB,QAAO,GAAIF,OAClBmD,UAAU5B,OAAOrB,QAAO,GAAIF,KAAK,CAACwD,OAAOC,QAAAA;AACvC,YAAI;AACF,kBAAQA,KAAAA;YACN,KAAK;YACL,KAAK;AACH,qBAAOD;YACT,KAAK;AACH,qBAAOE,OAAOC,KAAKH,KAAAA;YACrB;AACE,qBAAO,GAAGA,KAAAA;UACd;QACF,SAASrC,KAAK;AACZ,iBAAO,GAAGA,GAAAA;QACZ;MACF,CAAA;IACJ,EAAA;EACF;EAGQyC,kBAAkB;AACxB,WAAO,KAAKnE,MAAMoE;EACpB;EAEA,MAAMC,QAAQ;AACZ,UAAM,KAAKrF,SAASqF,MAAK;AACzB,UAAM,KAAKvF,eAAeuF,MAAK;AAC/B,UAAM,KAAKpG,KAAKqG,QAAO;EACzB;;;;EAMAC,SAASC,SAAoD;AAC3D,WAAO,KAAK1F,eAAeyF,SAASC,OAAAA;EACtC;EAEAC,gBAAgBD,SAAyC;AACvD,WAAO,KAAK1F,eAAe2F,gBAAgBD,OAAAA;EAC7C;EAEA,MAAME,cAAiC;AACrC,WAAO,KAAK5F,eAAe4F,YAAW;EACxC;;;;EAMAC,kBAAuC;AACrC,WAAO,KAAK/F,aAAa+F,gBAAe;EAC1C;EAEAC,gBAAgB/D,UAAqBS,WAAsB;AACzDV,IAAAA,KAAI,mBAAmB;MAAEC;MAAUS;IAAU,GAAA;;;;;;AAC7CuD,eAAW,KAAK1G,oBAAoB0C,UAAU,MAAM,IAAIiE,WAAWzG,UAAUC,IAAI,CAAA,EAAGyG,IAAIzD,SAAAA;EAC1F;AACF;;EAhNG0D,MAAMC,KAAI;GAPApH,cAAAA,WAAAA,WAAAA,MAAAA;;EAsJVmH,MAAMC,KAAK;IAAEC,OAAO;EAAK,CAAA;GAtJfrH,cAAAA,WAAAA,kBAAAA,IAAAA;;EAgLVmH,MAAMC,KAAK;IAAEC,OAAO;EAAK,CAAA;GAhLfrH,cAAAA,WAAAA,mBAAAA,IAAAA;AAAAA,gBAAAA,aAAAA;EADZmH,MAAMG,SAAQ;GACFtH,aAAAA;AA0Nb,IAAM0E,mBAAmB,CAACH,UAAAA;AACxB,QAAMgD,yBAAyB,oBAAI1G,IAAAA;AACnC,aAAW,EAAEa,KAAI,KAAM6C,MAAMiD,SAAS;AACpC,QAAI9F,KAAKiD,SAAS,GAAG;AACnB;IACF;AACA,YAAQjD,KAAK,CAAA,GAAE;MACb,KAAK;AACH,YAAIA,KAAKiD,UAAU,GAAG;AACpB4C,iCAAuBL,IAAIxF,KAAK,CAAA,CAAE;QACpC;AACA;IACJ;EACF;AACA,SAAO;OAAI6F;;AACb;AAEO,IAAMtE,qBAAqB,CAACP,QAAAA;AAEjC,QAAM+E,cAAc/E,IAAIgF,QAAQ1E,YAAYN,IAAIiF;AAChD,MAAIF,eAAe,MAAM;AACvB,WAAO;EACT;AAEA,SAAOG,OAAOH,WAAAA;AAChB;;;AKzRA,SAASI,aAAa;AAEtB,SAASC,yBAAuC;AAChD,SAASC,wBAAwB;AACjC,SAASC,aAAAA,kBAAiB;AAE1B,SAASC,OAAAA,YAAW;;AAyBb,IAAMC,8BAAN,MAAMA;EAcXC,YACmBC,WACAC,OACjB;SAFiBD,YAAAA;SACAC,QAAAA;SAfXC,sBAAkD;SAIzCC,yBAAyB,oBAAIC,IAAAA;SAK7BC,8BAA8B,oBAAIC,IAAAA;SAEnCC,yBAAyB,IAAId,MAAAA;EAK1C;EAEH,MAAae,uBAAuBC,KAAcC,YAAuC;AACvF,QAAI,KAAKR,uBAAuB,MAAM;AACpC;IACF;AACA,QAAI,CAACQ,WAAWC,SAAS;AACvBd,MAAAA,KAAIe,MAAM,mCAAmC;QAAEC,UAAU,KAAKb;MAAU,GAAA;;;;;;AACxE,WAAKc,qCAAqCL,GAAAA;IAC5C,OAAO;AACL,YAAMM,oBAAoB,MAAM,KAAKC,eAAeP,KAAKC,WAAWC,OAAO;AAC3E,YAAMM,MAAMF,kBAAkBG,QAAO;AACrCtB,MAAAA,WAAUqB,KAAAA,QAAAA;;;;;;;;;AACV,UAAIA,IAAIE,UAAU,MAAM;AACtB,aAAKC,eAAeL,iBAAAA;MACtB;AACA,WAAKb,sBAAsBa;IAC7B;EACF;EAEOM,mBAAmBC,UAAkB;AAC1C1B,IAAAA,WAAU,KAAKM,qBAAmB,QAAA;;;;;;;;;AAClC,QAAI,KAAKC,uBAAuBoB,IAAID,QAAAA,KAAa,KAAKjB,4BAA4BkB,IAAID,QAAAA,GAAW;AAC/F;IACF;AACA,UAAME,eAAe,KAAKtB,oBAAoBgB,QAAO;AACrDtB,IAAAA,WAAU4B,cAAAA,QAAAA;;;;;;;;;AACV,UAAMC,eAAeD,aAAaE,SAAS,CAAC,GAAGJ,QAAAA;AAC/C,QAAIG,eAAe,MAAM;AACvB,WAAKpB,4BAA4BsB,IAAIL,QAAAA;AACrCzB,MAAAA,KAAI+B,KAAK,sDAAsD;QAAEN;MAAS,GAAA;;;;;;AAC1E;IACF;AACA,SAAKO,mBAAmB;MAAE,CAACP,QAAAA,GAAWG;IAAY,CAAA;EACpD;EAEOK,qBAAqBJ,OAA2B;AACrD,QAAI,CAACA,OAAO;AACV;IACF;AACA,UAAMK,oBAAoBC,OAAOC,QAAQP,KAAAA,EAAOQ,OAAO,CAAC,CAACZ,QAAAA,MACvD,KAAKjB,4BAA4BkB,IAAID,QAAAA,CAAAA;AAEvC,SAAKO,mBAAmBG,OAAOG,YAAYJ,iBAAAA,CAAAA;AAC3CA,sBAAkBK,QAAQ,CAAC,CAACd,QAAAA,MAAc,KAAKjB,4BAA4BgC,OAAOf,QAAAA,CAAAA;EACpF;EAEOgB,wBAA6C;AAClD1C,IAAAA,WAAU,KAAKM,qBAAmB,QAAA;;;;;;;;;AAClC,WAAO,KAAKA;EACd;EAEOqC,wBAAwBjB,UAAuC;AACpE1B,IAAAA,WAAU,KAAKM,qBAAmB,QAAA;;;;;;;;;AAClC,UAAMsC,iBAAiB,KAAKvC,MAAMwC,OAAM;AACxC,SAAKrB,eAAeoB,cAAAA;AACpB,SAAKE,wBAAwBF,gBAAgBlB,QAAAA;AAC7C,SAAKpB,oBAAoByC,OAAO,CAACC,WAAAA;AAC/BA,aAAOlB,UAAU,CAAC;AAClBkB,aAAOlB,MAAMJ,QAAAA,IAAYkB,eAAeK;IAC1C,CAAA;AACA,WAAOL;EACT;EAEOE,wBAAwBI,QAA6BxB,UAAkB;AAC5E,SAAKnB,uBAAuB4C,IAAIzB,UAAUwB,MAAAA;EAC5C;EAEOE,wBAAkC;AACvC,UAAMC,qBAAqB;SAAI,KAAK9C,uBAAuB+C,KAAI;;AAC/D,SAAK/C,uBAAuBgD,MAAK;AACjC,SAAKjD,sBAAsB;AAC3B,WAAO+C;EACT;EAEQpB,mBAAmBH,OAA2B;AACpD,QAAI,CAACA,OAAO;AACV;IACF;AACA,eAAW,CAACJ,UAAU8B,YAAAA,KAAiBpB,OAAOC,QAAQP,KAAAA,GAAQ;AAC5D,YAAM2B,UAAU;QAAE/B;QAAU8B;MAAa;AACzC,YAAME,uBAAuB,KAAKnD,uBAAuBoD,IAAIjC,QAAAA;AAC7D,UAAIgC,wBAAwB,QAAQA,qBAAqBT,QAAQO,cAAc;AAC7EvD,QAAAA,KAAI2D,KAAK,qEAAqE;UAC5E,GAAGH;UACHI,mBAAmBH,qBAAqBT;QAC1C,GAAA;;;;;;AACA;MACF;AACA,UAAIS,sBAAsBT,QAAQO,cAAc;AAC9CvD,QAAAA,KAAI2D,KAAK,sCAAsCH,SAAAA;;;;;;AAC/C;MACF;AACA,YAAMP,SAAS,KAAK7C,MAAMyD,KAAeN,YAAAA;AACzCvD,MAAAA,KAAI8D,MAAM,8BAA8BN,SAAAA;;;;;;AACxC,WAAKlD,uBAAuB4C,IAAIzB,UAAUwB,MAAAA;AAC1C,WAAK,KAAKc,4BAA4Bd,QAAQxB,QAAAA;IAChD;EACF;EAEA,MAAcN,eAAeP,KAAcoC,KAAa;AACtD,UAAMgB,YAAY,KAAK5D,MAAMyD,KAAeb,GAAAA;AAC5C,WAAO,MAAM;AACX,UAAI;AACF,cAAMlD,iBAAiB,KAAO,iDAAiD,YAAA;AAC7E,gBAAMD,kBAAkBe,KAAKoD,UAAUC,UAAS,CAAA;QAClD,CAAA;AACA;MACF,SAASC,KAAK;AACZ,YAAI,GAAGA,GAAAA,GAAMC,SAAS,SAAA,GAAY;AAChCnE,UAAAA,KAAI+B,KAAK,cAAc;YAAEqC,IAAIJ,UAAUK;YAAYC,OAAON,UAAUM;UAAM,GAAA;;;;;;AAC1E;QACF;AAEA,cAAMJ;MACR;IACF;AAEA,QAAIF,UAAUM,UAAU,eAAe;AACrC,YAAM,IAAIC,MAAM,mCAAA;IAClB;AAEA,WAAOP;EACT;EAEQ/C,qCAAqCL,KAAc;AACzD,UAAMoD,YAAY,KAAK5D,MAAMwC,OAAM;AACnC,SAAKvC,sBAAsB2D;AAC3BpD,QAAI4D,UAAU,MAAA;AACZR,gBAAUxB,OAAM;AAChB,WAAKnC,sBAAsB;IAC7B,CAAA;EACF;EAEQkB,eAAe0B,QAA6B;AAClDA,WAAOH,OAAO,CAACC,WAAAA;AACbA,aAAOzB,WAAW;QAAEN,UAAU,KAAKb,UAAUsE,MAAK;MAAG;AACrD1B,aAAOzB,OAAON,WAAW,KAAKb,UAAUsE,MAAK;IAC/C,CAAA;EACF;EAEA,MAAcV,4BAA4Bd,QAA6BxB,UAAkB;AACvF,QAAI;AACF,YAAMwB,OAAO7B,IAAI;QAAC;OAAQ;AAC1B,YAAMoC,UAAU;QAAE/B;QAAUiD,QAAQzB,OAAOD;MAAI;AAC/C,UAAI,KAAKtC,uBAAuBiE,cAAa,MAAO,GAAG;AACrD3E,QAAAA,KAAI+B,KAAK,oDAAoDyB,SAAAA;;;;;;AAC7D;MACF;AACA,YAAMoB,kBAAkB,KAAKtE,uBAAuBoD,IAAIjC,QAAAA;AACxD,UAAImD,iBAAiB5B,QAAQC,OAAOD,KAAK;AACvChD,QAAAA,KAAI2D,KAAK,sEAAsEH,SAAAA;;;;;;AAC/E;MACF;AACA,WAAK9C,uBAAuBmE,KAAK;QAAE5B;QAAQxB;MAAS,CAAA;IACtD,SAASyC,KAAK;AACZ,YAAMY,qBAAqB,KAAKpE,uBAAuBiE,cAAa,IAAK;AACzE3E,MAAAA,KAAI2D,KAAK,6BAA6B;QACpClC;QACA8B,cAAcN,OAAOD;QACrB+B,cAAcD;QACdZ;MACF,GAAA;;;;;;AACA,UAAIY,oBAAoB;AACtB,cAAM,KAAKf,4BAA4Bd,QAAQxB,QAAAA;MACjD;IACF;EACF;AACF;",
|
|
6
|
-
"names": ["next", "automerge", "getHeads", "Repo", "IndexedDBStorageAdapter", "Context", "PublicKey", "log", "idCodec", "StorageType", "trace", "ComplexMap", "ComplexSet", "defaultMap", "mapValues", "arrayToBuffer", "bufferToArray", "AutomergeStorageAdapter", "constructor", "_directory", "_state", "load", "key", "undefined", "filename", "_getFilename", "file", "getOrCreateFile", "size", "stat", "buffer", "read", "bufferToArray", "save", "data", "write", "arrayToBuffer", "truncate", "length", "flush", "remove", "destroy", "loadRange", "keyPrefix", "entries", "list", "Promise", "all", "filter", "entry", "startsWith", "map", "_getKeyFromFilename", "removeRange", "close", "k", "replaceAll", "join", "split", "AutomergeStorageWrapper", "constructor", "storage", "callbacks", "_storage", "_callbacks", "load", "key", "save", "value", "beforeSave", "afterSave", "remove", "loadRange", "keyPrefix", "removeRange", "close", "AutomergeStorageAdapter", "Trigger", "NetworkAdapter", "cbor", "Stream", "invariant", "log", "LocalHostNetworkAdapter", "_peers", "Map", "_connected", "ready", "emit", "network", "connect", "peerId", "wake", "send", "message", "peer", "get", "targetId", "close", "forEach", "disconnect", "syncRepo", "id", "syncMessage", "_getPeerId", "next", "has", "set", "connected", "encode", "delete", "wait", "timeout", "then", "peerMetadata", "catch", "err", "sendSyncMessage", "decode", "getHostInfo", "Trigger", "NetworkAdapter", "cbor", "invariant", "log", "AutomergeReplicator", "MeshNetworkAdapter", "_extensions", "Map", "_connected", "ready", "emit", "network", "connect", "peerId", "wake", "send", "message", "receiverId", "targetId", "extension", "get", "sendSyncMessage", "payload", "encode", "catch", "err", "disconnect", "createExtension", "peerInfo", "onStartReplication", "info", "remotePeerId", "wait", "id", "thisPeerId", "toHex", "has", "set", "peerMetadata", "dxos_deviceKey", "onSyncMessage", "decode", "onClose", "delete", "AutomergeHost", "constructor", "directory", "metadata", "_ctx", "Context", "_authorizedDevices", "ComplexMap", "PublicKey", "hash", "_updatingMetadata", "Map", "_requestedDocs", "Set", "_metadata", "_meshNetwork", "MeshNetworkAdapter", "_clientNetwork", "LocalHostNetworkAdapter", "_storage", "AutomergeStorageWrapper", "storage", "type", "StorageType", "IDB", "IndexedDBStorageAdapter", "path", "AutomergeStorageAdapter", "callbacks", "beforeSave", "params", "_beforeSave", "_peerId", "random", "toHex", "_repo", "Repo", "peerId", "network", "sharePolicy", "documentId", "startsWith", "doc", "handles", "docSync", "isRequested", "has", "log", "spaceKey", "getSpaceKeyFromDoc", "authorizedDevices", "get", "from", "deviceKeyHex", "repo", "peerMetadataByPeerId", "dxos_deviceKey", "deviceKey", "isAuthorized", "localPeer", "remotePeer", "err", "catch", "ready", "listener", "handle", "_onDocument", "on", "onDispose", "off", "id", "event", "_onUpdate", "objectIds", "getInlineChanges", "length", "heads", "getHeads", "lastAvailableHash", "join", "encodedIds", "map", "objectId", "idCodec", "encode", "idToLastHash", "markingDirtyPromise", "markDirty", "then", "delete", "disposed", "set", "_automergeDocs", "mapValues", "state", "hasDoc", "automerge", "data", "value", "key", "
|
|
3
|
+
"sources": ["../../../src/automerge/automerge-host.ts", "../../../src/automerge/automerge-storage-adapter.ts", "../../../src/automerge/automerge-storage\u2013wrapper.ts", "../../../src/automerge/local-host-network-adapter.ts", "../../../src/automerge/mesh-network-adapter.ts", "../../../src/automerge/automerge-doc-loader.ts", "../../../src/automerge/reference.ts"],
|
|
4
|
+
"sourcesContent": ["//\n// Copyright 2023 DXOS.org\n//\n\nimport { next as automerge, getHeads } from '@dxos/automerge/automerge';\nimport {\n Repo,\n type PeerId,\n type DocumentId,\n type StorageKey,\n type DocHandle,\n type DocHandleChangePayload,\n} from '@dxos/automerge/automerge-repo';\nimport { IndexedDBStorageAdapter } from '@dxos/automerge/automerge-repo-storage-indexeddb';\nimport { type Stream } from '@dxos/codec-protobuf';\nimport { Context } from '@dxos/context';\nimport { PublicKey } from '@dxos/keys';\nimport { log } from '@dxos/log';\nimport { idCodec } from '@dxos/protocols';\nimport {\n type FlushRequest,\n type HostInfo,\n type SyncRepoRequest,\n type SyncRepoResponse,\n} from '@dxos/protocols/proto/dxos/echo/service';\nimport { StorageType, type Directory } from '@dxos/random-access-storage';\nimport { type AutomergeReplicator } from '@dxos/teleport-extension-automerge-replicator';\nimport { trace } from '@dxos/tracing';\nimport { ComplexMap, ComplexSet, defaultMap, mapValues } from '@dxos/util';\n\nimport { AutomergeStorageAdapter } from './automerge-storage-adapter';\nimport { AutomergeStorageWrapper } from './automerge-storage–wrapper';\nimport { LocalHostNetworkAdapter } from './local-host-network-adapter';\nimport { MeshNetworkAdapter } from './mesh-network-adapter';\n\nexport type { DocumentId };\n\nexport interface MetadataMethods {\n markDirty(idToLastHash: Map<string, string>): Promise<void>;\n}\n\nexport type AutomergeHostParams = {\n directory: Directory;\n metadata?: MetadataMethods;\n};\n\n@trace.resource()\nexport class AutomergeHost {\n private readonly _ctx = new Context();\n private readonly _repo: Repo;\n private readonly _meshNetwork: MeshNetworkAdapter;\n private readonly _clientNetwork: LocalHostNetworkAdapter;\n private readonly _storage: AutomergeStorageWrapper;\n\n @trace.info()\n private readonly _peerId: string;\n\n /**\n * spaceKey -> deviceKey[]\n */\n private readonly _authorizedDevices = new ComplexMap<PublicKey, ComplexSet<PublicKey>>(PublicKey.hash);\n\n private readonly _updatingMetadata = new Map<string, Promise<void>>();\n private readonly _metadata?: MetadataMethods;\n\n public _requestedDocs = new Set<string>();\n\n constructor({ directory, metadata }: AutomergeHostParams) {\n this._metadata = metadata;\n this._meshNetwork = new MeshNetworkAdapter();\n this._clientNetwork = new LocalHostNetworkAdapter();\n\n this._storage = new AutomergeStorageWrapper({\n storage:\n // TODO(mykola): Delete specific handling of IDB storage.\n directory.type === StorageType.IDB\n ? new IndexedDBStorageAdapter(directory.path, 'data')\n : new AutomergeStorageAdapter(directory),\n callbacks: { beforeSave: (params) => this._beforeSave(params) },\n });\n this._peerId = `host-${PublicKey.random().toHex()}` as PeerId;\n this._repo = new Repo({\n peerId: this._peerId as PeerId,\n network: [this._clientNetwork, this._meshNetwork],\n storage: this._storage,\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 sharePolicy: async (peerId /* device key */, documentId /* space key */) => {\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 doc = this._repo.handles[documentId]?.docSync();\n if (!doc) {\n const isRequested = this._requestedDocs.has(`automerge:${documentId}`);\n log('doc share policy check', { peerId, documentId, isRequested });\n return isRequested;\n }\n\n try {\n const spaceKey = getSpaceKeyFromDoc(doc);\n if (!spaceKey) {\n log('space key not found for share policy check', { peerId, documentId });\n return false;\n }\n\n const authorizedDevices = this._authorizedDevices.get(PublicKey.from(spaceKey));\n\n // TODO(mykola): Hack, stop abusing `peerMetadata` field.\n const deviceKeyHex = (this.repo.peerMetadataByPeerId[peerId] as any)?.dxos_deviceKey;\n if (!deviceKeyHex) {\n log('device key not found for share policy check', { peerId, documentId });\n return false;\n }\n const deviceKey = PublicKey.from(deviceKeyHex);\n\n const isAuthorized = authorizedDevices?.has(deviceKey) ?? false;\n log('share policy check', {\n localPeer: this._peerId,\n remotePeer: peerId,\n documentId,\n deviceKey,\n spaceKey,\n isAuthorized,\n });\n return isAuthorized;\n } catch (err) {\n log.catch(err);\n return false;\n }\n },\n });\n this._clientNetwork.ready();\n this._meshNetwork.ready();\n\n {\n const listener = ({ handle }: { handle: DocHandle<any> }) => this._onDocument(handle);\n this._repo.on('document', listener);\n this._ctx.onDispose(() => {\n this._repo.off('document', listener);\n Object.values(this._repo.handles).forEach((handle) => handle.off('change'));\n });\n }\n }\n\n get repo(): Repo {\n return this._repo;\n }\n\n private async _beforeSave(path: StorageKey) {\n const id = path[0];\n if (this._updatingMetadata.has(id)) {\n return this._updatingMetadata.get(id);\n }\n }\n\n private _onDocument(handle: DocHandle<any>) {\n const listener = (event: DocHandleChangePayload<any>) => this._onUpdate(event);\n handle.on('change', listener);\n }\n\n private _onUpdate(event: DocHandleChangePayload<any>) {\n if (this._metadata == null) {\n return;\n }\n\n const objectIds = getInlineChanges(event);\n if (objectIds.length === 0) {\n return;\n }\n\n const heads = getHeads(event.doc);\n const lastAvailableHash = heads.join('');\n if (!lastAvailableHash) {\n return;\n }\n\n const encodedIds = objectIds.map((objectId) => idCodec.encode({ documentId: event.handle.documentId, objectId }));\n const idToLastHash = new Map(encodedIds.map((id) => [id, lastAvailableHash]));\n const markingDirtyPromise = this._metadata\n .markDirty(idToLastHash)\n .then(() => {\n this._updatingMetadata.delete(event.handle.documentId);\n })\n .catch((err: Error) => {\n this._ctx.disposed && log.catch(err);\n });\n this._updatingMetadata.set(event.handle.documentId, markingDirtyPromise);\n }\n\n @trace.info({ depth: null })\n private _automergeDocs() {\n return mapValues(this._repo.handles, (handle) => ({\n state: handle.state,\n hasDoc: !!handle.docSync(),\n heads: handle.docSync() ? automerge.getHeads(handle.docSync()) : null,\n data:\n handle.docSync() &&\n mapValues(handle.docSync(), (value, key) => {\n try {\n switch (key) {\n case 'access':\n case 'links':\n return value;\n case 'objects':\n return Object.keys(value as any);\n default:\n return `${value}`;\n }\n } catch (err) {\n return `${err}`;\n }\n }),\n }));\n }\n\n @trace.info({ depth: null })\n private _automergePeers() {\n return this._repo.peers;\n }\n\n async close() {\n await this._storage.close();\n await this._clientNetwork.close();\n await this._ctx.dispose();\n }\n\n //\n // Methods for client-services.\n //\n\n async flush({ documentIds }: FlushRequest): Promise<void> {\n // Note: Wait for all requested documents to be loaded/synced from thin-client.\n await Promise.all(documentIds?.map((id) => this._repo.find(id as DocumentId).whenReady()) ?? []);\n await this._repo.flush(documentIds as DocumentId[]);\n }\n\n syncRepo(request: SyncRepoRequest): Stream<SyncRepoResponse> {\n return this._clientNetwork.syncRepo(request);\n }\n\n sendSyncMessage(request: SyncRepoRequest): Promise<void> {\n return this._clientNetwork.sendSyncMessage(request);\n }\n\n async getHostInfo(): Promise<HostInfo> {\n return this._clientNetwork.getHostInfo();\n }\n\n //\n // Mesh replication.\n //\n\n createExtension(): AutomergeReplicator {\n return this._meshNetwork.createExtension();\n }\n\n authorizeDevice(spaceKey: PublicKey, deviceKey: PublicKey) {\n log('authorizeDevice', { spaceKey, deviceKey });\n defaultMap(this._authorizedDevices, spaceKey, () => new ComplexSet(PublicKey.hash)).add(deviceKey);\n }\n}\n\n// TODO(mykola): Reconcile with `getInlineAndLinkChanges` in AutomergeDB.\nconst getInlineChanges = (event: DocHandleChangePayload<any>) => {\n const inlineChangedObjectIds = new Set<string>();\n for (const { path } of event.patches) {\n if (path.length < 2) {\n continue;\n }\n switch (path[0]) {\n case 'objects':\n if (path.length >= 2) {\n inlineChangedObjectIds.add(path[1]);\n }\n break;\n }\n }\n return [...inlineChangedObjectIds];\n};\n\nexport const getSpaceKeyFromDoc = (doc: any): 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.experimental_spaceKey;\n if (rawSpaceKey == null) {\n return null;\n }\n\n return String(rawSpaceKey);\n};\n", "//\n// Copyright 2024 DXOS.org\n//\n//\n// Copyright 2023 DXOS.org\n//\n\nimport { type Chunk, type StorageKey, type StorageAdapterInterface } from '@dxos/automerge/automerge-repo';\nimport { type Directory } from '@dxos/random-access-storage';\nimport { arrayToBuffer, bufferToArray } from '@dxos/util';\n\nexport class AutomergeStorageAdapter implements StorageAdapterInterface {\n // TODO(mykola): Hack for restricting automerge Repo to access storage if Host is `closed`.\n // Automerge Repo do not have any lifetime management.\n private _state: 'opened' | 'closed' = 'opened';\n\n constructor(private readonly _directory: Directory) {}\n\n async load(key: StorageKey): Promise<Uint8Array | undefined> {\n if (this._state !== 'opened') {\n return undefined;\n }\n const filename = this._getFilename(key);\n const file = this._directory.getOrCreateFile(filename);\n const { size } = await file.stat();\n if (!size || size === 0) {\n return undefined;\n }\n const buffer = await file.read(0, size);\n return bufferToArray(buffer);\n }\n\n async save(key: StorageKey, data: Uint8Array): Promise<void> {\n if (this._state !== 'opened') {\n return undefined;\n }\n const filename = this._getFilename(key);\n const file = this._directory.getOrCreateFile(filename);\n await file.write(0, arrayToBuffer(data));\n await file.truncate?.(data.length);\n\n await file.flush?.();\n }\n\n async remove(key: StorageKey): Promise<void> {\n if (this._state !== 'opened') {\n return undefined;\n }\n // TODO(dmaretskyi): Better deletion.\n const filename = this._getFilename(key);\n const file = this._directory.getOrCreateFile(filename);\n await file.destroy();\n }\n\n async loadRange(keyPrefix: StorageKey): Promise<Chunk[]> {\n if (this._state !== 'opened') {\n return [];\n }\n const filename = this._getFilename(keyPrefix);\n const entries = await this._directory.list();\n return Promise.all(\n entries\n .filter((entry) => entry.startsWith(filename))\n .map(async (entry): Promise<Chunk> => {\n const file = this._directory.getOrCreateFile(entry);\n const { size } = await file.stat();\n const buffer = await file.read(0, size);\n return {\n key: this._getKeyFromFilename(entry),\n data: bufferToArray(buffer),\n };\n }),\n );\n }\n\n async removeRange(keyPrefix: StorageKey): Promise<void> {\n if (this._state !== 'opened') {\n return undefined;\n }\n const filename = this._getFilename(keyPrefix);\n const entries = await this._directory.list();\n await Promise.all(\n entries\n .filter((entry) => entry.startsWith(filename))\n .map(async (entry): Promise<void> => {\n const file = this._directory.getOrCreateFile(entry);\n await file.destroy();\n }),\n );\n }\n\n async close(): Promise<void> {\n this._state = 'closed';\n }\n\n private _getFilename(key: StorageKey): string {\n return key.map((k) => k.replaceAll('%', '%25').replaceAll('-', '%2D')).join('-');\n }\n\n private _getKeyFromFilename(filename: string): StorageKey {\n return filename.split('-').map((k) => k.replaceAll('%2D', '-').replaceAll('%25', '%'));\n }\n}\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport { type StorageKey, type Chunk, type StorageAdapterInterface } from '@dxos/automerge/automerge-repo';\nimport { type MaybePromise } from '@dxos/util';\n\nimport { AutomergeStorageAdapter } from './automerge-storage-adapter';\n\nexport type StorageCallbacks = {\n beforeSave?: (path: string[]) => MaybePromise<void>;\n afterSave?: (path: string[]) => MaybePromise<void>;\n};\n\nexport type AutomergeStorageWrapperParams = {\n storage: StorageAdapterInterface;\n callbacks: StorageCallbacks;\n};\n\n/**\n * Wrapper for automerge storage that adds callback on save.\n */\nexport class AutomergeStorageWrapper implements StorageAdapterInterface {\n private readonly _storage: StorageAdapterInterface;\n private readonly _callbacks: StorageCallbacks;\n\n constructor({ storage, callbacks }: AutomergeStorageWrapperParams) {\n this._storage = storage;\n this._callbacks = callbacks;\n }\n\n async load(key: StorageKey): Promise<Uint8Array | undefined> {\n return this._storage.load(key);\n }\n\n async save(key: StorageKey, value: Uint8Array): Promise<void> {\n await this._callbacks.beforeSave?.(key);\n await this._storage.save(key, value);\n await this._callbacks.afterSave?.(key);\n }\n\n async remove(key: StorageKey): Promise<void> {\n return this._storage.remove(key);\n }\n\n async loadRange(keyPrefix: StorageKey): Promise<Chunk[]> {\n return this._storage.loadRange(keyPrefix);\n }\n\n async removeRange(keyPrefix: StorageKey): Promise<void> {\n return this._storage.removeRange(keyPrefix);\n }\n\n async close() {\n if (this._storage instanceof AutomergeStorageAdapter) {\n return this._storage.close();\n }\n }\n}\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport { Trigger } from '@dxos/async';\nimport { NetworkAdapter, type Message, type PeerId, cbor } from '@dxos/automerge/automerge-repo';\nimport { Stream } from '@dxos/codec-protobuf';\nimport { invariant } from '@dxos/invariant';\nimport { log } from '@dxos/log';\nimport { type HostInfo, type SyncRepoRequest, type SyncRepoResponse } from '@dxos/protocols/proto/dxos/echo/service';\n\ntype ClientSyncState = {\n connected: boolean;\n send: (message: Message) => void;\n disconnect: () => void;\n};\n\n/**\n * Used to replicate with apps running on the same device.\n */\nexport class LocalHostNetworkAdapter extends NetworkAdapter {\n private readonly _peers: Map<PeerId, ClientSyncState> = new Map();\n\n /**\n * Emits `ready` event. That signals to `Repo` that it can start using the adapter.\n */\n ready() {\n // NOTE: Emitting `ready` event in NetworkAdapter`s constructor causes a race condition\n // because `Repo` waits for `ready` event (which it never receives) before it starts using the adapter.\n this.emit('ready', {\n network: this,\n });\n }\n\n private _connected = new Trigger();\n\n override connect(peerId: PeerId): void {\n this.peerId = peerId;\n this._connected.wake();\n // No-op. Client always connects first\n }\n\n override send(message: Message): void {\n const peer = this._peers.get(message.targetId);\n invariant(peer, 'Peer not found.');\n peer.send(message);\n }\n\n async close() {\n this._peers.forEach((peer) => peer.disconnect());\n this.emit('close');\n }\n\n override disconnect(): void {\n // TODO(mykola): `disconnect` is not used anywhere in `Repo` from `@automerge/automerge-repo`. Should we remove it?\n // No-op\n }\n\n syncRepo({ id, syncMessage }: SyncRepoRequest): Stream<SyncRepoResponse> {\n const peerId = this._getPeerId(id);\n\n return new Stream(({ next, close }) => {\n invariant(!this._peers.has(peerId), 'Peer already connected.');\n this._peers.set(peerId, {\n connected: true,\n send: (message) => {\n next({\n syncMessage: cbor.encode(message),\n });\n },\n disconnect: () => {\n this._peers.delete(peerId);\n close();\n this.emit('peer-disconnected', {\n peerId,\n });\n },\n });\n\n this._connected\n .wait({ timeout: 1_000 })\n .then(() => {\n this.emit('peer-candidate', {\n peerMetadata: {},\n peerId,\n });\n })\n .catch((err) => log.catch(err));\n });\n }\n\n async sendSyncMessage({ id, syncMessage }: SyncRepoRequest): Promise<void> {\n await this._connected.wait({ timeout: 1_000 });\n const message = cbor.decode(syncMessage!) as Message;\n this.emit('message', message);\n }\n\n async getHostInfo(): Promise<HostInfo> {\n await this._connected.wait({ timeout: 1_000 });\n invariant(this.peerId, 'Peer id not set.');\n return {\n peerId: this.peerId,\n };\n }\n\n private _getPeerId(id: string): PeerId {\n return id as PeerId;\n }\n}\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport { Trigger } from '@dxos/async';\nimport { NetworkAdapter, type Message, type PeerId, cbor } from '@dxos/automerge/automerge-repo';\nimport { invariant } from '@dxos/invariant';\nimport { log } from '@dxos/log';\nimport { type PeerInfo } from '@dxos/protocols/proto/dxos/mesh/teleport/automerge';\nimport { AutomergeReplicator } from '@dxos/teleport-extension-automerge-replicator';\n\n/**\n * Used to replicate with other peers over the network.\n */\nexport class MeshNetworkAdapter extends NetworkAdapter {\n private readonly _extensions: Map<string, AutomergeReplicator> = new Map();\n private _connected = new Trigger();\n\n /**\n * Emits `ready` event. That signals to `Repo` that it can start using the adapter.\n */\n ready() {\n // NOTE: Emitting `ready` event in NetworkAdapter`s constructor causes a race condition\n // because `Repo` waits for `ready` event (which it never receives) before it starts using the adapter.\n this.emit('ready', {\n network: this,\n });\n }\n\n override connect(peerId: PeerId): void {\n this.peerId = peerId;\n this._connected.wake();\n }\n\n override send(message: Message): void {\n const receiverId = message.targetId;\n const extension = this._extensions.get(receiverId);\n invariant(extension, 'Extension not found.');\n extension.sendSyncMessage({ payload: cbor.encode(message) }).catch((err) => log.catch(err));\n }\n\n override disconnect(): void {\n // No-op\n }\n\n createExtension(): AutomergeReplicator {\n invariant(this.peerId, 'Peer id not set.');\n\n let peerInfo: PeerInfo;\n const extension = new AutomergeReplicator(\n {\n peerId: this.peerId,\n },\n {\n onStartReplication: async (info, remotePeerId /** Teleport ID */) => {\n await this._connected.wait();\n\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 log('onStartReplication', { id: info.id, thisPeerId: this.peerId, remotePeerId: remotePeerId.toHex() });\n if (!this._extensions.has(info.id)) {\n peerInfo = info;\n // TODO(mykola): Fix race condition?\n this._extensions.set(info.id, extension);\n\n log('peer-candidate', { id: info.id, thisPeerId: this.peerId, remotePeerId: remotePeerId.toHex() });\n this.emit('peer-candidate', {\n // TODO(mykola): Hack, stop abusing `peerMetadata` field.\n peerMetadata: {\n dxos_deviceKey: remotePeerId.toHex(),\n } as any,\n peerId: info.id as PeerId,\n });\n }\n },\n onSyncMessage: async ({ payload }) => {\n if (!peerInfo) {\n return;\n }\n const message = cbor.decode(payload) as Message;\n // Note: automerge Repo dedup messages.\n this.emit('message', message);\n },\n onClose: async () => {\n if (!peerInfo) {\n return;\n }\n this.emit('peer-disconnected', {\n peerId: peerInfo.id as PeerId,\n });\n this._extensions.delete(peerInfo.id);\n },\n },\n );\n return extension;\n }\n}\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport { Event } from '@dxos/async';\nimport { type DocHandle, type AutomergeUrl, type DocumentId, type Repo } from '@dxos/automerge/automerge-repo';\nimport { cancelWithContext, type Context } from '@dxos/context';\nimport { warnAfterTimeout } from '@dxos/debug';\nimport { invariant } from '@dxos/invariant';\nimport { type PublicKey } from '@dxos/keys';\nimport { log } from '@dxos/log';\n\nimport { type SpaceState, type SpaceDoc } from './types';\n\ntype SpaceDocumentLinks = SpaceDoc['links'];\n\nexport interface AutomergeDocumentLoader {\n onObjectDocumentLoaded: Event<ObjectDocumentLoaded>;\n\n getAllHandles(): DocHandle<SpaceDoc>[];\n\n loadSpaceRootDocHandle(ctx: Context, spaceState: SpaceState): Promise<void>;\n loadObjectDocument(objectId: string): void;\n getSpaceRootDocHandle(): DocHandle<SpaceDoc>;\n createDocumentForObject(objectId: string): DocHandle<SpaceDoc>;\n onObjectLinksUpdated(links: SpaceDocumentLinks): void;\n onObjectBoundToDocument(handle: DocHandle<SpaceDoc>, objectId: string): void;\n\n /**\n * @returns objectIds for which we had document handles or were loading one.\n */\n clearHandleReferences(): string[];\n}\n\n/**\n * Manages object <-> docHandle binding and automerge document loading.\n */\nexport class AutomergeDocumentLoaderImpl implements AutomergeDocumentLoader {\n private _spaceRootDocHandle: DocHandle<SpaceDoc> | null = null;\n /**\n * An object id pointer to a handle of the document where the object is stored inline.\n */\n private readonly _objectDocumentHandles = new Map<string, DocHandle<SpaceDoc>>();\n /**\n * If object was requested via loadObjectDocument but root document links weren't updated yet\n * loading will be triggered in onObjectLinksUpdated callback.\n */\n private readonly _objectsPendingDocumentLoad = new Set<string>();\n\n public readonly onObjectDocumentLoaded = new Event<ObjectDocumentLoaded>();\n\n constructor(\n private readonly _spaceKey: PublicKey,\n private readonly _repo: Repo,\n ) {}\n\n getAllHandles(): DocHandle<SpaceDoc>[] {\n return [...new Set(this._objectDocumentHandles.values())];\n }\n\n public async loadSpaceRootDocHandle(ctx: Context, spaceState: SpaceState): Promise<void> {\n if (this._spaceRootDocHandle != null) {\n return;\n }\n if (!spaceState.rootUrl) {\n log.error('Database opened with no rootUrl', { spaceKey: this._spaceKey });\n this._createContextBoundSpaceRootDocument(ctx);\n } else {\n const existingDocHandle = await this._initDocHandle(ctx, spaceState.rootUrl);\n const doc = existingDocHandle.docSync();\n invariant(doc);\n if (doc.access == null) {\n this._initDocAccess(existingDocHandle);\n }\n this._spaceRootDocHandle = existingDocHandle;\n }\n }\n\n public loadObjectDocument(objectId: string) {\n invariant(this._spaceRootDocHandle);\n if (this._objectDocumentHandles.has(objectId) || this._objectsPendingDocumentLoad.has(objectId)) {\n return;\n }\n const spaceRootDoc = this._spaceRootDocHandle.docSync();\n invariant(spaceRootDoc);\n const documentUrl = (spaceRootDoc.links ?? {})[objectId];\n if (documentUrl == null) {\n this._objectsPendingDocumentLoad.add(objectId);\n log.info('loading delayed until object links are initialized', { objectId });\n return;\n }\n this._loadLinkedObjects({ [objectId]: documentUrl });\n }\n\n public onObjectLinksUpdated(links: SpaceDocumentLinks) {\n if (!links) {\n return;\n }\n const linksAwaitingLoad = Object.entries(links).filter(([objectId]) =>\n this._objectsPendingDocumentLoad.has(objectId),\n );\n this._loadLinkedObjects(Object.fromEntries(linksAwaitingLoad));\n linksAwaitingLoad.forEach(([objectId]) => this._objectsPendingDocumentLoad.delete(objectId));\n }\n\n public getSpaceRootDocHandle(): DocHandle<SpaceDoc> {\n invariant(this._spaceRootDocHandle);\n return this._spaceRootDocHandle;\n }\n\n public createDocumentForObject(objectId: string): DocHandle<SpaceDoc> {\n invariant(this._spaceRootDocHandle);\n const spaceDocHandle = this._repo.create<SpaceDoc>();\n this._initDocAccess(spaceDocHandle);\n this.onObjectBoundToDocument(spaceDocHandle, objectId);\n this._spaceRootDocHandle.change((newDoc: SpaceDoc) => {\n newDoc.links ??= {};\n newDoc.links[objectId] = spaceDocHandle.url;\n });\n return spaceDocHandle;\n }\n\n public onObjectBoundToDocument(handle: DocHandle<SpaceDoc>, objectId: string) {\n this._objectDocumentHandles.set(objectId, handle);\n }\n\n public clearHandleReferences(): string[] {\n const objectsWithHandles = [...this._objectDocumentHandles.keys()];\n this._objectDocumentHandles.clear();\n this._spaceRootDocHandle = null;\n return objectsWithHandles;\n }\n\n private _loadLinkedObjects(links: SpaceDocumentLinks) {\n if (!links) {\n return;\n }\n for (const [objectId, automergeUrl] of Object.entries(links)) {\n const logMeta = { objectId, automergeUrl };\n const objectDocumentHandle = this._objectDocumentHandles.get(objectId);\n if (objectDocumentHandle != null && objectDocumentHandle.url !== automergeUrl) {\n log.warn('object already inlined in a different document, ignoring the link', {\n ...logMeta,\n actualDocumentUrl: objectDocumentHandle.url,\n });\n continue;\n }\n if (objectDocumentHandle?.url === automergeUrl) {\n log.warn('object document was already loaded', logMeta);\n continue;\n }\n const handle = this._repo.find<SpaceDoc>(automergeUrl as DocumentId);\n log.debug('document loading triggered', logMeta);\n this._objectDocumentHandles.set(objectId, handle);\n void this._createObjectOnDocumentLoad(handle, objectId);\n }\n }\n\n private async _initDocHandle(ctx: Context, url: string) {\n const docHandle = this._repo.find<SpaceDoc>(url as DocumentId);\n while (true) {\n try {\n await warnAfterTimeout(5_000, 'Automerge root doc load timeout (AutomergeDb)', async () => {\n await cancelWithContext(ctx, docHandle.whenReady()); // TODO(dmaretskyi): Temporary 5s timeout for debugging.\n });\n break;\n } catch (err) {\n if (`${err}`.includes('Timeout')) {\n log.info('wraparound', { id: docHandle.documentId, state: docHandle.state });\n continue;\n }\n\n throw err;\n }\n }\n\n if (docHandle.state === 'unavailable') {\n throw new Error('Automerge document is unavailable');\n }\n\n return docHandle;\n }\n\n private _createContextBoundSpaceRootDocument(ctx: Context) {\n const docHandle = this._repo.create<SpaceDoc>();\n this._spaceRootDocHandle = docHandle;\n ctx.onDispose(() => {\n docHandle.delete();\n this._spaceRootDocHandle = null;\n });\n }\n\n private _initDocAccess(handle: DocHandle<SpaceDoc>) {\n handle.change((newDoc: SpaceDoc) => {\n newDoc.access ??= { spaceKey: this._spaceKey.toHex() };\n newDoc.access.spaceKey = this._spaceKey.toHex();\n });\n }\n\n private async _createObjectOnDocumentLoad(handle: DocHandle<SpaceDoc>, objectId: string) {\n try {\n await handle.doc(['ready']);\n const logMeta = { objectId, docUrl: handle.url };\n if (this.onObjectDocumentLoaded.listenerCount() === 0) {\n log.info('document loaded after all listeners were removed', logMeta);\n return;\n }\n const objectDocHandle = this._objectDocumentHandles.get(objectId);\n if (objectDocHandle?.url !== handle.url) {\n log.warn('object was rebound while a document was loading, discarding handle', logMeta);\n return;\n }\n this.onObjectDocumentLoaded.emit({ handle, objectId });\n } catch (err) {\n const shouldRetryLoading = this.onObjectDocumentLoaded.listenerCount() > 0;\n log.warn('failed to load a document', {\n objectId,\n automergeUrl: handle.url,\n retryLoading: shouldRetryLoading,\n err,\n });\n if (shouldRetryLoading) {\n await this._createObjectOnDocumentLoad(handle, objectId);\n }\n }\n }\n}\n\nexport interface ObjectDocumentLoaded {\n handle: DocHandle<SpaceDoc>;\n objectId: string;\n}\n\nexport interface DocumentChanges {\n createdObjectIds: string[];\n updatedObjectIds: string[];\n objectsToRebind: string[];\n linkedDocuments: {\n [echoId: string]: AutomergeUrl;\n };\n}\n", "//\n// Copyright 2024 DXOS.org\n//\n\nimport { Reference } from '@dxos/echo-db';\n\nexport const REFERENCE_TYPE_TAG = 'dxos.echo.model.document.Reference';\n\n/**\n * Reference as it is stored in Automerge document.\n */\nexport type EncodedReferenceObject = {\n '@type': typeof REFERENCE_TYPE_TAG;\n itemId: string | null;\n protocol: string | null;\n host: string | null;\n};\n\nexport const encodeReference = (reference: Reference): EncodedReferenceObject => ({\n '@type': REFERENCE_TYPE_TAG,\n // NOTE: Automerge do not support undefined values, so we need to use null instead.\n itemId: reference.itemId ?? null,\n protocol: reference.protocol ?? null,\n host: reference.host ?? null,\n});\n\nexport const decodeReference = (value: any) =>\n new Reference(value.itemId, value.protocol ?? undefined, value.host ?? undefined);\n\nexport const isEncodedReferenceObject = (value: any): value is EncodedReferenceObject =>\n typeof value === 'object' && value !== null && value['@type'] === REFERENCE_TYPE_TAG;\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
|
+
"names": ["next", "automerge", "getHeads", "Repo", "IndexedDBStorageAdapter", "Context", "PublicKey", "log", "idCodec", "StorageType", "trace", "ComplexMap", "ComplexSet", "defaultMap", "mapValues", "arrayToBuffer", "bufferToArray", "AutomergeStorageAdapter", "constructor", "_directory", "_state", "load", "key", "undefined", "filename", "_getFilename", "file", "getOrCreateFile", "size", "stat", "buffer", "read", "bufferToArray", "save", "data", "write", "arrayToBuffer", "truncate", "length", "flush", "remove", "destroy", "loadRange", "keyPrefix", "entries", "list", "Promise", "all", "filter", "entry", "startsWith", "map", "_getKeyFromFilename", "removeRange", "close", "k", "replaceAll", "join", "split", "AutomergeStorageWrapper", "constructor", "storage", "callbacks", "_storage", "_callbacks", "load", "key", "save", "value", "beforeSave", "afterSave", "remove", "loadRange", "keyPrefix", "removeRange", "close", "AutomergeStorageAdapter", "Trigger", "NetworkAdapter", "cbor", "Stream", "invariant", "log", "LocalHostNetworkAdapter", "_peers", "Map", "_connected", "ready", "emit", "network", "connect", "peerId", "wake", "send", "message", "peer", "get", "targetId", "close", "forEach", "disconnect", "syncRepo", "id", "syncMessage", "_getPeerId", "next", "has", "set", "connected", "encode", "delete", "wait", "timeout", "then", "peerMetadata", "catch", "err", "sendSyncMessage", "decode", "getHostInfo", "Trigger", "NetworkAdapter", "cbor", "invariant", "log", "AutomergeReplicator", "MeshNetworkAdapter", "_extensions", "Map", "_connected", "ready", "emit", "network", "connect", "peerId", "wake", "send", "message", "receiverId", "targetId", "extension", "get", "sendSyncMessage", "payload", "encode", "catch", "err", "disconnect", "createExtension", "peerInfo", "onStartReplication", "info", "remotePeerId", "wait", "id", "thisPeerId", "toHex", "has", "set", "peerMetadata", "dxos_deviceKey", "onSyncMessage", "decode", "onClose", "delete", "AutomergeHost", "constructor", "directory", "metadata", "_ctx", "Context", "_authorizedDevices", "ComplexMap", "PublicKey", "hash", "_updatingMetadata", "Map", "_requestedDocs", "Set", "_metadata", "_meshNetwork", "MeshNetworkAdapter", "_clientNetwork", "LocalHostNetworkAdapter", "_storage", "AutomergeStorageWrapper", "storage", "type", "StorageType", "IDB", "IndexedDBStorageAdapter", "path", "AutomergeStorageAdapter", "callbacks", "beforeSave", "params", "_beforeSave", "_peerId", "random", "toHex", "_repo", "Repo", "peerId", "network", "sharePolicy", "documentId", "startsWith", "doc", "handles", "docSync", "isRequested", "has", "log", "spaceKey", "getSpaceKeyFromDoc", "authorizedDevices", "get", "from", "deviceKeyHex", "repo", "peerMetadataByPeerId", "dxos_deviceKey", "deviceKey", "isAuthorized", "localPeer", "remotePeer", "err", "catch", "ready", "listener", "handle", "_onDocument", "on", "onDispose", "off", "Object", "values", "forEach", "id", "event", "_onUpdate", "objectIds", "getInlineChanges", "length", "heads", "getHeads", "lastAvailableHash", "join", "encodedIds", "map", "objectId", "idCodec", "encode", "idToLastHash", "markingDirtyPromise", "markDirty", "then", "delete", "disposed", "set", "_automergeDocs", "mapValues", "state", "hasDoc", "automerge", "data", "value", "key", "keys", "_automergePeers", "peers", "close", "dispose", "flush", "documentIds", "Promise", "all", "find", "whenReady", "syncRepo", "request", "sendSyncMessage", "getHostInfo", "createExtension", "authorizeDevice", "defaultMap", "ComplexSet", "add", "trace", "info", "depth", "resource", "inlineChangedObjectIds", "patches", "rawSpaceKey", "access", "experimental_spaceKey", "String", "Event", "cancelWithContext", "warnAfterTimeout", "invariant", "log", "AutomergeDocumentLoaderImpl", "constructor", "_spaceKey", "_repo", "_spaceRootDocHandle", "_objectDocumentHandles", "Map", "_objectsPendingDocumentLoad", "Set", "onObjectDocumentLoaded", "getAllHandles", "values", "loadSpaceRootDocHandle", "ctx", "spaceState", "rootUrl", "error", "spaceKey", "_createContextBoundSpaceRootDocument", "existingDocHandle", "_initDocHandle", "doc", "docSync", "access", "_initDocAccess", "loadObjectDocument", "objectId", "has", "spaceRootDoc", "documentUrl", "links", "add", "info", "_loadLinkedObjects", "onObjectLinksUpdated", "linksAwaitingLoad", "Object", "entries", "filter", "fromEntries", "forEach", "delete", "getSpaceRootDocHandle", "createDocumentForObject", "spaceDocHandle", "create", "onObjectBoundToDocument", "change", "newDoc", "url", "handle", "set", "clearHandleReferences", "objectsWithHandles", "keys", "clear", "automergeUrl", "logMeta", "objectDocumentHandle", "get", "warn", "actualDocumentUrl", "find", "debug", "_createObjectOnDocumentLoad", "docHandle", "whenReady", "err", "includes", "id", "documentId", "state", "Error", "onDispose", "toHex", "docUrl", "listenerCount", "objectDocHandle", "emit", "shouldRetryLoading", "retryLoading", "Reference", "REFERENCE_TYPE_TAG", "encodeReference", "reference", "itemId", "protocol", "host", "decodeReference", "value", "Reference", "undefined", "isEncodedReferenceObject"]
|
|
7
7
|
}
|