@dxos/echo 0.8.4-staging.ac66bdf99f → 0.9.0
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 +7 -7
- package/dist/lib/neutral/Annotation.mjs +37 -6
- package/dist/lib/neutral/Database.mjs +6 -17
- package/dist/lib/neutral/Entity.mjs +31 -20
- package/dist/lib/neutral/Err.mjs +3 -3
- package/dist/lib/neutral/Feed.mjs +23 -19
- package/dist/lib/neutral/Filter.mjs +13 -15
- package/dist/lib/neutral/Format.mjs +23 -3
- package/dist/lib/neutral/JsonSchema.mjs +7 -8
- package/dist/lib/neutral/Key.mjs +9 -5
- package/dist/lib/neutral/Migration.mjs +11 -10
- package/dist/lib/neutral/Obj.mjs +29 -20
- package/dist/lib/neutral/Order.mjs +7 -3
- package/dist/lib/neutral/Query.mjs +17 -17
- package/dist/lib/neutral/QueryResult.mjs +1 -1
- package/dist/lib/neutral/Ref.mjs +10 -9
- package/dist/lib/neutral/Registry.mjs +14 -0
- package/dist/lib/neutral/Relation.mjs +28 -25
- package/dist/lib/neutral/Scope.mjs +12 -0
- package/dist/lib/neutral/Tag.mjs +17 -14
- package/dist/lib/neutral/Type.mjs +54 -26
- package/dist/lib/neutral/{chunk-OMUPQMLR.mjs → chunk-35INCYOE.mjs} +1 -1
- package/dist/lib/neutral/chunk-35INCYOE.mjs.map +7 -0
- package/dist/lib/neutral/chunk-3PBP4V4O.mjs +101 -0
- package/dist/lib/neutral/chunk-3PBP4V4O.mjs.map +7 -0
- package/dist/lib/neutral/chunk-4ZUHOTCG.mjs +184 -0
- package/dist/lib/neutral/chunk-4ZUHOTCG.mjs.map +7 -0
- package/dist/lib/neutral/chunk-5SMDBFVB.mjs +108 -0
- package/dist/lib/neutral/chunk-5SMDBFVB.mjs.map +7 -0
- package/dist/lib/neutral/{chunk-OS35K56T.mjs → chunk-5SUJPHAE.mjs} +3 -3
- package/dist/lib/neutral/{chunk-OS35K56T.mjs.map → chunk-5SUJPHAE.mjs.map} +2 -2
- package/dist/lib/neutral/{chunk-GZQTCRJB.mjs → chunk-6M2Z6WBH.mjs} +22 -2
- package/dist/lib/neutral/chunk-6M2Z6WBH.mjs.map +7 -0
- package/dist/lib/neutral/{chunk-V36VO5SS.mjs → chunk-6YDI3J37.mjs} +32 -40
- package/dist/lib/neutral/chunk-6YDI3J37.mjs.map +7 -0
- package/dist/lib/neutral/{chunk-MOR5ERFM.mjs → chunk-7FPIAJIV.mjs} +701 -1256
- package/dist/lib/neutral/chunk-7FPIAJIV.mjs.map +7 -0
- package/dist/lib/neutral/{chunk-ANHVGJI4.mjs → chunk-7LOUAPYZ.mjs} +9 -5
- package/dist/lib/neutral/chunk-7LOUAPYZ.mjs.map +7 -0
- package/dist/lib/neutral/{chunk-JUXPFOEI.mjs → chunk-7PI7C4EF.mjs} +48 -88
- package/dist/lib/neutral/chunk-7PI7C4EF.mjs.map +7 -0
- package/dist/lib/neutral/{chunk-UBEZSGXY.mjs → chunk-BBFJWWAV.mjs} +6 -6
- package/dist/lib/neutral/chunk-BBFJWWAV.mjs.map +7 -0
- package/dist/lib/neutral/{chunk-UI6MWK5W.mjs → chunk-EVK6XBXO.mjs} +16 -2
- package/dist/lib/neutral/chunk-EVK6XBXO.mjs.map +7 -0
- package/dist/lib/neutral/{chunk-7RO7CPBZ.mjs → chunk-IGK6FN65.mjs} +2 -2
- package/dist/lib/neutral/{chunk-HBUZJNZO.mjs → chunk-LWXVKPPW.mjs} +94 -99
- package/dist/lib/neutral/chunk-LWXVKPPW.mjs.map +7 -0
- package/dist/lib/neutral/{chunk-BVOFLCVF.mjs → chunk-MZ7K3MLL.mjs} +9 -6
- package/dist/lib/neutral/chunk-MZ7K3MLL.mjs.map +7 -0
- package/dist/lib/neutral/{chunk-TBKX6JQO.mjs → chunk-O6BH7EPN.mjs} +30 -3
- package/dist/lib/neutral/chunk-O6BH7EPN.mjs.map +7 -0
- package/dist/lib/neutral/{chunk-EAMSSLZC.mjs → chunk-QQIYS74I.mjs} +83 -46
- package/dist/lib/neutral/chunk-QQIYS74I.mjs.map +7 -0
- package/dist/lib/neutral/chunk-R5W6DXR4.mjs +678 -0
- package/dist/lib/neutral/chunk-R5W6DXR4.mjs.map +7 -0
- package/dist/lib/neutral/{chunk-WAK4DMFV.mjs → chunk-RIVWNMSF.mjs} +12 -7
- package/dist/lib/neutral/chunk-RIVWNMSF.mjs.map +7 -0
- package/dist/lib/neutral/{chunk-T6W2LEZU.mjs → chunk-SBVFRTST.mjs} +73 -38
- package/dist/lib/neutral/chunk-SBVFRTST.mjs.map +7 -0
- package/dist/lib/neutral/chunk-T6E37YIP.mjs +251 -0
- package/dist/lib/neutral/chunk-T6E37YIP.mjs.map +7 -0
- package/dist/lib/neutral/{chunk-DQYLD2RB.mjs → chunk-TFEWTY5A.mjs} +155 -129
- package/dist/lib/neutral/chunk-TFEWTY5A.mjs.map +7 -0
- package/dist/lib/neutral/{chunk-B4BASU6W.mjs → chunk-TYGKCRMK.mjs} +85 -76
- package/dist/lib/neutral/chunk-TYGKCRMK.mjs.map +7 -0
- package/dist/lib/neutral/{chunk-4OIBYSXE.mjs → chunk-UUP46KUQ.mjs} +78 -32
- package/dist/lib/neutral/chunk-UUP46KUQ.mjs.map +7 -0
- package/dist/lib/neutral/chunk-WISOH2XH.mjs +36 -0
- package/dist/lib/neutral/chunk-WISOH2XH.mjs.map +7 -0
- package/dist/lib/neutral/{chunk-G3IQMKF7.mjs → chunk-WTQJHC75.mjs} +111 -112
- package/dist/lib/neutral/chunk-WTQJHC75.mjs.map +7 -0
- package/dist/lib/neutral/chunk-WU3GIANS.mjs +31 -0
- package/dist/lib/neutral/chunk-WU3GIANS.mjs.map +7 -0
- package/dist/lib/neutral/{chunk-TU3GW67D.mjs → chunk-ZGNNFYHS.mjs} +40 -40
- package/dist/lib/neutral/chunk-ZGNNFYHS.mjs.map +7 -0
- package/dist/lib/neutral/index.mjs +47 -41
- package/dist/lib/neutral/internal/index.mjs +137 -72
- package/dist/lib/neutral/meta.json +1 -1
- package/dist/lib/neutral/testing/index.mjs +261 -178
- package/dist/lib/neutral/testing/index.mjs.map +4 -4
- package/dist/types/src/Annotation.d.ts +108 -4
- package/dist/types/src/Annotation.d.ts.map +1 -1
- package/dist/types/src/Annotation.test.d.ts +2 -0
- package/dist/types/src/Annotation.test.d.ts.map +1 -0
- package/dist/types/src/Collection.d.ts +2 -3
- package/dist/types/src/Collection.d.ts.map +1 -1
- package/dist/types/src/Database.d.ts +56 -49
- package/dist/types/src/Database.d.ts.map +1 -1
- package/dist/types/src/Dataset.d.ts +16 -6
- package/dist/types/src/Dataset.d.ts.map +1 -1
- package/dist/types/src/Entity.d.ts +101 -28
- package/dist/types/src/Entity.d.ts.map +1 -1
- package/dist/types/src/Err.d.ts +27 -27
- package/dist/types/src/Err.d.ts.map +1 -1
- package/dist/types/src/Feed.d.ts +66 -19
- package/dist/types/src/Feed.d.ts.map +1 -1
- package/dist/types/src/Filter.d.ts +38 -12
- package/dist/types/src/Filter.d.ts.map +1 -1
- package/dist/types/src/Format.d.ts +0 -2
- package/dist/types/src/Format.d.ts.map +1 -1
- package/dist/types/src/Hypergraph.d.ts +14 -9
- 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 +2 -2
- package/dist/types/src/JsonSchema.d.ts.map +1 -1
- package/dist/types/src/Key.d.ts +1 -1
- package/dist/types/src/Key.d.ts.map +1 -1
- package/dist/types/src/Migration.d.ts +26 -11
- package/dist/types/src/Migration.d.ts.map +1 -1
- package/dist/types/src/Obj.d.ts +104 -61
- package/dist/types/src/Obj.d.ts.map +1 -1
- package/dist/types/src/Order.d.ts +10 -0
- package/dist/types/src/Order.d.ts.map +1 -1
- package/dist/types/src/Query.d.ts +34 -12
- package/dist/types/src/Query.d.ts.map +1 -1
- package/dist/types/src/QueryResult.d.ts +21 -0
- package/dist/types/src/QueryResult.d.ts.map +1 -1
- package/dist/types/src/Ref.d.ts +15 -7
- package/dist/types/src/Ref.d.ts.map +1 -1
- package/dist/types/src/Registry.d.ts +131 -0
- package/dist/types/src/Registry.d.ts.map +1 -0
- package/dist/types/src/Relation.d.ts +73 -41
- package/dist/types/src/Relation.d.ts.map +1 -1
- package/dist/types/src/Scope.d.ts +35 -0
- package/dist/types/src/Scope.d.ts.map +1 -0
- package/dist/types/src/Tag.d.ts +21 -5
- package/dist/types/src/Tag.d.ts.map +1 -1
- package/dist/types/src/Type.d.ts +362 -95
- package/dist/types/src/Type.d.ts.map +1 -1
- package/dist/types/src/View.d.ts +9 -12
- 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 +4 -3
- package/dist/types/src/index.d.ts.map +1 -1
- package/dist/types/src/internal/Annotation/annotations.d.ts +79 -38
- package/dist/types/src/internal/Annotation/annotations.d.ts.map +1 -1
- package/dist/types/src/internal/Annotation/dictionary.d.ts +24 -0
- package/dist/types/src/internal/Annotation/dictionary.d.ts.map +1 -0
- package/dist/types/src/internal/Annotation/entity-dictionary.d.ts +14 -0
- package/dist/types/src/internal/Annotation/entity-dictionary.d.ts.map +1 -0
- package/dist/types/src/internal/Annotation/index.d.ts +4 -2
- package/dist/types/src/internal/Annotation/index.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 +14 -5
- package/dist/types/src/internal/Annotation/util.d.ts.map +1 -1
- package/dist/types/src/internal/Entity/api.d.ts +17 -3
- package/dist/types/src/internal/Entity/api.d.ts.map +1 -1
- package/dist/types/src/internal/Entity/entity.d.ts +72 -8
- package/dist/types/src/internal/Entity/entity.d.ts.map +1 -1
- package/dist/types/src/internal/Entity/guard.d.ts +10 -0
- package/dist/types/src/internal/Entity/guard.d.ts.map +1 -0
- package/dist/types/src/internal/Entity/index.d.ts +2 -0
- package/dist/types/src/internal/Entity/index.d.ts.map +1 -1
- package/dist/types/src/internal/Entity/model.d.ts +21 -17
- package/dist/types/src/internal/Entity/model.d.ts.map +1 -1
- package/dist/types/src/internal/Entity/object.d.ts +3 -3
- package/dist/types/src/internal/Entity/object.d.ts.map +1 -1
- package/dist/types/src/internal/Entity/relation.d.ts +30 -7
- package/dist/types/src/internal/Entity/relation.d.ts.map +1 -1
- package/dist/types/src/internal/Entity/type-kind.d.ts +24 -0
- package/dist/types/src/internal/Entity/type-kind.d.ts.map +1 -0
- package/dist/types/src/internal/Entity/type-uri.d.ts +24 -0
- package/dist/types/src/internal/Entity/type-uri.d.ts.map +1 -0
- 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 +3 -2
- package/dist/types/src/internal/Format/format.d.ts.map +1 -1
- package/dist/types/src/internal/Format/index.d.ts +2 -2
- package/dist/types/src/internal/Format/index.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 +3 -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 +34 -34
- package/dist/types/src/internal/JsonSchema/json-schema-type.d.ts.map +1 -1
- package/dist/types/src/internal/JsonSchema/json-schema.d.ts +3 -2
- package/dist/types/src/internal/JsonSchema/json-schema.d.ts.map +1 -1
- package/dist/types/src/internal/Obj/atoms.d.ts +38 -0
- package/dist/types/src/internal/Obj/atoms.d.ts.map +1 -0
- 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 +12 -12
- 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/index.d.ts +1 -1
- package/dist/types/src/internal/Obj/index.d.ts.map +1 -1
- package/dist/types/src/internal/Obj/json-serializer.d.ts +8 -8
- 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/Obj/typed-object.d.ts +1 -1
- 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.d.ts → Query/pretty.d.ts} +1 -1
- package/dist/types/src/internal/Query/pretty.d.ts.map +1 -0
- package/dist/types/src/internal/Ref/atoms.d.ts +10 -0
- package/dist/types/src/internal/Ref/atoms.d.ts.map +1 -0
- package/dist/types/src/internal/Ref/ref-array.d.ts +2 -2
- package/dist/types/src/internal/Ref/ref.d.ts +50 -19
- package/dist/types/src/internal/Ref/ref.d.ts.map +1 -1
- package/dist/types/src/internal/Ref/utils.d.ts +8 -0
- package/dist/types/src/internal/Ref/utils.d.ts.map +1 -0
- package/dist/types/src/internal/Type/compose.d.ts.map +1 -1
- package/dist/types/src/internal/Type/index.d.ts +1 -2
- package/dist/types/src/internal/Type/index.d.ts.map +1 -1
- package/dist/types/src/internal/Type/manipulation.d.ts +0 -1
- package/dist/types/src/internal/Type/manipulation.d.ts.map +1 -1
- package/dist/types/src/internal/Type/type-schema.d.ts +52 -0
- package/dist/types/src/internal/Type/type-schema.d.ts.map +1 -0
- package/dist/types/src/internal/common/api/meta.d.ts +14 -11
- 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 +11 -5
- 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 +18 -2
- package/dist/types/src/internal/common/proxy/typed-handler.d.ts.map +1 -1
- package/dist/types/src/internal/common/types/base.d.ts +4 -4
- package/dist/types/src/internal/common/types/base.d.ts.map +1 -1
- package/dist/types/src/internal/common/types/entity.d.ts +62 -5
- package/dist/types/src/internal/common/types/entity.d.ts.map +1 -1
- package/dist/types/src/internal/common/types/index.d.ts +1 -1
- package/dist/types/src/internal/common/types/index.d.ts.map +1 -1
- package/dist/types/src/internal/common/types/meta.d.ts +33 -12
- package/dist/types/src/internal/common/types/meta.d.ts.map +1 -1
- package/dist/types/src/internal/common/types/model-symbols.d.ts +15 -4
- package/dist/types/src/internal/common/types/model-symbols.d.ts.map +1 -1
- package/dist/types/src/internal/common/types/typename.d.ts +7 -0
- package/dist/types/src/internal/common/types/typename.d.ts.map +1 -1
- package/dist/types/src/internal/common/types/version.d.ts +1 -1
- package/dist/types/src/internal/common/types/well-known-types.d.ts +11 -0
- package/dist/types/src/internal/common/types/well-known-types.d.ts.map +1 -0
- package/dist/types/src/internal/index.d.ts +2 -2
- package/dist/types/src/internal/index.d.ts.map +1 -1
- package/dist/types/src/testing/index.d.ts +1 -0
- package/dist/types/src/testing/index.d.ts.map +1 -1
- package/dist/types/src/testing/registry.d.ts +9 -0
- package/dist/types/src/testing/registry.d.ts.map +1 -0
- 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 +83 -89
- package/dist/types/src/testing/test-schema.d.ts.map +1 -1
- package/dist/types/src/testing/util.d.ts +5 -3
- package/dist/types/src/testing/util.d.ts.map +1 -1
- package/dist/types/tsconfig.tsbuildinfo +1 -1
- package/package.json +26 -24
- package/src/Annotation.test.ts +439 -0
- package/src/Annotation.ts +158 -4
- package/src/Collection.ts +5 -9
- package/src/Database.ts +93 -100
- package/src/Dataset.ts +10 -2
- package/src/Entity.test.ts +116 -6
- package/src/Entity.ts +134 -32
- package/src/Err.ts +4 -4
- package/src/Feed.ts +92 -44
- package/src/Filter.ts +70 -40
- package/src/Format.ts +0 -4
- package/src/Hypergraph.ts +14 -9
- package/src/Json.test.ts +175 -0
- package/src/Json.ts +103 -0
- package/src/Key.ts +1 -1
- package/src/Migration.ts +39 -19
- package/src/Obj.test.ts +122 -20
- package/src/Obj.ts +168 -91
- package/src/Order.ts +22 -0
- package/src/Query.test.ts +183 -154
- package/src/Query.ts +172 -85
- package/src/QueryResult.ts +26 -0
- package/src/Ref.ts +22 -4
- package/src/Registry.ts +155 -0
- package/src/Relation.test.ts +10 -10
- package/src/Relation.ts +116 -69
- package/src/Scope.ts +50 -0
- package/src/Tag.md +88 -0
- package/src/Tag.ts +49 -6
- package/src/Type.test.ts +223 -18
- package/src/Type.ts +609 -131
- package/src/View.ts +14 -23
- package/src/exemplars.test.ts +21 -0
- package/src/index.ts +4 -4
- package/src/internal/Annotation/annotations.test.ts +31 -11
- package/src/internal/Annotation/annotations.ts +143 -111
- package/src/internal/Annotation/dictionary.ts +47 -0
- package/src/internal/Annotation/entity-dictionary.ts +74 -0
- package/src/internal/Annotation/index.ts +4 -2
- package/src/internal/Annotation/util.ts +17 -8
- package/src/internal/Entity/api.ts +54 -7
- package/src/internal/Entity/entity.ts +196 -47
- package/src/internal/Entity/guard.ts +26 -0
- package/src/internal/Entity/index.ts +2 -0
- package/src/internal/Entity/model.ts +38 -28
- package/src/internal/Entity/object.ts +21 -5
- package/src/internal/Entity/relation.ts +68 -34
- package/src/internal/Entity/type-kind.ts +75 -0
- package/src/internal/Entity/type-uri.ts +92 -0
- package/src/internal/Entity/util.ts +9 -9
- package/src/internal/Format/date.ts +0 -4
- package/src/internal/Format/format.test.ts +21 -0
- package/src/internal/Format/index.ts +2 -3
- package/src/internal/Format/object.ts +21 -4
- package/src/internal/Format/types.ts +1 -1
- package/src/internal/JsonSchema/annotations.ts +1 -1
- package/src/internal/JsonSchema/json-schema-type.ts +4 -4
- package/src/internal/JsonSchema/json-schema.test.ts +71 -145
- package/src/internal/JsonSchema/json-schema.ts +49 -35
- package/src/internal/Obj/atoms.ts +244 -0
- package/src/internal/Obj/clone.ts +9 -4
- package/src/internal/Obj/create-object.test.ts +12 -10
- package/src/internal/Obj/create-object.ts +68 -22
- package/src/internal/Obj/index.ts +1 -1
- package/src/internal/Obj/inspect.ts +5 -3
- package/src/internal/Obj/json-serializer.test.ts +101 -22
- package/src/internal/Obj/json-serializer.ts +89 -33
- package/src/internal/Obj/set-value.test.ts +22 -45
- package/src/internal/Obj/set-value.ts +12 -19
- package/src/internal/Obj/snapshot.ts +13 -4
- package/src/internal/Obj/typed-object.test.ts +9 -11
- package/src/internal/Obj/typed-object.ts +1 -1
- package/src/internal/Query/index.ts +5 -0
- package/src/internal/{Query.ts → Query/pretty.ts} +40 -12
- package/src/internal/Ref/atoms.ts +20 -0
- package/src/internal/Ref/ref-array.ts +3 -3
- package/src/internal/Ref/ref.test.ts +18 -27
- package/src/internal/Ref/ref.ts +137 -59
- package/src/internal/Ref/utils.ts +45 -0
- package/src/internal/Type/compose.test.ts +3 -1
- package/src/internal/Type/index.ts +1 -2
- package/src/internal/Type/manipulation.ts +0 -25
- package/src/internal/Type/type-schema.ts +60 -0
- package/src/internal/common/README.md +2 -2
- package/src/internal/common/api/meta.ts +19 -17
- package/src/internal/common/proxy/change-context.ts +1 -1
- package/src/internal/common/proxy/change.test.ts +91 -83
- package/src/internal/common/proxy/errors.ts +2 -2
- package/src/internal/common/proxy/handler.test.ts +1 -1
- package/src/internal/common/proxy/json-serializer.ts +27 -16
- package/src/internal/common/proxy/make-object.ts +44 -20
- package/src/internal/common/proxy/ownership.ts +2 -2
- 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 -2
- package/src/internal/common/proxy/schema.test.ts +48 -86
- package/src/internal/common/proxy/typed-handler.test.ts +12 -11
- package/src/internal/common/proxy/typed-handler.ts +78 -16
- package/src/internal/common/proxy/typed-object.test.ts +16 -28
- package/src/internal/common/types/base.ts +4 -4
- package/src/internal/common/types/entity.ts +80 -1
- package/src/internal/common/types/index.ts +6 -1
- package/src/internal/common/types/meta.ts +62 -20
- package/src/internal/common/types/model-symbols.ts +24 -4
- package/src/internal/common/types/typename.ts +39 -3
- package/src/internal/common/types/well-known-types.ts +15 -0
- package/src/internal/index.ts +6 -4
- package/src/testing/api.test.ts +15 -9
- package/src/testing/index.ts +1 -0
- package/src/testing/registry.ts +44 -0
- package/src/testing/test-data.ts +159 -99
- package/src/testing/test-schema.ts +22 -58
- package/src/testing/util.ts +14 -11
- package/dist/lib/neutral/Extension.mjs +0 -18
- package/dist/lib/neutral/SchemaRegistry.mjs +0 -2
- package/dist/lib/neutral/chunk-2KHZ36F5.mjs +0 -361
- package/dist/lib/neutral/chunk-2KHZ36F5.mjs.map +0 -7
- package/dist/lib/neutral/chunk-4OIBYSXE.mjs.map +0 -7
- package/dist/lib/neutral/chunk-4P3IXBLT.mjs +0 -45
- package/dist/lib/neutral/chunk-4P3IXBLT.mjs.map +0 -7
- package/dist/lib/neutral/chunk-ANHVGJI4.mjs.map +0 -7
- package/dist/lib/neutral/chunk-B4BASU6W.mjs.map +0 -7
- package/dist/lib/neutral/chunk-BNCCGLJN.mjs +0 -7
- package/dist/lib/neutral/chunk-BNCCGLJN.mjs.map +0 -7
- package/dist/lib/neutral/chunk-BVOFLCVF.mjs.map +0 -7
- package/dist/lib/neutral/chunk-DQYLD2RB.mjs.map +0 -7
- package/dist/lib/neutral/chunk-EAMSSLZC.mjs.map +0 -7
- package/dist/lib/neutral/chunk-G3IQMKF7.mjs.map +0 -7
- package/dist/lib/neutral/chunk-GZQTCRJB.mjs.map +0 -7
- package/dist/lib/neutral/chunk-HBUZJNZO.mjs.map +0 -7
- package/dist/lib/neutral/chunk-JUXPFOEI.mjs.map +0 -7
- package/dist/lib/neutral/chunk-MOR5ERFM.mjs.map +0 -7
- package/dist/lib/neutral/chunk-OMUPQMLR.mjs.map +0 -7
- package/dist/lib/neutral/chunk-PHU22NLC.mjs +0 -136
- package/dist/lib/neutral/chunk-PHU22NLC.mjs.map +0 -7
- package/dist/lib/neutral/chunk-ROG4RXXL.mjs +0 -97
- package/dist/lib/neutral/chunk-ROG4RXXL.mjs.map +0 -7
- package/dist/lib/neutral/chunk-T6W2LEZU.mjs.map +0 -7
- package/dist/lib/neutral/chunk-TBKX6JQO.mjs.map +0 -7
- package/dist/lib/neutral/chunk-TU3GW67D.mjs.map +0 -7
- package/dist/lib/neutral/chunk-UBEZSGXY.mjs.map +0 -7
- package/dist/lib/neutral/chunk-UI6MWK5W.mjs.map +0 -7
- package/dist/lib/neutral/chunk-V36VO5SS.mjs.map +0 -7
- package/dist/lib/neutral/chunk-WAK4DMFV.mjs.map +0 -7
- package/dist/lib/neutral/chunk-YAHXAYOW.mjs +0 -56
- package/dist/lib/neutral/chunk-YAHXAYOW.mjs.map +0 -7
- package/dist/lib/neutral/chunk-YS6Q3XAD.mjs +0 -50
- package/dist/lib/neutral/chunk-YS6Q3XAD.mjs.map +0 -7
- package/dist/types/src/Extension.d.ts +0 -80
- package/dist/types/src/Extension.d.ts.map +0 -1
- package/dist/types/src/Extension.test.d.ts +0 -2
- package/dist/types/src/Extension.test.d.ts.map +0 -1
- package/dist/types/src/SchemaRegistry.d.ts +0 -84
- package/dist/types/src/SchemaRegistry.d.ts.map +0 -1
- package/dist/types/src/internal/Obj/ids.d.ts +0 -6
- package/dist/types/src/internal/Obj/ids.d.ts.map +0 -1
- package/dist/types/src/internal/Query.d.ts.map +0 -1
- package/dist/types/src/internal/Type/echo-schema.d.ts +0 -181
- package/dist/types/src/internal/Type/echo-schema.d.ts.map +0 -1
- package/dist/types/src/internal/Type/persistent-schema.d.ts +0 -20
- package/dist/types/src/internal/Type/persistent-schema.d.ts.map +0 -1
- package/src/Extension.test.ts +0 -235
- package/src/Extension.ts +0 -122
- package/src/SchemaRegistry.ts +0 -106
- package/src/internal/Obj/ids.ts +0 -12
- package/src/internal/Type/echo-schema.ts +0 -423
- package/src/internal/Type/persistent-schema.ts +0 -33
- /package/dist/lib/neutral/{Extension.mjs.map → Registry.mjs.map} +0 -0
- /package/dist/lib/neutral/{SchemaRegistry.mjs.map → Scope.mjs.map} +0 -0
- /package/dist/lib/neutral/{chunk-7RO7CPBZ.mjs.map → chunk-IGK6FN65.mjs.map} +0 -0
package/src/Type.ts
CHANGED
|
@@ -4,70 +4,91 @@
|
|
|
4
4
|
|
|
5
5
|
// @import-as-namespace
|
|
6
6
|
|
|
7
|
-
import
|
|
7
|
+
import * as Schema from 'effect/Schema';
|
|
8
|
+
import type * as Types from 'effect/Types';
|
|
8
9
|
|
|
10
|
+
import { raise } from '@dxos/debug';
|
|
9
11
|
import { type EncodedReference } from '@dxos/echo-protocol';
|
|
10
|
-
import { invariant } from '@dxos/invariant';
|
|
11
|
-
import { type
|
|
12
|
-
import { type ToMutable } from '@dxos/util';
|
|
12
|
+
import { assertArgument, invariant } from '@dxos/invariant';
|
|
13
|
+
import { DXN, EID, type EntityId, type URI } from '@dxos/keys';
|
|
13
14
|
|
|
14
|
-
import type * as
|
|
15
|
+
import type * as Database from './Database';
|
|
16
|
+
import type * as EntityModule from './Entity';
|
|
15
17
|
import * as internal from './internal';
|
|
16
18
|
import * as typeInternal from './internal/Type';
|
|
17
|
-
import type * as ObjModule from './Obj';
|
|
18
19
|
import type * as RelationModule from './Relation';
|
|
19
20
|
|
|
20
|
-
/**
|
|
21
|
-
* Dynamic type that can be constructed, mutated, and persisted in the ECHO database.
|
|
22
|
-
*/
|
|
23
|
-
export const RuntimeType = typeInternal.EchoSchema;
|
|
24
|
-
|
|
25
|
-
/**
|
|
26
|
-
* Dynamic type that can be constructed, mutated, and persisted in the ECHO database.
|
|
27
|
-
*/
|
|
28
|
-
export type RuntimeType = typeInternal.EchoSchema;
|
|
29
|
-
|
|
30
21
|
//
|
|
31
22
|
// Internal types (not exported)
|
|
32
23
|
//
|
|
33
24
|
|
|
34
25
|
/**
|
|
35
|
-
*
|
|
36
|
-
*
|
|
26
|
+
* Structural base shared by the three sibling type-entity interfaces
|
|
27
|
+
* ({@link Obj}, {@link Relation}, {@link Type}). NOT exported — callers
|
|
28
|
+
* should constrain on {@link AnyEntity} when they want "any of the three"
|
|
29
|
+
* and on the specific kind interface otherwise.
|
|
37
30
|
*/
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
31
|
+
interface BaseTypeEntity<A> {
|
|
32
|
+
/**
|
|
33
|
+
* Entity-kind brand of the type-entity value itself — always `EntityKind.Type`.
|
|
34
|
+
* The kind of instance the type *describes* lives on `[SchemaKindId]`
|
|
35
|
+
* (Object / Relation / Type). Lets `Obj.isObject` / `Relation.isRelation`
|
|
36
|
+
* reject type entities by a single `[KindId]` check.
|
|
37
|
+
*/
|
|
38
|
+
readonly [internal.KindId]: internal.EntityKind.Type;
|
|
39
|
+
|
|
40
|
+
/**
|
|
41
|
+
* Object id. Like all ECHO entities, type entities always carry an id —
|
|
42
|
+
* stamped at construction for in-memory (static) declarations and assigned by
|
|
43
|
+
* the database once persisted. The id does NOT determine the entity's URI:
|
|
44
|
+
* static types resolve to their typename DXN, persisted types to `echo:/<id>`
|
|
45
|
+
* (see `getTypeURIFromSpecifier`).
|
|
46
|
+
*/
|
|
47
|
+
readonly id: EntityId;
|
|
48
|
+
|
|
49
|
+
readonly name?: string;
|
|
50
|
+
// NOTE: `typename` / `version` are intentionally NOT fields on any type-entity
|
|
51
|
+
// interface. Both static and persisted entities carry them in `EntityMeta`
|
|
52
|
+
// (`key` / `version`); read them via `Type.getTypename(self)` /
|
|
53
|
+
// `Type.getVersion(self)` — never as a direct property.
|
|
54
|
+
readonly jsonSchema: internal.JsonSchemaType;
|
|
55
|
+
readonly [InstancePhantomId]?: A;
|
|
56
|
+
}
|
|
41
57
|
|
|
42
58
|
//
|
|
43
|
-
// Obj
|
|
59
|
+
// Obj — `Type.Type` value for an ECHO object schema.
|
|
44
60
|
//
|
|
45
61
|
|
|
46
62
|
/**
|
|
47
|
-
* TypeScript type for an ECHO object
|
|
48
|
-
*
|
|
49
|
-
* `
|
|
63
|
+
* TypeScript type for an ECHO object type — a `Type.Type<A>` entity.
|
|
64
|
+
*
|
|
65
|
+
* `T` is the instance type produced by `Obj.make(Foo, props)`. `Fields` is
|
|
66
|
+
* retained as a structural hint (the runtime value still carries `.fields`),
|
|
67
|
+
* but consumers should derive instance/encoded types via `Type.InstanceType`.
|
|
68
|
+
*
|
|
69
|
+
* **Not a `Schema.Schema`.** `Foo.ast` / `Schema.Schema.Type<typeof Foo>` /
|
|
70
|
+
* `Schema.extend(Foo)` no longer typecheck — extract the Effect Schema via
|
|
71
|
+
* `Type.getSchema(Foo)` first, or derive instance types via
|
|
72
|
+
* `Type.InstanceType<typeof Foo>`.
|
|
50
73
|
*
|
|
51
74
|
* @example
|
|
52
75
|
* ```ts
|
|
53
|
-
* const
|
|
76
|
+
* const Person = Schema.Struct({
|
|
54
77
|
* name: Schema.String,
|
|
55
|
-
* }).pipe(Type.
|
|
78
|
+
* }).pipe(Type.makeObject(DXN.make('com.example.type.person', '0.1.0')));
|
|
56
79
|
*
|
|
57
|
-
*
|
|
58
|
-
* Object.keys(PersonSchema.fields); // ['name']
|
|
80
|
+
* type Person = Type.InstanceType<typeof Person>;
|
|
59
81
|
* ```
|
|
60
82
|
*/
|
|
61
|
-
export interface Obj<T, Fields extends Schema.Struct.Fields = Schema.Struct.Fields>
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
> {
|
|
83
|
+
export interface Obj<T, Fields extends Schema.Struct.Fields = Schema.Struct.Fields> extends BaseTypeEntity<
|
|
84
|
+
T & EntityModule.OfKind<typeof EntityModule.Kind.Object>
|
|
85
|
+
> {
|
|
86
|
+
/** Schema-kind brand (object). */
|
|
87
|
+
readonly [internal.SchemaKindId]: internal.EntityKind.Object;
|
|
88
|
+
|
|
89
|
+
/** Source Effect Schema — used internally by `Type.getSchema(self)`. */
|
|
90
|
+
readonly [internal.StaticTypeSchemaSlot]: Schema.Schema.AnyNoContext;
|
|
91
|
+
|
|
71
92
|
/**
|
|
72
93
|
* The fields defined in the original struct schema.
|
|
73
94
|
* Allows accessing field definitions for introspection.
|
|
@@ -76,61 +97,157 @@ export interface Obj<T, Fields extends Schema.Struct.Fields = Schema.Struct.Fiel
|
|
|
76
97
|
}
|
|
77
98
|
|
|
78
99
|
/**
|
|
79
|
-
*
|
|
80
|
-
*
|
|
81
|
-
* NOTE: Does not include the brand symbol to avoid TS4053 declaration portability issues.
|
|
82
|
-
* Use Type.isObjectSchema() for runtime type guards.
|
|
83
|
-
*/
|
|
84
|
-
type ObjectSchemaBase = Schema.Schema.AnyNoContext & {
|
|
85
|
-
readonly typename: string;
|
|
86
|
-
readonly version: string;
|
|
87
|
-
};
|
|
88
|
-
|
|
89
|
-
/**
|
|
90
|
-
* Type that represents any ECHO object schema.
|
|
91
|
-
* Accepts both static schemas (Type.object()) and mutable schemas (EchoSchema).
|
|
100
|
+
* Type that represents any ECHO object type — a `Type.Type` entity branded
|
|
101
|
+
* with the object entity kind, i.e. what `Type.makeObject(dxn)` produces.
|
|
92
102
|
*/
|
|
93
|
-
export type AnyObj =
|
|
103
|
+
export type AnyObj = Obj<unknown>;
|
|
94
104
|
|
|
95
105
|
/**
|
|
96
|
-
* Factory function to create an ECHO object
|
|
97
|
-
*
|
|
106
|
+
* Factory function to create an ECHO object type.
|
|
107
|
+
*
|
|
108
|
+
* Returns a `Type.Type` entity — a live, in-memory `TypeSchema` instance,
|
|
109
|
+
* NOT a `Schema.Schema`. Use `Type.InstanceType<typeof Foo>` for the instance
|
|
110
|
+
* type and `Type.getSchema(Foo)` to obtain the underlying Effect Schema.
|
|
111
|
+
*
|
|
112
|
+
* The entity's id defaults to `EntityId.deterministic(typename, version)` so
|
|
113
|
+
* constructing a type never reaches `crypto.getRandomValues()` — required for
|
|
114
|
+
* Cloudflare workerd, which forbids RNG calls in global (module-evaluation)
|
|
115
|
+
* scope. Pass `{ id }` to override (e.g. with `EntityId.random()` from a
|
|
116
|
+
* request handler).
|
|
98
117
|
*
|
|
99
118
|
* @example
|
|
100
119
|
* ```ts
|
|
101
120
|
* const Person = Schema.Struct({
|
|
102
121
|
* name: Schema.String,
|
|
103
|
-
* }).pipe(Type.
|
|
122
|
+
* }).pipe(Type.makeObject(DXN.make('com.example.type.person', '0.1.0')));
|
|
104
123
|
* ```
|
|
105
124
|
*/
|
|
106
|
-
export const
|
|
107
|
-
(
|
|
125
|
+
export const makeObject: {
|
|
126
|
+
(
|
|
127
|
+
dxn: DXN.DXN,
|
|
128
|
+
options?: { id?: EntityId },
|
|
129
|
+
): <Self extends Schema.Schema.Any>(self: Self) => Obj<Schema.Schema.Type<Self>>;
|
|
108
130
|
} = internal.EchoObjectSchema as any;
|
|
109
131
|
|
|
110
132
|
//
|
|
111
|
-
//
|
|
133
|
+
// Type — the ECHO entity that holds a schema and metadata.
|
|
134
|
+
// Persisted via `db.addType()`; subscribed to via `Filter.type(Type.Type)`.
|
|
112
135
|
//
|
|
113
136
|
|
|
114
|
-
|
|
137
|
+
/**
|
|
138
|
+
* ECHO meta-schema entity — stores `{ name?, typename, version, jsonSchema }`.
|
|
139
|
+
* Type-kind sibling of `Type.makeObject(...)` / `Type.makeRelation(...)` outputs.
|
|
140
|
+
* Stored types live under this entity; filter via `Filter.type(Type.Type)`.
|
|
141
|
+
*/
|
|
142
|
+
export const Type: Type<typeInternal.TypeSchema> = typeInternal.TypeSchema as any;
|
|
143
|
+
|
|
144
|
+
/**
|
|
145
|
+
* Default version stamped on draft (unnamed) types created via
|
|
146
|
+
* {@link makeObjectFromJsonSchema} / {@link makeRelationFromJsonSchema} when
|
|
147
|
+
* the caller does not supply one. Pure dynamic drafts surface as `'0.0.0'`
|
|
148
|
+
* until they are persisted, at which point automerge-heads suffix the version.
|
|
149
|
+
*/
|
|
150
|
+
const DRAFT_VERSION = '0.0.0';
|
|
115
151
|
|
|
116
|
-
|
|
152
|
+
/**
|
|
153
|
+
* Common props shared by the type-kind factories. Typename and version are
|
|
154
|
+
* optional — drafts omit typename and default version to {@link DRAFT_VERSION}.
|
|
155
|
+
*/
|
|
156
|
+
type MakeTypeProps = {
|
|
157
|
+
jsonSchema: internal.JsonSchemaType;
|
|
158
|
+
typename?: string;
|
|
159
|
+
version?: string;
|
|
160
|
+
name?: string;
|
|
161
|
+
id?: EntityId;
|
|
162
|
+
};
|
|
117
163
|
|
|
118
164
|
/**
|
|
119
|
-
*
|
|
120
|
-
*
|
|
121
|
-
*
|
|
122
|
-
* `
|
|
165
|
+
* Construct a new object-kind type entity from raw metadata — for cases where
|
|
166
|
+
* an Effect Schema isn't available (e.g. JSON-Schema arriving over the network
|
|
167
|
+
* or from a UI editor). Parallel to {@link makeObject} but takes pre-built
|
|
168
|
+
* `jsonSchema` instead of piping through an Effect schema.
|
|
169
|
+
*
|
|
170
|
+
* The returned entity is in-memory; persist it with `db.addType(entity)`.
|
|
123
171
|
*/
|
|
124
|
-
export
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
172
|
+
export const makeObjectFromJsonSchema = (props: MakeTypeProps): Type<typeInternal.TypeSchema> => {
|
|
173
|
+
const { typename, version, ...data } = props;
|
|
174
|
+
// `typename` / `version` are routed through `EntityMeta` (`key` / `version`)
|
|
175
|
+
// — the canonical registry-provenance pair — not data fields. Drafts default
|
|
176
|
+
// to `'0.0.0'`; the version is omitted from meta entirely when the caller
|
|
177
|
+
// doesn't supply one so the proxy projection can apply its own default.
|
|
178
|
+
return internal.makeObject(
|
|
179
|
+
internal.getStaticTypeSchema(typeInternal.TypeSchema) as any,
|
|
180
|
+
data as any,
|
|
181
|
+
{
|
|
182
|
+
keys: [],
|
|
183
|
+
key: typename,
|
|
184
|
+
version: version ?? DRAFT_VERSION,
|
|
185
|
+
},
|
|
186
|
+
typeInternal.TypeSchema,
|
|
187
|
+
) as unknown as Type<typeInternal.TypeSchema>;
|
|
188
|
+
};
|
|
189
|
+
|
|
190
|
+
/**
|
|
191
|
+
* Construct a new relation-kind type entity from raw metadata. Parallel to
|
|
192
|
+
* {@link makeRelation} but takes pre-built `jsonSchema` instead of piping
|
|
193
|
+
* through an Effect schema. `source` / `target` accept either a static
|
|
194
|
+
* `Type.Obj` entity or the well-known `Obj.Unknown` schema.
|
|
195
|
+
*
|
|
196
|
+
* The returned entity is in-memory; persist it with `db.addType(entity)`.
|
|
197
|
+
*/
|
|
198
|
+
export const makeRelationFromJsonSchema = (
|
|
199
|
+
props: MakeTypeProps & {
|
|
200
|
+
source: AnyObj | internal.UnknownTypeSchema<any, typeof EntityModule.Kind.Object>;
|
|
201
|
+
target: AnyObj | internal.UnknownTypeSchema<any, typeof EntityModule.Kind.Object>;
|
|
202
|
+
},
|
|
203
|
+
): Type<typeInternal.TypeSchema> => {
|
|
204
|
+
const { source, target, jsonSchema, typename, version, ...rest } = props;
|
|
205
|
+
// Embed source/target DXNs + relation entity-kind into the jsonSchema so the
|
|
206
|
+
// entity round-trips correctly through `toEffectSchema` / queries / refs.
|
|
207
|
+
const sourceURI = internal.getTypeURIFromSpecifier(source);
|
|
208
|
+
const targetURI = internal.getTypeURIFromSpecifier(target);
|
|
209
|
+
const enrichedJsonSchema: internal.JsonSchemaType = {
|
|
210
|
+
...jsonSchema,
|
|
211
|
+
entityKind: internal.EntityKind.Relation,
|
|
212
|
+
relationSource: { $ref: sourceURI },
|
|
213
|
+
relationTarget: { $ref: targetURI },
|
|
214
|
+
};
|
|
215
|
+
// `typename` / `version` route through `EntityMeta` (see
|
|
216
|
+
// {@link makeObjectFromJsonSchema}); drafts default version to `'0.0.0'`.
|
|
217
|
+
return internal.makeObject(
|
|
218
|
+
internal.getStaticTypeSchema(typeInternal.TypeSchema) as any,
|
|
219
|
+
{ ...rest, jsonSchema: enrichedJsonSchema } as any,
|
|
220
|
+
{
|
|
221
|
+
keys: [],
|
|
222
|
+
key: typename,
|
|
223
|
+
version: version ?? DRAFT_VERSION,
|
|
224
|
+
},
|
|
225
|
+
typeInternal.TypeSchema,
|
|
226
|
+
) as unknown as Type<typeInternal.TypeSchema>;
|
|
227
|
+
};
|
|
228
|
+
|
|
229
|
+
/**
|
|
230
|
+
* TypeScript type for an ECHO relation type — a `Type.Type<A>` entity.
|
|
231
|
+
*
|
|
232
|
+
* `T` is the instance-property type produced by `Relation.make(...)` (excluding
|
|
233
|
+
* source/target endpoints). `Source` and `Target` are the endpoint types.
|
|
234
|
+
*
|
|
235
|
+
* **Not a `Schema.Schema`.** See {@link Obj}'s note.
|
|
236
|
+
*/
|
|
237
|
+
export interface Relation<
|
|
238
|
+
T,
|
|
239
|
+
Source,
|
|
240
|
+
Target,
|
|
241
|
+
Fields extends Schema.Struct.Fields = Schema.Struct.Fields,
|
|
242
|
+
> extends BaseTypeEntity<
|
|
243
|
+
RelationModule.Endpoints<Source, Target> & T & EntityModule.OfKind<typeof EntityModule.Kind.Relation>
|
|
244
|
+
> {
|
|
245
|
+
/** Schema-kind brand (relation). */
|
|
246
|
+
readonly [internal.SchemaKindId]: internal.EntityKind.Relation;
|
|
247
|
+
|
|
248
|
+
/** Source Effect Schema — used internally by `Type.getSchema(self)`. */
|
|
249
|
+
readonly [internal.StaticTypeSchemaSlot]: Schema.Schema.AnyNoContext;
|
|
250
|
+
|
|
134
251
|
/**
|
|
135
252
|
* The fields defined in the original struct schema.
|
|
136
253
|
* Allows accessing field definitions for introspection.
|
|
@@ -139,21 +256,10 @@ export interface Relation<T, Source, Target, Fields extends Schema.Struct.Fields
|
|
|
139
256
|
}
|
|
140
257
|
|
|
141
258
|
/**
|
|
142
|
-
*
|
|
143
|
-
*
|
|
144
|
-
* NOTE: Does not include the brand symbol to avoid TS4053 declaration portability issues.
|
|
145
|
-
* Use Type.isRelationSchema() for runtime type guards.
|
|
259
|
+
* Type that represents any ECHO relation type — a `Type.Type` entity branded
|
|
260
|
+
* with the relation entity kind, i.e. what `Type.makeRelation(...)` produces.
|
|
146
261
|
*/
|
|
147
|
-
type
|
|
148
|
-
readonly typename: string;
|
|
149
|
-
readonly version: string;
|
|
150
|
-
};
|
|
151
|
-
|
|
152
|
-
/**
|
|
153
|
-
* Type that represents any ECHO relation schema.
|
|
154
|
-
* Accepts static schemas (Type.relation()).
|
|
155
|
-
*/
|
|
156
|
-
export type AnyRelation = RelationSchemaBase;
|
|
262
|
+
export type AnyRelation = Relation<unknown, unknown, unknown>;
|
|
157
263
|
|
|
158
264
|
/**
|
|
159
265
|
* Factory function to create an ECHO relation schema.
|
|
@@ -163,42 +269,93 @@ export type AnyRelation = RelationSchemaBase;
|
|
|
163
269
|
* ```ts
|
|
164
270
|
* const WorksFor = Schema.Struct({
|
|
165
271
|
* role: Schema.String,
|
|
166
|
-
* }).pipe(Type.
|
|
167
|
-
*
|
|
168
|
-
* version: '0.1.0',
|
|
272
|
+
* }).pipe(Type.makeRelation({
|
|
273
|
+
* dxn: DXN.make('com.example.type.worksFor', '0.1.0'),
|
|
169
274
|
* source: Person,
|
|
170
275
|
* target: Company,
|
|
171
276
|
* }));
|
|
172
277
|
* ```
|
|
173
278
|
*/
|
|
174
|
-
export const
|
|
175
|
-
<
|
|
176
|
-
|
|
177
|
-
|
|
279
|
+
export const makeRelation: {
|
|
280
|
+
<SourceInstance, TargetInstance>(opts: {
|
|
281
|
+
dxn: DXN.DXN;
|
|
282
|
+
source: Obj<SourceInstance, any> | internal.UnknownTypeSchema<SourceInstance, typeof EntityModule.Kind.Object>;
|
|
283
|
+
target: Obj<TargetInstance, any> | internal.UnknownTypeSchema<TargetInstance, typeof EntityModule.Kind.Object>;
|
|
284
|
+
/**
|
|
285
|
+
* Override the entity id. Defaults to `EntityId.deterministic(typename, version)`;
|
|
286
|
+
* see `Type.makeObject` for the workerd motivation.
|
|
287
|
+
*/
|
|
288
|
+
id?: EntityId;
|
|
289
|
+
}): <Self extends Schema.Schema.Any>(
|
|
178
290
|
self: Self,
|
|
179
|
-
) => Relation<
|
|
291
|
+
) => Relation<
|
|
292
|
+
Schema.Schema.Type<Self>,
|
|
293
|
+
SourceInstance & EntityModule.OfKind<typeof EntityModule.Kind.Object>,
|
|
294
|
+
TargetInstance & EntityModule.OfKind<typeof EntityModule.Kind.Object>
|
|
295
|
+
>;
|
|
180
296
|
} = internal.EchoRelationSchema as any;
|
|
181
297
|
|
|
182
298
|
/**
|
|
183
|
-
* Type
|
|
184
|
-
*
|
|
299
|
+
* Type that represents any ECHO type-kind entity — a `Type.Type` meta-schema
|
|
300
|
+
* value (static `Type.Type` or a persisted draft from `db.addType(...)`).
|
|
301
|
+
* Mirrors {@link AnyObj} / {@link AnyRelation} for the third sibling kind.
|
|
302
|
+
*/
|
|
303
|
+
export type AnyType = Type<unknown>;
|
|
304
|
+
|
|
305
|
+
/**
|
|
306
|
+
* Any ECHO type-entity — one of the three sibling kinds: object-kind, relation-kind,
|
|
307
|
+
* or type-kind (the meta-schema). APIs that want "any ECHO type" use this union;
|
|
308
|
+
* the underlying Effect Schema is retrieved via `Type.getSchema`.
|
|
185
309
|
*/
|
|
186
|
-
export type AnyEntity = AnyObj | AnyRelation;
|
|
310
|
+
export type AnyEntity = AnyObj | AnyRelation | AnyType;
|
|
187
311
|
|
|
188
312
|
/**
|
|
189
|
-
* Type guard
|
|
190
|
-
*
|
|
313
|
+
* Type guard: narrows a `Type.AnyEntity` to an object-kind entity. Checks
|
|
314
|
+
* ENTITIES, not instances — use `Obj.isObject` for instances. Raw
|
|
315
|
+
* `Schema.Schema` values (including the branded `Obj.Unknown` companion)
|
|
316
|
+
* are intentionally not accepted; inspect their `TypeAnnotation` directly.
|
|
191
317
|
*/
|
|
192
|
-
export const
|
|
193
|
-
return (
|
|
318
|
+
export const isObject = (entity: AnyEntity): entity is AnyObj => {
|
|
319
|
+
return internal.getSchemaKind(entity) === internal.EntityKind.Object;
|
|
194
320
|
};
|
|
195
321
|
|
|
196
322
|
/**
|
|
197
|
-
* Type guard
|
|
198
|
-
*
|
|
323
|
+
* Type guard: narrows a `Type.AnyEntity` to a relation-kind entity. Checks
|
|
324
|
+
* ENTITIES, not instances — use `Relation.isRelation` for instances.
|
|
199
325
|
*/
|
|
200
|
-
export const
|
|
201
|
-
return (
|
|
326
|
+
export const isRelation = (entity: AnyEntity): entity is AnyRelation => {
|
|
327
|
+
return internal.getSchemaKind(entity) === internal.EntityKind.Relation;
|
|
328
|
+
};
|
|
329
|
+
|
|
330
|
+
/**
|
|
331
|
+
* Type guard: narrows a `Type.AnyEntity` to the type-kind meta-schema
|
|
332
|
+
* (e.g. `Type.Type`). Mirrors {@link isObject} / {@link isRelation}.
|
|
333
|
+
*/
|
|
334
|
+
export const isTypeKind = (entity: AnyEntity): entity is Type => {
|
|
335
|
+
return internal.getSchemaKind(entity) === internal.EntityKind.Type;
|
|
336
|
+
};
|
|
337
|
+
|
|
338
|
+
/**
|
|
339
|
+
* Narrow a `Type.AnyEntity` (e.g. one returned from `schemaRegistry.query(...)`)
|
|
340
|
+
* to `AnyObj`, throwing if it describes a relation or the type-kind
|
|
341
|
+
* meta-schema. Use at call sites that need to pass the value to `Obj.make`,
|
|
342
|
+
* `Filter.type`, or other object-only APIs.
|
|
343
|
+
*/
|
|
344
|
+
export const assertObject = (entity: AnyEntity): AnyObj => {
|
|
345
|
+
assertArgument(isObject(entity), 'entity', 'Expected an object-kind Type entity.');
|
|
346
|
+
return entity;
|
|
347
|
+
};
|
|
348
|
+
|
|
349
|
+
/** Narrow a `Type.AnyEntity` to `AnyRelation`, throwing otherwise. */
|
|
350
|
+
export const expectRelation = (entity: AnyEntity): AnyRelation => {
|
|
351
|
+
assertArgument(isRelation(entity), 'entity', 'Expected a relation-kind Type entity.');
|
|
352
|
+
return entity;
|
|
353
|
+
};
|
|
354
|
+
|
|
355
|
+
/** Narrow a `Type.AnyEntity` to the `Type.Type` meta-schema, throwing otherwise. */
|
|
356
|
+
export const expectTypeKind = (entity: AnyEntity): Type => {
|
|
357
|
+
assertArgument(isTypeKind(entity), 'entity', 'Expected a type-kind Type entity.');
|
|
358
|
+
return entity;
|
|
202
359
|
};
|
|
203
360
|
|
|
204
361
|
/**
|
|
@@ -212,48 +369,369 @@ export type AnyRef = Schema.Schema<internal.Ref<any>, EncodedReference>;
|
|
|
212
369
|
//
|
|
213
370
|
|
|
214
371
|
/**
|
|
215
|
-
*
|
|
216
|
-
*
|
|
217
|
-
*
|
|
218
|
-
*
|
|
372
|
+
* Returns the URI identifying a type entity. Always defined.
|
|
373
|
+
*
|
|
374
|
+
* - Static `Type.Obj` / `Type.Relation` → typename DXN (e.g. `dxn:com.example.type.person:0.1.0`).
|
|
375
|
+
* - Persisted `Type.Type` instance (has `id`) → local `EID` (`echo:/<objectId>`).
|
|
376
|
+
* - In-memory `Type.Type` draft (has `id`, no typename) → local `EID`.
|
|
377
|
+
*
|
|
378
|
+
* Only accepts `Type.AnyEntity` entities. Raw `Schema.Schema` values and the
|
|
379
|
+
* branded `Obj.Unknown` / `Relation.Unknown` schemas are intentionally not
|
|
380
|
+
* supported — use `internal.getSchemaURI` or the schema's typename annotation
|
|
381
|
+
* directly when working at the schema level.
|
|
219
382
|
*/
|
|
220
|
-
export const
|
|
221
|
-
|
|
383
|
+
export const getURI = (input: AnyEntity): URI.URI => {
|
|
384
|
+
// For Type entities, route through `getTypeURIFromSpecifier` (id → EID,
|
|
385
|
+
// typename/version → DXN). For Obj/Relation entities, unwrap to the source
|
|
386
|
+
// Effect Schema first and read its annotations.
|
|
387
|
+
if (isType(input)) {
|
|
388
|
+
return internal.getTypeURIFromSpecifier(input);
|
|
389
|
+
}
|
|
390
|
+
return internal.getSchemaURI(getSchema(input)) ?? raise(new TypeError('Type entity has no URI'));
|
|
222
391
|
};
|
|
223
392
|
|
|
224
393
|
/**
|
|
225
|
-
* @
|
|
226
|
-
*
|
|
394
|
+
* @returns The typename. Example: `com.example.type.person`.
|
|
395
|
+
*
|
|
396
|
+
* Persisted `Type.Type` entities carry typename in `EntityMeta.key` (the
|
|
397
|
+
* canonical registry-provenance field); unnamed drafts fall back to the
|
|
398
|
+
* entity's object id so the helper always returns a string. Any `dxn:` or
|
|
399
|
+
* `echo:/` prefix is stripped — typename is a bare identifier, not a URI.
|
|
227
400
|
*/
|
|
228
|
-
|
|
229
|
-
|
|
230
|
-
|
|
401
|
+
// TODO(wittjosiah): For in-database types this should return the object id once the registry
|
|
402
|
+
// has more robust options for shadowing types (so callers can disambiguate db-stored copies).
|
|
403
|
+
export const getTypename = (input: AnyEntity): string => {
|
|
404
|
+
// Both in-memory and in-database entities carry typename in `EntityMeta.key`
|
|
405
|
+
// — the canonical registry-provenance field. In-memory entities attach meta
|
|
406
|
+
// eagerly (see `makeEchoTypeSchema`), so a single meta-backed read covers
|
|
407
|
+
// both forms.
|
|
408
|
+
const meta = internal.getMetaChecked(input);
|
|
409
|
+
let typename: string | undefined = meta.key as string | undefined;
|
|
410
|
+
// `meta.key` is a denormalized copy of the typename. The authoritative source
|
|
411
|
+
// for a type entity is its `jsonSchema.typename` (`getSchema` rebuilds the
|
|
412
|
+
// Effect Schema from `jsonSchema`). A type loaded from a snapshot can arrive
|
|
413
|
+
// with `meta.key` absent (the meta round-trip dropped the denormalized copy)
|
|
414
|
+
// while `jsonSchema.typename` is always present — consult it before the
|
|
415
|
+
// last-resort id fallback so callers never receive a bare object id where a
|
|
416
|
+
// typename is expected (e.g. `Filter.typename`, which rejects non-typenames).
|
|
417
|
+
if (typename == null) {
|
|
418
|
+
typename = input.jsonSchema?.typename;
|
|
419
|
+
}
|
|
420
|
+
// Unnamed drafts (no meta.key, no jsonSchema typename) fall back to the id.
|
|
421
|
+
typename ??= input.id as string;
|
|
422
|
+
// Typename is a bare identifier — strip URI prefixes if a caller seeded
|
|
423
|
+
// meta.key with one accidentally (or if a static entity carries a DXN-
|
|
424
|
+
// style typename).
|
|
425
|
+
typename = stripTypenamePrefix(typename);
|
|
426
|
+
invariant(typeof typename === 'string' && typename.length > 0, 'Invalid typename');
|
|
231
427
|
return typename;
|
|
232
428
|
};
|
|
233
429
|
|
|
234
430
|
/**
|
|
235
|
-
* Gets the version
|
|
431
|
+
* Gets the version.
|
|
236
432
|
* @example 0.1.0
|
|
433
|
+
* @example 0.1.0-<heads> (in-database, versioned by automerge heads)
|
|
434
|
+
*
|
|
435
|
+
* The registry-provenance semver lives in `EntityMeta.version`; unversioned
|
|
436
|
+
* drafts default to {@link DRAFT_VERSION} (`'0.0.0'`). In-database entities are
|
|
437
|
+
* additionally versioned by their automerge heads, which are exposed as the
|
|
438
|
+
* semver pre-release tag (`<semver>-<heads>`). In-memory declarations have no
|
|
439
|
+
* heads and surface the bare semver. Read the registry semver alone via
|
|
440
|
+
* `Type.getMeta(input).version`.
|
|
237
441
|
*/
|
|
238
|
-
export const getVersion = (
|
|
239
|
-
const
|
|
240
|
-
|
|
241
|
-
|
|
442
|
+
export const getVersion = (input: AnyEntity): string => {
|
|
443
|
+
const meta = internal.getMetaChecked(input);
|
|
444
|
+
// As with `getTypename`: `meta.version` is a denormalized copy; the
|
|
445
|
+
// authoritative semver lives in `jsonSchema.version`. Prefer meta, fall back
|
|
446
|
+
// to jsonSchema (always present on persisted types, survives serialization),
|
|
447
|
+
// then to `DRAFT_VERSION` for unversioned drafts.
|
|
448
|
+
const semver = (meta.version as string | undefined) ?? input.jsonSchema?.version ?? DRAFT_VERSION;
|
|
449
|
+
invariant(typeof semver === 'string' && semver.match(/^\d+\.\d+\.\d+$/), 'Invalid version');
|
|
450
|
+
// In-database entities are versioned by their automerge heads; expose them as
|
|
451
|
+
// the semver pre-release tag. In-memory drafts carry no heads → bare semver.
|
|
452
|
+
const heads = internal.version(input).automergeHeads;
|
|
453
|
+
if (heads != null && heads.length > 0) {
|
|
454
|
+
return `${semver}-${[...heads].sort().join('.')}`;
|
|
455
|
+
}
|
|
456
|
+
return semver;
|
|
242
457
|
};
|
|
243
458
|
|
|
244
459
|
/**
|
|
245
|
-
*
|
|
460
|
+
* Strip URI prefixes (`dxn:`, `echo:/`, `echo://`) from a typename string.
|
|
461
|
+
* Typename is a bare identifier — callers reading from meta or from a
|
|
462
|
+
* caller-supplied seed value shouldn't propagate URI prefixes downstream.
|
|
246
463
|
*/
|
|
247
|
-
|
|
464
|
+
const stripTypenamePrefix = (value: string): string => {
|
|
465
|
+
if (value.startsWith('dxn:')) {
|
|
466
|
+
return value.slice('dxn:'.length);
|
|
467
|
+
}
|
|
468
|
+
if (value.startsWith('echo://')) {
|
|
469
|
+
return value.slice('echo://'.length);
|
|
470
|
+
}
|
|
471
|
+
if (value.startsWith('echo:/')) {
|
|
472
|
+
return value.slice('echo:/'.length);
|
|
473
|
+
}
|
|
474
|
+
return value;
|
|
475
|
+
};
|
|
248
476
|
|
|
249
477
|
/**
|
|
250
|
-
*
|
|
478
|
+
* Type predicate: true iff the value is any type-kind ECHO entity — a static
|
|
479
|
+
* `Type.Obj` / `Type.Relation` produced by `Type.makeObject` / `Type.makeRelation`, a
|
|
480
|
+
* static meta `Type.Type`, or a persisted `Type.Type` returned by the database.
|
|
481
|
+
*
|
|
482
|
+
* All three branches stamp `[KindId] = Type`, so this is a single brand check.
|
|
483
|
+
* Use {@link isObject} / {@link isRelation} / {@link isTypeKind}
|
|
484
|
+
* when you need to discriminate further; use {@link getDatabase} when you mean
|
|
485
|
+
* "is this a db-attached type" (vs. an in-memory declaration).
|
|
486
|
+
*/
|
|
487
|
+
export const isType = (value: unknown): value is AnyEntity =>
|
|
488
|
+
internal.getEntityKindBrand(value) === internal.EntityKind.Type;
|
|
489
|
+
|
|
490
|
+
/**
|
|
491
|
+
* Get the database the type entity belongs to, or `undefined` if it is an
|
|
492
|
+
* in-memory declaration (`Type.makeObject` / `Type.makeRelation` result) not
|
|
493
|
+
* yet attached to a database. Mirrors `Obj.getDatabase` / `Relation.getDatabase`.
|
|
494
|
+
*
|
|
495
|
+
* Database attachment is the canonical discriminator between in-memory and
|
|
496
|
+
* in-database type entities — both are live reactive `TypeSchema` instances and
|
|
497
|
+
* are otherwise indistinguishable.
|
|
498
|
+
*/
|
|
499
|
+
export const getDatabase = (input: AnyEntity): Database.Database | undefined => internal.getDatabase(input);
|
|
500
|
+
|
|
501
|
+
/**
|
|
502
|
+
* Mutable meta type returned by `Type.getMeta` inside a `Type.update` callback.
|
|
503
|
+
* Mirrors `Obj.Meta` / `Relation.Meta` — `Type.Type` is an Entity like its
|
|
504
|
+
* siblings, so its meta is the same `EntityMeta` record:
|
|
505
|
+
* `{ keys, tags?, key?, version? }`.
|
|
506
|
+
*
|
|
507
|
+
* `key` / `version` here are the canonical registry-provenance pair
|
|
508
|
+
* (typename + semver) on persisted Type.Type entities; they are absent on
|
|
509
|
+
* unnamed drafts. Use {@link getTypename} / {@link getVersion} when you want
|
|
510
|
+
* a non-`undefined` value with id / {@link DRAFT_VERSION} fallbacks.
|
|
511
|
+
*/
|
|
512
|
+
export type Meta = internal.Meta;
|
|
513
|
+
|
|
514
|
+
/**
|
|
515
|
+
* Deeply read-only version of {@link Meta}.
|
|
516
|
+
* Prevents mutation at all nesting levels (e.g., `meta.keys.push()` is a TS error).
|
|
517
|
+
*/
|
|
518
|
+
export type ReadonlyMeta = internal.ReadonlyMeta;
|
|
519
|
+
|
|
520
|
+
/**
|
|
521
|
+
* Returns the entity's `EntityMeta`. Same semantics as `Obj.getMeta` /
|
|
522
|
+
* `Relation.getMeta` — `Type.Type` is an Entity and carries the canonical
|
|
523
|
+
* `EntityMeta` directly. Returns mutable meta when passed a mutable type
|
|
524
|
+
* (inside a `Type.update` callback), read-only meta otherwise.
|
|
525
|
+
*
|
|
526
|
+
* For persisted Type entities, `meta.key` holds the typename and
|
|
527
|
+
* `meta.version` holds the semver. Use {@link getTypename} / {@link getVersion}
|
|
528
|
+
* if you want the helpers' id / {@link DRAFT_VERSION} fallbacks for drafts.
|
|
529
|
+
*
|
|
530
|
+
* Both persisted and in-memory type entities (`Type.makeObject` /
|
|
531
|
+
* `Type.makeRelation` results) carry their `EntityMeta` via `[MetaId]`, so the
|
|
532
|
+
* lookup is uniform.
|
|
533
|
+
*/
|
|
534
|
+
export function getMeta(entity: internal.Mutable<AnyEntity>): Meta;
|
|
535
|
+
export function getMeta(entity: Mutable): Meta;
|
|
536
|
+
export function getMeta(entity: AnyEntity): ReadonlyMeta;
|
|
537
|
+
export function getMeta(entity: AnyEntity | internal.Mutable<AnyEntity> | Mutable): ReadonlyMeta | Meta {
|
|
538
|
+
// The `Mutable` overload accepts the narrowed view passed to `Type.update`
|
|
539
|
+
// callbacks; at runtime that draft IS the underlying persisted Type entity,
|
|
540
|
+
// so the same `MetaId` lookup works.
|
|
541
|
+
assertArgument(isType(entity), 'entity', 'Expected a Type entity.');
|
|
542
|
+
// Both persisted and in-memory type entities carry runtime `EntityMeta` via
|
|
543
|
+
// `[MetaId]`, so the lookup is uniform.
|
|
544
|
+
return internal.getMetaChecked(entity);
|
|
545
|
+
}
|
|
546
|
+
|
|
547
|
+
/**
|
|
548
|
+
* Get the display label of a type entity.
|
|
549
|
+
* Reads the field(s) nominated by the type's {@link LabelAnnotation} (e.g. `name` on persisted schemas).
|
|
550
|
+
* Returns `undefined` if no label field is populated.
|
|
551
|
+
*/
|
|
552
|
+
export const getLabel = (entity: AnyEntity, options?: internal.GetLabelOptions): string | undefined =>
|
|
553
|
+
internal.getLabel(entity, options);
|
|
554
|
+
|
|
555
|
+
/**
|
|
556
|
+
* String key used to phantom-carry the instance type produced by a `Type.Type`.
|
|
557
|
+
* Used by `Type.InstanceType<typeof Foo>` to recover the schema instance type
|
|
558
|
+
* since `Type.makeObject(dxn)` does not return a `Schema.Schema`.
|
|
559
|
+
*
|
|
560
|
+
* Re-exported from the internal types layer so both `Type.ts` and internal
|
|
561
|
+
* helpers (`makeObject`, `createObject`) reference the same phantom key.
|
|
562
|
+
*/
|
|
563
|
+
export const InstancePhantomId = internal.InstancePhantomId;
|
|
564
|
+
export type InstancePhantomId = internal.InstancePhantomId;
|
|
565
|
+
|
|
566
|
+
/**
|
|
567
|
+
* Sibling of {@link Obj} / {@link Relation} for the third ECHO entity kind:
|
|
568
|
+
* **type-kind** entities (meta-schemas). The singleton {@link Type} const is
|
|
569
|
+
* the canonical example — it describes stored type definitions themselves.
|
|
570
|
+
*
|
|
571
|
+
* Not a `Schema.Schema`. Use `Type.getSchema(value)` to obtain the underlying
|
|
572
|
+
* Effect Schema and `Type.update(value, draft => ...)` to mutate.
|
|
573
|
+
*
|
|
574
|
+
* `A` is the instance-type phantom — what `Obj.make(value, ...)` would produce.
|
|
575
|
+
* Merged with the `Type` const value via TypeScript declaration merging.
|
|
576
|
+
*/
|
|
577
|
+
export interface Type<A = unknown> extends BaseTypeEntity<A & EntityModule.OfKind<typeof EntityModule.Kind.Type>> {
|
|
578
|
+
/** Schema-kind brand (type — the meta-schema kind). */
|
|
579
|
+
readonly [internal.SchemaKindId]: internal.EntityKind.Type;
|
|
580
|
+
|
|
581
|
+
/** Source Effect Schema — used internally by `Type.getSchema(self)`. */
|
|
582
|
+
readonly [internal.StaticTypeSchemaSlot]: Schema.Schema.AnyNoContext;
|
|
583
|
+
}
|
|
584
|
+
|
|
585
|
+
/**
|
|
586
|
+
* Instance type produced by a Type entity.
|
|
587
|
+
*
|
|
588
|
+
* Accepts ONLY {@link AnyEntity} inputs — `Type.Obj`, `Type.Relation`, or
|
|
589
|
+
* `Type.Type`. Raw Effect `Schema.Schema` values are rejected: for those, use
|
|
590
|
+
* `Schema.Schema.Type<typeof Foo>` directly. This separation keeps the type
|
|
591
|
+
* system honest about which values represent ECHO entities versus plain
|
|
592
|
+
* Effect schemas.
|
|
593
|
+
*
|
|
594
|
+
* Dispatches on the entity kind:
|
|
595
|
+
* - `Relation<Props, S, T>` → `Endpoints<S,T> & Props & OfKind<Relation>`
|
|
596
|
+
* - `Obj<A>` → `A & OfKind<Object>`
|
|
597
|
+
* - `Type<A>` → `A & OfKind<Type>`
|
|
598
|
+
*/
|
|
599
|
+
export type InstanceType<T extends AnyEntity> =
|
|
600
|
+
T extends Relation<infer Props, infer Source, infer Target, any>
|
|
601
|
+
? RelationModule.Endpoints<Source, Target> & Props & EntityModule.OfKind<typeof EntityModule.Kind.Relation>
|
|
602
|
+
: T extends Obj<infer A, any>
|
|
603
|
+
? A & EntityModule.OfKind<typeof EntityModule.Kind.Object>
|
|
604
|
+
: T extends Type<infer A>
|
|
605
|
+
? A & EntityModule.OfKind<typeof EntityModule.Kind.Type>
|
|
606
|
+
: never;
|
|
607
|
+
|
|
608
|
+
/**
|
|
609
|
+
* Returns the Effect Schema for a type entity.
|
|
610
|
+
*
|
|
611
|
+
* - For static `Type.Obj` / `Type.Relation` entities the source Effect Schema is
|
|
612
|
+
* read from a hidden slot — these overloads preserve the instance type.
|
|
613
|
+
* - For `Type.Type` entities (the meta-schema kind) the schema is rebuilt from
|
|
614
|
+
* `type.jsonSchema`; the instance type isn't statically knowable so the wide
|
|
615
|
+
* `AnyEntity` overload widens to `Schema.Schema.AnyNoContext`.
|
|
616
|
+
*
|
|
617
|
+
* Always call this when you need to interact with the Effect Schema API
|
|
618
|
+
* (e.g. before passing to Effect.Schema functions). For ECHO-side APIs
|
|
619
|
+
* (`Obj.make`, `Filter.type`, `Ref`) pass the type entity directly.
|
|
620
|
+
*
|
|
621
|
+
* Only accepts `Type.AnyEntity` — raw `Schema.Schema` values can be used
|
|
622
|
+
* directly without unwrapping.
|
|
623
|
+
*/
|
|
624
|
+
export function getSchema<T extends AnyObj>(type: T): Schema.Schema<InstanceType<T>>;
|
|
625
|
+
export function getSchema<T extends AnyRelation>(type: T): Schema.Schema<InstanceType<T>>;
|
|
626
|
+
export function getSchema(type: AnyEntity): Schema.Schema.AnyNoContext;
|
|
627
|
+
export function getSchema(type: AnyEntity): Schema.Schema.AnyNoContext {
|
|
628
|
+
// Static `Type.Type` entities carry the source Effect Schema on a hidden
|
|
629
|
+
// slot so we can return it without round-tripping through JsonSchema.
|
|
630
|
+
const staticSchema = internal.getStaticTypeSchema(type);
|
|
631
|
+
if (staticSchema != null) {
|
|
632
|
+
return staticSchema;
|
|
633
|
+
}
|
|
634
|
+
assertArgument(isType(type), 'type', 'Expected a Type entity.');
|
|
635
|
+
// Persisted `Type.Type` entity — build the Effect Schema from its stored
|
|
636
|
+
// jsonSchema and re-attach the TypeIdentifierAnnotation so the rebuilt
|
|
637
|
+
// schema's URI (via getSchemaURI) matches the entity's local EID.
|
|
638
|
+
const rebuilt = internal.toEffectSchema(type.jsonSchema);
|
|
639
|
+
if (typeof type.id === 'string') {
|
|
640
|
+
return rebuilt.annotations({
|
|
641
|
+
[internal.TypeIdentifierAnnotationId]: EID.make({ entityId: type.id }),
|
|
642
|
+
});
|
|
643
|
+
}
|
|
644
|
+
return rebuilt;
|
|
645
|
+
}
|
|
646
|
+
|
|
647
|
+
/**
|
|
648
|
+
* Mutable view of a `Type.Type` — the shape passed to the `Type.update` callback.
|
|
649
|
+
* Outside `Type.update`, `Type.Type` fields are read-only (both at the type level
|
|
650
|
+
* and at runtime — direct assignment throws). Use this to constrain mutation to
|
|
651
|
+
* the change context, analogous to `Obj.update(obj, (draft) => ...)`.
|
|
652
|
+
*
|
|
653
|
+
* NOTE: `typename` and `version` are intentionally absent — they live in
|
|
654
|
+
* `EntityMeta` (`key` / `version` — the canonical registry-provenance pair).
|
|
655
|
+
* Read them via {@link getTypename} / {@link getVersion} / {@link getMeta};
|
|
656
|
+
* `typename` is treated as immutable on persisted entities.
|
|
657
|
+
*
|
|
658
|
+
* Unlike `Obj.update` — whose mutable view is inferred as `Mutable<A>` over the
|
|
659
|
+
* whole instance type because every data field is editable — a `Type.Type`
|
|
660
|
+
* exposes only `name` and `jsonSchema` for mutation. The rest of its shape
|
|
661
|
+
* (`id`, the `[KindId]` / `[SchemaKindId]` brands, and `typename` / `version`
|
|
662
|
+
* in meta) is immutable, so this view is declared explicitly rather than
|
|
663
|
+
* derived from `InstanceType<Type.Type>`.
|
|
251
664
|
*/
|
|
252
|
-
export
|
|
665
|
+
export interface Mutable {
|
|
666
|
+
name?: string;
|
|
667
|
+
// Deep-mutable within the change context — `Type.update`'s purpose is to allow
|
|
668
|
+
// mutation, so the draft exposes `jsonSchema` as writable (the readonly
|
|
669
|
+
// `JsonSchemaType` would force callers to cast).
|
|
670
|
+
jsonSchema: Types.DeepMutable<internal.JsonSchemaType>;
|
|
671
|
+
}
|
|
672
|
+
|
|
673
|
+
/**
|
|
674
|
+
* Perform mutations on a `Type.Type` within a change context.
|
|
675
|
+
*
|
|
676
|
+
* The callback receives a {@link Mutable} view of the type — direct mutation of
|
|
677
|
+
* a `Type.Type` outside `Type.update` throws at runtime, mirroring `Obj.update`.
|
|
678
|
+
* Delegates to the same automerge-transaction primitive `Obj.update(obj, cb)` uses.
|
|
679
|
+
*/
|
|
680
|
+
export const update = (type: AnyEntity, callback: (mutable: Mutable) => void): void => {
|
|
681
|
+
// `Type.Type` is an ECHO object; the change machinery is the same as `Obj.update`.
|
|
682
|
+
internal.change(type, callback as internal.ChangeCallback<AnyEntity>);
|
|
683
|
+
};
|
|
684
|
+
|
|
685
|
+
//
|
|
686
|
+
// Field-level helpers for mutating persisted types.
|
|
687
|
+
// These are thin wrappers over `Type.update` plus the JsonSchema manipulation
|
|
688
|
+
// utilities. Callers pass a persisted `Type.Type` (e.g. one returned by
|
|
689
|
+
// `db.addType(schemaEntity)`) and the helper drives the change context.
|
|
690
|
+
//
|
|
691
|
+
|
|
692
|
+
/**
|
|
693
|
+
* Add fields to a persisted type's schema.
|
|
694
|
+
* @throws if the type is not persisted.
|
|
695
|
+
*/
|
|
696
|
+
export const addFields = (type: AnyEntity, fields: Schema.Struct.Fields): void => {
|
|
697
|
+
const extended = typeInternal.addFieldsToSchema(getSchema(type), fields);
|
|
698
|
+
update(type, (draft) => {
|
|
699
|
+
draft.jsonSchema = internal.toJsonSchema(extended);
|
|
700
|
+
});
|
|
701
|
+
};
|
|
702
|
+
|
|
703
|
+
/**
|
|
704
|
+
* Replace existing fields on a persisted type's schema.
|
|
705
|
+
* @throws if the type is not persisted.
|
|
706
|
+
*/
|
|
707
|
+
export const updateFields = (type: AnyEntity, fields: Schema.Struct.Fields): void => {
|
|
708
|
+
const updated = typeInternal.updateFieldsInSchema(getSchema(type), fields);
|
|
709
|
+
update(type, (draft) => {
|
|
710
|
+
draft.jsonSchema = internal.toJsonSchema(updated);
|
|
711
|
+
});
|
|
712
|
+
};
|
|
713
|
+
|
|
714
|
+
/**
|
|
715
|
+
* Rename a field on a persisted type's schema.
|
|
716
|
+
* @throws if the type is not persisted.
|
|
717
|
+
*/
|
|
718
|
+
export const updateFieldPropertyName = (
|
|
719
|
+
type: Type,
|
|
720
|
+
{ before, after }: { before: PropertyKey; after: PropertyKey },
|
|
721
|
+
): void => {
|
|
722
|
+
const renamed = typeInternal.updateFieldNameInSchema(getSchema(type), { before, after });
|
|
723
|
+
update(type, (draft) => {
|
|
724
|
+
draft.jsonSchema = internal.toJsonSchema(renamed);
|
|
725
|
+
});
|
|
726
|
+
};
|
|
253
727
|
|
|
254
728
|
/**
|
|
255
|
-
*
|
|
729
|
+
* Remove fields from a persisted type's schema.
|
|
730
|
+
* @throws if the type is not persisted.
|
|
256
731
|
*/
|
|
257
|
-
export const
|
|
258
|
-
|
|
732
|
+
export const removeFields = (type: AnyEntity, fieldNames: string[]): void => {
|
|
733
|
+
const removed = typeInternal.removeFieldsFromSchema(getSchema(type), fieldNames);
|
|
734
|
+
update(type, (draft) => {
|
|
735
|
+
draft.jsonSchema = internal.toJsonSchema(removed);
|
|
736
|
+
});
|
|
259
737
|
};
|