@dxos/echo 0.8.4-main.b97322e → 0.8.4-main.c4373fc
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/README.md +1 -1
- package/dist/lib/browser/chunk-MWLA34S5.mjs +3843 -0
- package/dist/lib/browser/chunk-MWLA34S5.mjs.map +7 -0
- package/dist/lib/browser/chunk-OAZJQHVO.mjs +453 -0
- package/dist/lib/browser/chunk-OAZJQHVO.mjs.map +7 -0
- package/dist/lib/browser/chunk-ORIE2FMS.mjs +514 -0
- package/dist/lib/browser/chunk-ORIE2FMS.mjs.map +7 -0
- package/dist/lib/browser/index.mjs +26 -5
- package/dist/lib/browser/internal/index.mjs +326 -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 +23 -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-AXWVDOP7.mjs +453 -0
- package/dist/lib/node-esm/chunk-AXWVDOP7.mjs.map +7 -0
- package/dist/lib/node-esm/chunk-L4PBAJDP.mjs +514 -0
- package/dist/lib/node-esm/chunk-L4PBAJDP.mjs.map +7 -0
- package/dist/lib/node-esm/chunk-YTNLFBIK.mjs +3843 -0
- package/dist/lib/node-esm/chunk-YTNLFBIK.mjs.map +7 -0
- package/dist/lib/node-esm/index.mjs +26 -5
- package/dist/lib/node-esm/internal/index.mjs +326 -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 +23 -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 +111 -17
- 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 +11 -8
- package/dist/types/src/Relation.d.ts.map +1 -1
- package/dist/types/src/Type.d.ts +17 -18
- package/dist/types/src/Type.d.ts.map +1 -1
- package/dist/types/src/errors.d.ts +72 -0
- package/dist/types/src/errors.d.ts.map +1 -0
- package/dist/types/src/index.d.ts +3 -1
- 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 +119 -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 +32 -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 +29 -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 +22 -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 +17 -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/typed-relation.d.ts +32 -0
- package/dist/types/src/internal/object/typed-relation.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} +44 -15
- 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 +18 -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 +34 -16
- package/src/Obj.ts +250 -27
- package/src/Ref.ts +1 -2
- package/src/Relation.ts +25 -13
- package/src/Type.ts +32 -31
- package/src/errors.ts +18 -0
- package/src/index.ts +4 -1
- package/src/internal/ast/annotation-helper.ts +22 -0
- package/src/internal/ast/annotations.test.ts +98 -0
- package/src/internal/ast/annotations.ts +212 -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 +726 -0
- package/src/internal/json/json-schema.ts +548 -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 +115 -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 +204 -0
- package/src/internal/object/expando.ts +21 -0
- package/src/internal/object/ids.ts +12 -0
- package/src/internal/object/index.ts +20 -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 +62 -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 +81 -0
- package/src/internal/object/typed-relation.ts +85 -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 +86 -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/{dsl.test.ts → query.test.ts} +41 -2
- package/src/query/{dsl.ts → query.ts} +170 -32
- package/src/query/tag.ts +35 -0
- package/src/test/api.test.ts +17 -10
- 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-EUA7CM23.mjs +0 -619
- package/dist/lib/browser/chunk-EUA7CM23.mjs.map +0 -7
- package/dist/lib/node-esm/chunk-IV6BWGHK.mjs +0 -619
- package/dist/lib/node-esm/chunk-IV6BWGHK.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
|
@@ -0,0 +1,163 @@
|
|
|
1
|
+
//
|
|
2
|
+
// Copyright 2024 DXOS.org
|
|
3
|
+
//
|
|
4
|
+
|
|
5
|
+
import { inspect } from 'node:util';
|
|
6
|
+
|
|
7
|
+
import { describe, expect, test } from 'vitest';
|
|
8
|
+
|
|
9
|
+
import { registerSignalsRuntime } from '@dxos/echo-signals';
|
|
10
|
+
import type { Live } from '@dxos/live-object';
|
|
11
|
+
import { objectData } from '@dxos/live-object';
|
|
12
|
+
import { isNode } from '@dxos/util';
|
|
13
|
+
|
|
14
|
+
import { ATTR_META } from '..';
|
|
15
|
+
import { Testing, updateCounter } from '../testing';
|
|
16
|
+
|
|
17
|
+
import { live } from './reactive-object';
|
|
18
|
+
|
|
19
|
+
registerSignalsRuntime();
|
|
20
|
+
|
|
21
|
+
const TEST_OBJECT: Testing.TestSchema = {
|
|
22
|
+
string: 'foo',
|
|
23
|
+
number: 42,
|
|
24
|
+
boolean: true,
|
|
25
|
+
null: null,
|
|
26
|
+
stringArray: ['1', '2', '3'],
|
|
27
|
+
object: { field: 'bar' },
|
|
28
|
+
};
|
|
29
|
+
|
|
30
|
+
for (const schema of [undefined, Testing.TestSchemaWithClass]) {
|
|
31
|
+
const createObject = (props: Partial<Testing.TestSchemaWithClass> = {}): Live<Testing.TestSchemaWithClass> => {
|
|
32
|
+
return schema == null ? (live(props) as Testing.TestSchemaWithClass) : live(schema, props);
|
|
33
|
+
};
|
|
34
|
+
|
|
35
|
+
describe(`Non-echo specific proxy properties${schema == null ? '' : ' with schema'}`, () => {
|
|
36
|
+
test.skipIf(!isNode())('inspect', () => {
|
|
37
|
+
const obj = createObject({ string: 'bar' });
|
|
38
|
+
const str = inspect(obj, { colors: false });
|
|
39
|
+
expect(str).to.eq(`${schema == null ? '' : 'Typed '}{ string: 'bar' }`);
|
|
40
|
+
});
|
|
41
|
+
|
|
42
|
+
test('data symbol', async () => {
|
|
43
|
+
const obj = createObject({ ...TEST_OBJECT });
|
|
44
|
+
const objData: any = (obj as any)[objectData];
|
|
45
|
+
expect(objData).to.deep.contain({
|
|
46
|
+
'@type': `${schema ? 'Typed' : ''}ReactiveObject`,
|
|
47
|
+
...TEST_OBJECT,
|
|
48
|
+
});
|
|
49
|
+
});
|
|
50
|
+
|
|
51
|
+
test('can assign class instances', () => {
|
|
52
|
+
const obj = createObject();
|
|
53
|
+
|
|
54
|
+
const classInstance = new Testing.TestClass();
|
|
55
|
+
obj.classInstance = classInstance;
|
|
56
|
+
expect(obj.classInstance!.field).to.eq('value');
|
|
57
|
+
expect(obj.classInstance instanceof Testing.TestClass).to.eq(true);
|
|
58
|
+
expect(obj.classInstance === classInstance).to.be.true;
|
|
59
|
+
|
|
60
|
+
obj.classInstance!.field = 'baz';
|
|
61
|
+
expect(obj.classInstance!.field).to.eq('baz');
|
|
62
|
+
});
|
|
63
|
+
|
|
64
|
+
describe('class instance equality', () => {
|
|
65
|
+
test('toJSON', () => {
|
|
66
|
+
const original = { classInstance: new Testing.TestClass() };
|
|
67
|
+
const reactive = createObject(original);
|
|
68
|
+
if (!schema) {
|
|
69
|
+
expect(JSON.stringify(reactive)).to.eq(JSON.stringify(original));
|
|
70
|
+
} else {
|
|
71
|
+
expect(JSON.stringify(reactive)).to.eq(
|
|
72
|
+
JSON.stringify({
|
|
73
|
+
[ATTR_META]: {
|
|
74
|
+
keys: [],
|
|
75
|
+
},
|
|
76
|
+
...original,
|
|
77
|
+
}),
|
|
78
|
+
);
|
|
79
|
+
}
|
|
80
|
+
});
|
|
81
|
+
|
|
82
|
+
test('chai deep equal works', () => {
|
|
83
|
+
const original = { classInstance: new Testing.TestClass() };
|
|
84
|
+
const reactive = createObject(original);
|
|
85
|
+
expect(reactive).to.deep.eq(original);
|
|
86
|
+
expect(reactive).to.not.deep.eq({ ...original, number: 11 });
|
|
87
|
+
});
|
|
88
|
+
|
|
89
|
+
test('jest deep equal works', () => {
|
|
90
|
+
const original = { classInstance: new Testing.TestClass() };
|
|
91
|
+
const reactive = createObject(original);
|
|
92
|
+
expect(reactive).toEqual(original);
|
|
93
|
+
expect(reactive).not.toEqual({ ...original, number: 11 });
|
|
94
|
+
});
|
|
95
|
+
});
|
|
96
|
+
|
|
97
|
+
describe('signal updates', () => {
|
|
98
|
+
test('not in nested class instances', () => {
|
|
99
|
+
const obj = createObject({ classInstance: new Testing.TestClass() });
|
|
100
|
+
using updates = updateCounter(() => {
|
|
101
|
+
obj.classInstance!.field;
|
|
102
|
+
});
|
|
103
|
+
expect(updates.count, 'update count').to.eq(0);
|
|
104
|
+
|
|
105
|
+
obj.classInstance!.field = 'baz';
|
|
106
|
+
expect(updates.count, 'update count').to.eq(0);
|
|
107
|
+
});
|
|
108
|
+
});
|
|
109
|
+
});
|
|
110
|
+
}
|
|
111
|
+
|
|
112
|
+
describe('getters', () => {
|
|
113
|
+
test('add getter to object', () => {
|
|
114
|
+
let value = 'foo';
|
|
115
|
+
const obj = live({
|
|
116
|
+
get getter() {
|
|
117
|
+
return value;
|
|
118
|
+
},
|
|
119
|
+
});
|
|
120
|
+
expect(obj.getter).to.eq('foo');
|
|
121
|
+
|
|
122
|
+
value = 'bar';
|
|
123
|
+
expect(obj.getter).to.eq('bar');
|
|
124
|
+
});
|
|
125
|
+
|
|
126
|
+
test('signal updates', () => {
|
|
127
|
+
const innerObj = live({
|
|
128
|
+
string: 'bar',
|
|
129
|
+
});
|
|
130
|
+
|
|
131
|
+
const obj = live({
|
|
132
|
+
field: 1,
|
|
133
|
+
get getter() {
|
|
134
|
+
return innerObj.string;
|
|
135
|
+
},
|
|
136
|
+
});
|
|
137
|
+
|
|
138
|
+
using updates = updateCounter(() => {
|
|
139
|
+
const value = obj.getter;
|
|
140
|
+
expect(value).to.exist;
|
|
141
|
+
});
|
|
142
|
+
|
|
143
|
+
innerObj.string = 'baz';
|
|
144
|
+
expect(obj.getter).to.eq('baz');
|
|
145
|
+
expect(updates.count, 'update count').to.eq(1);
|
|
146
|
+
|
|
147
|
+
obj.field = 2;
|
|
148
|
+
expect(updates.count, 'update count').to.eq(1);
|
|
149
|
+
});
|
|
150
|
+
|
|
151
|
+
test('getter for array', () => {
|
|
152
|
+
const value = [1];
|
|
153
|
+
const obj = live({
|
|
154
|
+
get getter() {
|
|
155
|
+
return value;
|
|
156
|
+
},
|
|
157
|
+
});
|
|
158
|
+
expect(obj.getter).to.have.length(1);
|
|
159
|
+
|
|
160
|
+
value.push(2);
|
|
161
|
+
expect(obj.getter).to.have.length(2);
|
|
162
|
+
});
|
|
163
|
+
});
|
|
@@ -0,0 +1,108 @@
|
|
|
1
|
+
//
|
|
2
|
+
// Copyright 2024 DXOS.org
|
|
3
|
+
//
|
|
4
|
+
|
|
5
|
+
import type * as Schema from 'effect/Schema';
|
|
6
|
+
|
|
7
|
+
import { ObjectId } from '@dxos/keys';
|
|
8
|
+
import {
|
|
9
|
+
type Live,
|
|
10
|
+
UntypedReactiveHandler,
|
|
11
|
+
createProxy,
|
|
12
|
+
defineHiddenProperty,
|
|
13
|
+
isValidProxyTarget,
|
|
14
|
+
} from '@dxos/live-object';
|
|
15
|
+
|
|
16
|
+
import { getTypeAnnotation } from '../ast';
|
|
17
|
+
import { EntityKindId, Expando, MetaId, type ObjectMeta, ObjectMetaSchema, attachTypedJsonSerializer } from '../object';
|
|
18
|
+
import type { BaseObject, CreationProps } from '../types';
|
|
19
|
+
|
|
20
|
+
import { TypedReactiveHandler, prepareTypedTarget } from './typed-handler';
|
|
21
|
+
|
|
22
|
+
/**
|
|
23
|
+
* Creates a reactive object from a plain Javascript object.
|
|
24
|
+
* Optionally provides a TS-effect schema.
|
|
25
|
+
*
|
|
26
|
+
* @depreacted Use `Obj.make`.
|
|
27
|
+
*/
|
|
28
|
+
// TODO(dmaretskyi): Deep mutability.
|
|
29
|
+
// TODO(dmaretskyi): Invert generics (generic over schema) to have better error messages.
|
|
30
|
+
// TODO(dmaretskyi): Could mutate original object making it unusable.
|
|
31
|
+
// TODO(burdon): Use Schema.make() to handle defaults?
|
|
32
|
+
export const live: {
|
|
33
|
+
<T extends BaseObject>(obj: T): Live<T>;
|
|
34
|
+
<T extends BaseObject>(
|
|
35
|
+
schema: Schema.Schema<T, any, never>,
|
|
36
|
+
obj: NoInfer<CreationProps<T>>,
|
|
37
|
+
meta?: ObjectMeta,
|
|
38
|
+
): Live<T>;
|
|
39
|
+
} = <T extends BaseObject>(
|
|
40
|
+
objOrSchema: Schema.Schema<T, any> | T,
|
|
41
|
+
obj?: CreationProps<T>,
|
|
42
|
+
meta?: ObjectMeta,
|
|
43
|
+
): Live<T> => {
|
|
44
|
+
// TODO(dmaretskyi): Remove Expando special case.
|
|
45
|
+
if (obj && (objOrSchema as any) !== Expando) {
|
|
46
|
+
return createReactiveObject<T>({ ...obj } as T, meta, objOrSchema as Schema.Schema<T, any>);
|
|
47
|
+
} else if (obj && (objOrSchema as any) === Expando) {
|
|
48
|
+
return createReactiveObject<T>({ ...obj } as T, meta, undefined, { expando: true });
|
|
49
|
+
} else {
|
|
50
|
+
return createReactiveObject<T>(objOrSchema as T, meta);
|
|
51
|
+
}
|
|
52
|
+
};
|
|
53
|
+
|
|
54
|
+
const createReactiveObject = <T extends BaseObject>(
|
|
55
|
+
obj: T,
|
|
56
|
+
meta?: ObjectMeta,
|
|
57
|
+
schema?: Schema.Schema<T>,
|
|
58
|
+
options?: { expando?: boolean },
|
|
59
|
+
): Live<T> => {
|
|
60
|
+
if (!isValidProxyTarget(obj)) {
|
|
61
|
+
throw new Error('Value cannot be made into a reactive object.');
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
if (schema) {
|
|
65
|
+
const annotation = getTypeAnnotation(schema);
|
|
66
|
+
const shouldGenerateId = options?.expando || !!annotation;
|
|
67
|
+
if (shouldGenerateId) {
|
|
68
|
+
setIdOnTarget(obj);
|
|
69
|
+
}
|
|
70
|
+
if (annotation) {
|
|
71
|
+
defineHiddenProperty(obj, EntityKindId, annotation.kind);
|
|
72
|
+
}
|
|
73
|
+
initMeta(obj, meta);
|
|
74
|
+
prepareTypedTarget(obj, schema);
|
|
75
|
+
attachTypedJsonSerializer(obj);
|
|
76
|
+
return createProxy<T>(obj, TypedReactiveHandler.instance);
|
|
77
|
+
} else {
|
|
78
|
+
if (options?.expando) {
|
|
79
|
+
setIdOnTarget(obj);
|
|
80
|
+
}
|
|
81
|
+
initMeta(obj, meta);
|
|
82
|
+
return createProxy<T>(obj, UntypedReactiveHandler.instance);
|
|
83
|
+
}
|
|
84
|
+
};
|
|
85
|
+
|
|
86
|
+
/**
|
|
87
|
+
* Set ID on ECHO object targets during creation.
|
|
88
|
+
* Used for objects with schema and the ones explicitly marked as Expando.
|
|
89
|
+
*/
|
|
90
|
+
const setIdOnTarget = (target: any) => {
|
|
91
|
+
// invariant(!('id' in target), 'Object already has an `id` field, which is reserved.');
|
|
92
|
+
if ('id' in target && target.id !== undefined && target.id !== null) {
|
|
93
|
+
if (!ObjectId.isValid(target.id)) {
|
|
94
|
+
throw new Error('Invalid object id format.');
|
|
95
|
+
}
|
|
96
|
+
} else {
|
|
97
|
+
target.id = ObjectId.random();
|
|
98
|
+
}
|
|
99
|
+
};
|
|
100
|
+
|
|
101
|
+
/**
|
|
102
|
+
* Set metadata on object.
|
|
103
|
+
*/
|
|
104
|
+
// TODO(dmaretskyi): Move to echo-schema.
|
|
105
|
+
const initMeta = <T>(obj: T, meta: ObjectMeta = { keys: [] }) => {
|
|
106
|
+
prepareTypedTarget(meta, ObjectMetaSchema);
|
|
107
|
+
defineHiddenProperty(obj, MetaId, createProxy(meta, TypedReactiveHandler.instance as any));
|
|
108
|
+
};
|
|
@@ -0,0 +1,136 @@
|
|
|
1
|
+
//
|
|
2
|
+
// Copyright 2024 DXOS.org
|
|
3
|
+
//
|
|
4
|
+
|
|
5
|
+
import * as Schema from 'effect/Schema';
|
|
6
|
+
import * as SchemaAST from 'effect/SchemaAST';
|
|
7
|
+
import { describe, expect, test } from 'vitest';
|
|
8
|
+
|
|
9
|
+
import { PropertyMeta, TypedObject, getPropertyMetaAnnotation, getTypeAnnotation } from '..';
|
|
10
|
+
import { createEchoSchema } from '../../testing/echo-schema';
|
|
11
|
+
|
|
12
|
+
// TODO(dmaretskyi): Comment.
|
|
13
|
+
class EmptySchemaType extends TypedObject({
|
|
14
|
+
typename: 'example.com/type/Empty',
|
|
15
|
+
version: '0.1.0',
|
|
16
|
+
})({}) {}
|
|
17
|
+
|
|
18
|
+
describe('dynamic schema', () => {
|
|
19
|
+
test('getProperties filters out id and unwraps optionality', async () => {
|
|
20
|
+
class TestSchema extends TypedObject({
|
|
21
|
+
typename: 'example.com/type/Test',
|
|
22
|
+
version: '0.1.0',
|
|
23
|
+
})({
|
|
24
|
+
field1: Schema.String,
|
|
25
|
+
field2: Schema.Boolean,
|
|
26
|
+
}) {}
|
|
27
|
+
|
|
28
|
+
const registered = createEchoSchema(TestSchema);
|
|
29
|
+
expect(registered.getProperties().map((p) => [p.name, p.type])).to.deep.eq([
|
|
30
|
+
['field1', SchemaAST.stringKeyword],
|
|
31
|
+
['field2', SchemaAST.booleanKeyword],
|
|
32
|
+
]);
|
|
33
|
+
});
|
|
34
|
+
|
|
35
|
+
test('addColumns', async () => {
|
|
36
|
+
class TestSchema extends TypedObject({
|
|
37
|
+
typename: 'example.com/type/Test',
|
|
38
|
+
version: '0.1.0',
|
|
39
|
+
})({
|
|
40
|
+
field1: Schema.String,
|
|
41
|
+
}) {}
|
|
42
|
+
|
|
43
|
+
const registered = createEchoSchema(TestSchema);
|
|
44
|
+
registered.addFields({ field2: Schema.Boolean });
|
|
45
|
+
expect(registered.getProperties().map((p) => [p.name, p.type])).to.deep.eq([
|
|
46
|
+
['field1', SchemaAST.stringKeyword],
|
|
47
|
+
['field2', SchemaAST.booleanKeyword],
|
|
48
|
+
]);
|
|
49
|
+
});
|
|
50
|
+
|
|
51
|
+
test('updateColumns preserves order of existing and appends new fields', async () => {
|
|
52
|
+
const registered = createEchoSchema(EmptySchemaType);
|
|
53
|
+
registered.addFields({ field1: Schema.String });
|
|
54
|
+
registered.addFields({ field2: Schema.Boolean });
|
|
55
|
+
registered.addFields({ field3: Schema.Number });
|
|
56
|
+
registered.updateFields({ field4: Schema.Boolean, field2: Schema.String });
|
|
57
|
+
expect(registered.getProperties().map((p) => [p.name, p.type])).to.deep.eq([
|
|
58
|
+
['field1', SchemaAST.stringKeyword],
|
|
59
|
+
['field2', SchemaAST.stringKeyword],
|
|
60
|
+
['field3', SchemaAST.numberKeyword],
|
|
61
|
+
['field4', SchemaAST.booleanKeyword],
|
|
62
|
+
]);
|
|
63
|
+
});
|
|
64
|
+
|
|
65
|
+
test('removeColumns', async () => {
|
|
66
|
+
const registered = createEchoSchema(EmptySchemaType);
|
|
67
|
+
registered.addFields({ field1: Schema.String });
|
|
68
|
+
registered.addFields({ field2: Schema.Boolean });
|
|
69
|
+
registered.addFields({ field3: Schema.Number });
|
|
70
|
+
registered.removeFields(['field2']);
|
|
71
|
+
expect(registered.getProperties().map((p) => [p.name, p.type])).to.deep.eq([
|
|
72
|
+
['field1', SchemaAST.stringKeyword],
|
|
73
|
+
['field3', SchemaAST.numberKeyword],
|
|
74
|
+
]);
|
|
75
|
+
});
|
|
76
|
+
|
|
77
|
+
test('schema manipulations preserve annotations', async () => {
|
|
78
|
+
const metaNamespace = 'dxos.test';
|
|
79
|
+
const metaInfo = { maxLength: 10 };
|
|
80
|
+
const registered = createEchoSchema(EmptySchemaType);
|
|
81
|
+
registered.addFields({
|
|
82
|
+
field1: Schema.String.pipe(PropertyMeta(metaNamespace, metaInfo)),
|
|
83
|
+
field2: Schema.String,
|
|
84
|
+
});
|
|
85
|
+
registered.addFields({ field3: Schema.String });
|
|
86
|
+
registered.updateFields({ field3: Schema.Boolean });
|
|
87
|
+
registered.removeFields(['field2']);
|
|
88
|
+
expect(getTypeAnnotation(registered)).to.deep.contain({
|
|
89
|
+
typename: 'example.com/type/Empty',
|
|
90
|
+
version: '0.1.0',
|
|
91
|
+
});
|
|
92
|
+
expect(getPropertyMetaAnnotation(registered.getProperties()[0], metaNamespace)).to.deep.eq(metaInfo);
|
|
93
|
+
});
|
|
94
|
+
|
|
95
|
+
test('updates typename', async ({ expect }) => {
|
|
96
|
+
// Create schema with some fields and annotations.
|
|
97
|
+
const registered = createEchoSchema(EmptySchemaType);
|
|
98
|
+
const originalVersion = registered.storedSchema.version;
|
|
99
|
+
registered.addFields({
|
|
100
|
+
name: Schema.String.pipe(PropertyMeta('test', { maxLength: 10 })),
|
|
101
|
+
age: Schema.Number,
|
|
102
|
+
});
|
|
103
|
+
|
|
104
|
+
// First update.
|
|
105
|
+
const newTypename1 = 'example.com/type/Individual';
|
|
106
|
+
registered.updateTypename(newTypename1);
|
|
107
|
+
|
|
108
|
+
// Basic typename update checks.
|
|
109
|
+
expect(registered.typename).toBe(newTypename1);
|
|
110
|
+
expect(registered.jsonSchema.$id).toBe(`dxn:type:${newTypename1}`);
|
|
111
|
+
expect(registered.jsonSchema.typename).toBe(newTypename1);
|
|
112
|
+
|
|
113
|
+
// Version preservation check.
|
|
114
|
+
expect(registered.storedSchema.version).toBe(originalVersion);
|
|
115
|
+
|
|
116
|
+
// Field preservation check.
|
|
117
|
+
const properties = registered.getProperties();
|
|
118
|
+
expect(properties).toHaveLength(2);
|
|
119
|
+
expect(properties[0].name).toBe('name');
|
|
120
|
+
|
|
121
|
+
// Annotation preservation check.
|
|
122
|
+
const nameMeta = getPropertyMetaAnnotation(properties[0], 'test');
|
|
123
|
+
expect(nameMeta).toEqual({ maxLength: 10 });
|
|
124
|
+
|
|
125
|
+
// Second update to ensure multiple updates work.
|
|
126
|
+
const newTypename2 = 'example.com/type/Person';
|
|
127
|
+
registered.updateTypename(newTypename2);
|
|
128
|
+
expect(registered.typename).toBe(newTypename2);
|
|
129
|
+
expect(registered.jsonSchema.$id).toBe(`dxn:type:${newTypename2}`);
|
|
130
|
+
expect(registered.jsonSchema.typename).toBe(newTypename2);
|
|
131
|
+
expect(getTypeAnnotation(registered)).to.deep.contain({
|
|
132
|
+
typename: 'example.com/type/Person',
|
|
133
|
+
version: '0.1.0',
|
|
134
|
+
});
|
|
135
|
+
});
|
|
136
|
+
});
|
|
@@ -0,0 +1,102 @@
|
|
|
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 { Ref, TypedObject, create, foreignKey, getMeta, getSchema, isInstanceOf } from '..';
|
|
9
|
+
import { Testing } from '../testing';
|
|
10
|
+
|
|
11
|
+
import { live } from './reactive-object';
|
|
12
|
+
|
|
13
|
+
describe('complex schema validations', () => {
|
|
14
|
+
const setValue = (target: any, prop: string, value: any) => {
|
|
15
|
+
target[prop] = value;
|
|
16
|
+
};
|
|
17
|
+
|
|
18
|
+
test('any', () => {
|
|
19
|
+
const schema = Schema.Struct({ field: Schema.Any });
|
|
20
|
+
const object = live(schema, { field: { nested: { value: 100 } } });
|
|
21
|
+
expect(() => setValue(object, 'field', { any: 'value' })).not.to.throw();
|
|
22
|
+
});
|
|
23
|
+
|
|
24
|
+
test('meta', () => {
|
|
25
|
+
const source = 'test';
|
|
26
|
+
const schema = Schema.Struct({ field: Schema.Number });
|
|
27
|
+
const object = live(schema, { field: 42 }, { keys: [foreignKey(source, '123')] });
|
|
28
|
+
expect(getMeta(object).keys).to.deep.eq([foreignKey(source, '123')]);
|
|
29
|
+
});
|
|
30
|
+
|
|
31
|
+
test('object', () => {
|
|
32
|
+
const schema = Schema.Struct({ field: Schema.optional(Schema.Object) });
|
|
33
|
+
const object = live(schema, { field: { nested: { value: 100 } } });
|
|
34
|
+
expect(() => setValue(object, 'field', { any: 'value' })).not.to.throw();
|
|
35
|
+
});
|
|
36
|
+
|
|
37
|
+
test('references', () => {
|
|
38
|
+
class Foo extends TypedObject({ typename: 'example.com/type/Foo', version: '0.1.0' })({ field: Schema.String }) {}
|
|
39
|
+
class Bar extends TypedObject({ typename: 'example.com/type/Bar', version: '0.1.0' })({ fooRef: Ref(Foo) }) {}
|
|
40
|
+
const field = 'hello';
|
|
41
|
+
expect(() => live(Bar, { fooRef: { id: '1', field } as any })).to.throw();
|
|
42
|
+
expect(() => live(Bar, { fooRef: undefined as any })).to.throw(); // Unresolved reference.
|
|
43
|
+
const bar = live(Bar, { fooRef: Ref.make(live(Foo, { field })) });
|
|
44
|
+
expect(bar.fooRef.target?.field).to.eq(field);
|
|
45
|
+
});
|
|
46
|
+
|
|
47
|
+
test('index signatures', () => {
|
|
48
|
+
const schema = Schema.Struct({}, { key: Schema.String, value: Schema.Number });
|
|
49
|
+
const object = live(schema, { unknownField: 1 });
|
|
50
|
+
expect(() => setValue(object, 'field', '42')).to.throw();
|
|
51
|
+
expect(() => setValue(object, 'unknown_field', 42)).not.to.throw();
|
|
52
|
+
});
|
|
53
|
+
|
|
54
|
+
test('suspend', () => {
|
|
55
|
+
const schema = Schema.Struct({
|
|
56
|
+
array: Schema.optional(Schema.suspend(() => Schema.Array(Schema.Union(Schema.Null, Schema.Number)))),
|
|
57
|
+
object: Schema.optional(Schema.suspend(() => Schema.Union(Schema.Null, Schema.Struct({ field: Schema.Number })))),
|
|
58
|
+
});
|
|
59
|
+
|
|
60
|
+
const object = live(schema, { array: [1, 2, null], object: { field: 3 } });
|
|
61
|
+
expect(() => setValue(object, 'object', { field: 4 })).not.to.throw();
|
|
62
|
+
expect(() => setValue(object.object, 'field', 4)).not.to.throw();
|
|
63
|
+
expect(() => setValue(object.array, '0', 4)).not.to.throw();
|
|
64
|
+
expect(() => setValue(object.array, '0', '4')).to.throw();
|
|
65
|
+
});
|
|
66
|
+
|
|
67
|
+
test('nesting static objects with schema in the live object', () => {
|
|
68
|
+
const contact1 = create(Testing.Contact, {
|
|
69
|
+
name: 'Robert Smith',
|
|
70
|
+
email: 'robert@example.com',
|
|
71
|
+
} as any);
|
|
72
|
+
const contact2 = create(Testing.Contact, {
|
|
73
|
+
name: 'Katy Perry',
|
|
74
|
+
email: 'katy@example.com',
|
|
75
|
+
} as any);
|
|
76
|
+
|
|
77
|
+
const contactBook = live({
|
|
78
|
+
contacts: [contact1],
|
|
79
|
+
});
|
|
80
|
+
|
|
81
|
+
expect(isInstanceOf(Testing.Contact, contactBook.contacts[0])).to.eq(true);
|
|
82
|
+
expect(getSchema(contactBook.contacts[0])).to.eq(Testing.Contact);
|
|
83
|
+
|
|
84
|
+
contactBook.contacts.push(contact2);
|
|
85
|
+
expect(isInstanceOf(Testing.Contact, contactBook.contacts[1])).to.eq(true);
|
|
86
|
+
expect(getSchema(contactBook.contacts[1])).to.eq(Testing.Contact);
|
|
87
|
+
});
|
|
88
|
+
|
|
89
|
+
test('creating an object with data from another object', () => {
|
|
90
|
+
const contact = live(Testing.Contact, {
|
|
91
|
+
name: 'Robert Smith',
|
|
92
|
+
email: 'robert@example.com',
|
|
93
|
+
});
|
|
94
|
+
const TestSchema = Schema.Struct({
|
|
95
|
+
value: Schema.Unknown,
|
|
96
|
+
});
|
|
97
|
+
const data = live(TestSchema, {
|
|
98
|
+
value: contact,
|
|
99
|
+
});
|
|
100
|
+
expect((data.value as any).name).to.eq('Robert Smith');
|
|
101
|
+
});
|
|
102
|
+
});
|