@dxos/echo 0.8.4-main.e098934 → 0.8.4-main.ead640a
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-HKFCK2GL.mjs +175 -0
- package/dist/lib/browser/chunk-HKFCK2GL.mjs.map +7 -0
- package/dist/lib/browser/{chunk-EIXXFUN5.mjs → chunk-MAAYELT7.mjs} +477 -486
- package/dist/lib/browser/chunk-MAAYELT7.mjs.map +7 -0
- package/dist/lib/browser/chunk-MB6MMNFP.mjs +3857 -0
- package/dist/lib/browser/chunk-MB6MMNFP.mjs.map +7 -0
- package/dist/lib/browser/index.mjs +12 -7
- package/dist/lib/browser/internal/index.mjs +332 -0
- package/dist/lib/browser/internal/index.mjs.map +7 -0
- package/dist/lib/browser/meta.json +1 -1
- package/dist/lib/browser/query/index.mjs +15 -0
- package/dist/lib/browser/query/index.mjs.map +7 -0
- package/dist/lib/browser/testing/index.mjs +260 -32
- package/dist/lib/browser/testing/index.mjs.map +4 -4
- package/dist/lib/node-esm/{chunk-TCY7IVTS.mjs → chunk-5NWDGIBT.mjs} +477 -486
- package/dist/lib/node-esm/chunk-5NWDGIBT.mjs.map +7 -0
- package/dist/lib/node-esm/chunk-AUAH4E2J.mjs +175 -0
- package/dist/lib/node-esm/chunk-AUAH4E2J.mjs.map +7 -0
- package/dist/lib/node-esm/chunk-BQRA4VLX.mjs +3857 -0
- package/dist/lib/node-esm/chunk-BQRA4VLX.mjs.map +7 -0
- package/dist/lib/node-esm/index.mjs +12 -7
- package/dist/lib/node-esm/internal/index.mjs +332 -0
- package/dist/lib/node-esm/internal/index.mjs.map +7 -0
- package/dist/lib/node-esm/meta.json +1 -1
- package/dist/lib/node-esm/query/index.mjs +15 -0
- package/dist/lib/node-esm/query/index.mjs.map +7 -0
- package/dist/lib/node-esm/testing/index.mjs +260 -32
- package/dist/lib/node-esm/testing/index.mjs.map +4 -4
- package/dist/types/src/Obj.d.ts +22 -5
- package/dist/types/src/Obj.d.ts.map +1 -1
- package/dist/types/src/Ref.d.ts +1 -1
- package/dist/types/src/Ref.d.ts.map +1 -1
- package/dist/types/src/Relation.d.ts +5 -4
- package/dist/types/src/Relation.d.ts.map +1 -1
- package/dist/types/src/Type.d.ts +16 -17
- package/dist/types/src/Type.d.ts.map +1 -1
- package/dist/types/src/errors.d.ts +8 -8
- package/dist/types/src/index.d.ts +2 -2
- package/dist/types/src/index.d.ts.map +1 -1
- package/dist/types/src/internal/ast/annotation-helper.d.ts +8 -0
- package/dist/types/src/internal/ast/annotation-helper.d.ts.map +1 -0
- package/dist/types/src/internal/ast/annotations.d.ts +125 -0
- package/dist/types/src/internal/ast/annotations.d.ts.map +1 -0
- package/dist/types/src/internal/ast/annotations.test.d.ts +2 -0
- package/dist/types/src/internal/ast/annotations.test.d.ts.map +1 -0
- package/dist/types/src/internal/ast/entity-kind.d.ts +10 -0
- package/dist/types/src/internal/ast/entity-kind.d.ts.map +1 -0
- package/dist/types/src/internal/ast/index.d.ts +5 -0
- package/dist/types/src/internal/ast/index.d.ts.map +1 -0
- package/dist/types/src/internal/ast/types.d.ts +6 -0
- package/dist/types/src/internal/ast/types.d.ts.map +1 -0
- package/dist/types/src/internal/formats/date.d.ts +63 -0
- package/dist/types/src/internal/formats/date.d.ts.map +1 -0
- package/dist/types/src/internal/formats/date.test.d.ts +2 -0
- package/dist/types/src/internal/formats/date.test.d.ts.map +1 -0
- package/dist/types/src/internal/formats/format.d.ts +30 -0
- package/dist/types/src/internal/formats/format.d.ts.map +1 -0
- package/dist/types/src/internal/formats/format.test.d.ts +2 -0
- package/dist/types/src/internal/formats/format.test.d.ts.map +1 -0
- package/dist/types/src/internal/formats/index.d.ts +8 -0
- package/dist/types/src/internal/formats/index.d.ts.map +1 -0
- package/dist/types/src/internal/formats/number.d.ts +31 -0
- package/dist/types/src/internal/formats/number.d.ts.map +1 -0
- package/dist/types/src/internal/formats/object.d.ts +35 -0
- package/dist/types/src/internal/formats/object.d.ts.map +1 -0
- package/dist/types/src/internal/formats/select.d.ts +11 -0
- package/dist/types/src/internal/formats/select.d.ts.map +1 -0
- package/dist/types/src/internal/formats/string.d.ts +38 -0
- package/dist/types/src/internal/formats/string.d.ts.map +1 -0
- package/dist/types/src/internal/formats/types.d.ts +68 -0
- package/dist/types/src/internal/formats/types.d.ts.map +1 -0
- package/dist/types/src/internal/index.d.ts +15 -0
- package/dist/types/src/internal/index.d.ts.map +1 -0
- package/dist/types/src/internal/json/annotations.d.ts +19 -0
- package/dist/types/src/internal/json/annotations.d.ts.map +1 -0
- package/dist/types/src/internal/json/effect-schema.test.d.ts +2 -0
- package/dist/types/src/internal/json/effect-schema.test.d.ts.map +1 -0
- package/dist/types/src/internal/json/index.d.ts +2 -0
- package/dist/types/src/internal/json/index.d.ts.map +1 -0
- package/dist/types/src/internal/json/json-schema.d.ts +28 -0
- package/dist/types/src/internal/json/json-schema.d.ts.map +1 -0
- package/dist/types/src/internal/json/json-schema.test.d.ts +2 -0
- package/dist/types/src/internal/json/json-schema.test.d.ts.map +1 -0
- package/dist/types/src/internal/json-schema/index.d.ts +3 -0
- package/dist/types/src/internal/json-schema/index.d.ts.map +1 -0
- package/dist/types/src/internal/json-schema/json-schema-normalize.d.ts +7 -0
- package/dist/types/src/internal/json-schema/json-schema-normalize.d.ts.map +1 -0
- package/dist/types/src/internal/json-schema/json-schema-type.d.ts +250 -0
- package/dist/types/src/internal/json-schema/json-schema-type.d.ts.map +1 -0
- package/dist/types/src/internal/object/accessors.d.ts +37 -0
- package/dist/types/src/internal/object/accessors.d.ts.map +1 -0
- package/dist/types/src/internal/object/common.d.ts +18 -0
- package/dist/types/src/internal/object/common.d.ts.map +1 -0
- package/dist/types/src/internal/object/create.d.ts +40 -0
- package/dist/types/src/internal/object/create.d.ts.map +1 -0
- package/dist/types/src/internal/object/create.test.d.ts +2 -0
- package/dist/types/src/internal/object/create.test.d.ts.map +1 -0
- package/dist/types/src/internal/object/deleted.d.ts +6 -0
- package/dist/types/src/internal/object/deleted.d.ts.map +1 -0
- package/dist/types/src/internal/object/entity.d.ts +33 -0
- package/dist/types/src/internal/object/entity.d.ts.map +1 -0
- package/dist/types/src/internal/object/expando.d.ts +14 -0
- package/dist/types/src/internal/object/expando.d.ts.map +1 -0
- package/dist/types/src/internal/object/ids.d.ts +6 -0
- package/dist/types/src/internal/object/ids.d.ts.map +1 -0
- package/dist/types/src/internal/object/index.d.ts +16 -0
- package/dist/types/src/internal/object/index.d.ts.map +1 -0
- package/dist/types/src/internal/object/inspect.d.ts +2 -0
- package/dist/types/src/internal/object/inspect.d.ts.map +1 -0
- package/dist/types/src/internal/object/json-serializer.d.ts +32 -0
- package/dist/types/src/internal/object/json-serializer.d.ts.map +1 -0
- package/dist/types/src/internal/object/json-serializer.test.d.ts +2 -0
- package/dist/types/src/internal/object/json-serializer.test.d.ts.map +1 -0
- package/dist/types/src/internal/object/meta.d.ts +31 -0
- package/dist/types/src/internal/object/meta.d.ts.map +1 -0
- package/dist/types/src/internal/object/model.d.ts +117 -0
- package/dist/types/src/internal/object/model.d.ts.map +1 -0
- package/dist/types/src/internal/object/relation.d.ts +17 -0
- package/dist/types/src/internal/object/relation.d.ts.map +1 -0
- package/dist/types/src/internal/object/schema-validator.d.ts +15 -0
- package/dist/types/src/internal/object/schema-validator.d.ts.map +1 -0
- package/dist/types/src/internal/object/schema-validator.test.d.ts +2 -0
- package/dist/types/src/internal/object/schema-validator.test.d.ts.map +1 -0
- package/dist/types/src/internal/object/typed-object.d.ts +31 -0
- package/dist/types/src/internal/object/typed-object.d.ts.map +1 -0
- package/dist/types/src/internal/object/typed-object.test.d.ts +2 -0
- package/dist/types/src/internal/object/typed-object.test.d.ts.map +1 -0
- package/dist/types/src/internal/object/typename.d.ts +15 -0
- package/dist/types/src/internal/object/typename.d.ts.map +1 -0
- package/dist/types/src/internal/object/version.d.ts +14 -0
- package/dist/types/src/internal/object/version.d.ts.map +1 -0
- package/dist/types/src/internal/projection/compose.d.ts +6 -0
- package/dist/types/src/internal/projection/compose.d.ts.map +1 -0
- package/dist/types/src/internal/projection/compose.test.d.ts +2 -0
- package/dist/types/src/internal/projection/compose.test.d.ts.map +1 -0
- package/dist/types/src/internal/projection/index.d.ts +2 -0
- package/dist/types/src/internal/projection/index.d.ts.map +1 -0
- package/dist/types/src/internal/proxy/handler.test.d.ts +2 -0
- package/dist/types/src/internal/proxy/handler.test.d.ts.map +1 -0
- package/dist/types/src/internal/proxy/reactive-object.d.ts +15 -0
- package/dist/types/src/internal/proxy/reactive-object.d.ts.map +1 -0
- package/dist/types/src/internal/proxy/schema.test.d.ts +2 -0
- package/dist/types/src/internal/proxy/schema.test.d.ts.map +1 -0
- package/dist/types/src/internal/proxy/typed-handler.d.ts +44 -0
- package/dist/types/src/internal/proxy/typed-handler.d.ts.map +1 -0
- package/dist/types/src/internal/proxy/typed-handler.test.d.ts +2 -0
- package/dist/types/src/internal/proxy/typed-handler.test.d.ts.map +1 -0
- package/dist/types/src/internal/proxy/typed-object.test.d.ts +2 -0
- package/dist/types/src/internal/proxy/typed-object.test.d.ts.map +1 -0
- package/dist/types/src/internal/query/index.d.ts +2 -0
- package/dist/types/src/internal/query/index.d.ts.map +1 -0
- package/dist/types/src/internal/query/query.d.ts +17 -0
- package/dist/types/src/internal/query/query.d.ts.map +1 -0
- package/dist/types/src/internal/ref/index.d.ts +3 -0
- package/dist/types/src/internal/ref/index.d.ts.map +1 -0
- package/dist/types/src/internal/ref/ref-array.d.ts +21 -0
- package/dist/types/src/internal/ref/ref-array.d.ts.map +1 -0
- package/dist/types/src/internal/ref/ref.d.ts +206 -0
- package/dist/types/src/internal/ref/ref.d.ts.map +1 -0
- package/dist/types/src/internal/ref/ref.test.d.ts +2 -0
- package/dist/types/src/internal/ref/ref.test.d.ts.map +1 -0
- package/dist/types/src/internal/schema/echo-schema.d.ts +168 -0
- package/dist/types/src/internal/schema/echo-schema.d.ts.map +1 -0
- package/dist/types/src/internal/schema/index.d.ts +7 -0
- package/dist/types/src/internal/schema/index.d.ts.map +1 -0
- package/dist/types/src/internal/schema/manipulation.d.ts +10 -0
- package/dist/types/src/internal/schema/manipulation.d.ts.map +1 -0
- package/dist/types/src/internal/schema/runtime-schema-registry.d.ts +18 -0
- package/dist/types/src/internal/schema/runtime-schema-registry.d.ts.map +1 -0
- package/dist/types/src/internal/schema/snapshot.d.ts +6 -0
- package/dist/types/src/internal/schema/snapshot.d.ts.map +1 -0
- package/dist/types/src/internal/schema/stored-schema.d.ts +13 -0
- package/dist/types/src/internal/schema/stored-schema.d.ts.map +1 -0
- package/dist/types/src/internal/testing/index.d.ts +3 -0
- package/dist/types/src/internal/testing/index.d.ts.map +1 -0
- package/dist/types/src/internal/testing/types.d.ts +455 -0
- package/dist/types/src/internal/testing/types.d.ts.map +1 -0
- package/dist/types/src/internal/testing/utils.d.ts +10 -0
- package/dist/types/src/internal/testing/utils.d.ts.map +1 -0
- package/dist/types/src/internal/types/index.d.ts +3 -0
- package/dist/types/src/internal/types/index.d.ts.map +1 -0
- package/dist/types/src/internal/types/types.d.ts +79 -0
- package/dist/types/src/internal/types/types.d.ts.map +1 -0
- package/dist/types/src/internal/types/types.test.d.ts +2 -0
- package/dist/types/src/internal/types/types.test.d.ts.map +1 -0
- package/dist/types/src/internal/types/util.d.ts +5 -0
- package/dist/types/src/internal/types/util.d.ts.map +1 -0
- package/dist/types/src/query/index.d.ts +2 -1
- package/dist/types/src/query/index.d.ts.map +1 -1
- package/dist/types/src/query/{dsl.d.ts → query.d.ts} +28 -23
- package/dist/types/src/query/query.d.ts.map +1 -0
- package/dist/types/src/query/query.test.d.ts +2 -0
- package/dist/types/src/query/query.test.d.ts.map +1 -0
- package/dist/types/src/query/tag.d.ts +17 -0
- package/dist/types/src/query/tag.d.ts.map +1 -0
- package/dist/types/src/testing/echo-schema.d.ts +7 -0
- package/dist/types/src/testing/echo-schema.d.ts.map +1 -0
- package/dist/types/src/testing/index.d.ts +2 -0
- package/dist/types/src/testing/index.d.ts.map +1 -1
- package/dist/types/src/testing/types.d.ts +18 -18
- package/dist/types/src/testing/types.d.ts.map +1 -1
- package/dist/types/tsconfig.tsbuildinfo +1 -1
- package/package.json +29 -18
- package/src/Obj.ts +77 -12
- package/src/Ref.ts +1 -2
- package/src/Relation.ts +15 -5
- package/src/Type.ts +29 -30
- package/src/index.ts +2 -2
- package/src/internal/ast/annotation-helper.ts +22 -0
- package/src/internal/ast/annotations.test.ts +98 -0
- package/src/internal/ast/annotations.ts +219 -0
- package/src/internal/ast/entity-kind.ts +15 -0
- package/src/internal/ast/index.ts +8 -0
- package/src/internal/ast/types.ts +17 -0
- package/src/internal/formats/date.test.ts +56 -0
- package/src/internal/formats/date.ts +217 -0
- package/src/internal/formats/format.test.ts +77 -0
- package/src/internal/formats/format.ts +52 -0
- package/src/internal/formats/index.ts +12 -0
- package/src/internal/formats/number.ts +89 -0
- package/src/internal/formats/object.ts +80 -0
- package/src/internal/formats/select.ts +16 -0
- package/src/internal/formats/string.ts +76 -0
- package/src/internal/formats/types.ts +175 -0
- package/src/internal/index.ts +22 -0
- package/src/internal/json/annotations.ts +50 -0
- package/src/internal/json/effect-schema.test.ts +143 -0
- package/src/internal/json/index.ts +5 -0
- package/src/internal/json/json-schema.test.ts +857 -0
- package/src/internal/json/json-schema.ts +519 -0
- package/src/internal/json-schema/index.ts +6 -0
- package/src/internal/json-schema/json-schema-normalize.ts +109 -0
- package/src/internal/json-schema/json-schema-type.ts +403 -0
- package/src/internal/object/accessors.ts +153 -0
- package/src/internal/object/common.ts +76 -0
- package/src/internal/object/create.test.ts +118 -0
- package/src/internal/object/create.ts +96 -0
- package/src/internal/object/deleted.ts +19 -0
- package/src/internal/object/entity.ts +248 -0
- package/src/internal/object/expando.ts +21 -0
- package/src/internal/object/ids.ts +12 -0
- package/src/internal/object/index.ts +19 -0
- package/src/internal/object/inspect.ts +48 -0
- package/src/internal/object/json-serializer.test.ts +99 -0
- package/src/internal/object/json-serializer.ts +225 -0
- package/src/internal/object/meta.ts +61 -0
- package/src/internal/object/model.ts +170 -0
- package/src/internal/object/relation.ts +24 -0
- package/src/internal/object/schema-validator.test.ts +186 -0
- package/src/internal/object/schema-validator.ts +241 -0
- package/src/internal/object/typed-object.test.ts +34 -0
- package/src/internal/object/typed-object.ts +88 -0
- package/src/internal/object/typename.ts +61 -0
- package/src/internal/object/version.ts +22 -0
- package/src/internal/projection/compose.test.ts +43 -0
- package/src/internal/projection/compose.ts +36 -0
- package/src/internal/projection/index.ts +5 -0
- package/src/internal/proxy/handler.test.ts +163 -0
- package/src/internal/proxy/reactive-object.ts +108 -0
- package/src/internal/proxy/schema.test.ts +136 -0
- package/src/internal/proxy/typed-handler.test.ts +102 -0
- package/src/internal/proxy/typed-handler.ts +228 -0
- package/src/internal/proxy/typed-object.test.ts +100 -0
- package/src/internal/query/index.ts +5 -0
- package/src/internal/query/query.ts +23 -0
- package/src/internal/ref/index.ts +6 -0
- package/src/internal/ref/ref-array.ts +39 -0
- package/src/internal/ref/ref.test.ts +100 -0
- package/src/internal/ref/ref.ts +521 -0
- package/src/internal/schema/echo-schema.ts +383 -0
- package/src/internal/schema/index.ts +10 -0
- package/src/internal/schema/manipulation.ts +92 -0
- package/src/internal/schema/runtime-schema-registry.ts +78 -0
- package/src/internal/schema/snapshot.ts +25 -0
- package/src/internal/schema/stored-schema.ts +26 -0
- package/src/internal/testing/index.ts +6 -0
- package/src/internal/testing/types.ts +211 -0
- package/src/internal/testing/utils.ts +54 -0
- package/src/internal/types/index.ts +6 -0
- package/src/internal/types/types.test.ts +48 -0
- package/src/internal/types/types.ts +176 -0
- package/src/internal/types/util.ts +9 -0
- package/src/query/index.ts +2 -1
- package/src/query/query.test.ts +401 -0
- package/src/query/{dsl.ts → query.ts} +82 -45
- package/src/query/tag.ts +37 -0
- package/src/test/api.test.ts +9 -9
- package/src/testing/echo-schema.ts +39 -0
- package/src/testing/index.ts +2 -0
- package/src/testing/types.ts +1 -1
- package/dist/lib/browser/chunk-EIXXFUN5.mjs.map +0 -7
- package/dist/lib/node-esm/chunk-TCY7IVTS.mjs.map +0 -7
- package/dist/types/src/experimental/database.d.ts +0 -8
- package/dist/types/src/experimental/database.d.ts.map +0 -1
- package/dist/types/src/experimental/index.d.ts +0 -1
- package/dist/types/src/experimental/index.d.ts.map +0 -1
- package/dist/types/src/experimental/queue.d.ts +0 -8
- package/dist/types/src/experimental/queue.d.ts.map +0 -1
- package/dist/types/src/experimental/space.d.ts +0 -8
- package/dist/types/src/experimental/space.d.ts.map +0 -1
- package/dist/types/src/query/dsl.d.ts.map +0 -1
- package/dist/types/src/query/dsl.test.d.ts +0 -2
- package/dist/types/src/query/dsl.test.d.ts.map +0 -1
- package/src/experimental/database.ts +0 -11
- package/src/experimental/index.ts +0 -7
- package/src/experimental/queue.ts +0 -11
- package/src/experimental/space.ts +0 -11
- package/src/query/dsl.test.ts +0 -362
|
@@ -0,0 +1,225 @@
|
|
|
1
|
+
//
|
|
2
|
+
// Copyright 2025 DXOS.org
|
|
3
|
+
//
|
|
4
|
+
|
|
5
|
+
import * as Schema from 'effect/Schema';
|
|
6
|
+
|
|
7
|
+
import { raise } from '@dxos/debug';
|
|
8
|
+
import { type EncodedReference, type ObjectMeta, isEncodedReference } from '@dxos/echo-protocol';
|
|
9
|
+
import { assertArgument, invariant } from '@dxos/invariant';
|
|
10
|
+
import { DXN, ObjectId } from '@dxos/keys';
|
|
11
|
+
import { defineHiddenProperty } from '@dxos/live-object';
|
|
12
|
+
import { assumeType, deepMapValues, visitValues } from '@dxos/util';
|
|
13
|
+
|
|
14
|
+
import { EntityKind } from '../ast';
|
|
15
|
+
import { Ref, type RefResolver, refFromEncodedReference, setRefResolver } from '../ref';
|
|
16
|
+
import { type AnyEchoObject } from '../types';
|
|
17
|
+
|
|
18
|
+
import { setSchema } from './accessors';
|
|
19
|
+
import { ObjectMetaSchema } from './meta';
|
|
20
|
+
import {
|
|
21
|
+
ATTR_DELETED,
|
|
22
|
+
ATTR_META,
|
|
23
|
+
ATTR_RELATION_SOURCE,
|
|
24
|
+
ATTR_RELATION_TARGET,
|
|
25
|
+
ATTR_SELF_DXN,
|
|
26
|
+
ATTR_TYPE,
|
|
27
|
+
EntityKindId,
|
|
28
|
+
MetaId,
|
|
29
|
+
type ObjectJSON,
|
|
30
|
+
RelationSourceDXNId,
|
|
31
|
+
RelationSourceId,
|
|
32
|
+
RelationTargetDXNId,
|
|
33
|
+
RelationTargetId,
|
|
34
|
+
SelfDXNId,
|
|
35
|
+
TypeId,
|
|
36
|
+
assertObjectModelShape,
|
|
37
|
+
} from './model';
|
|
38
|
+
import { getType, setTypename } from './typename';
|
|
39
|
+
|
|
40
|
+
type DeepReplaceRef<T> =
|
|
41
|
+
T extends Ref<any> ? EncodedReference : T extends object ? { [K in keyof T]: DeepReplaceRef<T[K]> } : T;
|
|
42
|
+
|
|
43
|
+
type SerializedObject<T extends { id: string }> = { [K in keyof T]: DeepReplaceRef<T[K]> } & ObjectJSON;
|
|
44
|
+
|
|
45
|
+
/**
|
|
46
|
+
* Converts object to it's JSON representation.
|
|
47
|
+
*/
|
|
48
|
+
export const objectToJSON = <T extends AnyEchoObject>(obj: T): SerializedObject<T> => {
|
|
49
|
+
const typename = getType(obj)?.toString();
|
|
50
|
+
invariant(typename && typeof typename === 'string');
|
|
51
|
+
return typedJsonSerializer.call(obj);
|
|
52
|
+
};
|
|
53
|
+
|
|
54
|
+
/**
|
|
55
|
+
* Creates an object from it's json representation.
|
|
56
|
+
* Performs schema validation.
|
|
57
|
+
* References and schema will be resolvable if the `refResolver` is provided.
|
|
58
|
+
*
|
|
59
|
+
* The function need to be async to support resolving the schema as well as the relation endpoints.
|
|
60
|
+
*/
|
|
61
|
+
export const objectFromJSON = async (
|
|
62
|
+
jsonData: unknown,
|
|
63
|
+
{ refResolver, dxn }: { refResolver?: RefResolver; dxn?: DXN } = {},
|
|
64
|
+
): Promise<AnyEchoObject> => {
|
|
65
|
+
assumeType<ObjectJSON>(jsonData);
|
|
66
|
+
assertArgument(typeof jsonData === 'object' && jsonData !== null, 'jsonData', 'expect object');
|
|
67
|
+
assertArgument(typeof jsonData[ATTR_TYPE] === 'string', 'jsonData[ATTR_TYPE]', 'expected object to have a type');
|
|
68
|
+
assertArgument(typeof jsonData.id === 'string', 'jsonData.id', 'expected object to have an id');
|
|
69
|
+
|
|
70
|
+
const type = DXN.parse(jsonData[ATTR_TYPE]);
|
|
71
|
+
const schema = await refResolver?.resolveSchema(type);
|
|
72
|
+
invariant(schema === undefined || Schema.isSchema(schema));
|
|
73
|
+
|
|
74
|
+
let obj: any;
|
|
75
|
+
if (schema != null) {
|
|
76
|
+
obj = await schema.pipe(Schema.decodeUnknownPromise)(jsonData);
|
|
77
|
+
if (refResolver) {
|
|
78
|
+
setRefResolverOnData(obj, refResolver);
|
|
79
|
+
}
|
|
80
|
+
} else {
|
|
81
|
+
obj = decodeGeneric(jsonData, { refResolver });
|
|
82
|
+
}
|
|
83
|
+
|
|
84
|
+
invariant(ObjectId.isValid(obj.id), 'Invalid object id');
|
|
85
|
+
|
|
86
|
+
setTypename(obj, type);
|
|
87
|
+
if (schema) {
|
|
88
|
+
setSchema(obj, schema);
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
const isRelation =
|
|
92
|
+
typeof jsonData[ATTR_RELATION_SOURCE] === 'string' || typeof jsonData[ATTR_RELATION_TARGET] === 'string';
|
|
93
|
+
if (isRelation) {
|
|
94
|
+
const sourceDxn: DXN = DXN.parse(jsonData[ATTR_RELATION_SOURCE] ?? raise(new TypeError('Missing relation source')));
|
|
95
|
+
const targetDxn: DXN = DXN.parse(jsonData[ATTR_RELATION_TARGET] ?? raise(new TypeError('Missing relation target')));
|
|
96
|
+
|
|
97
|
+
// TODO(dmaretskyi): Async!
|
|
98
|
+
const source = (await refResolver?.resolve(sourceDxn)) as AnyEchoObject | undefined;
|
|
99
|
+
const target = (await refResolver?.resolve(targetDxn)) as AnyEchoObject | undefined;
|
|
100
|
+
|
|
101
|
+
defineHiddenProperty(obj, EntityKindId, EntityKind.Relation);
|
|
102
|
+
defineHiddenProperty(obj, RelationSourceDXNId, sourceDxn);
|
|
103
|
+
defineHiddenProperty(obj, RelationTargetDXNId, targetDxn);
|
|
104
|
+
defineHiddenProperty(obj, RelationSourceId, source);
|
|
105
|
+
defineHiddenProperty(obj, RelationTargetId, target);
|
|
106
|
+
} else {
|
|
107
|
+
defineHiddenProperty(obj, EntityKindId, EntityKind.Object);
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
if (typeof jsonData[ATTR_META] === 'object') {
|
|
111
|
+
const meta = await ObjectMetaSchema.pipe(Schema.decodeUnknownPromise)(jsonData[ATTR_META]);
|
|
112
|
+
|
|
113
|
+
// Defensive programming.
|
|
114
|
+
invariant(Array.isArray(meta.keys));
|
|
115
|
+
|
|
116
|
+
defineHiddenProperty(obj, MetaId, meta);
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
if (dxn) {
|
|
120
|
+
defineHiddenProperty(obj, SelfDXNId, dxn);
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
assertObjectModelShape(obj);
|
|
124
|
+
invariant((obj as any)[ATTR_TYPE] === undefined, 'Invalid object model');
|
|
125
|
+
invariant((obj as any)[ATTR_SELF_DXN] === undefined, 'Invalid object model');
|
|
126
|
+
invariant((obj as any)[ATTR_DELETED] === undefined, 'Invalid object model');
|
|
127
|
+
invariant((obj as any)[ATTR_RELATION_SOURCE] === undefined, 'Invalid object model');
|
|
128
|
+
invariant((obj as any)[ATTR_RELATION_TARGET] === undefined, 'Invalid object model');
|
|
129
|
+
invariant((obj as any)[ATTR_META] === undefined, 'Invalid object model');
|
|
130
|
+
return obj;
|
|
131
|
+
};
|
|
132
|
+
|
|
133
|
+
const decodeGeneric = (jsonData: unknown, options: { refResolver?: RefResolver }) => {
|
|
134
|
+
const {
|
|
135
|
+
[ATTR_TYPE]: _type,
|
|
136
|
+
[ATTR_META]: _meta,
|
|
137
|
+
[ATTR_DELETED]: _deleted,
|
|
138
|
+
[ATTR_RELATION_SOURCE]: _relationSource,
|
|
139
|
+
[ATTR_RELATION_TARGET]: _relationTarget,
|
|
140
|
+
[ATTR_SELF_DXN]: _selfDxn,
|
|
141
|
+
...props
|
|
142
|
+
} = jsonData as any;
|
|
143
|
+
|
|
144
|
+
return deepMapValues(props, (value, recurse) => {
|
|
145
|
+
if (isEncodedReference(value)) {
|
|
146
|
+
return refFromEncodedReference(value, options.refResolver);
|
|
147
|
+
}
|
|
148
|
+
return recurse(value);
|
|
149
|
+
});
|
|
150
|
+
};
|
|
151
|
+
|
|
152
|
+
export const setRefResolverOnData = (obj: AnyEchoObject, refResolver: RefResolver) => {
|
|
153
|
+
const go = (value: unknown) => {
|
|
154
|
+
if (Ref.isRef(value)) {
|
|
155
|
+
setRefResolver(value, refResolver);
|
|
156
|
+
} else {
|
|
157
|
+
visitValues(value, go);
|
|
158
|
+
}
|
|
159
|
+
};
|
|
160
|
+
|
|
161
|
+
go(obj);
|
|
162
|
+
};
|
|
163
|
+
|
|
164
|
+
export const attachTypedJsonSerializer = (obj: any) => {
|
|
165
|
+
const descriptor = Object.getOwnPropertyDescriptor(obj, 'toJSON');
|
|
166
|
+
if (descriptor) {
|
|
167
|
+
return;
|
|
168
|
+
}
|
|
169
|
+
|
|
170
|
+
Object.defineProperty(obj, 'toJSON', {
|
|
171
|
+
value: typedJsonSerializer,
|
|
172
|
+
writable: false,
|
|
173
|
+
enumerable: false,
|
|
174
|
+
// Setting `configurable` to false breaks proxy invariants, should be fixable.
|
|
175
|
+
configurable: true,
|
|
176
|
+
});
|
|
177
|
+
};
|
|
178
|
+
|
|
179
|
+
// NOTE: KEEP as function.
|
|
180
|
+
const typedJsonSerializer = function (this: any) {
|
|
181
|
+
const { id, ...rest } = this;
|
|
182
|
+
const result: any = {
|
|
183
|
+
id,
|
|
184
|
+
};
|
|
185
|
+
|
|
186
|
+
if (this[TypeId]) {
|
|
187
|
+
result[ATTR_TYPE] = this[TypeId].toString();
|
|
188
|
+
}
|
|
189
|
+
|
|
190
|
+
if (this[SelfDXNId]) {
|
|
191
|
+
result[ATTR_SELF_DXN] = this[SelfDXNId].toString();
|
|
192
|
+
}
|
|
193
|
+
|
|
194
|
+
if (this[RelationSourceDXNId]) {
|
|
195
|
+
const sourceDXN = this[RelationSourceDXNId];
|
|
196
|
+
invariant(sourceDXN instanceof DXN);
|
|
197
|
+
result[ATTR_RELATION_SOURCE] = sourceDXN.toString();
|
|
198
|
+
}
|
|
199
|
+
if (this[RelationTargetDXNId]) {
|
|
200
|
+
const targetDXN = this[RelationTargetDXNId];
|
|
201
|
+
invariant(targetDXN instanceof DXN);
|
|
202
|
+
result[ATTR_RELATION_TARGET] = targetDXN.toString();
|
|
203
|
+
}
|
|
204
|
+
|
|
205
|
+
if (this[MetaId]) {
|
|
206
|
+
result[ATTR_META] = serializeMeta(this[MetaId]);
|
|
207
|
+
}
|
|
208
|
+
|
|
209
|
+
Object.assign(result, serializeData(rest));
|
|
210
|
+
return result;
|
|
211
|
+
};
|
|
212
|
+
|
|
213
|
+
const serializeData = (data: unknown) => {
|
|
214
|
+
return deepMapValues(data, (value, recurse) => {
|
|
215
|
+
if (Ref.isRef(value)) {
|
|
216
|
+
// TODO(dmaretskyi): Should this be configurable?
|
|
217
|
+
return value.noInline().encode();
|
|
218
|
+
}
|
|
219
|
+
return recurse(value);
|
|
220
|
+
});
|
|
221
|
+
};
|
|
222
|
+
|
|
223
|
+
const serializeMeta = (meta: ObjectMeta) => {
|
|
224
|
+
return deepMapValues(meta, (value, recurse) => recurse(value));
|
|
225
|
+
};
|
|
@@ -0,0 +1,61 @@
|
|
|
1
|
+
//
|
|
2
|
+
// Copyright 2024 DXOS.org
|
|
3
|
+
//
|
|
4
|
+
|
|
5
|
+
import * as Schema from 'effect/Schema';
|
|
6
|
+
|
|
7
|
+
import { ForeignKey } from '@dxos/echo-protocol';
|
|
8
|
+
import { invariant } from '@dxos/invariant';
|
|
9
|
+
import { type Comparator, intersection } from '@dxos/util';
|
|
10
|
+
|
|
11
|
+
import { type BaseObject } from '../types';
|
|
12
|
+
|
|
13
|
+
import { MetaId } from './model';
|
|
14
|
+
|
|
15
|
+
//
|
|
16
|
+
// ObjectMeta
|
|
17
|
+
//
|
|
18
|
+
|
|
19
|
+
// TODO(dmaretskyi): Rename to ObjectMeta
|
|
20
|
+
export const ObjectMetaSchema = Schema.mutable(
|
|
21
|
+
Schema.Struct({
|
|
22
|
+
keys: Schema.mutable(Schema.Array(ForeignKey)),
|
|
23
|
+
|
|
24
|
+
/**
|
|
25
|
+
* A set of tags.
|
|
26
|
+
* Tags are arbitrary application-defined strings.
|
|
27
|
+
* ECHO makes no assumptions about the tag structure.
|
|
28
|
+
*/
|
|
29
|
+
// TODO(dmaretskyi): Has to be optional for compatibility with old data.
|
|
30
|
+
// Defaulting to an empty array is possible but requires a bit more work.
|
|
31
|
+
tags: Schema.optional(Schema.mutable(Schema.Array(Schema.String))),
|
|
32
|
+
}),
|
|
33
|
+
);
|
|
34
|
+
export type ObjectMeta = Schema.Schema.Type<typeof ObjectMetaSchema>;
|
|
35
|
+
|
|
36
|
+
export const foreignKey = (source: string, id: string): ForeignKey => ({ source, id });
|
|
37
|
+
export const foreignKeyEquals = (a: ForeignKey, b: ForeignKey) => a.source === b.source && a.id === b.id;
|
|
38
|
+
|
|
39
|
+
/**
|
|
40
|
+
* Get metadata from object.
|
|
41
|
+
* Only callable on the object root.
|
|
42
|
+
* @deprecated Use {@link getMeta}.
|
|
43
|
+
*/
|
|
44
|
+
// TODO(dmaretskyi): Remove.
|
|
45
|
+
export const getObjectMeta = (obj: any): ObjectMeta => {
|
|
46
|
+
return getMeta(obj);
|
|
47
|
+
};
|
|
48
|
+
|
|
49
|
+
/*
|
|
50
|
+
* Get metadata from object.
|
|
51
|
+
* Only callable on the object root.
|
|
52
|
+
*/
|
|
53
|
+
export const getMeta = (obj: BaseObject): ObjectMeta => {
|
|
54
|
+
const metadata = (obj as any)[MetaId];
|
|
55
|
+
invariant(metadata, 'ObjectMeta not found.');
|
|
56
|
+
return metadata;
|
|
57
|
+
};
|
|
58
|
+
|
|
59
|
+
// TODO(dmaretskyi): Move to echo-schema.
|
|
60
|
+
export const compareForeignKeys: Comparator<BaseObject> = (a: BaseObject, b: BaseObject) =>
|
|
61
|
+
intersection(getMeta(a).keys, getMeta(b).keys, foreignKeyEquals).length > 0;
|
|
@@ -0,0 +1,170 @@
|
|
|
1
|
+
//
|
|
2
|
+
// Copyright 2025 DXOS.org
|
|
3
|
+
//
|
|
4
|
+
|
|
5
|
+
import type * as Schema from 'effect/Schema';
|
|
6
|
+
|
|
7
|
+
import { type ForeignKey } from '@dxos/echo-protocol';
|
|
8
|
+
import { invariant } from '@dxos/invariant';
|
|
9
|
+
import { DXN, ObjectId } from '@dxos/keys';
|
|
10
|
+
import { assumeType } from '@dxos/util';
|
|
11
|
+
|
|
12
|
+
import { EntityKind } from '../ast';
|
|
13
|
+
|
|
14
|
+
import { type ObjectMeta } from './meta';
|
|
15
|
+
import type { Version } from './version';
|
|
16
|
+
|
|
17
|
+
//
|
|
18
|
+
// Defines the internal model of the echo object.
|
|
19
|
+
//
|
|
20
|
+
|
|
21
|
+
/**
|
|
22
|
+
* Entity kind.
|
|
23
|
+
*/
|
|
24
|
+
export const EntityKindId = Symbol('@dxos/echo/EntityKind');
|
|
25
|
+
|
|
26
|
+
/**
|
|
27
|
+
* DXN to the object itself.
|
|
28
|
+
*/
|
|
29
|
+
export const SelfDXNId = Symbol('@dxos/echo/DXN');
|
|
30
|
+
|
|
31
|
+
/**
|
|
32
|
+
* Property name for self DXN when object is serialized to JSON.
|
|
33
|
+
*/
|
|
34
|
+
export const ATTR_SELF_DXN = '@dxn';
|
|
35
|
+
|
|
36
|
+
/**
|
|
37
|
+
* DXN to the object type.
|
|
38
|
+
*/
|
|
39
|
+
export const TypeId = Symbol('@dxos/echo/Type');
|
|
40
|
+
|
|
41
|
+
/**
|
|
42
|
+
* Property name for typename when object is serialized to JSON.
|
|
43
|
+
*/
|
|
44
|
+
export const ATTR_TYPE = '@type';
|
|
45
|
+
|
|
46
|
+
/**
|
|
47
|
+
* Reference to the object schema.
|
|
48
|
+
*/
|
|
49
|
+
export const SchemaId = Symbol('@dxos/echo/Schema');
|
|
50
|
+
|
|
51
|
+
/**
|
|
52
|
+
* Deletion marker.
|
|
53
|
+
*/
|
|
54
|
+
export const DeletedId = Symbol('@dxos/echo/Deleted');
|
|
55
|
+
|
|
56
|
+
/**
|
|
57
|
+
* Property name for deleted when object is serialized to JSON.
|
|
58
|
+
*/
|
|
59
|
+
export const ATTR_DELETED = '@deleted';
|
|
60
|
+
|
|
61
|
+
/**
|
|
62
|
+
* Metadata section.
|
|
63
|
+
*/
|
|
64
|
+
export const MetaId = Symbol('@dxos/echo/Meta');
|
|
65
|
+
|
|
66
|
+
/**
|
|
67
|
+
* Property name for meta when object is serialized to JSON.
|
|
68
|
+
*/
|
|
69
|
+
export const ATTR_META = '@meta';
|
|
70
|
+
|
|
71
|
+
/**
|
|
72
|
+
* Used to access relation source ref on live ECHO objects.
|
|
73
|
+
* Reading this symbol must return `Live<EchoObject<any>>` or a DXN.
|
|
74
|
+
*/
|
|
75
|
+
export const RelationSourceDXNId: unique symbol = Symbol('@dxos/echo/RelationSourceDXN');
|
|
76
|
+
|
|
77
|
+
/**
|
|
78
|
+
* Property name for relation source when object is serialized to JSON.
|
|
79
|
+
*/
|
|
80
|
+
export const ATTR_RELATION_SOURCE = '@relationSource';
|
|
81
|
+
|
|
82
|
+
/**
|
|
83
|
+
* Used to access relation target ref on live ECHO objects.
|
|
84
|
+
* Reading this symbol must return `Live<EchoObject<any>>` or a DXN.
|
|
85
|
+
*/
|
|
86
|
+
export const RelationTargetDXNId: unique symbol = Symbol('@dxos/echo/RelationTargetDXN');
|
|
87
|
+
|
|
88
|
+
/**
|
|
89
|
+
* Property name for relation target when object is serialized to JSON.
|
|
90
|
+
*/
|
|
91
|
+
export const ATTR_RELATION_TARGET = '@relationTarget';
|
|
92
|
+
|
|
93
|
+
/**
|
|
94
|
+
* Used to access relation source ref on live ECHO objects.
|
|
95
|
+
* Reading this symbol must return `Live<EchoObject<any>>` or a DXN.
|
|
96
|
+
*/
|
|
97
|
+
export const RelationSourceId: unique symbol = Symbol('@dxos/echo/RelationSource');
|
|
98
|
+
|
|
99
|
+
/**
|
|
100
|
+
* Used to access relation target ref on live ECHO objects.
|
|
101
|
+
* Reading this symbol must return `Live<EchoObject<any>>` or a DXN.
|
|
102
|
+
*/
|
|
103
|
+
export const RelationTargetId: unique symbol = Symbol('@dxos/echo/RelationTarget');
|
|
104
|
+
|
|
105
|
+
/**
|
|
106
|
+
* Getter to get object version.
|
|
107
|
+
*/
|
|
108
|
+
export const ObjectVersionId: unique symbol = Symbol('@dxos/echo/ObjectVersion');
|
|
109
|
+
|
|
110
|
+
/**
|
|
111
|
+
* Internal runtime representation of an object.
|
|
112
|
+
* The fields are accessed through getter functions.
|
|
113
|
+
*/
|
|
114
|
+
export interface InternalObjectProps {
|
|
115
|
+
id: ObjectId;
|
|
116
|
+
readonly [SelfDXNId]?: DXN;
|
|
117
|
+
|
|
118
|
+
// Echo supports untyped objects O_O.
|
|
119
|
+
readonly [TypeId]?: DXN;
|
|
120
|
+
|
|
121
|
+
/**
|
|
122
|
+
* Returns the schema for the object.
|
|
123
|
+
*/
|
|
124
|
+
readonly [SchemaId]?: Schema.Schema.AnyNoContext;
|
|
125
|
+
readonly [EntityKindId]: EntityKind;
|
|
126
|
+
readonly [DeletedId]?: boolean;
|
|
127
|
+
readonly [MetaId]?: ObjectMeta;
|
|
128
|
+
readonly [RelationSourceDXNId]?: DXN;
|
|
129
|
+
readonly [RelationTargetDXNId]?: DXN;
|
|
130
|
+
readonly [RelationSourceId]?: InternalObjectProps;
|
|
131
|
+
readonly [RelationTargetId]?: InternalObjectProps;
|
|
132
|
+
readonly [ObjectVersionId]?: Version;
|
|
133
|
+
}
|
|
134
|
+
|
|
135
|
+
/**
|
|
136
|
+
* JSON representation of an object or relation.
|
|
137
|
+
*/
|
|
138
|
+
export interface ObjectJSON {
|
|
139
|
+
id: string;
|
|
140
|
+
[ATTR_SELF_DXN]?: DXN.String;
|
|
141
|
+
[ATTR_TYPE]: DXN.String;
|
|
142
|
+
[ATTR_DELETED]?: boolean;
|
|
143
|
+
[ATTR_META]?: ObjectMetaJSON;
|
|
144
|
+
[ATTR_RELATION_SOURCE]?: DXN.String;
|
|
145
|
+
[ATTR_RELATION_TARGET]?: DXN.String;
|
|
146
|
+
}
|
|
147
|
+
|
|
148
|
+
export interface ObjectMetaJSON {
|
|
149
|
+
keys: ForeignKey[];
|
|
150
|
+
tags?: string[];
|
|
151
|
+
}
|
|
152
|
+
|
|
153
|
+
// NOTE: Keep as `function` to avoid type inference issues.
|
|
154
|
+
// eslint-disable-next-line prefer-arrow-functions/prefer-arrow-functions
|
|
155
|
+
export function assertObjectModelShape(obj: unknown): asserts obj is InternalObjectProps {
|
|
156
|
+
invariant(typeof obj === 'object' && obj !== null, 'Invalid object model: not an object');
|
|
157
|
+
assumeType<InternalObjectProps>(obj);
|
|
158
|
+
invariant(ObjectId.isValid(obj.id), 'Invalid object model: invalid id');
|
|
159
|
+
invariant(obj[TypeId] === undefined || obj[TypeId] instanceof DXN, 'Invalid object model: invalid type');
|
|
160
|
+
invariant(
|
|
161
|
+
obj[EntityKindId] === EntityKind.Object || obj[EntityKindId] === EntityKind.Relation,
|
|
162
|
+
'Invalid object model: invalid entity kind',
|
|
163
|
+
);
|
|
164
|
+
if (obj[EntityKindId] === EntityKind.Relation) {
|
|
165
|
+
invariant(obj[RelationSourceDXNId] instanceof DXN, 'Invalid object model: invalid relation source');
|
|
166
|
+
invariant(obj[RelationTargetDXNId] instanceof DXN, 'Invalid object model: invalid relation target');
|
|
167
|
+
invariant(!(obj[RelationSourceId] instanceof DXN), 'Invalid object model: source pointer is a DXN');
|
|
168
|
+
invariant(!(obj[RelationTargetId] instanceof DXN), 'Invalid object model: target pointer is a DXN');
|
|
169
|
+
}
|
|
170
|
+
}
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
//
|
|
2
|
+
// Copyright 2025 DXOS.org
|
|
3
|
+
//
|
|
4
|
+
|
|
5
|
+
import type { RelationSourceId, RelationTargetId } from './model';
|
|
6
|
+
|
|
7
|
+
/**
|
|
8
|
+
* Source and target props on relations.
|
|
9
|
+
*/
|
|
10
|
+
// TODO(burdon): any?
|
|
11
|
+
export type RelationSourceTargetRefs<Source = any, Target = any> = {
|
|
12
|
+
/**
|
|
13
|
+
* Source ECHO live object.
|
|
14
|
+
*/
|
|
15
|
+
[RelationSourceId]: Source;
|
|
16
|
+
|
|
17
|
+
/**
|
|
18
|
+
* Target ECHO live object.
|
|
19
|
+
*/
|
|
20
|
+
[RelationTargetId]: Target;
|
|
21
|
+
};
|
|
22
|
+
|
|
23
|
+
export type RelationSource<R> = R extends RelationSourceTargetRefs<infer Source, infer _Target> ? Source : never;
|
|
24
|
+
export type RelationTarget<R> = R extends RelationSourceTargetRefs<infer _Source, infer Target> ? Target : never;
|
|
@@ -0,0 +1,186 @@
|
|
|
1
|
+
//
|
|
2
|
+
// Copyright 2024 DXOS.org
|
|
3
|
+
//
|
|
4
|
+
|
|
5
|
+
import * as Schema from 'effect/Schema';
|
|
6
|
+
import { describe, expect, test } from 'vitest';
|
|
7
|
+
|
|
8
|
+
import { getDeep } from '@dxos/util';
|
|
9
|
+
|
|
10
|
+
import { SchemaValidator } from './schema-validator';
|
|
11
|
+
|
|
12
|
+
describe('schema-validator', () => {
|
|
13
|
+
describe('validateSchema', () => {
|
|
14
|
+
test('throws on ambiguous discriminated type union', () => {
|
|
15
|
+
const TestSchema = Schema.Struct({
|
|
16
|
+
union: Schema.Union(Schema.Struct({ a: Schema.Number }), Schema.Struct({ b: Schema.String })),
|
|
17
|
+
});
|
|
18
|
+
|
|
19
|
+
expect(() => SchemaValidator.validateSchema(TestSchema)).to.throw();
|
|
20
|
+
});
|
|
21
|
+
});
|
|
22
|
+
|
|
23
|
+
describe('hasPropertyAnnotation', () => {
|
|
24
|
+
test('has annotation', () => {
|
|
25
|
+
const annotationId = Symbol('foo');
|
|
26
|
+
const annotationValue = 'bar';
|
|
27
|
+
const TestSchema: Schema.Schema.AnyNoContext = Schema.Struct({
|
|
28
|
+
name: Schema.String.annotations({ [annotationId]: annotationValue }),
|
|
29
|
+
parent: Schema.optional(Schema.suspend(() => TestSchema.annotations({ [annotationId]: annotationValue }))),
|
|
30
|
+
friends: Schema.suspend(() =>
|
|
31
|
+
Schema.mutable(Schema.Array(TestSchema.annotations({ [annotationId]: annotationValue }))),
|
|
32
|
+
),
|
|
33
|
+
});
|
|
34
|
+
expect(SchemaValidator.hasTypeAnnotation(TestSchema, 'name', annotationId)).to.be.true;
|
|
35
|
+
expect(SchemaValidator.hasTypeAnnotation(TestSchema, 'parent', annotationId)).to.be.true;
|
|
36
|
+
expect(SchemaValidator.hasTypeAnnotation(TestSchema, 'friends', annotationId)).to.be.true;
|
|
37
|
+
});
|
|
38
|
+
|
|
39
|
+
test('no annotation', () => {
|
|
40
|
+
const annotationId = Symbol('foo');
|
|
41
|
+
const Person: Schema.Schema.AnyNoContext = Schema.Struct({
|
|
42
|
+
name: Schema.String,
|
|
43
|
+
parent: Schema.optional(Schema.suspend(() => Person)),
|
|
44
|
+
friends: Schema.suspend(() => Schema.mutable(Schema.Array(Person))),
|
|
45
|
+
});
|
|
46
|
+
expect(SchemaValidator.hasTypeAnnotation(Person, 'name', annotationId)).to.be.false;
|
|
47
|
+
expect(SchemaValidator.hasTypeAnnotation(Person, 'parent', annotationId)).to.be.false;
|
|
48
|
+
expect(SchemaValidator.hasTypeAnnotation(Person, 'friends', annotationId)).to.be.false;
|
|
49
|
+
});
|
|
50
|
+
});
|
|
51
|
+
|
|
52
|
+
describe('getPropertySchema', () => {
|
|
53
|
+
const validateValueToAssign = (args: {
|
|
54
|
+
schema: Schema.Schema.AnyNoContext;
|
|
55
|
+
target: any;
|
|
56
|
+
path: string[];
|
|
57
|
+
valueToAssign: any;
|
|
58
|
+
expectToThrow?: boolean;
|
|
59
|
+
}) => {
|
|
60
|
+
const expectation = expect(() => {
|
|
61
|
+
const nestedSchema = SchemaValidator.getPropertySchema(args.schema, args.path, (path) => {
|
|
62
|
+
return getDeep(args.target, path);
|
|
63
|
+
});
|
|
64
|
+
Schema.validateSync(nestedSchema)(args.valueToAssign);
|
|
65
|
+
});
|
|
66
|
+
if (args.expectToThrow) {
|
|
67
|
+
expectation.to.throw();
|
|
68
|
+
} else {
|
|
69
|
+
expectation.not.to.throw();
|
|
70
|
+
}
|
|
71
|
+
};
|
|
72
|
+
|
|
73
|
+
test('basic', () => {
|
|
74
|
+
for (const value of [42, '42']) {
|
|
75
|
+
validateValueToAssign({
|
|
76
|
+
schema: Schema.Struct({ object: Schema.Struct({ field: Schema.Number }) }),
|
|
77
|
+
target: {},
|
|
78
|
+
path: ['object', 'field'],
|
|
79
|
+
valueToAssign: value,
|
|
80
|
+
expectToThrow: typeof value !== 'number',
|
|
81
|
+
});
|
|
82
|
+
}
|
|
83
|
+
});
|
|
84
|
+
|
|
85
|
+
test('preserves annotations', () => {
|
|
86
|
+
const annotationId = Symbol('foo');
|
|
87
|
+
const annotationValue = 'bar';
|
|
88
|
+
const Person: Schema.Schema.AnyNoContext = Schema.Struct({
|
|
89
|
+
parent: Schema.optional(Schema.suspend(() => Person.annotations({ [annotationId]: annotationValue }))),
|
|
90
|
+
friends: Schema.suspend(() =>
|
|
91
|
+
Schema.mutable(Schema.Array(Person.annotations({ [annotationId]: annotationValue }))),
|
|
92
|
+
),
|
|
93
|
+
});
|
|
94
|
+
expect(SchemaValidator.getPropertySchema(Person, ['parent']).ast.annotations[annotationId]).to.eq(
|
|
95
|
+
annotationValue,
|
|
96
|
+
);
|
|
97
|
+
expect(SchemaValidator.getPropertySchema(Person, ['friends', '0']).ast.annotations[annotationId]).to.eq(
|
|
98
|
+
annotationValue,
|
|
99
|
+
);
|
|
100
|
+
});
|
|
101
|
+
|
|
102
|
+
test('discriminated union', () => {
|
|
103
|
+
const square = Schema.Struct({ type: Schema.Literal('square'), side: Schema.Number });
|
|
104
|
+
const circle = Schema.Struct({ type: Schema.Literal('circle'), radius: Schema.Number });
|
|
105
|
+
const shape = Schema.Union(square, circle);
|
|
106
|
+
validateValueToAssign({
|
|
107
|
+
schema: shape,
|
|
108
|
+
target: { type: 'square' },
|
|
109
|
+
path: ['side'],
|
|
110
|
+
valueToAssign: 1,
|
|
111
|
+
});
|
|
112
|
+
validateValueToAssign({
|
|
113
|
+
schema: shape,
|
|
114
|
+
target: { type: 'circle' },
|
|
115
|
+
path: ['side'],
|
|
116
|
+
valueToAssign: 1,
|
|
117
|
+
expectToThrow: true,
|
|
118
|
+
});
|
|
119
|
+
validateValueToAssign({
|
|
120
|
+
schema: shape,
|
|
121
|
+
target: { type: 'square' },
|
|
122
|
+
path: ['radius'],
|
|
123
|
+
valueToAssign: 1,
|
|
124
|
+
expectToThrow: true,
|
|
125
|
+
});
|
|
126
|
+
});
|
|
127
|
+
|
|
128
|
+
test('any', () => {
|
|
129
|
+
validateValueToAssign({
|
|
130
|
+
schema: Schema.Any,
|
|
131
|
+
target: { field: { nested: { value: Schema.Number } } },
|
|
132
|
+
path: ['field', 'nested'],
|
|
133
|
+
valueToAssign: { any: 'value' },
|
|
134
|
+
});
|
|
135
|
+
});
|
|
136
|
+
test('index signatures', () => {
|
|
137
|
+
for (const value of [42, '42']) {
|
|
138
|
+
validateValueToAssign({
|
|
139
|
+
schema: Schema.Struct({ field: Schema.String }, { key: Schema.String, value: Schema.Number }),
|
|
140
|
+
target: {},
|
|
141
|
+
path: ['unknownField'],
|
|
142
|
+
valueToAssign: value,
|
|
143
|
+
expectToThrow: typeof value !== 'number',
|
|
144
|
+
});
|
|
145
|
+
}
|
|
146
|
+
});
|
|
147
|
+
|
|
148
|
+
test('index signature from optional record', () => {
|
|
149
|
+
for (const value of [42, '42']) {
|
|
150
|
+
validateValueToAssign({
|
|
151
|
+
schema: Schema.Struct({
|
|
152
|
+
field: Schema.optional(Schema.Record({ key: Schema.String, value: Schema.Number })),
|
|
153
|
+
}),
|
|
154
|
+
target: {},
|
|
155
|
+
path: ['field', 'unknownField'],
|
|
156
|
+
valueToAssign: value,
|
|
157
|
+
expectToThrow: typeof value !== 'number',
|
|
158
|
+
});
|
|
159
|
+
}
|
|
160
|
+
});
|
|
161
|
+
|
|
162
|
+
test('suspend', () => {
|
|
163
|
+
const schemaWithSuspend = Schema.Struct({
|
|
164
|
+
array: Schema.optional(Schema.suspend(() => Schema.Array(Schema.Union(Schema.Null, Schema.Number)))),
|
|
165
|
+
object: Schema.optional(
|
|
166
|
+
Schema.suspend(() => Schema.Union(Schema.Null, Schema.Struct({ field: Schema.Number }))),
|
|
167
|
+
),
|
|
168
|
+
});
|
|
169
|
+
const target: any = { array: [1, 2, null], object: { field: 3 } };
|
|
170
|
+
for (const value of [42, '42']) {
|
|
171
|
+
for (const path of [
|
|
172
|
+
['array', '0'],
|
|
173
|
+
['object', 'field'],
|
|
174
|
+
]) {
|
|
175
|
+
validateValueToAssign({
|
|
176
|
+
schema: schemaWithSuspend,
|
|
177
|
+
target,
|
|
178
|
+
path,
|
|
179
|
+
valueToAssign: value,
|
|
180
|
+
expectToThrow: typeof value !== 'number',
|
|
181
|
+
});
|
|
182
|
+
}
|
|
183
|
+
}
|
|
184
|
+
});
|
|
185
|
+
});
|
|
186
|
+
});
|