@dxos/echo 0.8.4-main.5ea62a8 → 0.8.4-main.66e292d

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.
Files changed (328) hide show
  1. package/dist/lib/browser/chunk-7GH6RXJ3.mjs +3683 -0
  2. package/dist/lib/browser/chunk-7GH6RXJ3.mjs.map +7 -0
  3. package/dist/lib/browser/chunk-E4UTVJNF.mjs +1111 -0
  4. package/dist/lib/browser/chunk-E4UTVJNF.mjs.map +7 -0
  5. package/dist/lib/browser/index.mjs +24 -1
  6. package/dist/lib/browser/internal/index.mjs +336 -0
  7. package/dist/lib/browser/internal/index.mjs.map +7 -0
  8. package/dist/lib/browser/meta.json +1 -1
  9. package/dist/lib/browser/testing/index.mjs +289 -31
  10. package/dist/lib/browser/testing/index.mjs.map +4 -4
  11. package/dist/lib/node-esm/chunk-JE5RXM2I.mjs +1111 -0
  12. package/dist/lib/node-esm/chunk-JE5RXM2I.mjs.map +7 -0
  13. package/dist/lib/node-esm/chunk-M4B6BMD2.mjs +3683 -0
  14. package/dist/lib/node-esm/chunk-M4B6BMD2.mjs.map +7 -0
  15. package/dist/lib/node-esm/index.mjs +24 -1
  16. package/dist/lib/node-esm/internal/index.mjs +336 -0
  17. package/dist/lib/node-esm/internal/index.mjs.map +7 -0
  18. package/dist/lib/node-esm/meta.json +1 -1
  19. package/dist/lib/node-esm/testing/index.mjs +289 -31
  20. package/dist/lib/node-esm/testing/index.mjs.map +4 -4
  21. package/dist/types/src/Annotation.d.ts +2 -0
  22. package/dist/types/src/Annotation.d.ts.map +1 -0
  23. package/dist/types/src/Database.d.ts +137 -0
  24. package/dist/types/src/Database.d.ts.map +1 -0
  25. package/dist/types/src/Entity.d.ts +36 -0
  26. package/dist/types/src/Entity.d.ts.map +1 -0
  27. package/dist/types/src/Format.d.ts +4 -0
  28. package/dist/types/src/Format.d.ts.map +1 -0
  29. package/dist/types/src/JsonSchema.d.ts +9 -0
  30. package/dist/types/src/JsonSchema.d.ts.map +1 -0
  31. package/dist/types/src/Key.d.ts +1 -0
  32. package/dist/types/src/Key.d.ts.map +1 -1
  33. package/dist/types/src/Obj.d.ts +121 -50
  34. package/dist/types/src/Obj.d.ts.map +1 -1
  35. package/dist/types/src/Ref.d.ts +10 -10
  36. package/dist/types/src/Ref.d.ts.map +1 -1
  37. package/dist/types/src/Relation.d.ts +18 -14
  38. package/dist/types/src/Relation.d.ts.map +1 -1
  39. package/dist/types/src/Tag.d.ts +17 -0
  40. package/dist/types/src/Tag.d.ts.map +1 -0
  41. package/dist/types/src/Type.d.ts +39 -50
  42. package/dist/types/src/Type.d.ts.map +1 -1
  43. package/dist/types/src/errors.d.ts +68 -0
  44. package/dist/types/src/errors.d.ts.map +1 -0
  45. package/dist/types/src/index.d.ts +9 -3
  46. package/dist/types/src/index.d.ts.map +1 -1
  47. package/dist/types/src/internal/annotations/annotations.d.ts +174 -0
  48. package/dist/types/src/internal/annotations/annotations.d.ts.map +1 -0
  49. package/dist/types/src/internal/annotations/annotations.test.d.ts +2 -0
  50. package/dist/types/src/internal/annotations/annotations.test.d.ts.map +1 -0
  51. package/dist/types/src/internal/annotations/index.d.ts +3 -0
  52. package/dist/types/src/internal/annotations/index.d.ts.map +1 -0
  53. package/dist/types/src/internal/annotations/util.d.ts +26 -0
  54. package/dist/types/src/internal/annotations/util.d.ts.map +1 -0
  55. package/dist/types/src/internal/entities/entity.d.ts +10 -0
  56. package/dist/types/src/internal/entities/entity.d.ts.map +1 -0
  57. package/dist/types/src/internal/entities/expando.d.ts +16 -0
  58. package/dist/types/src/internal/entities/expando.d.ts.map +1 -0
  59. package/dist/types/src/internal/entities/index.d.ts +6 -0
  60. package/dist/types/src/internal/entities/index.d.ts.map +1 -0
  61. package/dist/types/src/internal/entities/model.d.ts +70 -0
  62. package/dist/types/src/internal/entities/model.d.ts.map +1 -0
  63. package/dist/types/src/internal/entities/object.d.ts +11 -0
  64. package/dist/types/src/internal/entities/object.d.ts.map +1 -0
  65. package/dist/types/src/internal/entities/relation.d.ts +55 -0
  66. package/dist/types/src/internal/entities/relation.d.ts.map +1 -0
  67. package/dist/types/src/internal/entities/util.d.ts +2 -0
  68. package/dist/types/src/internal/entities/util.d.ts.map +1 -0
  69. package/dist/types/src/internal/formats/date.d.ts +63 -0
  70. package/dist/types/src/internal/formats/date.d.ts.map +1 -0
  71. package/dist/types/src/internal/formats/date.test.d.ts +2 -0
  72. package/dist/types/src/internal/formats/date.test.d.ts.map +1 -0
  73. package/dist/types/src/internal/formats/format.d.ts +32 -0
  74. package/dist/types/src/internal/formats/format.d.ts.map +1 -0
  75. package/dist/types/src/internal/formats/format.test.d.ts +2 -0
  76. package/dist/types/src/internal/formats/format.test.d.ts.map +1 -0
  77. package/dist/types/src/internal/formats/index.d.ts +8 -0
  78. package/dist/types/src/internal/formats/index.d.ts.map +1 -0
  79. package/dist/types/src/internal/formats/number.d.ts +31 -0
  80. package/dist/types/src/internal/formats/number.d.ts.map +1 -0
  81. package/dist/types/src/internal/formats/object.d.ts +35 -0
  82. package/dist/types/src/internal/formats/object.d.ts.map +1 -0
  83. package/dist/types/src/internal/formats/select.d.ts +11 -0
  84. package/dist/types/src/internal/formats/select.d.ts.map +1 -0
  85. package/dist/types/src/internal/formats/string.d.ts +38 -0
  86. package/dist/types/src/internal/formats/string.d.ts.map +1 -0
  87. package/dist/types/src/internal/formats/types.d.ts +68 -0
  88. package/dist/types/src/internal/formats/types.d.ts.map +1 -0
  89. package/dist/types/src/internal/index.d.ts +11 -0
  90. package/dist/types/src/internal/index.d.ts.map +1 -0
  91. package/dist/types/src/internal/json-schema/annotations.d.ts +19 -0
  92. package/dist/types/src/internal/json-schema/annotations.d.ts.map +1 -0
  93. package/dist/types/src/internal/json-schema/effect-schema.test.d.ts +2 -0
  94. package/dist/types/src/internal/json-schema/effect-schema.test.d.ts.map +1 -0
  95. package/dist/types/src/internal/json-schema/index.d.ts +5 -0
  96. package/dist/types/src/internal/json-schema/index.d.ts.map +1 -0
  97. package/dist/types/src/internal/json-schema/json-schema-normalize.d.ts +7 -0
  98. package/dist/types/src/internal/json-schema/json-schema-normalize.d.ts.map +1 -0
  99. package/dist/types/src/internal/json-schema/json-schema-type.d.ts +250 -0
  100. package/dist/types/src/internal/json-schema/json-schema-type.d.ts.map +1 -0
  101. package/dist/types/src/internal/json-schema/json-schema.d.ts +29 -0
  102. package/dist/types/src/internal/json-schema/json-schema.d.ts.map +1 -0
  103. package/dist/types/src/internal/json-schema/json-schema.test.d.ts +2 -0
  104. package/dist/types/src/internal/json-schema/json-schema.test.d.ts.map +1 -0
  105. package/dist/types/src/internal/object/common.d.ts +18 -0
  106. package/dist/types/src/internal/object/common.d.ts.map +1 -0
  107. package/dist/types/src/internal/object/create-object.d.ts +39 -0
  108. package/dist/types/src/internal/object/create-object.d.ts.map +1 -0
  109. package/dist/types/src/internal/object/create-object.test.d.ts +2 -0
  110. package/dist/types/src/internal/object/create-object.test.d.ts.map +1 -0
  111. package/dist/types/src/internal/object/deleted.d.ts +6 -0
  112. package/dist/types/src/internal/object/deleted.d.ts.map +1 -0
  113. package/dist/types/src/internal/object/ids.d.ts +6 -0
  114. package/dist/types/src/internal/object/ids.d.ts.map +1 -0
  115. package/dist/types/src/internal/object/index.d.ts +8 -0
  116. package/dist/types/src/internal/object/index.d.ts.map +1 -0
  117. package/dist/types/src/internal/object/inspect.d.ts +2 -0
  118. package/dist/types/src/internal/object/inspect.d.ts.map +1 -0
  119. package/dist/types/src/internal/object/json-serializer.d.ts +31 -0
  120. package/dist/types/src/internal/object/json-serializer.d.ts.map +1 -0
  121. package/dist/types/src/internal/object/json-serializer.test.d.ts +2 -0
  122. package/dist/types/src/internal/object/json-serializer.test.d.ts.map +1 -0
  123. package/dist/types/src/internal/object/schema-validator.d.ts +15 -0
  124. package/dist/types/src/internal/object/schema-validator.d.ts.map +1 -0
  125. package/dist/types/src/internal/object/schema-validator.test.d.ts +2 -0
  126. package/dist/types/src/internal/object/schema-validator.test.d.ts.map +1 -0
  127. package/dist/types/src/internal/object/typed-object.d.ts +31 -0
  128. package/dist/types/src/internal/object/typed-object.d.ts.map +1 -0
  129. package/dist/types/src/internal/object/typed-object.test.d.ts +2 -0
  130. package/dist/types/src/internal/object/typed-object.test.d.ts.map +1 -0
  131. package/dist/types/src/internal/proxy/handler.test.d.ts +2 -0
  132. package/dist/types/src/internal/proxy/handler.test.d.ts.map +1 -0
  133. package/dist/types/src/internal/proxy/index.d.ts +3 -0
  134. package/dist/types/src/internal/proxy/index.d.ts.map +1 -0
  135. package/dist/types/src/internal/proxy/make-object.d.ts +16 -0
  136. package/dist/types/src/internal/proxy/make-object.d.ts.map +1 -0
  137. package/dist/types/src/internal/proxy/schema.test.d.ts +2 -0
  138. package/dist/types/src/internal/proxy/schema.test.d.ts.map +1 -0
  139. package/dist/types/src/internal/proxy/typed-handler.d.ts +44 -0
  140. package/dist/types/src/internal/proxy/typed-handler.d.ts.map +1 -0
  141. package/dist/types/src/internal/proxy/typed-handler.test.d.ts +2 -0
  142. package/dist/types/src/internal/proxy/typed-handler.test.d.ts.map +1 -0
  143. package/dist/types/src/internal/proxy/typed-object.test.d.ts +2 -0
  144. package/dist/types/src/internal/proxy/typed-object.test.d.ts.map +1 -0
  145. package/dist/types/src/internal/ref/index.d.ts +3 -0
  146. package/dist/types/src/internal/ref/index.d.ts.map +1 -0
  147. package/dist/types/src/internal/ref/ref-array.d.ts +21 -0
  148. package/dist/types/src/internal/ref/ref-array.d.ts.map +1 -0
  149. package/dist/types/src/internal/ref/ref.d.ts +209 -0
  150. package/dist/types/src/internal/ref/ref.d.ts.map +1 -0
  151. package/dist/types/src/internal/ref/ref.test.d.ts +2 -0
  152. package/dist/types/src/internal/ref/ref.test.d.ts.map +1 -0
  153. package/dist/types/src/internal/schema/compose.d.ts +6 -0
  154. package/dist/types/src/internal/schema/compose.d.ts.map +1 -0
  155. package/dist/types/src/internal/schema/compose.test.d.ts +2 -0
  156. package/dist/types/src/internal/schema/compose.test.d.ts.map +1 -0
  157. package/dist/types/src/internal/schema/echo-schema.d.ts +168 -0
  158. package/dist/types/src/internal/schema/echo-schema.d.ts.map +1 -0
  159. package/dist/types/src/internal/schema/index.d.ts +6 -0
  160. package/dist/types/src/internal/schema/index.d.ts.map +1 -0
  161. package/dist/types/src/internal/schema/manipulation.d.ts +10 -0
  162. package/dist/types/src/internal/schema/manipulation.d.ts.map +1 -0
  163. package/dist/types/src/internal/schema/persistent-schema.d.ts +18 -0
  164. package/dist/types/src/internal/schema/persistent-schema.d.ts.map +1 -0
  165. package/dist/types/src/internal/schema/runtime-schema-registry.d.ts +18 -0
  166. package/dist/types/src/internal/schema/runtime-schema-registry.d.ts.map +1 -0
  167. package/dist/types/src/internal/schema/snapshot.d.ts +6 -0
  168. package/dist/types/src/internal/schema/snapshot.d.ts.map +1 -0
  169. package/dist/types/src/internal/types/base.d.ts +37 -0
  170. package/dist/types/src/internal/types/base.d.ts.map +1 -0
  171. package/dist/types/src/internal/types/entity.d.ts +12 -0
  172. package/dist/types/src/internal/types/entity.d.ts.map +1 -0
  173. package/dist/types/src/internal/types/index.d.ts +6 -0
  174. package/dist/types/src/internal/types/index.d.ts.map +1 -0
  175. package/dist/types/src/internal/types/meta.d.ts +40 -0
  176. package/dist/types/src/internal/types/meta.d.ts.map +1 -0
  177. package/dist/types/src/internal/types/typename.d.ts +13 -0
  178. package/dist/types/src/internal/types/typename.d.ts.map +1 -0
  179. package/dist/types/src/internal/types/version.d.ts +15 -0
  180. package/dist/types/src/internal/types/version.d.ts.map +1 -0
  181. package/dist/types/src/query/filter.d.ts +167 -0
  182. package/dist/types/src/query/filter.d.ts.map +1 -0
  183. package/dist/types/src/query/index.d.ts +4 -1
  184. package/dist/types/src/query/index.d.ts.map +1 -1
  185. package/dist/types/src/query/order.d.ts +12 -0
  186. package/dist/types/src/query/order.d.ts.map +1 -0
  187. package/dist/types/src/query/query.d.ts +112 -0
  188. package/dist/types/src/query/query.d.ts.map +1 -0
  189. package/dist/types/src/query/query.test.d.ts +2 -0
  190. package/dist/types/src/query/query.test.d.ts.map +1 -0
  191. package/dist/types/src/query/testing.d.ts +51 -0
  192. package/dist/types/src/query/testing.d.ts.map +1 -0
  193. package/dist/types/src/query/types.d.ts +17 -0
  194. package/dist/types/src/query/types.d.ts.map +1 -0
  195. package/dist/types/src/query/util.d.ts +8 -0
  196. package/dist/types/src/query/util.d.ts.map +1 -0
  197. package/dist/types/src/{test → testing}/api.test.d.ts.map +1 -1
  198. package/dist/types/src/testing/index.d.ts +3 -1
  199. package/dist/types/src/testing/index.d.ts.map +1 -1
  200. package/dist/types/src/testing/test-data.d.ts +18 -0
  201. package/dist/types/src/testing/test-data.d.ts.map +1 -0
  202. package/dist/types/src/testing/test-schema.d.ts +337 -0
  203. package/dist/types/src/testing/test-schema.d.ts.map +1 -0
  204. package/dist/types/src/testing/util.d.ts +16 -0
  205. package/dist/types/src/testing/util.d.ts.map +1 -0
  206. package/dist/types/tsconfig.tsbuildinfo +1 -1
  207. package/package.json +49 -29
  208. package/src/Annotation.ts +17 -0
  209. package/src/Database.ts +189 -0
  210. package/src/Entity.ts +51 -0
  211. package/src/Format.ts +11 -0
  212. package/src/JsonSchema.ts +16 -0
  213. package/src/Key.ts +3 -0
  214. package/src/Obj.ts +353 -107
  215. package/src/Ref.ts +9 -10
  216. package/src/Relation.ts +68 -43
  217. package/src/Tag.ts +40 -0
  218. package/src/Type.ts +97 -85
  219. package/src/errors.ts +18 -0
  220. package/src/index.ts +13 -4
  221. package/src/internal/README.md +83 -0
  222. package/src/internal/annotations/annotations.test.ts +96 -0
  223. package/src/internal/annotations/annotations.ts +463 -0
  224. package/src/internal/annotations/index.ts +6 -0
  225. package/src/internal/annotations/util.ts +70 -0
  226. package/src/internal/entities/entity.ts +109 -0
  227. package/src/internal/entities/expando.ts +23 -0
  228. package/src/internal/entities/index.ts +9 -0
  229. package/src/internal/entities/model.ts +129 -0
  230. package/src/internal/entities/object.ts +45 -0
  231. package/src/internal/entities/relation.ts +155 -0
  232. package/src/internal/entities/util.ts +33 -0
  233. package/src/internal/formats/date.test.ts +56 -0
  234. package/src/internal/formats/date.ts +217 -0
  235. package/src/internal/formats/format.test.ts +77 -0
  236. package/src/internal/formats/format.ts +55 -0
  237. package/src/internal/formats/index.ts +12 -0
  238. package/src/internal/formats/number.ts +89 -0
  239. package/src/internal/formats/object.ts +80 -0
  240. package/src/internal/formats/select.ts +16 -0
  241. package/src/internal/formats/string.ts +76 -0
  242. package/src/internal/formats/types.ts +180 -0
  243. package/src/internal/index.ts +38 -0
  244. package/src/internal/json-schema/annotations.ts +50 -0
  245. package/src/internal/json-schema/effect-schema.test.ts +143 -0
  246. package/src/internal/json-schema/index.ts +8 -0
  247. package/src/internal/json-schema/json-schema-normalize.ts +109 -0
  248. package/src/internal/json-schema/json-schema-type.ts +404 -0
  249. package/src/internal/json-schema/json-schema.test.ts +859 -0
  250. package/src/internal/json-schema/json-schema.ts +528 -0
  251. package/src/internal/object/common.ts +75 -0
  252. package/src/internal/object/create-object.test.ts +116 -0
  253. package/src/internal/object/create-object.ts +95 -0
  254. package/src/internal/object/deleted.ts +19 -0
  255. package/src/internal/object/ids.ts +12 -0
  256. package/src/internal/object/index.ts +11 -0
  257. package/src/internal/object/inspect.ts +46 -0
  258. package/src/internal/object/json-serializer.test.ts +94 -0
  259. package/src/internal/object/json-serializer.ts +230 -0
  260. package/src/internal/object/schema-validator.test.ts +186 -0
  261. package/src/internal/object/schema-validator.ts +242 -0
  262. package/src/internal/object/typed-object.test.ts +34 -0
  263. package/src/internal/object/typed-object.ts +94 -0
  264. package/src/internal/proxy/handler.test.ts +173 -0
  265. package/src/internal/proxy/index.ts +6 -0
  266. package/src/internal/proxy/make-object.ts +113 -0
  267. package/src/internal/proxy/schema.test.ts +137 -0
  268. package/src/internal/proxy/typed-handler.test.ts +102 -0
  269. package/src/internal/proxy/typed-handler.ts +233 -0
  270. package/src/internal/proxy/typed-object.test.ts +105 -0
  271. package/src/internal/ref/index.ts +6 -0
  272. package/src/internal/ref/ref-array.ts +39 -0
  273. package/src/internal/ref/ref.test.ts +101 -0
  274. package/src/internal/ref/ref.ts +525 -0
  275. package/src/internal/schema/compose.test.ts +42 -0
  276. package/src/internal/schema/compose.ts +37 -0
  277. package/src/internal/schema/echo-schema.ts +385 -0
  278. package/src/internal/schema/index.ts +9 -0
  279. package/src/internal/schema/manipulation.ts +92 -0
  280. package/src/internal/schema/persistent-schema.ts +28 -0
  281. package/src/internal/schema/runtime-schema-registry.ts +78 -0
  282. package/src/internal/schema/snapshot.ts +25 -0
  283. package/src/internal/types/base.ts +58 -0
  284. package/src/internal/types/entity.ts +23 -0
  285. package/src/internal/types/index.ts +9 -0
  286. package/src/internal/types/meta.ts +76 -0
  287. package/src/internal/types/typename.ts +45 -0
  288. package/src/internal/types/version.ts +20 -0
  289. package/src/query/filter.ts +455 -0
  290. package/src/query/index.ts +5 -1
  291. package/src/query/order.ts +34 -0
  292. package/src/query/query.test.ts +334 -0
  293. package/src/query/query.ts +303 -0
  294. package/src/query/testing.ts +64 -0
  295. package/src/query/types.ts +23 -0
  296. package/src/query/util.ts +25 -0
  297. package/src/testing/api.test.ts +100 -0
  298. package/src/testing/index.ts +3 -1
  299. package/src/testing/test-data.ts +130 -0
  300. package/src/testing/test-schema.ts +213 -0
  301. package/src/testing/util.ts +78 -0
  302. package/dist/lib/browser/chunk-2KNTYMIW.mjs +0 -697
  303. package/dist/lib/browser/chunk-2KNTYMIW.mjs.map +0 -7
  304. package/dist/lib/node-esm/chunk-EXNNC62J.mjs +0 -697
  305. package/dist/lib/node-esm/chunk-EXNNC62J.mjs.map +0 -7
  306. package/dist/types/src/experimental/database.d.ts +0 -8
  307. package/dist/types/src/experimental/database.d.ts.map +0 -1
  308. package/dist/types/src/experimental/index.d.ts +0 -1
  309. package/dist/types/src/experimental/index.d.ts.map +0 -1
  310. package/dist/types/src/experimental/queue.d.ts +0 -8
  311. package/dist/types/src/experimental/queue.d.ts.map +0 -1
  312. package/dist/types/src/experimental/space.d.ts +0 -8
  313. package/dist/types/src/experimental/space.d.ts.map +0 -1
  314. package/dist/types/src/query/dsl.d.ts +0 -218
  315. package/dist/types/src/query/dsl.d.ts.map +0 -1
  316. package/dist/types/src/query/dsl.test.d.ts +0 -2
  317. package/dist/types/src/query/dsl.test.d.ts.map +0 -1
  318. package/dist/types/src/testing/types.d.ts +0 -113
  319. package/dist/types/src/testing/types.d.ts.map +0 -1
  320. package/src/experimental/database.ts +0 -11
  321. package/src/experimental/index.ts +0 -7
  322. package/src/experimental/queue.ts +0 -11
  323. package/src/experimental/space.ts +0 -11
  324. package/src/query/dsl.test.ts +0 -324
  325. package/src/query/dsl.ts +0 -646
  326. package/src/test/api.test.ts +0 -173
  327. package/src/testing/types.ts +0 -91
  328. /package/dist/types/src/{test → testing}/api.test.d.ts +0 -0
