@dxos/echo 0.8.4-main.8360d9e660 → 0.8.4-main.8baae0fced
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/LICENSE +102 -5
- package/README.md +3 -3
- package/dist/lib/neutral/Annotation.mjs +5 -3
- package/dist/lib/neutral/Database.mjs +8 -4
- package/dist/lib/neutral/Entity.mjs +16 -14
- package/dist/lib/neutral/Err.mjs +1 -1
- package/dist/lib/neutral/Extension.mjs +18 -0
- package/dist/lib/neutral/Extension.mjs.map +7 -0
- package/dist/lib/neutral/Feed.mjs +23 -18
- package/dist/lib/neutral/Filter.mjs +23 -13
- package/dist/lib/neutral/Format.mjs +3 -3
- package/dist/lib/neutral/JsonSchema.mjs +8 -8
- package/dist/lib/neutral/Key.mjs +1 -1
- package/dist/lib/neutral/Migration.mjs +17 -0
- package/dist/lib/neutral/Migration.mjs.map +7 -0
- package/dist/lib/neutral/Obj.mjs +16 -13
- package/dist/lib/neutral/Order.mjs +1 -1
- package/dist/lib/neutral/Query.mjs +19 -17
- package/dist/lib/neutral/QueryResult.mjs +1 -1
- package/dist/lib/neutral/Ref.mjs +9 -7
- package/dist/lib/neutral/Relation.mjs +15 -14
- package/dist/lib/neutral/SchemaRegistry.mjs +1 -1
- package/dist/lib/neutral/Tag.mjs +14 -13
- package/dist/lib/neutral/Type.mjs +10 -10
- package/dist/lib/neutral/{chunk-FZO7LQO7.mjs → chunk-44HT3MEC.mjs} +2 -2
- package/dist/lib/neutral/{chunk-ROAGDPV7.mjs → chunk-4A2GS5LQ.mjs} +12 -8
- package/dist/lib/neutral/chunk-4A2GS5LQ.mjs.map +7 -0
- package/dist/lib/neutral/{chunk-OENWMTE6.mjs → chunk-5SL5LDLD.mjs} +4 -2
- package/dist/lib/neutral/chunk-5SL5LDLD.mjs.map +7 -0
- package/dist/lib/neutral/{chunk-ANHVGJI4.mjs → chunk-7RVZT53K.mjs} +1 -1
- package/dist/lib/neutral/{chunk-YWXWXIE5.mjs → chunk-APHSOTIX.mjs} +2 -2
- package/dist/lib/neutral/{chunk-YWXWXIE5.mjs.map → chunk-APHSOTIX.mjs.map} +2 -2
- package/dist/lib/neutral/{chunk-6DNYDXCV.mjs → chunk-B2P7IVG3.mjs} +61 -14
- package/dist/lib/neutral/chunk-B2P7IVG3.mjs.map +7 -0
- package/dist/lib/neutral/{chunk-BNCCGLJN.mjs → chunk-BICZKPQG.mjs} +1 -1
- package/dist/lib/neutral/{chunk-43Y5DOS6.mjs → chunk-BMB7IHGB.mjs} +16 -66
- package/dist/lib/neutral/chunk-BMB7IHGB.mjs.map +7 -0
- package/dist/lib/neutral/{chunk-6K2MVI2O.mjs → chunk-BUBEC474.mjs} +4 -4
- package/dist/lib/neutral/{chunk-YMNSMKKW.mjs → chunk-C4PSESGN.mjs} +6 -22
- package/dist/lib/neutral/{chunk-YMNSMKKW.mjs.map → chunk-C4PSESGN.mjs.map} +3 -3
- package/dist/lib/neutral/chunk-FIWO2FZK.mjs +36 -0
- package/dist/lib/neutral/chunk-FIWO2FZK.mjs.map +7 -0
- package/dist/lib/neutral/chunk-GWFFC34K.mjs +50 -0
- package/dist/lib/neutral/chunk-GWFFC34K.mjs.map +7 -0
- package/dist/lib/neutral/{chunk-NXMFBIT5.mjs → chunk-HKETO4L4.mjs} +72 -6
- package/dist/lib/neutral/chunk-HKETO4L4.mjs.map +7 -0
- package/dist/lib/neutral/{chunk-NEGC54NE.mjs → chunk-I2DARWPX.mjs} +17 -19
- package/dist/lib/neutral/chunk-I2DARWPX.mjs.map +7 -0
- package/dist/lib/neutral/{chunk-7VNVH63N.mjs → chunk-IVSI7QO6.mjs} +50 -20
- package/dist/lib/neutral/chunk-IVSI7QO6.mjs.map +7 -0
- package/dist/lib/neutral/{chunk-BOZZPUXE.mjs → chunk-MLS7U7AT.mjs} +12 -65
- package/dist/lib/neutral/chunk-MLS7U7AT.mjs.map +7 -0
- package/dist/lib/neutral/{chunk-WYOKA6AE.mjs → chunk-N4B7FHQT.mjs} +2 -2
- package/dist/lib/neutral/{chunk-WYOKA6AE.mjs.map → chunk-N4B7FHQT.mjs.map} +1 -1
- package/dist/lib/neutral/{chunk-FXEG7EOK.mjs → chunk-N7VOEPSV.mjs} +6 -3
- package/dist/lib/neutral/{chunk-FXEG7EOK.mjs.map → chunk-N7VOEPSV.mjs.map} +3 -3
- package/dist/lib/neutral/{chunk-UTBRYVQC.mjs → chunk-QRZ2I3ZM.mjs} +2 -2
- package/dist/lib/neutral/{chunk-SEMVAGBM.mjs → chunk-TNBK56IN.mjs} +19 -24
- package/dist/lib/neutral/chunk-TNBK56IN.mjs.map +7 -0
- package/dist/lib/neutral/{chunk-B5OXLWZL.mjs → chunk-TRPZU2HV.mjs} +2 -2
- package/dist/lib/neutral/{chunk-UI6MWK5W.mjs → chunk-TTCSATUD.mjs} +1 -1
- package/dist/lib/neutral/{chunk-OMUPQMLR.mjs → chunk-V72DY6LU.mjs} +1 -1
- package/dist/lib/neutral/{chunk-6GPU7XC3.mjs → chunk-VW42HESL.mjs} +54 -17
- package/dist/lib/neutral/chunk-VW42HESL.mjs.map +7 -0
- package/dist/lib/neutral/{chunk-4JRI2ZJI.mjs → chunk-X3356HPV.mjs} +120 -12
- package/dist/lib/neutral/chunk-X3356HPV.mjs.map +7 -0
- package/dist/lib/neutral/{chunk-W47JKR3X.mjs → chunk-XEXM5HWQ.mjs} +18 -46
- package/dist/lib/neutral/chunk-XEXM5HWQ.mjs.map +7 -0
- package/dist/lib/neutral/{chunk-C4JZK4J7.mjs → chunk-Z5GKP74O.mjs} +231 -479
- package/dist/lib/neutral/chunk-Z5GKP74O.mjs.map +7 -0
- package/dist/lib/neutral/{chunk-UBEZSGXY.mjs → chunk-ZISMEVKD.mjs} +1 -1
- package/dist/lib/neutral/{chunk-UBEZSGXY.mjs.map → chunk-ZISMEVKD.mjs.map} +2 -2
- package/dist/lib/neutral/index.mjs +38 -28
- package/dist/lib/neutral/internal/index.mjs +15 -9
- package/dist/lib/neutral/meta.json +1 -1
- package/dist/lib/neutral/testing/index.mjs +185 -129
- package/dist/lib/neutral/testing/index.mjs.map +3 -3
- package/dist/types/src/Annotation.d.ts +1 -1
- package/dist/types/src/Annotation.d.ts.map +1 -1
- package/dist/types/src/Collection.d.ts.map +1 -1
- package/dist/types/src/Database.d.ts +14 -2
- package/dist/types/src/Database.d.ts.map +1 -1
- package/dist/types/src/Dataset.d.ts +2 -1
- package/dist/types/src/Dataset.d.ts.map +1 -1
- package/dist/types/src/Entity.d.ts +17 -11
- package/dist/types/src/Entity.d.ts.map +1 -1
- package/dist/types/src/Err.d.ts +18 -18
- package/dist/types/src/Err.d.ts.map +1 -1
- package/dist/types/src/Extension.d.ts +80 -0
- package/dist/types/src/Extension.d.ts.map +1 -0
- package/dist/types/src/Extension.test.d.ts +2 -0
- package/dist/types/src/Extension.test.d.ts.map +1 -0
- package/dist/types/src/Feed.d.ts +62 -21
- package/dist/types/src/Feed.d.ts.map +1 -1
- package/dist/types/src/Filter.d.ts +54 -4
- package/dist/types/src/Filter.d.ts.map +1 -1
- package/dist/types/src/Filter.test.d.ts +2 -0
- package/dist/types/src/Filter.test.d.ts.map +1 -0
- package/dist/types/src/Hypergraph.d.ts +3 -3
- package/dist/types/src/Hypergraph.d.ts.map +1 -1
- package/dist/types/src/Json.d.ts +33 -0
- package/dist/types/src/Json.d.ts.map +1 -0
- package/dist/types/src/Json.test.d.ts +2 -0
- package/dist/types/src/Json.test.d.ts.map +1 -0
- package/dist/types/src/JsonSchema.d.ts +1 -1
- package/dist/types/src/Migration.d.ts +69 -0
- package/dist/types/src/Migration.d.ts.map +1 -0
- package/dist/types/src/Obj.d.ts +42 -28
- package/dist/types/src/Obj.d.ts.map +1 -1
- package/dist/types/src/Order.d.ts.map +1 -1
- package/dist/types/src/Query.d.ts +13 -2
- package/dist/types/src/Query.d.ts.map +1 -1
- package/dist/types/src/Ref.d.ts +1 -0
- package/dist/types/src/Ref.d.ts.map +1 -1
- package/dist/types/src/Relation.d.ts +17 -18
- package/dist/types/src/Relation.d.ts.map +1 -1
- package/dist/types/src/Tag.d.ts +2 -2
- package/dist/types/src/Tag.d.ts.map +1 -1
- package/dist/types/src/Type.d.ts +3 -3
- package/dist/types/src/Type.d.ts.map +1 -1
- package/dist/types/src/View.d.ts +1 -1
- package/dist/types/src/View.d.ts.map +1 -1
- package/dist/types/src/exemplars.test.d.ts +2 -0
- package/dist/types/src/exemplars.test.d.ts.map +1 -0
- package/dist/types/src/index.d.ts +3 -0
- package/dist/types/src/index.d.ts.map +1 -1
- package/dist/types/src/internal/Annotation/annotations.d.ts +12 -2
- package/dist/types/src/internal/Annotation/annotations.d.ts.map +1 -1
- package/dist/types/src/internal/Annotation/sorting.d.ts.map +1 -1
- package/dist/types/src/internal/Annotation/util.d.ts +1 -1
- package/dist/types/src/internal/Annotation/util.d.ts.map +1 -1
- package/dist/types/src/internal/Entity/api.d.ts.map +1 -1
- package/dist/types/src/internal/Entity/model.d.ts +2 -0
- package/dist/types/src/internal/Entity/model.d.ts.map +1 -1
- package/dist/types/src/internal/Entity/object.d.ts.map +1 -1
- package/dist/types/src/internal/Entity/relation.d.ts.map +1 -1
- package/dist/types/src/internal/Entity/version.d.ts.map +1 -1
- package/dist/types/src/internal/Format/date.d.ts.map +1 -1
- package/dist/types/src/internal/Format/format.d.ts.map +1 -1
- package/dist/types/src/internal/Format/number.d.ts.map +1 -1
- package/dist/types/src/internal/Format/object.d.ts.map +1 -1
- package/dist/types/src/internal/Format/types.d.ts.map +1 -1
- package/dist/types/src/internal/JsonSchema/json-schema-normalize.d.ts.map +1 -1
- package/dist/types/src/internal/JsonSchema/json-schema-type.d.ts +28 -28
- package/dist/types/src/internal/JsonSchema/json-schema-type.d.ts.map +1 -1
- package/dist/types/src/internal/JsonSchema/json-schema.d.ts +1 -1
- package/dist/types/src/internal/JsonSchema/json-schema.d.ts.map +1 -1
- package/dist/types/src/internal/Obj/clone.d.ts.map +1 -1
- package/dist/types/src/internal/Obj/common.d.ts.map +1 -1
- package/dist/types/src/internal/Obj/create-object.d.ts.map +1 -1
- package/dist/types/src/internal/Obj/deleted.d.ts.map +1 -1
- package/dist/types/src/internal/Obj/ids.d.ts +1 -1
- package/dist/types/src/internal/Obj/ids.d.ts.map +1 -1
- package/dist/types/src/internal/Obj/json-serializer.d.ts +4 -3
- package/dist/types/src/internal/Obj/json-serializer.d.ts.map +1 -1
- package/dist/types/src/internal/Obj/set-value.d.ts +1 -1
- package/dist/types/src/internal/Obj/set-value.d.ts.map +1 -1
- package/dist/types/src/internal/Obj/snapshot.d.ts.map +1 -1
- package/dist/types/src/internal/Query.d.ts +10 -0
- package/dist/types/src/internal/Query.d.ts.map +1 -0
- package/dist/types/src/internal/Ref/ref-array.d.ts.map +1 -1
- package/dist/types/src/internal/Ref/ref.d.ts +14 -1
- package/dist/types/src/internal/Ref/ref.d.ts.map +1 -1
- package/dist/types/src/internal/Type/compose.d.ts.map +1 -1
- package/dist/types/src/internal/Type/echo-schema.d.ts +2 -2
- package/dist/types/src/internal/Type/echo-schema.d.ts.map +1 -1
- package/dist/types/src/internal/Type/manipulation.d.ts.map +1 -1
- package/dist/types/src/internal/common/api/meta.d.ts +3 -3
- package/dist/types/src/internal/common/api/meta.d.ts.map +1 -1
- package/dist/types/src/internal/common/proxy/change-context.d.ts +1 -1
- package/dist/types/src/internal/common/proxy/change-context.d.ts.map +1 -1
- package/dist/types/src/internal/common/proxy/define-hidden-property.d.ts.map +1 -1
- package/dist/types/src/internal/common/proxy/errors.d.ts +1 -1
- package/dist/types/src/internal/common/proxy/errors.d.ts.map +1 -1
- package/dist/types/src/internal/common/proxy/event-batch.d.ts.map +1 -1
- package/dist/types/src/internal/common/proxy/json-serializer.d.ts.map +1 -1
- package/dist/types/src/internal/common/proxy/make-object.d.ts.map +1 -1
- package/dist/types/src/internal/common/proxy/ownership.d.ts.map +1 -1
- package/dist/types/src/internal/common/proxy/proxy-utils.d.ts.map +1 -1
- package/dist/types/src/internal/common/proxy/reactive-array.d.ts +1 -1
- package/dist/types/src/internal/common/proxy/reactive.d.ts +1 -1
- package/dist/types/src/internal/common/proxy/reactive.d.ts.map +1 -1
- package/dist/types/src/internal/common/proxy/reactive.test.d.ts +2 -0
- package/dist/types/src/internal/common/proxy/reactive.test.d.ts.map +1 -0
- package/dist/types/src/internal/common/proxy/schema-validator.d.ts.map +1 -1
- package/dist/types/src/internal/common/proxy/typed-handler.d.ts.map +1 -1
- package/dist/types/src/internal/common/types/base.d.ts.map +1 -1
- package/dist/types/src/internal/common/types/entity.d.ts +4 -4
- package/dist/types/src/internal/common/types/entity.d.ts.map +1 -1
- package/dist/types/src/internal/common/types/meta.d.ts +10 -0
- package/dist/types/src/internal/common/types/meta.d.ts.map +1 -1
- package/dist/types/src/internal/common/types/version.d.ts +1 -1
- package/dist/types/src/internal/index.d.ts +1 -0
- package/dist/types/src/internal/index.d.ts.map +1 -1
- package/dist/types/src/testing/test-data.d.ts +8 -8
- package/dist/types/src/testing/test-data.d.ts.map +1 -1
- package/dist/types/src/testing/test-schema.d.ts +53 -53
- package/dist/types/src/testing/test-schema.d.ts.map +1 -1
- package/dist/types/src/testing/util.d.ts.map +1 -1
- package/dist/types/tsconfig.tsbuildinfo +1 -1
- package/package.json +25 -15
- package/src/Annotation.ts +1 -0
- package/src/Collection.ts +2 -2
- package/src/Database.ts +50 -15
- package/src/Entity.ts +18 -12
- package/src/Extension.test.ts +235 -0
- package/src/Extension.ts +122 -0
- package/src/Feed.ts +107 -34
- package/src/Filter.test.ts +90 -0
- package/src/Filter.ts +97 -3
- package/src/Hypergraph.ts +3 -3
- package/src/Json.test.ts +175 -0
- package/src/Json.ts +102 -0
- package/src/Migration.ts +106 -0
- package/src/Obj.test.ts +105 -13
- package/src/Obj.ts +154 -33
- package/src/Query.test.ts +199 -9
- package/src/Query.ts +58 -8
- package/src/Ref.ts +2 -0
- package/src/Relation.ts +24 -20
- package/src/Type.ts +1 -1
- package/src/View.ts +1 -1
- package/src/exemplars.test.ts +21 -0
- package/src/index.ts +4 -0
- package/src/internal/Annotation/annotations.test.ts +51 -2
- package/src/internal/Annotation/annotations.ts +33 -14
- package/src/internal/Annotation/sorting.ts +0 -1
- package/src/internal/Entity/api.ts +0 -1
- package/src/internal/Entity/model.ts +2 -0
- package/src/internal/Entity/object.ts +0 -1
- package/src/internal/Entity/version.ts +0 -1
- package/src/internal/Format/date.test.ts +0 -1
- package/src/internal/Format/format.test.ts +0 -1
- package/src/internal/JsonSchema/json-schema-type.ts +1 -1
- package/src/internal/JsonSchema/json-schema.test.ts +1 -2
- package/src/internal/JsonSchema/json-schema.ts +1 -2
- package/src/internal/Obj/clone.ts +1 -1
- package/src/internal/Obj/create-object.test.ts +2 -4
- package/src/internal/Obj/create-object.ts +2 -3
- package/src/internal/Obj/deleted.ts +1 -1
- package/src/internal/Obj/ids.ts +1 -1
- package/src/internal/Obj/json-serializer.test.ts +49 -5
- package/src/internal/Obj/json-serializer.ts +47 -25
- package/src/internal/Obj/set-value.test.ts +24 -24
- package/src/internal/Obj/set-value.ts +1 -1
- package/src/internal/Query.ts +156 -0
- package/src/internal/Ref/ref-array.ts +0 -1
- package/src/internal/Ref/ref.test.ts +0 -1
- package/src/internal/Ref/ref.ts +18 -1
- package/src/internal/Type/compose.test.ts +0 -1
- package/src/internal/Type/echo-schema.ts +3 -4
- package/src/internal/common/README.md +1 -1
- package/src/internal/common/api/meta.ts +3 -3
- package/src/internal/common/proxy/change-context.ts +1 -1
- package/src/internal/common/proxy/change.test.ts +94 -94
- package/src/internal/common/proxy/errors.ts +2 -2
- package/src/internal/common/proxy/handler.test.ts +0 -2
- package/src/internal/common/proxy/json-serializer.ts +4 -1
- package/src/internal/common/proxy/make-object.ts +0 -1
- package/src/internal/common/proxy/ownership.ts +0 -1
- package/src/internal/common/proxy/reactive-array.ts +1 -1
- package/src/internal/common/proxy/reactive.test.ts +54 -0
- package/src/internal/common/proxy/reactive.ts +11 -3
- package/src/internal/common/proxy/typed-handler.test.ts +0 -1
- package/src/internal/common/proxy/typed-handler.ts +8 -10
- package/src/internal/common/proxy/typed-object.test.ts +2 -3
- package/src/internal/common/types/entity.ts +1 -1
- package/src/internal/common/types/meta.ts +12 -1
- package/src/internal/index.ts +1 -0
- package/src/testing/api.test.ts +0 -1
- package/src/testing/test-data.ts +157 -98
- package/dist/lib/neutral/chunk-43Y5DOS6.mjs.map +0 -7
- package/dist/lib/neutral/chunk-4JRI2ZJI.mjs.map +0 -7
- package/dist/lib/neutral/chunk-6DNYDXCV.mjs.map +0 -7
- package/dist/lib/neutral/chunk-6GPU7XC3.mjs.map +0 -7
- package/dist/lib/neutral/chunk-7VNVH63N.mjs.map +0 -7
- package/dist/lib/neutral/chunk-BOZZPUXE.mjs.map +0 -7
- package/dist/lib/neutral/chunk-C4JZK4J7.mjs.map +0 -7
- package/dist/lib/neutral/chunk-NEGC54NE.mjs.map +0 -7
- package/dist/lib/neutral/chunk-NXMFBIT5.mjs.map +0 -7
- package/dist/lib/neutral/chunk-OENWMTE6.mjs.map +0 -7
- package/dist/lib/neutral/chunk-ROAGDPV7.mjs.map +0 -7
- package/dist/lib/neutral/chunk-SEMVAGBM.mjs.map +0 -7
- package/dist/lib/neutral/chunk-W47JKR3X.mjs.map +0 -7
- /package/dist/lib/neutral/{chunk-FZO7LQO7.mjs.map → chunk-44HT3MEC.mjs.map} +0 -0
- /package/dist/lib/neutral/{chunk-ANHVGJI4.mjs.map → chunk-7RVZT53K.mjs.map} +0 -0
- /package/dist/lib/neutral/{chunk-BNCCGLJN.mjs.map → chunk-BICZKPQG.mjs.map} +0 -0
- /package/dist/lib/neutral/{chunk-6K2MVI2O.mjs.map → chunk-BUBEC474.mjs.map} +0 -0
- /package/dist/lib/neutral/{chunk-UTBRYVQC.mjs.map → chunk-QRZ2I3ZM.mjs.map} +0 -0
- /package/dist/lib/neutral/{chunk-B5OXLWZL.mjs.map → chunk-TRPZU2HV.mjs.map} +0 -0
- /package/dist/lib/neutral/{chunk-UI6MWK5W.mjs.map → chunk-TTCSATUD.mjs.map} +0 -0
- /package/dist/lib/neutral/{chunk-OMUPQMLR.mjs.map → chunk-V72DY6LU.mjs.map} +0 -0
|
@@ -2,9 +2,8 @@
|
|
|
2
2
|
// Copyright 2025 DXOS.org
|
|
3
3
|
//
|
|
4
4
|
|
|
5
|
-
import { inspect } from 'util';
|
|
6
|
-
|
|
7
5
|
import * as Schema from 'effect/Schema';
|
|
6
|
+
import { inspect } from 'util';
|
|
8
7
|
import { describe, expect, test } from 'vitest';
|
|
9
8
|
|
|
10
9
|
import { DXN } from '@dxos/keys';
|
|
@@ -12,9 +11,8 @@ import { DXN } from '@dxos/keys';
|
|
|
12
11
|
import { Relation } from '../../index';
|
|
13
12
|
import { TestSchema } from '../../testing';
|
|
14
13
|
import { getSchemaDXN, getTypeDXN, isInstanceOf } from '../Annotation';
|
|
15
|
-
import { ATTR_RELATION_SOURCE, ATTR_RELATION_TARGET } from '../Entity';
|
|
16
14
|
import { ATTR_META, ATTR_TYPE, getSchema } from '../common/types';
|
|
17
|
-
|
|
15
|
+
import { ATTR_RELATION_SOURCE, ATTR_RELATION_TARGET } from '../Entity';
|
|
18
16
|
import { createObject } from './create-object';
|
|
19
17
|
import { objectToJSON } from './json-serializer';
|
|
20
18
|
|
|
@@ -9,6 +9,8 @@ import { assertArgument, failedInvariant } from '@dxos/invariant';
|
|
|
9
9
|
import { ObjectId } from '@dxos/keys';
|
|
10
10
|
|
|
11
11
|
import { getSchemaDXN, getTypeAnnotation, setTypename } from '../Annotation';
|
|
12
|
+
import { defineHiddenProperty } from '../common/proxy';
|
|
13
|
+
import { EntityKind, KindId, MetaId, setSchema } from '../common/types';
|
|
12
14
|
import {
|
|
13
15
|
RelationSourceDXNId,
|
|
14
16
|
RelationSourceId,
|
|
@@ -17,9 +19,6 @@ import {
|
|
|
17
19
|
assertObjectModel,
|
|
18
20
|
getObjectDXN,
|
|
19
21
|
} from '../Entity';
|
|
20
|
-
import { defineHiddenProperty } from '../common/proxy';
|
|
21
|
-
import { EntityKind, KindId, MetaId, setSchema } from '../common/types';
|
|
22
|
-
|
|
23
22
|
import { attachedTypedObjectInspector } from './inspect';
|
|
24
23
|
import { attachTypedJsonSerializer } from './json-serializer';
|
|
25
24
|
|
package/src/internal/Obj/ids.ts
CHANGED
|
@@ -5,7 +5,7 @@
|
|
|
5
5
|
import { DXN, ObjectId, QueueSubspaceTags, SpaceId } from '@dxos/keys';
|
|
6
6
|
|
|
7
7
|
/**
|
|
8
|
-
* @deprecated Use `db.
|
|
8
|
+
* @deprecated Use `Feed.make(...)` + `db.add(feed)` then `Feed.getQueueDxn(feed)`.
|
|
9
9
|
*/
|
|
10
10
|
// TODO(burdon): Move to @dxos/keys.
|
|
11
11
|
export const createQueueDXN = (spaceId = SpaceId.random(), queueId = ObjectId.random()) =>
|
|
@@ -2,27 +2,28 @@
|
|
|
2
2
|
// Copyright 2025 DXOS.org
|
|
3
3
|
//
|
|
4
4
|
|
|
5
|
+
import * as Schema from 'effect/Schema';
|
|
5
6
|
import { describe, expect, test } from 'vitest';
|
|
6
7
|
|
|
7
8
|
import { DXN } from '@dxos/keys';
|
|
8
9
|
|
|
9
10
|
import * as Obj from '../../Obj';
|
|
10
11
|
import { TestSchema } from '../../testing';
|
|
12
|
+
import * as Type from '../../Type';
|
|
11
13
|
import { getSchemaDXN, getSchemaTypename, getTypeDXN, getTypename } from '../Annotation';
|
|
12
14
|
import { getMetaChecked } from '../common/api';
|
|
13
|
-
import { RelationSourceId, RelationTargetId, getObjectDXN } from '../Entity';
|
|
14
15
|
import { makeObject } from '../common/proxy';
|
|
15
|
-
import { Ref, StaticRefResolver } from '../Ref';
|
|
16
16
|
import { ATTR_TYPE, EntityKind, KindId, MetaId, TypeId, getSchema } from '../common/types';
|
|
17
|
-
|
|
17
|
+
import { RelationSourceId, RelationTargetId, getObjectDXN } from '../Entity';
|
|
18
|
+
import { Ref, StaticRefResolver } from '../Ref';
|
|
18
19
|
import { createObject } from './create-object';
|
|
19
20
|
import { objectFromJSON, objectToJSON } from './json-serializer';
|
|
20
21
|
|
|
21
22
|
describe('Object JSON serializer', () => {
|
|
22
23
|
test('should serialize and deserialize object', async () => {
|
|
23
24
|
const contact = makeObject(TestSchema.Person, { name: 'Alice' });
|
|
24
|
-
Obj.
|
|
25
|
-
getMetaChecked(
|
|
25
|
+
Obj.update(contact, (contact) => {
|
|
26
|
+
getMetaChecked(contact).keys.push({ id: '12345', source: 'example.com' });
|
|
26
27
|
});
|
|
27
28
|
|
|
28
29
|
const task = createObject(TestSchema.Task, {
|
|
@@ -118,4 +119,47 @@ describe('Object JSON serializer', () => {
|
|
|
118
119
|
expect(expandoFromJson.message).toBe('local-only');
|
|
119
120
|
expect((expandoFromJson as any)[ATTR_TYPE]).toBeUndefined();
|
|
120
121
|
});
|
|
122
|
+
|
|
123
|
+
describe('Uint8Array', () => {
|
|
124
|
+
const Blob = Schema.Struct({
|
|
125
|
+
name: Schema.String,
|
|
126
|
+
bytes: Schema.Uint8ArrayFromSelf,
|
|
127
|
+
}).pipe(
|
|
128
|
+
Type.object({
|
|
129
|
+
typename: 'com.example.type.blob',
|
|
130
|
+
version: '0.1.0',
|
|
131
|
+
}),
|
|
132
|
+
);
|
|
133
|
+
interface Blob extends Schema.Schema.Type<typeof Blob> {}
|
|
134
|
+
|
|
135
|
+
test('round-trips Uint8Array field through JSON with schema', async ({ expect }) => {
|
|
136
|
+
const bytes = new Uint8Array([0, 1, 2, 3, 250, 251, 252, 253, 254, 255]);
|
|
137
|
+
const blob = Obj.make(Blob, { name: 'blob', bytes });
|
|
138
|
+
|
|
139
|
+
const blobJson = objectToJSON(blob);
|
|
140
|
+
// JSON must round-trip through stringify/parse without loss.
|
|
141
|
+
const roundTripped = JSON.parse(JSON.stringify(blobJson));
|
|
142
|
+
|
|
143
|
+
const refResolver = new StaticRefResolver().addSchema(Blob);
|
|
144
|
+
const blobFromJson = (await objectFromJSON(roundTripped, { refResolver })) as Blob;
|
|
145
|
+
|
|
146
|
+
expect(blobFromJson.name).toBe('blob');
|
|
147
|
+
expect(blobFromJson.bytes).toBeInstanceOf(Uint8Array);
|
|
148
|
+
expect(Array.from(blobFromJson.bytes)).toEqual(Array.from(bytes));
|
|
149
|
+
});
|
|
150
|
+
|
|
151
|
+
test('round-trips Uint8Array field through JSON without schema resolver', async ({ expect }) => {
|
|
152
|
+
const bytes = new Uint8Array([10, 20, 30, 40, 50]);
|
|
153
|
+
const blob = Obj.make(Blob, { name: 'blob', bytes });
|
|
154
|
+
|
|
155
|
+
const blobJson = objectToJSON(blob);
|
|
156
|
+
const roundTripped = JSON.parse(JSON.stringify(blobJson));
|
|
157
|
+
|
|
158
|
+
const blobFromJson = (await objectFromJSON(roundTripped)) as Blob;
|
|
159
|
+
|
|
160
|
+
expect(blobFromJson.name).toBe('blob');
|
|
161
|
+
expect(blobFromJson.bytes).toBeInstanceOf(Uint8Array);
|
|
162
|
+
expect(Array.from(blobFromJson.bytes)).toEqual(Array.from(bytes));
|
|
163
|
+
});
|
|
164
|
+
});
|
|
121
165
|
});
|
|
@@ -8,11 +8,24 @@ import { raise } from '@dxos/debug';
|
|
|
8
8
|
import { type EncodedReference, ObjectStructure, isEncodedReference } from '@dxos/echo-protocol';
|
|
9
9
|
import { assertArgument, invariant } from '@dxos/invariant';
|
|
10
10
|
import { DXN, ObjectId } from '@dxos/keys';
|
|
11
|
-
import { assumeType, deepMapValues, visitValues } from '@dxos/util';
|
|
11
|
+
import { assumeType, decodeUint8ArrayFromJson, deepMapValues, isEncodedUint8Array, visitValues } from '@dxos/util';
|
|
12
12
|
|
|
13
13
|
import type * as Database from '../../Database';
|
|
14
14
|
import type * as Obj from '../../Obj';
|
|
15
15
|
import { getTypeDXN, setTypename } from '../Annotation';
|
|
16
|
+
import { attachTypedJsonSerializer, defineHiddenProperty, typedJsonSerializer } from '../common/proxy';
|
|
17
|
+
import {
|
|
18
|
+
ATTR_META,
|
|
19
|
+
ATTR_PARENT,
|
|
20
|
+
ATTR_TYPE,
|
|
21
|
+
type AnyEntity,
|
|
22
|
+
EntityKind,
|
|
23
|
+
KindId,
|
|
24
|
+
MetaId,
|
|
25
|
+
ObjectMetaSchema,
|
|
26
|
+
ParentId,
|
|
27
|
+
setSchema,
|
|
28
|
+
} from '../common/types';
|
|
16
29
|
import {
|
|
17
30
|
ATTR_DELETED,
|
|
18
31
|
ATTR_RELATION_SOURCE,
|
|
@@ -27,20 +40,7 @@ import {
|
|
|
27
40
|
SelfDXNId,
|
|
28
41
|
assertObjectModel,
|
|
29
42
|
} from '../Entity';
|
|
30
|
-
import { attachTypedJsonSerializer, defineHiddenProperty, typedJsonSerializer } from '../common/proxy';
|
|
31
43
|
import { Ref, type RefResolver, refFromEncodedReference, setRefResolver } from '../Ref';
|
|
32
|
-
import {
|
|
33
|
-
ATTR_META,
|
|
34
|
-
ATTR_PARENT,
|
|
35
|
-
ATTR_TYPE,
|
|
36
|
-
type AnyEntity,
|
|
37
|
-
EntityKind,
|
|
38
|
-
KindId,
|
|
39
|
-
MetaId,
|
|
40
|
-
ObjectMetaSchema,
|
|
41
|
-
ParentId,
|
|
42
|
-
setSchema,
|
|
43
|
-
} from '../common/types';
|
|
44
44
|
|
|
45
45
|
// Re-export for backward compatibility.
|
|
46
46
|
export { attachTypedJsonSerializer };
|
|
@@ -80,7 +80,12 @@ export const objectToJSON = <T extends AnyEntity>(obj: T): SerializedObject<T> =
|
|
|
80
80
|
*/
|
|
81
81
|
export const objectFromJSON = async (
|
|
82
82
|
jsonData: unknown,
|
|
83
|
-
{
|
|
83
|
+
{
|
|
84
|
+
refResolver,
|
|
85
|
+
dxn,
|
|
86
|
+
database,
|
|
87
|
+
parent,
|
|
88
|
+
}: { refResolver?: RefResolver; dxn?: DXN; database?: Database.Database; parent?: Obj.Unknown } = {},
|
|
84
89
|
): Promise<AnyEntity> => {
|
|
85
90
|
assumeType<ObjectJSON>(jsonData);
|
|
86
91
|
assertArgument(typeof jsonData === 'object' && jsonData !== null, 'jsonData', 'expect object');
|
|
@@ -90,7 +95,7 @@ export const objectFromJSON = async (
|
|
|
90
95
|
const type = DXN.parse(jsonData[ATTR_TYPE]);
|
|
91
96
|
const schema = await refResolver?.resolveSchema(type);
|
|
92
97
|
invariant(schema === undefined || Schema.isSchema(schema));
|
|
93
|
-
const decodedInput = stripInternalJsonKeys(jsonData);
|
|
98
|
+
const decodedInput = restoreUint8Arrays(stripInternalJsonKeys(jsonData));
|
|
94
99
|
|
|
95
100
|
let obj: any;
|
|
96
101
|
if (schema != null) {
|
|
@@ -111,15 +116,15 @@ export const objectFromJSON = async (
|
|
|
111
116
|
const isRelation =
|
|
112
117
|
typeof jsonData[ATTR_RELATION_SOURCE] === 'string' || typeof jsonData[ATTR_RELATION_TARGET] === 'string';
|
|
113
118
|
if (isRelation) {
|
|
114
|
-
const
|
|
115
|
-
const
|
|
119
|
+
const sourceDXN: DXN = DXN.parse(jsonData[ATTR_RELATION_SOURCE] ?? raise(new TypeError('Missing relation source')));
|
|
120
|
+
const targetDXN: DXN = DXN.parse(jsonData[ATTR_RELATION_TARGET] ?? raise(new TypeError('Missing relation target')));
|
|
116
121
|
|
|
117
|
-
const source = (await refResolver?.resolve(
|
|
118
|
-
const target = (await refResolver?.resolve(
|
|
122
|
+
const source = (await refResolver?.resolve(sourceDXN)) as AnyEntity | undefined;
|
|
123
|
+
const target = (await refResolver?.resolve(targetDXN)) as AnyEntity | undefined;
|
|
119
124
|
|
|
120
125
|
defineHiddenProperty(obj, KindId, EntityKind.Relation);
|
|
121
|
-
defineHiddenProperty(obj, RelationSourceDXNId,
|
|
122
|
-
defineHiddenProperty(obj, RelationTargetDXNId,
|
|
126
|
+
defineHiddenProperty(obj, RelationSourceDXNId, sourceDXN);
|
|
127
|
+
defineHiddenProperty(obj, RelationTargetDXNId, targetDXN);
|
|
123
128
|
defineHiddenProperty(obj, RelationSourceId, source);
|
|
124
129
|
defineHiddenProperty(obj, RelationTargetId, target);
|
|
125
130
|
} else {
|
|
@@ -137,8 +142,10 @@ export const objectFromJSON = async (
|
|
|
137
142
|
}
|
|
138
143
|
|
|
139
144
|
if (jsonData[ATTR_PARENT]) {
|
|
140
|
-
const
|
|
141
|
-
const
|
|
145
|
+
const parentDXN = DXN.parse(jsonData[ATTR_PARENT]);
|
|
146
|
+
const resolvedParent = (await refResolver?.resolve(parentDXN)) as Obj.Unknown | undefined;
|
|
147
|
+
defineHiddenProperty(obj, ParentId, resolvedParent);
|
|
148
|
+
} else if (parent) {
|
|
142
149
|
defineHiddenProperty(obj, ParentId, parent);
|
|
143
150
|
}
|
|
144
151
|
|
|
@@ -167,17 +174,32 @@ const decodeGeneric = (jsonData: unknown, options: { refResolver?: RefResolver }
|
|
|
167
174
|
if (isEncodedReference(value)) {
|
|
168
175
|
return refFromEncodedReference(value, options.refResolver);
|
|
169
176
|
}
|
|
177
|
+
if (isEncodedUint8Array(value)) {
|
|
178
|
+
return decodeUint8ArrayFromJson(value);
|
|
179
|
+
}
|
|
170
180
|
|
|
171
181
|
return visitor(value);
|
|
172
182
|
});
|
|
173
183
|
};
|
|
174
184
|
|
|
185
|
+
/**
|
|
186
|
+
* Recursively replaces encoded `Uint8Array` JSON markers with actual `Uint8Array` instances.
|
|
187
|
+
* Runs before schema decoding so `Schema.Uint8ArrayFromSelf` sees real bytes.
|
|
188
|
+
*/
|
|
189
|
+
const restoreUint8Arrays = (data: unknown): any =>
|
|
190
|
+
deepMapValues(data, (value, recurse) => {
|
|
191
|
+
if (isEncodedUint8Array(value)) {
|
|
192
|
+
return decodeUint8ArrayFromJson(value);
|
|
193
|
+
}
|
|
194
|
+
return recurse(value);
|
|
195
|
+
});
|
|
196
|
+
|
|
175
197
|
const stripInternalJsonKeys = (jsonData: unknown) => {
|
|
176
198
|
const {
|
|
177
199
|
[ATTR_TYPE]: _type,
|
|
178
200
|
[ATTR_META]: _meta,
|
|
179
201
|
[ATTR_DELETED]: _deleted,
|
|
180
|
-
[ATTR_SELF_DXN]:
|
|
202
|
+
[ATTR_SELF_DXN]: _selfDXN,
|
|
181
203
|
[ATTR_RELATION_SOURCE]: _relationSource,
|
|
182
204
|
[ATTR_RELATION_TARGET]: _relationTarget,
|
|
183
205
|
...props
|
|
@@ -17,7 +17,7 @@ describe('Obj.setValue', () => {
|
|
|
17
17
|
email: 'john@example.com',
|
|
18
18
|
});
|
|
19
19
|
|
|
20
|
-
Obj.
|
|
20
|
+
Obj.update(person, (person) => Obj.setValue(person, ['address', 'city'], 'NYC'));
|
|
21
21
|
|
|
22
22
|
expect(person.address).toBeDefined();
|
|
23
23
|
expect(person.address?.city).toBe('NYC');
|
|
@@ -30,7 +30,7 @@ describe('Obj.setValue', () => {
|
|
|
30
30
|
email: 'john@example.com',
|
|
31
31
|
});
|
|
32
32
|
|
|
33
|
-
Obj.
|
|
33
|
+
Obj.update(person, (person) => Obj.setValue(person, ['fields', 0, 'label'], 'Phone'));
|
|
34
34
|
|
|
35
35
|
expect(Array.isArray(person.fields)).toBe(true);
|
|
36
36
|
expect(person.fields?.[0].label).toBe('Phone');
|
|
@@ -45,7 +45,7 @@ describe('Obj.setValue', () => {
|
|
|
45
45
|
email: 'john@example.com',
|
|
46
46
|
});
|
|
47
47
|
|
|
48
|
-
Obj.
|
|
48
|
+
Obj.update(person, (person) => Obj.setValue(person, ['address', 'coordinates', 'lat'], 40.7128));
|
|
49
49
|
|
|
50
50
|
expect(person.address).toBeDefined();
|
|
51
51
|
expect(person.address?.coordinates).toBeDefined();
|
|
@@ -69,10 +69,10 @@ describe('Obj.setValue', () => {
|
|
|
69
69
|
|
|
70
70
|
const container = Obj.make(Container, { name: 'box' });
|
|
71
71
|
|
|
72
|
-
Obj.
|
|
73
|
-
Obj.setValue(
|
|
74
|
-
Obj.setValue(
|
|
75
|
-
Obj.setValue(
|
|
72
|
+
Obj.update(container, (container) => {
|
|
73
|
+
Obj.setValue(container, ['items', 0, 'value'], 10);
|
|
74
|
+
Obj.setValue(container, ['items', 1, 'value'], 20);
|
|
75
|
+
Obj.setValue(container, ['items', 2, 'value'], 30);
|
|
76
76
|
});
|
|
77
77
|
|
|
78
78
|
expect(container.items).toHaveLength(3);
|
|
@@ -88,7 +88,7 @@ describe('Obj.setValue', () => {
|
|
|
88
88
|
email: 'john@example.com',
|
|
89
89
|
});
|
|
90
90
|
|
|
91
|
-
Obj.
|
|
91
|
+
Obj.update(person, (person) => Obj.setValue(person, ['age'], 25));
|
|
92
92
|
|
|
93
93
|
expect(person.age).toBe(25);
|
|
94
94
|
});
|
|
@@ -100,7 +100,7 @@ describe('Obj.setValue', () => {
|
|
|
100
100
|
email: 'john@example.com',
|
|
101
101
|
});
|
|
102
102
|
|
|
103
|
-
Obj.
|
|
103
|
+
Obj.update(person, (person) => Obj.setValue(person, ['address', 'city'], 'NYC'));
|
|
104
104
|
|
|
105
105
|
expect(person.address).toBeDefined();
|
|
106
106
|
expect(person.address?.city).toBe('NYC');
|
|
@@ -114,8 +114,8 @@ describe('Obj.setValue', () => {
|
|
|
114
114
|
});
|
|
115
115
|
|
|
116
116
|
let result: any;
|
|
117
|
-
Obj.
|
|
118
|
-
result = Obj.setValue(
|
|
117
|
+
Obj.update(person, (person) => {
|
|
118
|
+
result = Obj.setValue(person, ['age'], 30);
|
|
119
119
|
});
|
|
120
120
|
|
|
121
121
|
expect(result).toBe(30);
|
|
@@ -129,7 +129,7 @@ describe('Obj.setValue', () => {
|
|
|
129
129
|
age: 25,
|
|
130
130
|
});
|
|
131
131
|
|
|
132
|
-
Obj.
|
|
132
|
+
Obj.update(person, (person) => Obj.setValue(person, ['age'], 30));
|
|
133
133
|
|
|
134
134
|
expect(person.age).toBe(30);
|
|
135
135
|
});
|
|
@@ -142,7 +142,7 @@ describe('Obj.setValue', () => {
|
|
|
142
142
|
address: { city: 'Boston', state: 'MA', coordinates: {} },
|
|
143
143
|
});
|
|
144
144
|
|
|
145
|
-
Obj.
|
|
145
|
+
Obj.update(person, (person) => Obj.setValue(person, ['address', 'city'], 'NYC'));
|
|
146
146
|
|
|
147
147
|
expect(person.address?.city).toBe('NYC');
|
|
148
148
|
expect(person.address?.state).toBe('MA');
|
|
@@ -160,11 +160,11 @@ describe('Obj.setValue', () => {
|
|
|
160
160
|
|
|
161
161
|
const matrix = Obj.make(Matrix, {});
|
|
162
162
|
|
|
163
|
-
Obj.
|
|
164
|
-
Obj.setValue(
|
|
165
|
-
Obj.setValue(
|
|
166
|
-
Obj.setValue(
|
|
167
|
-
Obj.setValue(
|
|
163
|
+
Obj.update(matrix, (matrix) => {
|
|
164
|
+
Obj.setValue(matrix, ['values', 0, 0], 1);
|
|
165
|
+
Obj.setValue(matrix, ['values', 0, 1], 2);
|
|
166
|
+
Obj.setValue(matrix, ['values', 1, 0], 3);
|
|
167
|
+
Obj.setValue(matrix, ['values', 1, 1], 4);
|
|
168
168
|
});
|
|
169
169
|
|
|
170
170
|
expect(matrix.values?.[0][0]).toBe(1);
|
|
@@ -180,8 +180,8 @@ describe('Obj.setValue', () => {
|
|
|
180
180
|
email: 'john@example.com',
|
|
181
181
|
});
|
|
182
182
|
|
|
183
|
-
Obj.
|
|
184
|
-
expect(() => Obj.setValue(
|
|
183
|
+
Obj.update(person, (person) => {
|
|
184
|
+
expect(() => Obj.setValue(person, [], 'value')).toThrow('Path must not be empty');
|
|
185
185
|
});
|
|
186
186
|
});
|
|
187
187
|
|
|
@@ -192,7 +192,7 @@ describe('Obj.setValue', () => {
|
|
|
192
192
|
email: 'john@example.com',
|
|
193
193
|
});
|
|
194
194
|
|
|
195
|
-
Obj.
|
|
195
|
+
Obj.update(person, (person) => Obj.setValue(person, ['age'], 30));
|
|
196
196
|
|
|
197
197
|
expect(person.age).toBe(30);
|
|
198
198
|
});
|
|
@@ -215,7 +215,7 @@ describe('Obj.setValue', () => {
|
|
|
215
215
|
const container = Obj.make(Container, { name: 'box' });
|
|
216
216
|
|
|
217
217
|
// Using string '0' for array index.
|
|
218
|
-
Obj.
|
|
218
|
+
Obj.update(container, (container) => Obj.setValue(container, ['items', '0', 'value'], 42));
|
|
219
219
|
|
|
220
220
|
expect(container.items?.[0].value).toBe(42);
|
|
221
221
|
});
|
|
@@ -243,7 +243,7 @@ describe('Obj.setValue', () => {
|
|
|
243
243
|
|
|
244
244
|
// This should work: setting a nested property on an array element.
|
|
245
245
|
// The required 'id' field should be initialized with a default value.
|
|
246
|
-
Obj.
|
|
246
|
+
Obj.update(todoList, (todoList) => Obj.setValue(todoList, ['tasks', 0, 'title'], 'Buy groceries'));
|
|
247
247
|
|
|
248
248
|
expect(todoList.tasks?.[0].id).toBe(''); // Default value for required String
|
|
249
249
|
expect(todoList.tasks?.[0].title).toBe('Buy groceries');
|
|
@@ -270,7 +270,7 @@ describe('Obj.setValue', () => {
|
|
|
270
270
|
|
|
271
271
|
const container = Obj.make(Container, { name: 'box' });
|
|
272
272
|
|
|
273
|
-
Obj.
|
|
273
|
+
Obj.update(container, (container) => Obj.setValue(container, ['items', 0, 'label'], 'First Item'));
|
|
274
274
|
|
|
275
275
|
// All required primitive fields should have default values.
|
|
276
276
|
expect(container.items?.[0].id).toBe('');
|
|
@@ -21,7 +21,7 @@ import { getSchema } from '../common/types';
|
|
|
21
21
|
|
|
22
22
|
/**
|
|
23
23
|
* Set a deeply nested property on an object.
|
|
24
|
-
* Must be called within an Obj.
|
|
24
|
+
* Must be called within an Obj.update or Relation.update callback.
|
|
25
25
|
*/
|
|
26
26
|
export const setValue = (obj: Mutable<any>, path: readonly (string | number)[], value: any): void => {
|
|
27
27
|
invariant(path.length > 0, 'Path must not be empty');
|
|
@@ -0,0 +1,156 @@
|
|
|
1
|
+
//
|
|
2
|
+
// Copyright 2025 DXOS.org
|
|
3
|
+
//
|
|
4
|
+
|
|
5
|
+
import { type QueryAST } from '@dxos/echo-protocol';
|
|
6
|
+
|
|
7
|
+
/**
|
|
8
|
+
* Returns a human-readable string representation of a Filter AST.
|
|
9
|
+
*/
|
|
10
|
+
export const prettyFilter = (filter: QueryAST.Filter): string => {
|
|
11
|
+
switch (filter.type) {
|
|
12
|
+
case 'object': {
|
|
13
|
+
// A type-less object filter with only a meta-key constraint is `Filter.key(...)`.
|
|
14
|
+
if (
|
|
15
|
+
filter.typename === null &&
|
|
16
|
+
(filter.id === undefined || filter.id.length === 0) &&
|
|
17
|
+
Object.keys(filter.props).length === 0 &&
|
|
18
|
+
(filter.foreignKeys === undefined || filter.foreignKeys.length === 0) &&
|
|
19
|
+
filter.metaKey !== undefined
|
|
20
|
+
) {
|
|
21
|
+
return filter.metaVersion !== undefined
|
|
22
|
+
? `Filter.key(${JSON.stringify(filter.metaKey)}, { version: ${JSON.stringify(filter.metaVersion)} })`
|
|
23
|
+
: `Filter.key(${JSON.stringify(filter.metaKey)})`;
|
|
24
|
+
}
|
|
25
|
+
const parts: string[] = [];
|
|
26
|
+
if (filter.typename !== null) {
|
|
27
|
+
parts.push(String(filter.typename));
|
|
28
|
+
}
|
|
29
|
+
if (filter.id !== undefined && filter.id.length > 0) {
|
|
30
|
+
parts.push(`id: [${filter.id.join(', ')}]`);
|
|
31
|
+
}
|
|
32
|
+
const propEntries = Object.entries(filter.props);
|
|
33
|
+
if (propEntries.length > 0) {
|
|
34
|
+
const propsStr = propEntries.map(([k, v]) => `${k}: ${prettyFilter(v)}`).join(', ');
|
|
35
|
+
parts.push(`{ ${propsStr} }`);
|
|
36
|
+
}
|
|
37
|
+
if (filter.foreignKeys !== undefined && filter.foreignKeys.length > 0) {
|
|
38
|
+
parts.push(`foreignKeys: [${filter.foreignKeys.map((fk) => JSON.stringify(fk)).join(', ')}]`);
|
|
39
|
+
}
|
|
40
|
+
if (filter.metaKey !== undefined) {
|
|
41
|
+
parts.push(
|
|
42
|
+
filter.metaVersion !== undefined
|
|
43
|
+
? `metaKey: ${JSON.stringify(filter.metaKey)} (${filter.metaVersion})`
|
|
44
|
+
: `metaKey: ${JSON.stringify(filter.metaKey)}`,
|
|
45
|
+
);
|
|
46
|
+
}
|
|
47
|
+
return parts.length > 0 ? `Filter.type(${parts.join(', ')})` : 'Filter.everything()';
|
|
48
|
+
}
|
|
49
|
+
case 'compare':
|
|
50
|
+
return `Filter.${filter.operator}(${JSON.stringify(filter.value)})`;
|
|
51
|
+
case 'in':
|
|
52
|
+
return `Filter.in([${filter.values.map((v) => JSON.stringify(v)).join(', ')}])`;
|
|
53
|
+
case 'contains':
|
|
54
|
+
return `Filter.contains(${JSON.stringify(filter.value)})`;
|
|
55
|
+
case 'tag':
|
|
56
|
+
return `Filter.tag(${JSON.stringify(filter.tag)})`;
|
|
57
|
+
case 'range':
|
|
58
|
+
return `Filter.range(${JSON.stringify(filter.from)}, ${JSON.stringify(filter.to)})`;
|
|
59
|
+
case 'text-search':
|
|
60
|
+
return filter.searchKind
|
|
61
|
+
? `Filter.textSearch(${JSON.stringify(filter.text)}, ${JSON.stringify(filter.searchKind)})`
|
|
62
|
+
: `Filter.textSearch(${JSON.stringify(filter.text)})`;
|
|
63
|
+
case 'timestamp':
|
|
64
|
+
return `Filter.${filter.field}.${filter.operator}(${filter.value})`;
|
|
65
|
+
case 'child-of':
|
|
66
|
+
return `Filter.childOf([${filter.parents.map((p) => JSON.stringify(p)).join(', ')}], { transitive: ${filter.transitive} })`;
|
|
67
|
+
case 'not':
|
|
68
|
+
return `Filter.not(${prettyFilter(filter.filter)})`;
|
|
69
|
+
case 'and':
|
|
70
|
+
return `Filter.and(${filter.filters.map(prettyFilter).join(', ')})`;
|
|
71
|
+
case 'or':
|
|
72
|
+
return `Filter.or(${filter.filters.map(prettyFilter).join(', ')})`;
|
|
73
|
+
}
|
|
74
|
+
};
|
|
75
|
+
|
|
76
|
+
/**
|
|
77
|
+
* Returns a human-readable string representation of a Query AST.
|
|
78
|
+
*/
|
|
79
|
+
export const prettyQuery = (query: QueryAST.Query): string => {
|
|
80
|
+
switch (query.type) {
|
|
81
|
+
case 'select':
|
|
82
|
+
return `Query.select(${prettyFilter(query.filter)})`;
|
|
83
|
+
case 'filter':
|
|
84
|
+
return `${prettyQuery(query.selection)}.select(${prettyFilter(query.filter)})`;
|
|
85
|
+
case 'reference-traversal':
|
|
86
|
+
return `${prettyQuery(query.anchor)}.reference(${JSON.stringify(query.property)})`;
|
|
87
|
+
case 'incoming-references': {
|
|
88
|
+
const args: string[] = [];
|
|
89
|
+
if (query.typename !== null) {
|
|
90
|
+
args.push(String(query.typename));
|
|
91
|
+
}
|
|
92
|
+
if (query.property !== null) {
|
|
93
|
+
args.push(JSON.stringify(query.property));
|
|
94
|
+
}
|
|
95
|
+
return `${prettyQuery(query.anchor)}.referencedBy(${args.join(', ')})`;
|
|
96
|
+
}
|
|
97
|
+
case 'relation': {
|
|
98
|
+
const method =
|
|
99
|
+
query.direction === 'outgoing' ? 'sourceOf' : query.direction === 'incoming' ? 'targetOf' : 'relationOf';
|
|
100
|
+
const filterStr = query.filter !== undefined ? prettyFilter(query.filter) : '';
|
|
101
|
+
return `${prettyQuery(query.anchor)}.${method}(${filterStr})`;
|
|
102
|
+
}
|
|
103
|
+
case 'relation-traversal':
|
|
104
|
+
return `${prettyQuery(query.anchor)}.${query.direction}()`;
|
|
105
|
+
case 'hierarchy-traversal':
|
|
106
|
+
return query.direction === 'to-parent'
|
|
107
|
+
? `${prettyQuery(query.anchor)}.parent()`
|
|
108
|
+
: `${prettyQuery(query.anchor)}.children()`;
|
|
109
|
+
case 'union':
|
|
110
|
+
return `Query.all(${query.queries.map(prettyQuery).join(', ')})`;
|
|
111
|
+
case 'set-difference':
|
|
112
|
+
return `Query.without(${prettyQuery(query.source)}, ${prettyQuery(query.exclude)})`;
|
|
113
|
+
case 'order': {
|
|
114
|
+
const orders = query.order.map((o) => {
|
|
115
|
+
if (o.kind === 'natural') {
|
|
116
|
+
return 'Order.natural()';
|
|
117
|
+
} else if (o.kind === 'rank') {
|
|
118
|
+
return `Order.rank(${JSON.stringify(o.direction)})`;
|
|
119
|
+
} else {
|
|
120
|
+
return `Order.property(${JSON.stringify(o.property)}, ${JSON.stringify(o.direction)})`;
|
|
121
|
+
}
|
|
122
|
+
});
|
|
123
|
+
return `${prettyQuery(query.query)}.orderBy(${orders.join(', ')})`;
|
|
124
|
+
}
|
|
125
|
+
case 'options': {
|
|
126
|
+
const opts = query.options;
|
|
127
|
+
const parts: string[] = [];
|
|
128
|
+
if (opts.deleted !== undefined) {
|
|
129
|
+
parts.push(`deleted: ${JSON.stringify(opts.deleted)}`);
|
|
130
|
+
}
|
|
131
|
+
if (opts.debugLabel !== undefined) {
|
|
132
|
+
parts.push(`debugLabel: ${JSON.stringify(opts.debugLabel)}`);
|
|
133
|
+
}
|
|
134
|
+
return `${prettyQuery(query.query)}.options({ ${parts.join(', ')} })`;
|
|
135
|
+
}
|
|
136
|
+
case 'from': {
|
|
137
|
+
if (query.from._tag === 'scope') {
|
|
138
|
+
const scope = query.from.scope;
|
|
139
|
+
const parts: string[] = [];
|
|
140
|
+
if (scope.spaceIds !== undefined) {
|
|
141
|
+
parts.push(`spaceIds: [${scope.spaceIds.map((s) => JSON.stringify(s)).join(', ')}]`);
|
|
142
|
+
}
|
|
143
|
+
if (scope.feeds !== undefined) {
|
|
144
|
+
parts.push(`feeds: [${scope.feeds.map(String).join(', ')}]`);
|
|
145
|
+
}
|
|
146
|
+
if (scope.allFeedsFromSpaces !== undefined) {
|
|
147
|
+
parts.push(`allFeedsFromSpaces: ${scope.allFeedsFromSpaces}`);
|
|
148
|
+
}
|
|
149
|
+
return `${prettyQuery(query.query)}.from({ ${parts.join(', ')} })`;
|
|
150
|
+
}
|
|
151
|
+
return `${prettyQuery(query.query)}.from(${prettyQuery(query.from.query)})`;
|
|
152
|
+
}
|
|
153
|
+
case 'limit':
|
|
154
|
+
return `${prettyQuery(query.query)}.limit(${query.limit})`;
|
|
155
|
+
}
|
|
156
|
+
};
|
package/src/internal/Ref/ref.ts
CHANGED
|
@@ -20,8 +20,8 @@ import { DXN, ObjectId } from '@dxos/keys';
|
|
|
20
20
|
|
|
21
21
|
import * as Database from '../../Database';
|
|
22
22
|
import { ReferenceAnnotationId, getSchemaDXN, getTypeAnnotation, getTypeIdentifierAnnotation } from '../Annotation';
|
|
23
|
-
import { type JsonSchemaType } from '../JsonSchema';
|
|
24
23
|
import type { AnyEntity, AnyProperties } from '../common/types';
|
|
24
|
+
import { type JsonSchemaType } from '../JsonSchema';
|
|
25
25
|
|
|
26
26
|
/**
|
|
27
27
|
* The `$id` and `$ref` fields for an ECHO reference schema.
|
|
@@ -164,6 +164,16 @@ export interface Ref<T> extends Pipeable.Pipeable {
|
|
|
164
164
|
|
|
165
165
|
tryLoad(): Promise<T | undefined>;
|
|
166
166
|
|
|
167
|
+
/**
|
|
168
|
+
* Subscribe to the ref's resolution event.
|
|
169
|
+
* The callback fires when the target object becomes available in the working set
|
|
170
|
+
* (e.g. when its document is loaded after sibling-client mutation).
|
|
171
|
+
* Note: the resolver only schedules a notification when the target is requested
|
|
172
|
+
* via {@link target} while it is not yet loaded.
|
|
173
|
+
* @returns Function that unsubscribes the callback.
|
|
174
|
+
*/
|
|
175
|
+
onResolved(callback: () => void): () => void;
|
|
176
|
+
|
|
167
177
|
/**
|
|
168
178
|
* Do not inline the target object in the reference.
|
|
169
179
|
* Makes .target unavailable unless the reference is connected to a database context.
|
|
@@ -420,6 +430,13 @@ export class RefImpl<T> implements Ref<T> {
|
|
|
420
430
|
return (await this.#resolver.resolve(this.#dxn)) as T | undefined;
|
|
421
431
|
}
|
|
422
432
|
|
|
433
|
+
/**
|
|
434
|
+
* @inheritdoc
|
|
435
|
+
*/
|
|
436
|
+
onResolved(callback: () => void): () => void {
|
|
437
|
+
return this.#resolved.on(callback);
|
|
438
|
+
}
|
|
439
|
+
|
|
423
440
|
/**
|
|
424
441
|
* Do not inline the target object in the reference.
|
|
425
442
|
* Makes .target unavailable unless the reference is connected to a database context.
|
|
@@ -9,7 +9,6 @@ import { FieldPath } from '../Annotation';
|
|
|
9
9
|
import { EchoObjectSchema } from '../Entity';
|
|
10
10
|
import { FormatAnnotation, TypeFormat } from '../Format';
|
|
11
11
|
import { ECHO_ANNOTATIONS_NS_KEY, toJsonSchema } from '../JsonSchema';
|
|
12
|
-
|
|
13
12
|
import { composeSchema } from './compose';
|
|
14
13
|
|
|
15
14
|
describe('schema composition', () => {
|