@dxos/echo-protocol 0.8.4-main.b97322e → 0.8.4-main.bd9b33e6c8

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.
@@ -12,13 +12,15 @@ import { visitValues } from "@dxos/util";
12
12
  import { assertArgument } from "@dxos/invariant";
13
13
  import { DXN, LOCAL_SPACE_TAG } from "@dxos/keys";
14
14
  var Reference = class _Reference {
15
- static {
16
- /**
17
- * Protocol references to runtime registered types.
18
- * @deprecated
19
- */
20
- this.TYPE_PROTOCOL = "protobuf";
21
- }
15
+ _objectId;
16
+ _protocol;
17
+ _host;
18
+ _dxn;
19
+ /**
20
+ * Protocol references to runtime registered types.
21
+ * @deprecated
22
+ */
23
+ static TYPE_PROTOCOL = "protobuf";
22
24
  static fromDXN(dxn) {
23
25
  switch (dxn.kind) {
24
26
  case DXN.kind.TYPE:
@@ -33,8 +35,8 @@ var Reference = class _Reference {
33
35
  return new _Reference(dxn.parts[0], void 0, dxn.parts[0], dxn);
34
36
  }
35
37
  }
36
- static fromValue(value) {
37
- return new _Reference(value.objectId, value.protocol, value.host);
38
+ static fromValue(value2) {
39
+ return new _Reference(value2.objectId, value2.protocol, value2.host);
38
40
  }
39
41
  /**
40
42
  * Reference an object in the local space.
@@ -122,28 +124,35 @@ var REFERENCE_TYPE_TAG = "dxos.echo.model.document.Reference";
122
124
  var encodeReference = (reference) => ({
123
125
  "/": reference.toDXN().toString()
124
126
  });
125
- var decodeReference = (value) => {
126
- if (typeof value !== "object" || value === null || typeof value["/"] !== "string") {
127
+ var decodeReference = (value2) => {
128
+ if (typeof value2 !== "object" || value2 === null || typeof value2["/"] !== "string") {
127
129
  throw new Error("Invalid reference");
128
130
  }
129
- const dxnString = value["/"];
131
+ const dxnString = value2["/"];
130
132
  if (dxnString.length % 2 === 0 && dxnString.slice(0, dxnString.length / 2) === dxnString.slice(dxnString.length / 2) && dxnString.includes("dxn:echo")) {
131
133
  throw new Error("Automerge bug detected!");
132
134
  }
133
135
  return Reference.fromDXN(DXN.parse(dxnString));
134
136
  };
135
- var isEncodedReference = (value) => typeof value === "object" && value !== null && Object.keys(value).length === 1 && typeof value["/"] === "string";
137
+ var isEncodedReference = (value2) => typeof value2 === "object" && value2 !== null && Object.keys(value2).length === 1 && typeof value2["/"] === "string";
136
138
  var EncodedReference = Object.freeze({
137
139
  isEncodedReference,
138
- getReferenceString: (value) => {
139
- assertArgument(isEncodedReference(value), "invalid reference");
140
- return value["/"];
140
+ getReferenceString: (value2) => {
141
+ assertArgument(isEncodedReference(value2), "value", "invalid reference");
142
+ return value2["/"];
141
143
  },
142
- toDXN: (value) => {
143
- return DXN.parse(EncodedReference.getReferenceString(value));
144
+ toDXN: (value2) => {
145
+ return DXN.parse(EncodedReference.getReferenceString(value2));
144
146
  },
145
147
  fromDXN: (dxn) => {
146
- return encodeReference(Reference.fromDXN(dxn));
148
+ return {
149
+ "/": dxn.toString()
150
+ };
151
+ },
152
+ fromLegacyTypename: (typename) => {
153
+ return {
154
+ "/": DXN.fromTypename(typename).toString()
155
+ };
147
156
  }
148
157
  });
149
158
 
@@ -216,29 +225,35 @@ var ObjectStructure = Object.freeze({
216
225
  getRelationTarget: (object) => {
217
226
  return object.system?.target;
218
227
  },
228
+ getParent: (object) => {
229
+ return object.system?.parent;
230
+ },
219
231
  /**
220
232
  * @returns All references in the data section of the object.
221
233
  */
222
234
  getAllOutgoingReferences: (object) => {
223
235
  const references = [];
224
- const visit2 = (path, value) => {
225
- if (isEncodedReference(value)) {
236
+ const visit2 = (path, value2) => {
237
+ if (isEncodedReference(value2)) {
226
238
  references.push({
227
239
  path,
228
- reference: value
240
+ reference: value2
229
241
  });
230
242
  } else {
231
- visitValues(value, (value2, key) => visit2([
243
+ visitValues(value2, (value3, key) => visit2([
232
244
  ...path,
233
245
  String(key)
234
- ], value2));
246
+ ], value3));
235
247
  }
236
248
  };
237
- visitValues(object.data, (value, key) => visit2([
249
+ visitValues(object.data, (value2, key) => visit2([
238
250
  String(key)
239
- ], value));
251
+ ], value2));
240
252
  return references;
241
253
  },
254
+ getTags: (object) => {
255
+ return object.meta.tags ?? [];
256
+ },
242
257
  makeObject: ({ type, data, keys }) => {
243
258
  return {
244
259
  system: {
@@ -274,37 +289,63 @@ var ObjectStructure = Object.freeze({
274
289
  var PROPERTY_ID = "id";
275
290
  var DATA_NAMESPACE = "data";
276
291
 
277
- // src/space-doc-version.ts
278
- var SpaceDocVersion = Object.freeze({
292
+ // src/echo-feed-codec.ts
293
+ import { FeedProtocol } from "@dxos/protocols";
294
+ var ATTR_META = "@meta";
295
+ var EchoFeedCodec = class _EchoFeedCodec {
296
+ static #encoder = new TextEncoder();
297
+ static #decoder = new TextDecoder();
279
298
  /**
280
- * For the documents created before the versioning was introduced.
299
+ * Prepares a value for feed storage (strips queue position from metadata) and encodes to bytes.
281
300
  */
282
- LEGACY: 0,
301
+ static encode(value2) {
302
+ const prepared = _EchoFeedCodec.#stripQueuePosition(value2);
303
+ return _EchoFeedCodec.#encoder.encode(JSON.stringify(prepared));
304
+ }
283
305
  /**
284
- * Current version.
306
+ * Decodes feed block bytes to a JSON value.
307
+ * If position is provided, injects queue position into the decoded object's metadata.
285
308
  */
286
- CURRENT: 1
287
- });
288
-
289
- // src/space-id.ts
290
- import { subtleCrypto } from "@dxos/crypto";
291
- import { PublicKey, SpaceId } from "@dxos/keys";
292
- import { ComplexMap } from "@dxos/util";
293
- var SPACE_IDS_CACHE = new ComplexMap(PublicKey.hash);
294
- var createIdFromSpaceKey = async (spaceKey) => {
295
- const cachedValue = SPACE_IDS_CACHE.get(spaceKey);
296
- if (cachedValue !== void 0) {
297
- return cachedValue;
309
+ static decode(data, position) {
310
+ const decoded = JSON.parse(_EchoFeedCodec.#decoder.decode(data));
311
+ if (position !== void 0 && typeof decoded === "object" && decoded !== null) {
312
+ _EchoFeedCodec.#setQueuePosition(decoded, position);
313
+ }
314
+ return decoded;
315
+ }
316
+ static #stripQueuePosition(value2) {
317
+ if (typeof value2 !== "object" || value2 === null) {
318
+ return value2;
319
+ }
320
+ const obj = structuredClone(value2);
321
+ const meta = obj[ATTR_META];
322
+ if (meta?.keys?.some((key) => key.source === FeedProtocol.KEY_QUEUE_POSITION)) {
323
+ meta.keys = meta.keys.filter((key) => key.source !== FeedProtocol.KEY_QUEUE_POSITION);
324
+ }
325
+ return obj;
326
+ }
327
+ static #setQueuePosition(obj, position) {
328
+ obj[ATTR_META] ??= {
329
+ keys: []
330
+ };
331
+ obj[ATTR_META].keys ??= [];
332
+ const keys = obj[ATTR_META].keys;
333
+ for (let i = 0; i < keys.length; i++) {
334
+ if (keys[i].source === FeedProtocol.KEY_QUEUE_POSITION) {
335
+ keys.splice(i, 1);
336
+ i--;
337
+ }
338
+ }
339
+ keys.push({
340
+ source: FeedProtocol.KEY_QUEUE_POSITION,
341
+ id: position.toString()
342
+ });
298
343
  }
299
- const digest = await subtleCrypto.digest("SHA-256", spaceKey.asUint8Array());
300
- const bytes = new Uint8Array(digest).slice(0, SpaceId.byteLength);
301
- const spaceId = SpaceId.encode(bytes);
302
- SPACE_IDS_CACHE.set(spaceKey, spaceId);
303
- return spaceId;
304
344
  };
305
345
 
306
346
  // src/foreign-key.ts
307
- import { Schema, SchemaAST } from "effect";
347
+ import * as Schema from "effect/Schema";
348
+ import * as SchemaAST from "effect/SchemaAST";
308
349
  var ForeignKey_ = Schema.Struct({
309
350
  /**
310
351
  * Name of the foreign database/system.
@@ -315,9 +356,9 @@ var ForeignKey_ = Schema.Struct({
315
356
  * Id within the foreign database.
316
357
  */
317
358
  // TODO(wittjosiah): This annotation is currently used to ensure id field shows up in forms.
318
- // TODO(dmaretskyi): `false` is not a valid value for the annotation.
359
+ // TODO(dmaretskyi): `false` is not a valid value for the annotation. Use a different annotation.
319
360
  id: Schema.String.annotations({
320
- [SchemaAST.IdentifierAnnotationId]: false
361
+ [SchemaAST.IdentifierAnnotationId]: "false"
321
362
  })
322
363
  });
323
364
  var ForeignKey = ForeignKey_;
@@ -328,29 +369,43 @@ __export(ast_exports, {
328
369
  Filter: () => Filter,
329
370
  FilterAnd: () => FilterAnd,
330
371
  FilterCompare: () => FilterCompare,
372
+ FilterContains: () => FilterContains,
331
373
  FilterIn: () => FilterIn,
332
374
  FilterNot: () => FilterNot,
333
375
  FilterObject: () => FilterObject,
334
376
  FilterOr: () => FilterOr,
335
377
  FilterRange: () => FilterRange,
378
+ FilterTag: () => FilterTag,
336
379
  FilterTextSearch: () => FilterTextSearch,
380
+ FilterTimestamp: () => FilterTimestamp,
381
+ Order: () => Order,
382
+ OrderDirection: () => OrderDirection,
337
383
  Query: () => Query,
338
384
  QueryFilterClause: () => QueryFilterClause,
385
+ QueryFromClause: () => QueryFromClause,
386
+ QueryFromClause_: () => QueryFromClause_,
387
+ QueryHierarchyTraversalClause: () => QueryHierarchyTraversalClause,
339
388
  QueryIncomingReferencesClause: () => QueryIncomingReferencesClause,
389
+ QueryLimitClause: () => QueryLimitClause,
340
390
  QueryOptions: () => QueryOptions,
341
391
  QueryOptionsClause: () => QueryOptionsClause,
392
+ QueryOrderClause: () => QueryOrderClause,
342
393
  QueryReferenceTraversalClause: () => QueryReferenceTraversalClause,
343
394
  QueryRelationClause: () => QueryRelationClause,
344
395
  QueryRelationTraversalClause: () => QueryRelationTraversalClause,
345
396
  QuerySelectClause: () => QuerySelectClause,
346
397
  QuerySetDifferenceClause: () => QuerySetDifferenceClause,
347
398
  QueryUnionClause: () => QueryUnionClause,
399
+ Scope: () => Scope,
400
+ fold: () => fold,
401
+ map: () => map,
348
402
  visit: () => visit
349
403
  });
350
- import { Schema as Schema2 } from "effect";
404
+ import * as Match from "effect/Match";
405
+ import * as Schema2 from "effect/Schema";
351
406
  import { DXN as DXN2, ObjectId } from "@dxos/keys";
352
407
  var TypenameSpecifier = Schema2.Union(DXN2.Schema, Schema2.Null).annotations({
353
- description: "DXN or null. Null means any type will match"
408
+ description: "DXN or null; null matches any type"
354
409
  });
355
410
  var FilterObject_ = Schema2.Struct({
356
411
  type: Schema2.Literal("object"),
@@ -383,12 +438,29 @@ var FilterIn_ = Schema2.Struct({
383
438
  values: Schema2.Array(Schema2.Any)
384
439
  });
385
440
  var FilterIn = FilterIn_;
441
+ var FilterContains_ = Schema2.Struct({
442
+ type: Schema2.Literal("contains"),
443
+ value: Schema2.Any
444
+ });
445
+ var FilterContains = FilterContains_;
446
+ var FilterTag_ = Schema2.Struct({
447
+ type: Schema2.Literal("tag"),
448
+ tag: Schema2.String
449
+ });
450
+ var FilterTag = FilterTag_;
386
451
  var FilterRange_ = Schema2.Struct({
387
452
  type: Schema2.Literal("range"),
388
453
  from: Schema2.Any,
389
454
  to: Schema2.Any
390
455
  });
391
456
  var FilterRange = FilterRange_;
457
+ var FilterTimestamp_ = Schema2.Struct({
458
+ type: Schema2.Literal("timestamp"),
459
+ field: Schema2.Literal("createdAt", "updatedAt"),
460
+ operator: Schema2.Literal("gt", "gte", "lt", "lte"),
461
+ value: Schema2.Number
462
+ });
463
+ var FilterTimestamp = FilterTimestamp_;
392
464
  var FilterTextSearch_ = Schema2.Struct({
393
465
  type: Schema2.Literal("text-search"),
394
466
  text: Schema2.String,
@@ -410,7 +482,9 @@ var FilterOr_ = Schema2.Struct({
410
482
  filters: Schema2.Array(Schema2.suspend(() => Filter))
411
483
  });
412
484
  var FilterOr = FilterOr_;
413
- var Filter = Schema2.Union(FilterObject, FilterTextSearch, FilterCompare, FilterIn, FilterRange, FilterNot, FilterAnd, FilterOr);
485
+ var Filter = Schema2.Union(FilterObject, FilterCompare, FilterIn, FilterContains, FilterTag, FilterRange, FilterTimestamp, FilterTextSearch, FilterNot, FilterAnd, FilterOr).annotations({
486
+ identifier: "org.dxos.schema.filter"
487
+ });
414
488
  var QuerySelectClause_ = Schema2.Struct({
415
489
  type: Schema2.Literal("select"),
416
490
  filter: Schema2.suspend(() => Filter)
@@ -431,7 +505,11 @@ var QueryReferenceTraversalClause = QueryReferenceTraversalClause_;
431
505
  var QueryIncomingReferencesClause_ = Schema2.Struct({
432
506
  type: Schema2.Literal("incoming-references"),
433
507
  anchor: Schema2.suspend(() => Query),
434
- property: Schema2.String,
508
+ /**
509
+ * Property path where the reference is located.
510
+ * If null, matches references from any property.
511
+ */
512
+ property: Schema2.NullOr(Schema2.String),
435
513
  typename: TypenameSpecifier
436
514
  });
437
515
  var QueryIncomingReferencesClause = QueryIncomingReferencesClause_;
@@ -453,6 +531,16 @@ var QueryRelationTraversalClause_ = Schema2.Struct({
453
531
  direction: Schema2.Literal("source", "target", "both")
454
532
  });
455
533
  var QueryRelationTraversalClause = QueryRelationTraversalClause_;
534
+ var QueryHierarchyTraversalClause_ = Schema2.Struct({
535
+ type: Schema2.Literal("hierarchy-traversal"),
536
+ anchor: Schema2.suspend(() => Query),
537
+ /**
538
+ * to-parent: traverse from child to parent.
539
+ * to-children: traverse from parent to children.
540
+ */
541
+ direction: Schema2.Literal("to-parent", "to-children")
542
+ });
543
+ var QueryHierarchyTraversalClause = QueryHierarchyTraversalClause_;
456
544
  var QueryUnionClause_ = Schema2.Struct({
457
545
  type: Schema2.Literal("union"),
458
546
  queries: Schema2.Array(Schema2.suspend(() => Query))
@@ -464,50 +552,257 @@ var QuerySetDifferenceClause_ = Schema2.Struct({
464
552
  exclude: Schema2.suspend(() => Query)
465
553
  });
466
554
  var QuerySetDifferenceClause = QuerySetDifferenceClause_;
555
+ var OrderDirection = Schema2.Literal("asc", "desc");
556
+ var Order_ = Schema2.Union(Schema2.Struct({
557
+ // How database wants to order them (in practice - by id).
558
+ kind: Schema2.Literal("natural")
559
+ }), Schema2.Struct({
560
+ kind: Schema2.Literal("property"),
561
+ property: Schema2.String,
562
+ direction: OrderDirection
563
+ }), Schema2.Struct({
564
+ // Order by relevance rank (for FTS/vector search results).
565
+ // Default direction is 'desc' (higher rank = better match first).
566
+ kind: Schema2.Literal("rank"),
567
+ direction: OrderDirection
568
+ }));
569
+ var Order = Order_;
570
+ var QueryOrderClause_ = Schema2.Struct({
571
+ type: Schema2.Literal("order"),
572
+ query: Schema2.suspend(() => Query),
573
+ order: Schema2.Array(Order)
574
+ });
575
+ var QueryOrderClause = QueryOrderClause_;
467
576
  var QueryOptionsClause_ = Schema2.Struct({
468
577
  type: Schema2.Literal("options"),
469
578
  query: Schema2.suspend(() => Query),
470
579
  options: Schema2.suspend(() => QueryOptions)
471
580
  });
472
581
  var QueryOptionsClause = QueryOptionsClause_;
473
- var Query_ = Schema2.Union(QuerySelectClause, QueryFilterClause, QueryReferenceTraversalClause, QueryIncomingReferencesClause, QueryRelationClause, QueryRelationTraversalClause, QueryUnionClause, QuerySetDifferenceClause, QueryOptionsClause);
582
+ var QueryLimitClause_ = Schema2.Struct({
583
+ type: Schema2.Literal("limit"),
584
+ query: Schema2.suspend(() => Query),
585
+ limit: Schema2.Number
586
+ });
587
+ var QueryLimitClause = QueryLimitClause_;
588
+ var QueryFromClause_ = Schema2.Struct({
589
+ type: Schema2.Literal("from"),
590
+ query: Schema2.suspend(() => Query),
591
+ from: Schema2.Union(Schema2.TaggedStruct("scope", {
592
+ scope: Schema2.suspend(() => Scope)
593
+ }), Schema2.TaggedStruct("query", {
594
+ query: Schema2.suspend(() => Query)
595
+ }))
596
+ });
597
+ var QueryFromClause = QueryFromClause_;
598
+ var Query_ = Schema2.Union(QuerySelectClause, QueryFilterClause, QueryReferenceTraversalClause, QueryIncomingReferencesClause, QueryRelationClause, QueryRelationTraversalClause, QueryHierarchyTraversalClause, QueryUnionClause, QuerySetDifferenceClause, QueryOrderClause, QueryOptionsClause, QueryLimitClause, QueryFromClause).annotations({
599
+ identifier: "org.dxos.schema.query"
600
+ });
474
601
  var Query = Query_;
475
602
  var QueryOptions = Schema2.Struct({
476
- spaceIds: Schema2.optional(Schema2.Array(Schema2.String)),
603
+ /**
604
+ * Nested select statements will use this option to filter deleted objects.
605
+ */
477
606
  deleted: Schema2.optional(Schema2.Literal("include", "exclude", "only"))
478
607
  });
608
+ var Scope = Schema2.Struct({
609
+ /**
610
+ * The nested select statemets will select from the given spaces.
611
+ *
612
+ * NOTE: Spaces and queues are unioned together if both are specified.
613
+ */
614
+ spaceIds: Schema2.optional(Schema2.Array(Schema2.String)),
615
+ /**
616
+ * If true, the nested select statements will select from all queues in the spaces specified by `spaceIds`.
617
+ */
618
+ allQueuesFromSpaces: Schema2.optional(Schema2.Boolean),
619
+ /**
620
+ * The nested select statemets will select from the given queues.
621
+ *
622
+ * NOTE: Spaces and queues are unioned together if both are specified.
623
+ */
624
+ queues: Schema2.optional(Schema2.Array(DXN2.Schema))
625
+ });
479
626
  var visit = (query, visitor) => {
480
- switch (query.type) {
481
- case "filter":
482
- visit(query.selection, visitor);
483
- break;
484
- case "reference-traversal":
485
- visit(query.anchor, visitor);
486
- break;
487
- case "incoming-references":
488
- visit(query.anchor, visitor);
489
- break;
490
- case "relation":
491
- visit(query.anchor, visitor);
492
- break;
493
- case "options":
494
- visit(query.query, visitor);
495
- break;
496
- case "relation-traversal":
497
- visit(query.anchor, visitor);
498
- break;
499
- case "union":
500
- query.queries.forEach((q) => visit(q, visitor));
501
- break;
502
- case "set-difference":
503
- visit(query.source, visitor);
504
- visit(query.exclude, visitor);
505
- break;
627
+ visitor(query);
628
+ Match.value(query).pipe(Match.when({
629
+ type: "filter"
630
+ }, ({ selection }) => visit(selection, visitor)), Match.when({
631
+ type: "reference-traversal"
632
+ }, ({ anchor }) => visit(anchor, visitor)), Match.when({
633
+ type: "incoming-references"
634
+ }, ({ anchor }) => visit(anchor, visitor)), Match.when({
635
+ type: "relation"
636
+ }, ({ anchor }) => visit(anchor, visitor)), Match.when({
637
+ type: "options"
638
+ }, ({ query: query2 }) => visit(query2, visitor)), Match.when({
639
+ type: "relation-traversal"
640
+ }, ({ anchor }) => visit(anchor, visitor)), Match.when({
641
+ type: "hierarchy-traversal"
642
+ }, ({ anchor }) => visit(anchor, visitor)), Match.when({
643
+ type: "union"
644
+ }, ({ queries }) => queries.forEach((q) => visit(q, visitor))), Match.when({
645
+ type: "set-difference"
646
+ }, ({ source, exclude }) => {
647
+ visit(source, visitor);
648
+ visit(exclude, visitor);
649
+ }), Match.when({
650
+ type: "order"
651
+ }, ({ query: query2 }) => visit(query2, visitor)), Match.when({
652
+ type: "limit"
653
+ }, ({ query: query2 }) => visit(query2, visitor)), Match.when({
654
+ type: "from"
655
+ }, (node) => {
656
+ visit(node.query, visitor);
657
+ if (node.from._tag === "query") {
658
+ visit(node.from.query, visitor);
659
+ }
660
+ }), Match.when({
661
+ type: "select"
662
+ }, () => {
663
+ }), Match.exhaustive);
664
+ };
665
+ var map = (query, mapper) => {
666
+ const mapped = Match.value(query).pipe(Match.when({
667
+ type: "filter"
668
+ }, (node) => ({
669
+ ...node,
670
+ selection: map(node.selection, mapper)
671
+ })), Match.when({
672
+ type: "reference-traversal"
673
+ }, (node) => ({
674
+ ...node,
675
+ anchor: map(node.anchor, mapper)
676
+ })), Match.when({
677
+ type: "incoming-references"
678
+ }, (node) => ({
679
+ ...node,
680
+ anchor: map(node.anchor, mapper)
681
+ })), Match.when({
682
+ type: "relation"
683
+ }, (node) => ({
684
+ ...node,
685
+ anchor: map(node.anchor, mapper)
686
+ })), Match.when({
687
+ type: "relation-traversal"
688
+ }, (node) => ({
689
+ ...node,
690
+ anchor: map(node.anchor, mapper)
691
+ })), Match.when({
692
+ type: "hierarchy-traversal"
693
+ }, (node) => ({
694
+ ...node,
695
+ anchor: map(node.anchor, mapper)
696
+ })), Match.when({
697
+ type: "options"
698
+ }, (node) => ({
699
+ ...node,
700
+ query: map(node.query, mapper)
701
+ })), Match.when({
702
+ type: "order"
703
+ }, (node) => ({
704
+ ...node,
705
+ query: map(node.query, mapper)
706
+ })), Match.when({
707
+ type: "limit"
708
+ }, (node) => ({
709
+ ...node,
710
+ query: map(node.query, mapper)
711
+ })), Match.when({
712
+ type: "from"
713
+ }, (node) => ({
714
+ ...node,
715
+ query: map(node.query, mapper),
716
+ ...node.from._tag === "query" ? {
717
+ from: {
718
+ _tag: "query",
719
+ query: map(node.from.query, mapper)
720
+ }
721
+ } : {}
722
+ })), Match.when({
723
+ type: "union"
724
+ }, (node) => ({
725
+ ...node,
726
+ queries: node.queries.map((q) => map(q, mapper))
727
+ })), Match.when({
728
+ type: "set-difference"
729
+ }, (node) => ({
730
+ ...node,
731
+ source: map(node.source, mapper),
732
+ exclude: map(node.exclude, mapper)
733
+ })), Match.when({
734
+ type: "select"
735
+ }, (node) => node), Match.exhaustive);
736
+ return mapper(mapped);
737
+ };
738
+ var fold = (query, reducer) => {
739
+ return Match.value(query).pipe(Match.withReturnType(), Match.when({
740
+ type: "filter"
741
+ }, ({ selection }) => fold(selection, reducer)), Match.when({
742
+ type: "reference-traversal"
743
+ }, ({ anchor }) => fold(anchor, reducer)), Match.when({
744
+ type: "incoming-references"
745
+ }, ({ anchor }) => fold(anchor, reducer)), Match.when({
746
+ type: "relation"
747
+ }, ({ anchor }) => fold(anchor, reducer)), Match.when({
748
+ type: "options"
749
+ }, ({ query: query2 }) => fold(query2, reducer)), Match.when({
750
+ type: "relation-traversal"
751
+ }, ({ anchor }) => fold(anchor, reducer)), Match.when({
752
+ type: "hierarchy-traversal"
753
+ }, ({ anchor }) => fold(anchor, reducer)), Match.when({
754
+ type: "union"
755
+ }, ({ queries }) => queries.flatMap((q) => fold(q, reducer))), Match.when({
756
+ type: "set-difference"
757
+ }, ({ source, exclude }) => fold(source, reducer).concat(fold(exclude, reducer))), Match.when({
758
+ type: "order"
759
+ }, ({ query: query2 }) => fold(query2, reducer)), Match.when({
760
+ type: "limit"
761
+ }, ({ query: query2 }) => fold(query2, reducer)), Match.when({
762
+ type: "from"
763
+ }, (node) => {
764
+ const results = fold(node.query, reducer);
765
+ if (node.from._tag === "query") {
766
+ return results.concat(fold(node.from.query, reducer));
767
+ }
768
+ return results;
769
+ }), Match.when({
770
+ type: "select"
771
+ }, () => []), Match.exhaustive);
772
+ };
773
+
774
+ // src/space-doc-version.ts
775
+ var SpaceDocVersion = Object.freeze({
776
+ /**
777
+ * For the documents created before the versioning was introduced.
778
+ */
779
+ LEGACY: 0,
780
+ /**
781
+ * Current version.
782
+ */
783
+ CURRENT: 1
784
+ });
785
+
786
+ // src/space-id.ts
787
+ import { subtleCrypto } from "@dxos/crypto";
788
+ import { PublicKey, SpaceId } from "@dxos/keys";
789
+ import { ComplexMap } from "@dxos/util";
790
+ var SPACE_IDS_CACHE = new ComplexMap(PublicKey.hash);
791
+ var createIdFromSpaceKey = async (spaceKey) => {
792
+ const cachedValue = SPACE_IDS_CACHE.get(spaceKey);
793
+ if (cachedValue !== void 0) {
794
+ return cachedValue;
506
795
  }
796
+ const digest = await subtleCrypto.digest("SHA-256", spaceKey.asUint8Array());
797
+ const bytes = new Uint8Array(digest).slice(0, SpaceId.byteLength);
798
+ const spaceId = SpaceId.encode(bytes);
799
+ SPACE_IDS_CACHE.set(spaceKey, spaceId);
800
+ return spaceId;
507
801
  };
508
802
  export {
509
803
  DATA_NAMESPACE,
510
804
  DatabaseDirectory,
805
+ EchoFeedCodec,
511
806
  EncodedReference,
512
807
  ForeignKey,
513
808
  ObjectStructure,