@@ -0,0 +1,525 @@
1
+ //
2
+ // Copyright 2024 DXOS.org
3
+ //
4
+
5
+ import * as Effect from 'effect/Effect';
6
+ import * as Option from 'effect/Option';
7
+ import * as ParseResult from 'effect/ParseResult';
8
+ import * as Schema from 'effect/Schema';
9
+ import * as SchemaAST from 'effect/SchemaAST';
10
+
11
+ import { type EncodedReference, Reference } from '@dxos/echo-protocol';
12
+ import { compositeRuntime } from '@dxos/echo-signals/runtime';
13
+ import { assertArgument, invariant } from '@dxos/invariant';
14
+ import { DXN, ObjectId } from '@dxos/keys';
15
+
16
+ import { ReferenceAnnotationId, getSchemaDXN, getTypeAnnotation, getTypeIdentifierAnnotation } from '../annotations';
17
+ import { type JsonSchemaType } from '../json-schema';
18
+ import type { AnyProperties, WithId } from '../types';
19
+
20
+ /**
21
+ * The `$id` and `$ref` fields for an ECHO reference schema.
22
+ */
23
+ export const JSON_SCHEMA_ECHO_REF_ID = '/schemas/echo/ref';
24
+
25
+ export const getSchemaReference = (property: JsonSchemaType): { typename: string } | undefined => {
26
+ const { $id, reference: { schema: { $ref } = {} } = {} } = property;
27
+ if ($id === JSON_SCHEMA_ECHO_REF_ID && $ref) {
28
+ return { typename: DXN.parse($ref).typename };
29
+ }
30
+ };
31
+
32
+ export const createSchemaReference = (typename: string): JsonSchemaType => {
33
+ return {
34
+ $id: JSON_SCHEMA_ECHO_REF_ID,
35
+ reference: {
36
+ schema: {
37
+ $ref: DXN.fromTypename(typename).toString(),
38
+ },
39
+ },
40
+ };
41
+ };
42
+
43
+ /**
44
+ * Runtime type-info for a reference extracted from effect AST.
45
+ */
46
+ export type RefereneAST = {
47
+ /**
48
+ * Typename of linked schema.
49
+ */
50
+ typename: string;
51
+
52
+ /**
53
+ * Version of linked schema.
54
+ */
55
+ version: string;
56
+ };
57
+
58
+ export const getReferenceAst = (ast: SchemaAST.AST): RefereneAST | undefined => {
59
+ if (ast._tag !== 'Declaration' || !ast.annotations[ReferenceAnnotationId]) {
60
+ return undefined;
61
+ }
62
+ return {
63
+ typename: (ast.annotations[ReferenceAnnotationId] as any).typename,
64
+ version: (ast.annotations[ReferenceAnnotationId] as any).version,
65
+ };
66
+ };
67
+
68
+ export const RefTypeId: unique symbol = Symbol('@dxos/echo/internal/Ref');
69
+
70
+ /**
71
+ * Reference Schema.
72
+ */
73
+ export interface RefSchema<T extends WithId> extends Schema.SchemaClass<Ref<T>, EncodedReference> {}
74
+
75
+ /**
76
+ * Type of the `Ref` function and extra methods attached to it.
77
+ */
78
+ export interface RefFn {
79
+ <S extends Schema.Schema.Any>(schema: S): RefSchema<Schema.Schema.Type<S>>;
80
+
81
+ /**
82
+ * @returns True if the object is a reference.
83
+ */
84
+ isRef: (obj: unknown) => obj is Ref<any>;
85
+
86
+ /**
87
+ * @returns True if the reference points to the given object id.
88
+ */
89
+ hasObjectId: (id: ObjectId) => (ref: Ref<any>) => boolean;
90
+
91
+ /**
92
+ * @returns True if the schema is a reference schema.
93
+ */
94
+ isRefSchema: (schema: Schema.Schema<any, any>) => schema is RefSchema<any>;
95
+
96
+ /**
97
+ * @returns True if the schema AST is a reference schema.
98
+ */
99
+ isRefSchemaAST: (ast: SchemaAST.AST) => boolean;
100
+
101
+ /**
102
+ * Constructs a reference that points to the given object.
103
+ */
104
+ // TODO(burdon): Narrow to Obj.Any?
105
+ make: <T extends WithId>(object: T) => Ref<T>;
106
+
107
+ /**
108
+ * Constructs a reference that points to the object specified by the provided DXN.
109
+ */
110
+ fromDXN: (dxn: DXN) => Ref<any>;
111
+ }
112
+
113
+ /**
114
+ * Schema builder for references.
115
+ */
116
+ export const Ref: RefFn = <S extends Schema.Schema.Any>(schema: S): RefSchema<Schema.Schema.Type<S>> => {
117
+ assertArgument(Schema.isSchema(schema), 'schema', 'Must call with an instance of effect-schema');
118
+
119
+ const annotation = getTypeAnnotation(schema);
120
+ if (annotation == null) {
121
+ throw new Error('Reference target must be an ECHO schema.');
122
+ }
123
+
124
+ return createEchoReferenceSchema(
125
+ getTypeIdentifierAnnotation(schema),
126
+ annotation.typename,
127
+ annotation.version,
128
+ getSchemaExpectedName(schema.ast),
129
+ );
130
+ };
131
+
132
+ /**
133
+ * Represents materialized reference to a target.
134
+ * This is the data type for the fields marked as ref.
135
+ */
136
+ export interface Ref<T> {
137
+ /**
138
+ * Target object DXN.
139
+ */
140
+ get dxn(): DXN;
141
+
142
+ /**
143
+ * @returns The reference target.
144
+ * May return `undefined` if the object is not loaded in the working set.
145
+ * Accessing this property, even if it returns `undefined` will trigger the object to be loaded to the working set.
146
+ *
147
+ * @reactive Supports signal subscriptions.
148
+ */
149
+ get target(): T | undefined;
150
+
151
+ /**
152
+ * @returns Promise that will resolves with the target object.
153
+ * Will load the object from disk if it is not present in the working set.
154
+ * @throws If the object is not available locally.
155
+ */
156
+ load(): Promise<T>;
157
+
158
+ /**
159
+ * @returns Promise that will resolves with the target object or undefined if the object is not loaded locally.
160
+ */
161
+ tryLoad(): Promise<T | undefined>;
162
+
163
+ /**
164
+ * Do not inline the target object in the reference.
165
+ * Makes .target unavailable unless the reference is connected to a database context.
166
+ *
167
+ * When serialized with toJSON, the difference is between:
168
+ * `{ "/": "dxn:..." }`
169
+ * and
170
+ * `{ "/": "dxn:...", "target": { ... } }`
171
+ *
172
+ * Clones the reference object.
173
+ */
174
+ noInline(): Ref<T>;
175
+
176
+ /**
177
+ * Serializes the reference to a JSON object.
178
+ * The serialization format is compatible with the IPLD-style encoded references.
179
+ * When a reference has a saved target (i.e. the target or object holding the reference is not in the database),
180
+ * the target is included in the serialized object.
181
+ *
182
+ * Examples:
183
+ * `{ "/": "dxn:..." }`
184
+ * `{ "/": "dxn:...", "target": { ... } }`
185
+ */
186
+ encode(): EncodedReference;
187
+
188
+ [RefTypeId]: {
189
+ _T: T;
190
+ };
191
+ }
192
+
193
+ export declare namespace Ref {
194
+ /**
195
+ * Target of the reference.
196
+ */
197
+ export type Target<R> = R extends Ref<infer U> ? U : never;
198
+ }
199
+
200
+ Ref.isRef = (obj: any): obj is Ref<any> => {
201
+ return obj && typeof obj === 'object' && RefTypeId in obj;
202
+ };
203
+
204
+ Ref.hasObjectId = (id: ObjectId) => (ref: Ref<any>) => ref.dxn.isLocalObjectId() && ref.dxn.parts[1] === id;
205
+
206
+ Ref.isRefSchema = (schema: Schema.Schema<any, any>): schema is RefSchema<any> => {
207
+ return Ref.isRefSchemaAST(schema.ast);
208
+ };
209
+
210
+ Ref.isRefSchemaAST = (ast: SchemaAST.AST): boolean => {
211
+ return SchemaAST.getAnnotation(ast, ReferenceAnnotationId).pipe(Option.isSome);
212
+ };
213
+
214
+ Ref.make = <T extends AnyProperties>(obj: T): Ref<T> => {
215
+ if (typeof obj !== 'object' || obj === null) {
216
+ throw new TypeError('Expected: ECHO object.');
217
+ }
218
+
219
+ // TODO(dmaretskyi): Extract to `getObjectDXN` function.
220
+ const id = obj.id;
221
+ invariant(ObjectId.isValid(id), 'Invalid object ID');
222
+ const dxn = Reference.localObjectReference(id).toDXN();
223
+ return new RefImpl(dxn, obj);
224
+ };
225
+
226
+ Ref.fromDXN = (dxn: DXN): Ref<any> => {
227
+ assertArgument(dxn instanceof DXN, 'dxn', 'Expected DXN');
228
+ return new RefImpl(dxn);
229
+ };
230
+
231
+ /**
232
+ * `reference` field on the schema object.
233
+ */
234
+ export type JsonSchemaReferenceInfo = {
235
+ schema: { $ref: string };
236
+ schemaVersion?: string;
237
+ };
238
+
239
+ /**
240
+ * @internal
241
+ */
242
+ // TODO(burdon): Move to json schema and make private?
243
+ export const createEchoReferenceSchema = (
244
+ echoId: string | undefined,
245
+ typename: string | undefined,
246
+ version: string | undefined,
247
+ schemaName?: string, // TODO(burdon): Not used.
248
+ ): Schema.SchemaClass<Ref<any>, EncodedReference> => {
249
+ if (!echoId && !typename) {
250
+ throw new TypeError('Either echoId or typename must be provided.');
251
+ }
252
+
253
+ const referenceInfo: JsonSchemaReferenceInfo = {
254
+ schema: {
255
+ // TODO(dmaretskyi): Include version?
256
+ $ref: echoId ?? DXN.fromTypename(typename!).toString(),
257
+ },
258
+ schemaVersion: version,
259
+ };
260
+
261
+ // TODO(dmaretskyi): Add name and description.
262
+ const refSchema = Schema.declare<Ref<any>, EncodedReference, []>(
263
+ [],
264
+ {
265
+ encode: () => {
266
+ return (value) => {
267
+ return Effect.succeed({
268
+ '/': (value as Ref<any>).dxn.toString(),
269
+ });
270
+ };
271
+ },
272
+ decode: () => {
273
+ return (value) => {
274
+ // TODO(dmaretskyi): This branch seems to be taken by Schema.is
275
+ if (Ref.isRef(value)) {
276
+ return Effect.succeed(value);
277
+ }
278
+
279
+ if (typeof value !== 'object' || value == null || typeof (value as any)['/'] !== 'string') {
280
+ return Effect.fail(new ParseResult.Unexpected(value, 'reference'));
281
+ }
282
+
283
+ return Effect.succeed(Ref.fromDXN(DXN.parse((value as any)['/'])));
284
+ };
285
+ },
286
+ },
287
+ {
288
+ jsonSchema: {
289
+ // TODO(dmaretskyi): We should remove `$id` and keep `$ref` with a fully qualified name.
290
+ $id: JSON_SCHEMA_ECHO_REF_ID,
291
+ $ref: JSON_SCHEMA_ECHO_REF_ID,
292
+ reference: referenceInfo,
293
+ },
294
+ [ReferenceAnnotationId]: {
295
+ typename: typename ?? '',
296
+ version,
297
+ },
298
+ },
299
+ );
300
+
301
+ return refSchema;
302
+ };
303
+
304
+ const getSchemaExpectedName = (ast: SchemaAST.Annotated): string | undefined => {
305
+ return SchemaAST.getIdentifierAnnotation(ast).pipe(
306
+ Option.orElse(() => SchemaAST.getTitleAnnotation(ast)),
307
+ Option.orElse(() => SchemaAST.getDescriptionAnnotation(ast)),
308
+ Option.getOrElse(() => undefined),
309
+ );
310
+ };
311
+
312
+ export interface RefResolver {
313
+ /**
314
+ * Resolve ref synchronously from the objects in the working set.
315
+ *
316
+ * @param dxn
317
+ * @param load If true the resolver should attempt to load the object from disk.
318
+ * @param onLoad Callback to call when the object is loaded.
319
+ */
320
+ resolveSync(dxn: DXN, load: boolean, onLoad?: () => void): AnyProperties | undefined;
321
+
322
+ /**
323
+ * Resolver ref asynchronously.
324
+ */
325
+ resolve(dxn: DXN): Promise<AnyProperties | undefined>;
326
+
327
+ // TODO(dmaretskyi): Combine with `resolve`.
328
+ resolveSchema(dxn: DXN): Promise<Schema.Schema.AnyNoContext | undefined>;
329
+ }
330
+
331
+ export class RefImpl<T> implements Ref<T> {
332
+ #dxn: DXN;
333
+ #resolver?: RefResolver = undefined;
334
+ #signal = compositeRuntime.createSignal();
335
+
336
+ /**
337
+ * Target is set when the reference is created from a specific object.
338
+ * In this case, the target might not be in the database.
339
+ */
340
+ #target: T | undefined = undefined;
341
+
342
+ /**
343
+ * Callback to issue a reactive notification when object is resolved.
344
+ */
345
+ #resolverCallback = () => {
346
+ this.#signal.notifyWrite();
347
+ };
348
+
349
+ constructor(dxn: DXN, target?: T) {
350
+ this.#dxn = dxn;
351
+ this.#target = target;
352
+ }
353
+
354
+ /**
355
+ * @inheritdoc
356
+ */
357
+ get dxn(): DXN {
358
+ return this.#dxn;
359
+ }
360
+
361
+ /**
362
+ * @inheritdoc
363
+ */
364
+ get target(): T | undefined {
365
+ this.#signal.notifyRead();
366
+ if (this.#target) {
367
+ return this.#target;
368
+ }
369
+
370
+ invariant(this.#resolver, 'Resolver is not set');
371
+ return this.#resolver.resolveSync(this.#dxn, true, this.#resolverCallback) as T | undefined;
372
+ }
373
+
374
+ /**
375
+ * @inheritdoc
376
+ */
377
+ async load(): Promise<T> {
378
+ if (this.#target) {
379
+ return this.#target;
380
+ }
381
+ invariant(this.#resolver, 'Resolver is not set');
382
+ const obj = await this.#resolver.resolve(this.#dxn);
383
+ if (obj == null) {
384
+ throw new Error('Object not found');
385
+ }
386
+ return obj as T;
387
+ }
388
+
389
+ /**
390
+ * @inheritdoc
391
+ */
392
+ async tryLoad(): Promise<T | undefined> {
393
+ invariant(this.#resolver, 'Resolver is not set');
394
+ return (await this.#resolver.resolve(this.#dxn)) as T | undefined;
395
+ }
396
+
397
+ /**
398
+ * Do not inline the target object in the reference.
399
+ * Makes .target unavailable unless the reference is connected to a database context.
400
+ * Clones the reference object.
401
+ */
402
+ noInline(): RefImpl<T> {
403
+ const ref = new RefImpl<T>(this.#dxn, undefined);
404
+ ref.#resolver = this.#resolver;
405
+ return ref;
406
+ }
407
+
408
+ encode(): EncodedReference {
409
+ return {
410
+ '/': this.#dxn.toString(),
411
+ ...(this.#target ? { target: this.#target } : {}),
412
+ };
413
+ }
414
+
415
+ /**
416
+ * Serializes the reference to a JSON object.
417
+ * The serialization format is compatible with the IPLD-style encoded references.
418
+ * When a reference has a saved target (i.e. the target or object holding the reference is not in the database),
419
+ * the target is included in the serialized object.
420
+ */
421
+ toJSON(): EncodedReference {
422
+ return this.encode();
423
+ }
424
+
425
+ toString(): string {
426
+ if (this.#target) {
427
+ return `Ref(${this.#target.toString()})`;
428
+ }
429
+
430
+ return `Ref(${this.#dxn.toString()})`;
431
+ }
432
+
433
+ [RefTypeId] = refVariance;
434
+
435
+ /**
436
+ * Internal method to set the resolver.
437
+ *
438
+ * @internal
439
+ */
440
+ _setResolver(resolver: RefResolver): void {
441
+ this.#resolver = resolver;
442
+ }
443
+
444
+ /**
445
+ * Internal method to get the saved target.
446
+ * Not the same as `target` which is resolved from the resolver.
447
+ *
448
+ * @internal
449
+ */
450
+ _getSavedTarget(): T | undefined {
451
+ return this.#target;
452
+ }
453
+ }
454
+
455
+ /**
456
+ * Internal API for setting the reference resolver.
457
+ */
458
+ export const setRefResolver = (ref: Ref<any>, resolver: RefResolver) => {
459
+ invariant(ref instanceof RefImpl, 'Ref is not an instance of RefImpl');
460
+ ref._setResolver(resolver);
461
+ };
462
+
463
+ /**
464
+ * Internal API for getting the saved target on a reference.
465
+ */
466
+ export const getRefSavedTarget = (ref: Ref<any>): AnyProperties | undefined => {
467
+ invariant(ref instanceof RefImpl, 'Ref is not an instance of RefImpl');
468
+ return ref._getSavedTarget();
469
+ };
470
+
471
+ // Used to validate reference target type.
472
+ const refVariance: Ref<any>[typeof RefTypeId] = {
473
+ _T: null as any,
474
+ };
475
+
476
+ export const refFromEncodedReference = (encodedReference: EncodedReference, resolver?: RefResolver): Ref<any> => {
477
+ const dxn = DXN.parse(encodedReference['/']);
478
+ const ref = new RefImpl(dxn);
479
+
480
+ // TODO(dmaretskyi): Handle inline target in the encoded reference.
481
+
482
+ if (resolver) {
483
+ setRefResolver(ref, resolver);
484
+ }
485
+ return ref;
486
+ };
487
+
488
+ export class StaticRefResolver implements RefResolver {
489
+ public objects = new Map<ObjectId, AnyProperties>();
490
+ public schemas = new Map<DXN.String, Schema.Schema.AnyNoContext>();
491
+
492
+ addObject(obj: AnyProperties): this {
493
+ this.objects.set(obj.id, obj);
494
+ return this;
495
+ }
496
+
497
+ addSchema(schema: Schema.Schema.AnyNoContext): this {
498
+ const dxn = getSchemaDXN(schema);
499
+ invariant(dxn, 'Schema has no DXN');
500
+ this.schemas.set(dxn.toString(), schema);
501
+ return this;
502
+ }
503
+
504
+ resolveSync(dxn: DXN, _load: boolean, _onLoad?: () => void): AnyProperties | undefined {
505
+ const id = dxn?.asEchoDXN()?.echoId;
506
+ if (id == null) {
507
+ return undefined;
508
+ }
509
+
510
+ return this.objects.get(id);
511
+ }
512
+
513
+ async resolve(dxn: DXN): Promise<AnyProperties | undefined> {
514
+ const id = dxn?.asEchoDXN()?.echoId;
515
+ if (id == null) {
516
+ return undefined;
517
+ }
518
+
519
+ return this.objects.get(id);
520
+ }
521
+
522
+ async resolveSchema(dxn: DXN): Promise<Schema.Schema.AnyNoContext | undefined> {
523
+ return this.schemas.get(dxn.toString());
524
+ }
525
+ }
@@ -0,0 +1,42 @@
1
+ //
2
+ // Copyright 2024 DXOS.org
3
+ //
4
+
5
+ import * as Schema from 'effect/Schema';
6
+ import { describe, test } from 'vitest';
7
+
8
+ import { FieldPath } from '../annotations';
9
+ import { FormatAnnotation, TypeFormat } from '../formats';
10
+ import { ECHO_ANNOTATIONS_NS_KEY, toJsonSchema } from '../json-schema';
11
+ import { TypedObject } from '../object';
12
+
13
+ import { composeSchema } from './compose';
14
+
15
+ describe('schema composition', () => {
16
+ test('schema composition', ({ expect }) => {
17
+ class BaseType extends TypedObject({ typename: 'example.com/Person', version: '0.1.0' })({
18
+ name: Schema.String,
19
+ email: Schema.String,
20
+ }) {}
21
+
22
+ const OverlaySchema = Schema.Struct({
23
+ email: Schema.String.pipe(FieldPath('$.email'), FormatAnnotation.set(TypeFormat.Email)),
24
+ });
25
+
26
+ const baseSchema = toJsonSchema(BaseType);
27
+ const overlaySchema = toJsonSchema(OverlaySchema);
28
+ const composedSchema = composeSchema(baseSchema, overlaySchema);
29
+ expect(composedSchema.properties).to.deep.eq({
30
+ email: {
31
+ type: 'string',
32
+ format: TypeFormat.Email,
33
+ // TODO(dmaretskyi): Should use the new field.
34
+ [ECHO_ANNOTATIONS_NS_KEY]: {
35
+ meta: {
36
+ path: '$.email',
37
+ },
38
+ },
39
+ },
40
+ });
41
+ });
42
+ });
@@ -0,0 +1,37 @@
1
+ //
2
+ // Copyright 2024 DXOS.org
3
+ //
4
+
5
+ import { invariant } from '@dxos/invariant';
6
+
7
+ import { type JsonSchemaType } from '../json-schema';
8
+
9
+ import { getSnapshot } from './snapshot';
10
+
11
+ /**
12
+ * Creates a composite schema from the source and projection schemas.
13
+ */
14
+ // TODO(burdon): Use effect schema projections.
15
+ // TODO(burdon): Can avoid having to call this every time we modify any property on the view?
16
+ export const composeSchema = (source: JsonSchemaType, target: JsonSchemaType): JsonSchemaType => {
17
+ const result: JsonSchemaType = getSnapshot(target);
18
+ invariant('type' in result && result.type === 'object', 'source schema must be an object');
19
+ invariant('type' in source && source.type === 'object', 'target schema must be an object');
20
+
21
+ for (const prop in result.properties) {
22
+ const propSchema = source.properties![prop]; // TODO(dmaretskyi): Find by json-path instead.
23
+ const annotations = (propSchema as JsonSchemaType)?.annotations?.meta;
24
+ if (annotations) {
25
+ (result.properties[prop] as JsonSchemaType).annotations ??= {};
26
+ (result.properties[prop] as JsonSchemaType).annotations!.meta ??= {};
27
+ for (const key in annotations) {
28
+ (result.properties[prop] as JsonSchemaType).annotations!.meta![key] ??= {};
29
+ Object.assign((result.properties[prop] as JsonSchemaType).annotations!.meta![key], annotations[key], {
30
+ ...(result.properties[prop] as JsonSchemaType).annotations!.meta![key],
31
+ });
32
+ }
33
+ }
34
+ }
35
+
36
+ return result;
37
+ };