@dxos/echo 0.8.4-main.67995b8 → 0.8.4-main.70d3990

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 (329) hide show
  1. package/README.md +1 -1
  2. package/dist/lib/browser/chunk-7GH6RXJ3.mjs +3683 -0
  3. package/dist/lib/browser/chunk-7GH6RXJ3.mjs.map +7 -0
  4. package/dist/lib/browser/chunk-E4UTVJNF.mjs +1111 -0
  5. package/dist/lib/browser/chunk-E4UTVJNF.mjs.map +7 -0
  6. package/dist/lib/browser/index.mjs +24 -1
  7. package/dist/lib/browser/internal/index.mjs +336 -0
  8. package/dist/lib/browser/internal/index.mjs.map +7 -0
  9. package/dist/lib/browser/meta.json +1 -1
  10. package/dist/lib/browser/testing/index.mjs +289 -31
  11. package/dist/lib/browser/testing/index.mjs.map +4 -4
  12. package/dist/lib/node-esm/chunk-JE5RXM2I.mjs +1111 -0
  13. package/dist/lib/node-esm/chunk-JE5RXM2I.mjs.map +7 -0
  14. package/dist/lib/node-esm/chunk-M4B6BMD2.mjs +3683 -0
  15. package/dist/lib/node-esm/chunk-M4B6BMD2.mjs.map +7 -0
  16. package/dist/lib/node-esm/index.mjs +24 -1
  17. package/dist/lib/node-esm/internal/index.mjs +336 -0
  18. package/dist/lib/node-esm/internal/index.mjs.map +7 -0
  19. package/dist/lib/node-esm/meta.json +1 -1
  20. package/dist/lib/node-esm/testing/index.mjs +289 -31
  21. package/dist/lib/node-esm/testing/index.mjs.map +4 -4
  22. package/dist/types/src/Annotation.d.ts +2 -0
  23. package/dist/types/src/Annotation.d.ts.map +1 -0
  24. package/dist/types/src/Database.d.ts +137 -0
  25. package/dist/types/src/Database.d.ts.map +1 -0
  26. package/dist/types/src/Entity.d.ts +36 -0
  27. package/dist/types/src/Entity.d.ts.map +1 -0
  28. package/dist/types/src/Format.d.ts +4 -0
  29. package/dist/types/src/Format.d.ts.map +1 -0
  30. package/dist/types/src/JsonSchema.d.ts +9 -0
  31. package/dist/types/src/JsonSchema.d.ts.map +1 -0
  32. package/dist/types/src/Key.d.ts +1 -0
  33. package/dist/types/src/Key.d.ts.map +1 -1
  34. package/dist/types/src/Obj.d.ts +152 -35
  35. package/dist/types/src/Obj.d.ts.map +1 -1
  36. package/dist/types/src/Ref.d.ts +10 -10
  37. package/dist/types/src/Ref.d.ts.map +1 -1
  38. package/dist/types/src/Relation.d.ts +20 -14
  39. package/dist/types/src/Relation.d.ts.map +1 -1
  40. package/dist/types/src/Tag.d.ts +17 -0
  41. package/dist/types/src/Tag.d.ts.map +1 -0
  42. package/dist/types/src/Type.d.ts +39 -50
  43. package/dist/types/src/Type.d.ts.map +1 -1
  44. package/dist/types/src/errors.d.ts +68 -0
  45. package/dist/types/src/errors.d.ts.map +1 -0
  46. package/dist/types/src/index.d.ts +9 -3
  47. package/dist/types/src/index.d.ts.map +1 -1
  48. package/dist/types/src/internal/annotations/annotations.d.ts +174 -0
  49. package/dist/types/src/internal/annotations/annotations.d.ts.map +1 -0
  50. package/dist/types/src/internal/annotations/annotations.test.d.ts +2 -0
  51. package/dist/types/src/internal/annotations/annotations.test.d.ts.map +1 -0
  52. package/dist/types/src/internal/annotations/index.d.ts +3 -0
  53. package/dist/types/src/internal/annotations/index.d.ts.map +1 -0
  54. package/dist/types/src/internal/annotations/util.d.ts +26 -0
  55. package/dist/types/src/internal/annotations/util.d.ts.map +1 -0
  56. package/dist/types/src/internal/entities/entity.d.ts +10 -0
  57. package/dist/types/src/internal/entities/entity.d.ts.map +1 -0
  58. package/dist/types/src/internal/entities/expando.d.ts +16 -0
  59. package/dist/types/src/internal/entities/expando.d.ts.map +1 -0
  60. package/dist/types/src/internal/entities/index.d.ts +6 -0
  61. package/dist/types/src/internal/entities/index.d.ts.map +1 -0
  62. package/dist/types/src/internal/entities/model.d.ts +70 -0
  63. package/dist/types/src/internal/entities/model.d.ts.map +1 -0
  64. package/dist/types/src/internal/entities/object.d.ts +11 -0
  65. package/dist/types/src/internal/entities/object.d.ts.map +1 -0
  66. package/dist/types/src/internal/entities/relation.d.ts +55 -0
  67. package/dist/types/src/internal/entities/relation.d.ts.map +1 -0
  68. package/dist/types/src/internal/entities/util.d.ts +2 -0
  69. package/dist/types/src/internal/entities/util.d.ts.map +1 -0
  70. package/dist/types/src/internal/formats/date.d.ts +63 -0
  71. package/dist/types/src/internal/formats/date.d.ts.map +1 -0
  72. package/dist/types/src/internal/formats/date.test.d.ts +2 -0
  73. package/dist/types/src/internal/formats/date.test.d.ts.map +1 -0
  74. package/dist/types/src/internal/formats/format.d.ts +32 -0
  75. package/dist/types/src/internal/formats/format.d.ts.map +1 -0
  76. package/dist/types/src/internal/formats/format.test.d.ts +2 -0
  77. package/dist/types/src/internal/formats/format.test.d.ts.map +1 -0
  78. package/dist/types/src/internal/formats/index.d.ts +8 -0
  79. package/dist/types/src/internal/formats/index.d.ts.map +1 -0
  80. package/dist/types/src/internal/formats/number.d.ts +31 -0
  81. package/dist/types/src/internal/formats/number.d.ts.map +1 -0
  82. package/dist/types/src/internal/formats/object.d.ts +35 -0
  83. package/dist/types/src/internal/formats/object.d.ts.map +1 -0
  84. package/dist/types/src/internal/formats/select.d.ts +11 -0
  85. package/dist/types/src/internal/formats/select.d.ts.map +1 -0
  86. package/dist/types/src/internal/formats/string.d.ts +38 -0
  87. package/dist/types/src/internal/formats/string.d.ts.map +1 -0
  88. package/dist/types/src/internal/formats/types.d.ts +68 -0
  89. package/dist/types/src/internal/formats/types.d.ts.map +1 -0
  90. package/dist/types/src/internal/index.d.ts +11 -0
  91. package/dist/types/src/internal/index.d.ts.map +1 -0
  92. package/dist/types/src/internal/json-schema/annotations.d.ts +19 -0
  93. package/dist/types/src/internal/json-schema/annotations.d.ts.map +1 -0
  94. package/dist/types/src/internal/json-schema/effect-schema.test.d.ts +2 -0
  95. package/dist/types/src/internal/json-schema/effect-schema.test.d.ts.map +1 -0
  96. package/dist/types/src/internal/json-schema/index.d.ts +5 -0
  97. package/dist/types/src/internal/json-schema/index.d.ts.map +1 -0
  98. package/dist/types/src/internal/json-schema/json-schema-normalize.d.ts +7 -0
  99. package/dist/types/src/internal/json-schema/json-schema-normalize.d.ts.map +1 -0
  100. package/dist/types/src/internal/json-schema/json-schema-type.d.ts +250 -0
  101. package/dist/types/src/internal/json-schema/json-schema-type.d.ts.map +1 -0
  102. package/dist/types/src/internal/json-schema/json-schema.d.ts +29 -0
  103. package/dist/types/src/internal/json-schema/json-schema.d.ts.map +1 -0
  104. package/dist/types/src/internal/json-schema/json-schema.test.d.ts +2 -0
  105. package/dist/types/src/internal/json-schema/json-schema.test.d.ts.map +1 -0
  106. package/dist/types/src/internal/object/common.d.ts +18 -0
  107. package/dist/types/src/internal/object/common.d.ts.map +1 -0
  108. package/dist/types/src/internal/object/create-object.d.ts +39 -0
  109. package/dist/types/src/internal/object/create-object.d.ts.map +1 -0
  110. package/dist/types/src/internal/object/create-object.test.d.ts +2 -0
  111. package/dist/types/src/internal/object/create-object.test.d.ts.map +1 -0
  112. package/dist/types/src/internal/object/deleted.d.ts +6 -0
  113. package/dist/types/src/internal/object/deleted.d.ts.map +1 -0
  114. package/dist/types/src/internal/object/ids.d.ts +6 -0
  115. package/dist/types/src/internal/object/ids.d.ts.map +1 -0
  116. package/dist/types/src/internal/object/index.d.ts +8 -0
  117. package/dist/types/src/internal/object/index.d.ts.map +1 -0
  118. package/dist/types/src/internal/object/inspect.d.ts +2 -0
  119. package/dist/types/src/internal/object/inspect.d.ts.map +1 -0
  120. package/dist/types/src/internal/object/json-serializer.d.ts +31 -0
  121. package/dist/types/src/internal/object/json-serializer.d.ts.map +1 -0
  122. package/dist/types/src/internal/object/json-serializer.test.d.ts +2 -0
  123. package/dist/types/src/internal/object/json-serializer.test.d.ts.map +1 -0
  124. package/dist/types/src/internal/object/schema-validator.d.ts +15 -0
  125. package/dist/types/src/internal/object/schema-validator.d.ts.map +1 -0
  126. package/dist/types/src/internal/object/schema-validator.test.d.ts +2 -0
  127. package/dist/types/src/internal/object/schema-validator.test.d.ts.map +1 -0
  128. package/dist/types/src/internal/object/typed-object.d.ts +31 -0
  129. package/dist/types/src/internal/object/typed-object.d.ts.map +1 -0
  130. package/dist/types/src/internal/object/typed-object.test.d.ts +2 -0
  131. package/dist/types/src/internal/object/typed-object.test.d.ts.map +1 -0
  132. package/dist/types/src/internal/proxy/handler.test.d.ts +2 -0
  133. package/dist/types/src/internal/proxy/handler.test.d.ts.map +1 -0
  134. package/dist/types/src/internal/proxy/index.d.ts +3 -0
  135. package/dist/types/src/internal/proxy/index.d.ts.map +1 -0
  136. package/dist/types/src/internal/proxy/make-object.d.ts +16 -0
  137. package/dist/types/src/internal/proxy/make-object.d.ts.map +1 -0
  138. package/dist/types/src/internal/proxy/schema.test.d.ts +2 -0
  139. package/dist/types/src/internal/proxy/schema.test.d.ts.map +1 -0
  140. package/dist/types/src/internal/proxy/typed-handler.d.ts +44 -0
  141. package/dist/types/src/internal/proxy/typed-handler.d.ts.map +1 -0
  142. package/dist/types/src/internal/proxy/typed-handler.test.d.ts +2 -0
  143. package/dist/types/src/internal/proxy/typed-handler.test.d.ts.map +1 -0
  144. package/dist/types/src/internal/proxy/typed-object.test.d.ts +2 -0
  145. package/dist/types/src/internal/proxy/typed-object.test.d.ts.map +1 -0
  146. package/dist/types/src/internal/ref/index.d.ts +3 -0
  147. package/dist/types/src/internal/ref/index.d.ts.map +1 -0
  148. package/dist/types/src/internal/ref/ref-array.d.ts +21 -0
  149. package/dist/types/src/internal/ref/ref-array.d.ts.map +1 -0
  150. package/dist/types/src/internal/ref/ref.d.ts +209 -0
  151. package/dist/types/src/internal/ref/ref.d.ts.map +1 -0
  152. package/dist/types/src/internal/ref/ref.test.d.ts +2 -0
  153. package/dist/types/src/internal/ref/ref.test.d.ts.map +1 -0
  154. package/dist/types/src/internal/schema/compose.d.ts +6 -0
  155. package/dist/types/src/internal/schema/compose.d.ts.map +1 -0
  156. package/dist/types/src/internal/schema/compose.test.d.ts +2 -0
  157. package/dist/types/src/internal/schema/compose.test.d.ts.map +1 -0
  158. package/dist/types/src/internal/schema/echo-schema.d.ts +168 -0
  159. package/dist/types/src/internal/schema/echo-schema.d.ts.map +1 -0
  160. package/dist/types/src/internal/schema/index.d.ts +6 -0
  161. package/dist/types/src/internal/schema/index.d.ts.map +1 -0
  162. package/dist/types/src/internal/schema/manipulation.d.ts +10 -0
  163. package/dist/types/src/internal/schema/manipulation.d.ts.map +1 -0
  164. package/dist/types/src/internal/schema/persistent-schema.d.ts +18 -0
  165. package/dist/types/src/internal/schema/persistent-schema.d.ts.map +1 -0
  166. package/dist/types/src/internal/schema/runtime-schema-registry.d.ts +18 -0
  167. package/dist/types/src/internal/schema/runtime-schema-registry.d.ts.map +1 -0
  168. package/dist/types/src/internal/schema/snapshot.d.ts +6 -0
  169. package/dist/types/src/internal/schema/snapshot.d.ts.map +1 -0
  170. package/dist/types/src/internal/types/base.d.ts +37 -0
  171. package/dist/types/src/internal/types/base.d.ts.map +1 -0
  172. package/dist/types/src/internal/types/entity.d.ts +12 -0
  173. package/dist/types/src/internal/types/entity.d.ts.map +1 -0
  174. package/dist/types/src/internal/types/index.d.ts +6 -0
  175. package/dist/types/src/internal/types/index.d.ts.map +1 -0
  176. package/dist/types/src/internal/types/meta.d.ts +40 -0
  177. package/dist/types/src/internal/types/meta.d.ts.map +1 -0
  178. package/dist/types/src/internal/types/typename.d.ts +13 -0
  179. package/dist/types/src/internal/types/typename.d.ts.map +1 -0
  180. package/dist/types/src/internal/types/version.d.ts +15 -0
  181. package/dist/types/src/internal/types/version.d.ts.map +1 -0
  182. package/dist/types/src/query/filter.d.ts +167 -0
  183. package/dist/types/src/query/filter.d.ts.map +1 -0
  184. package/dist/types/src/query/index.d.ts +4 -1
  185. package/dist/types/src/query/index.d.ts.map +1 -1
  186. package/dist/types/src/query/order.d.ts +12 -0
  187. package/dist/types/src/query/order.d.ts.map +1 -0
  188. package/dist/types/src/query/query.d.ts +112 -0
  189. package/dist/types/src/query/query.d.ts.map +1 -0
  190. package/dist/types/src/query/query.test.d.ts +2 -0
  191. package/dist/types/src/query/query.test.d.ts.map +1 -0
  192. package/dist/types/src/query/testing.d.ts +51 -0
  193. package/dist/types/src/query/testing.d.ts.map +1 -0
  194. package/dist/types/src/query/types.d.ts +17 -0
  195. package/dist/types/src/query/types.d.ts.map +1 -0
  196. package/dist/types/src/query/util.d.ts +8 -0
  197. package/dist/types/src/query/util.d.ts.map +1 -0
  198. package/dist/types/src/{test → testing}/api.test.d.ts.map +1 -1
  199. package/dist/types/src/testing/index.d.ts +3 -1
  200. package/dist/types/src/testing/index.d.ts.map +1 -1
  201. package/dist/types/src/testing/test-data.d.ts +18 -0
  202. package/dist/types/src/testing/test-data.d.ts.map +1 -0
  203. package/dist/types/src/testing/test-schema.d.ts +337 -0
  204. package/dist/types/src/testing/test-schema.d.ts.map +1 -0
  205. package/dist/types/src/testing/util.d.ts +16 -0
  206. package/dist/types/src/testing/util.d.ts.map +1 -0
  207. package/dist/types/tsconfig.tsbuildinfo +1 -1
  208. package/package.json +57 -37
  209. package/src/Annotation.ts +17 -0
  210. package/src/Database.ts +189 -0
  211. package/src/Entity.ts +51 -0
  212. package/src/Format.ts +11 -0
  213. package/src/JsonSchema.ts +16 -0
  214. package/src/Key.ts +3 -0
  215. package/src/Obj.ts +407 -71
  216. package/src/Ref.ts +9 -10
  217. package/src/Relation.ts +72 -46
  218. package/src/Tag.ts +40 -0
  219. package/src/Type.ts +98 -84
  220. package/src/errors.ts +18 -0
  221. package/src/index.ts +13 -4
  222. package/src/internal/README.md +83 -0
  223. package/src/internal/annotations/annotations.test.ts +96 -0
  224. package/src/internal/annotations/annotations.ts +463 -0
  225. package/src/internal/annotations/index.ts +6 -0
  226. package/src/internal/annotations/util.ts +70 -0
  227. package/src/internal/entities/entity.ts +109 -0
  228. package/src/internal/entities/expando.ts +23 -0
  229. package/src/internal/entities/index.ts +9 -0
  230. package/src/internal/entities/model.ts +129 -0
  231. package/src/internal/entities/object.ts +45 -0
  232. package/src/internal/entities/relation.ts +155 -0
  233. package/src/internal/entities/util.ts +33 -0
  234. package/src/internal/formats/date.test.ts +56 -0
  235. package/src/internal/formats/date.ts +217 -0
  236. package/src/internal/formats/format.test.ts +77 -0
  237. package/src/internal/formats/format.ts +55 -0
  238. package/src/internal/formats/index.ts +12 -0
  239. package/src/internal/formats/number.ts +89 -0
  240. package/src/internal/formats/object.ts +80 -0
  241. package/src/internal/formats/select.ts +16 -0
  242. package/src/internal/formats/string.ts +76 -0
  243. package/src/internal/formats/types.ts +180 -0
  244. package/src/internal/index.ts +38 -0
  245. package/src/internal/json-schema/annotations.ts +50 -0
  246. package/src/internal/json-schema/effect-schema.test.ts +143 -0
  247. package/src/internal/json-schema/index.ts +8 -0
  248. package/src/internal/json-schema/json-schema-normalize.ts +109 -0
  249. package/src/internal/json-schema/json-schema-type.ts +404 -0
  250. package/src/internal/json-schema/json-schema.test.ts +859 -0
  251. package/src/internal/json-schema/json-schema.ts +528 -0
  252. package/src/internal/object/common.ts +75 -0
  253. package/src/internal/object/create-object.test.ts +116 -0
  254. package/src/internal/object/create-object.ts +95 -0
  255. package/src/internal/object/deleted.ts +19 -0
  256. package/src/internal/object/ids.ts +12 -0
  257. package/src/internal/object/index.ts +11 -0
  258. package/src/internal/object/inspect.ts +46 -0
  259. package/src/internal/object/json-serializer.test.ts +94 -0
  260. package/src/internal/object/json-serializer.ts +230 -0
  261. package/src/internal/object/schema-validator.test.ts +186 -0
  262. package/src/internal/object/schema-validator.ts +242 -0
  263. package/src/internal/object/typed-object.test.ts +34 -0
  264. package/src/internal/object/typed-object.ts +94 -0
  265. package/src/internal/proxy/handler.test.ts +173 -0
  266. package/src/internal/proxy/index.ts +6 -0
  267. package/src/internal/proxy/make-object.ts +113 -0
  268. package/src/internal/proxy/schema.test.ts +137 -0
  269. package/src/internal/proxy/typed-handler.test.ts +102 -0
  270. package/src/internal/proxy/typed-handler.ts +233 -0
  271. package/src/internal/proxy/typed-object.test.ts +105 -0
  272. package/src/internal/ref/index.ts +6 -0
  273. package/src/internal/ref/ref-array.ts +39 -0
  274. package/src/internal/ref/ref.test.ts +101 -0
  275. package/src/internal/ref/ref.ts +525 -0
  276. package/src/internal/schema/compose.test.ts +42 -0
  277. package/src/internal/schema/compose.ts +37 -0
  278. package/src/internal/schema/echo-schema.ts +385 -0
  279. package/src/internal/schema/index.ts +9 -0
  280. package/src/internal/schema/manipulation.ts +92 -0
  281. package/src/internal/schema/persistent-schema.ts +28 -0
  282. package/src/internal/schema/runtime-schema-registry.ts +78 -0
  283. package/src/internal/schema/snapshot.ts +25 -0
  284. package/src/internal/types/base.ts +58 -0
  285. package/src/internal/types/entity.ts +23 -0
  286. package/src/internal/types/index.ts +9 -0
  287. package/src/internal/types/meta.ts +76 -0
  288. package/src/internal/types/typename.ts +45 -0
  289. package/src/internal/types/version.ts +20 -0
  290. package/src/query/filter.ts +455 -0
  291. package/src/query/index.ts +5 -1
  292. package/src/query/order.ts +34 -0
  293. package/src/query/query.test.ts +334 -0
  294. package/src/query/query.ts +303 -0
  295. package/src/query/testing.ts +64 -0
  296. package/src/query/types.ts +23 -0
  297. package/src/query/util.ts +25 -0
  298. package/src/testing/api.test.ts +100 -0
  299. package/src/testing/index.ts +3 -1
  300. package/src/testing/test-data.ts +130 -0
  301. package/src/testing/test-schema.ts +213 -0
  302. package/src/testing/util.ts +78 -0
  303. package/dist/lib/browser/chunk-EUA7CM23.mjs +0 -619
  304. package/dist/lib/browser/chunk-EUA7CM23.mjs.map +0 -7
  305. package/dist/lib/node-esm/chunk-IV6BWGHK.mjs +0 -619
  306. package/dist/lib/node-esm/chunk-IV6BWGHK.mjs.map +0 -7
  307. package/dist/types/src/experimental/database.d.ts +0 -8
  308. package/dist/types/src/experimental/database.d.ts.map +0 -1
  309. package/dist/types/src/experimental/index.d.ts +0 -1
  310. package/dist/types/src/experimental/index.d.ts.map +0 -1
  311. package/dist/types/src/experimental/queue.d.ts +0 -8
  312. package/dist/types/src/experimental/queue.d.ts.map +0 -1
  313. package/dist/types/src/experimental/space.d.ts +0 -8
  314. package/dist/types/src/experimental/space.d.ts.map +0 -1
  315. package/dist/types/src/query/dsl.d.ts +0 -218
  316. package/dist/types/src/query/dsl.d.ts.map +0 -1
  317. package/dist/types/src/query/dsl.test.d.ts +0 -2
  318. package/dist/types/src/query/dsl.test.d.ts.map +0 -1
  319. package/dist/types/src/testing/types.d.ts +0 -113
  320. package/dist/types/src/testing/types.d.ts.map +0 -1
  321. package/src/experimental/database.ts +0 -11
  322. package/src/experimental/index.ts +0 -7
  323. package/src/experimental/queue.ts +0 -11
  324. package/src/experimental/space.ts +0 -11
  325. package/src/query/dsl.test.ts +0 -323
  326. package/src/query/dsl.ts +0 -646
  327. package/src/test/api.test.ts +0 -173
  328. package/src/testing/types.ts +0 -91
  329. /package/dist/types/src/{test → testing}/api.test.d.ts +0 -0
package/src/Obj.ts CHANGED
@@ -2,161 +2,497 @@
2
2
  // Copyright 2025 DXOS.org
3
3
  //
4
4
 
5
- import { Schema } from 'effect';
5
+ import * as Function from 'effect/Function';
6
+ import * as Schema from 'effect/Schema';
6
7
 
7
- import * as EchoSchema from '@dxos/echo-schema';
8
+ import { type ForeignKey } from '@dxos/echo-protocol';
8
9
  import { assertArgument, invariant } from '@dxos/invariant';
9
- import { type DXN } from '@dxos/keys';
10
- import type * as LiveObject from '@dxos/live-object';
11
- import { live } from '@dxos/live-object';
12
- import { assumeType } from '@dxos/util';
13
-
14
- import type * as Ref from './Ref';
15
- import type * as Relation from './Relation';
16
- import type * as Type from './Type';
17
-
18
- // NOTE: Don't export: Obj.Any and Obj.Obj form the public API.
19
- interface ObjBase extends Type.OfKind<EchoSchema.EntityKind.Object> {
20
- readonly id: EchoSchema.ObjectId;
21
- }
10
+ import { type DXN, ObjectId } from '@dxos/keys';
11
+ import { getSnapshot as getSnapshot$ } from '@dxos/live-object';
12
+ import { assumeType, deepMapValues } from '@dxos/util';
13
+
14
+ import * as Entity from './Entity';
15
+ import {
16
+ type AnyEchoObject,
17
+ type AnyProperties,
18
+ type InternalObjectProps,
19
+ MetaId,
20
+ type ObjectJSON,
21
+ type ObjectMeta,
22
+ ObjectVersionId,
23
+ VersionTypeId,
24
+ getDescription as getDescription$,
25
+ getLabel as getLabel$,
26
+ getMeta as getMeta$,
27
+ getObjectDXN,
28
+ getSchema as getSchema$,
29
+ getSchemaTypename,
30
+ getTypeAnnotation,
31
+ getTypeDXN as getTypeDXN$,
32
+ isDeleted as isDeleted$,
33
+ isInstanceOf,
34
+ makeObject,
35
+ objectFromJSON,
36
+ objectToJSON,
37
+ setDescription as setDescription$,
38
+ setLabel as setLabel$,
39
+ } from './internal';
40
+ import * as Ref from './Ref';
41
+ import * as Type from './Type';
42
+
43
+ /**
44
+ * Base type for all ECHO objects.
45
+ * @private
46
+ */
47
+ interface BaseObj extends AnyEchoObject, Entity.OfKind<typeof Entity.Kind.Object> {}
48
+
49
+ /**
50
+ * Base type for all Obj objects.
51
+ */
52
+ export interface Any extends BaseObj {}
53
+
54
+ export const Any = Schema.Struct({}).pipe(
55
+ Type.Obj({
56
+ typename: 'dxos.org/type/Any',
57
+ version: '0.1.0',
58
+ }),
59
+ );
22
60
 
23
61
  /**
24
62
  * Object type with specific properties.
25
63
  */
26
- export type Obj<Props> = ObjBase & Props;
64
+ export type Obj<Props> = BaseObj & Props;
27
65
 
28
66
  /**
29
- * Base type for all ECHO objects.
67
+ * Object with arbitrary properties.
68
+ *
69
+ * NOTE: Due to how typescript works, this type is not assignable to a specific schema type.
70
+ * In that case, use `Obj.instanceOf` to check if an object is of a specific type.
30
71
  */
31
- export interface Any extends ObjBase {}
72
+ export interface AnyProps extends BaseObj, AnyProperties {}
73
+
74
+ const defaultMeta: ObjectMeta = {
75
+ keys: [],
76
+ };
32
77
 
33
- type Props<T> = { id?: EchoSchema.ObjectId } & Type.Properties<T>;
78
+ type Props<T = any> = {
79
+ id?: ObjectId;
80
+ [Meta]?: Partial<ObjectMeta>;
81
+ } & Type.Properties<T>;
34
82
 
35
- export type MakeProps<S extends Type.Obj.Any> = NoInfer<Props<Schema.Schema.Type<S>>>;
83
+ // TODO(burdon): Should we allow the caller to set the id?
84
+ export type MakeProps<T extends Schema.Schema.AnyNoContext> = {
85
+ id?: ObjectId;
86
+ [Meta]?: Partial<ObjectMeta>;
87
+ } & NoInfer<Props<Schema.Schema.Type<T>>>;
36
88
 
37
89
  /**
38
- * Creates new object.
90
+ * Creates a new object of the given types.
91
+ * @param schema - Object schema.
92
+ * @param props - Object properties.
93
+ * @param meta - Object metadata (deprecated) -- pass with Obj.Meta.
94
+ *
95
+ * Meta can be passed as a symbol in `props`.
96
+ *
97
+ * Example:
98
+ * ```ts
99
+ * const obj = Obj.make(Person, { [Obj.Meta]: { keys: [...] }, name: 'John' });
100
+ * ```
39
101
  */
40
- // TODO(dmaretskyi): Move meta into props.
41
- export const make = <S extends Type.Obj.Any>(
102
+ export const make = <S extends Schema.Schema.AnyNoContext>(
42
103
  schema: S,
43
104
  props: MakeProps<S>,
44
- meta?: EchoSchema.ObjectMeta,
45
- ): LiveObject.Live<Schema.Schema.Type<S>> => {
46
- assertArgument(
47
- EchoSchema.getTypeAnnotation(schema)?.kind === EchoSchema.EntityKind.Object,
48
- 'Expected an object schema',
49
- );
105
+ meta?: Partial<ObjectMeta>,
106
+ ): Obj<Schema.Schema.Type<S>> => {
107
+ assertArgument(getTypeAnnotation(schema)?.kind === Entity.Kind.Object, 'schema', 'Expected an object schema');
50
108
 
51
- if (props[EchoSchema.MetaId] != null) {
52
- meta = props[EchoSchema.MetaId] as any;
53
- delete props[EchoSchema.MetaId];
109
+ // Set default fields on meta on creation.
110
+ if (props[MetaId] != null) {
111
+ meta = { ...structuredClone(defaultMeta), ...props[MetaId] };
112
+ delete props[MetaId];
54
113
  }
55
114
 
56
- return live<Schema.Schema.Type<S>>(schema, props as any, meta);
115
+ // Filter undefined values.
116
+ const filterUndefined = Object.fromEntries(Object.entries(props).filter(([_, v]) => v !== undefined));
117
+
118
+ return makeObject<Schema.Schema.Type<S>>(schema, filterUndefined as any, {
119
+ ...defaultMeta,
120
+ ...meta,
121
+ });
57
122
  };
58
123
 
124
+ /**
125
+ * Determine if object is an ECHO object.
126
+ */
59
127
  export const isObject = (obj: unknown): obj is Any => {
60
- assumeType<EchoSchema.InternalObjectProps>(obj);
61
- return typeof obj === 'object' && obj !== null && obj[EchoSchema.EntityKindId] === EchoSchema.EntityKind.Object;
128
+ assumeType<InternalObjectProps>(obj);
129
+ return typeof obj === 'object' && obj !== null && obj[Entity.KindId] === Entity.Kind.Object;
130
+ };
131
+
132
+ //
133
+ // Snapshot
134
+ //
135
+
136
+ /**
137
+ * Returns an immutable snapshot of an object.
138
+ */
139
+ export const getSnapshot: <T extends Any>(obj: Obj<T>) => T = getSnapshot$;
140
+
141
+ export type CloneOptions = {
142
+ /**
143
+ * Retain the original object's ID.
144
+ * @default false
145
+ */
146
+ retainId?: boolean;
147
+ };
148
+
149
+ /**
150
+ * Clones an object or relation.
151
+ * This does not clone referenced objects, only the properties in the object.
152
+ * @returns A new object with the same schema and properties.
153
+ */
154
+ export const clone = <T extends Any>(obj: T, opts?: CloneOptions): T => {
155
+ const { id, ...data } = obj;
156
+ const schema = getSchema$(obj);
157
+ invariant(schema != null, 'Object should have a schema');
158
+ const props: any = deepMapValues(data, (value, recurse) => {
159
+ if (Ref.isRef(value)) {
160
+ return value;
161
+ }
162
+ return recurse(value);
163
+ });
164
+
165
+ if (opts?.retainId) {
166
+ props.id = id;
167
+ }
168
+ const meta = getMeta(obj);
169
+ props[MetaId] = deepMapValues(meta, (value, recurse) => {
170
+ if (Ref.isRef(value)) {
171
+ return value;
172
+ }
173
+ return recurse(value);
174
+ });
175
+
176
+ return make(schema as Type.Obj.Any, props);
62
177
  };
63
178
 
179
+ //
180
+ // Type
181
+ //
182
+
183
+ // TODO(burdon): To discuss: prefer over ObjectId or Key.ObjectId or Type.ID?
184
+ export const ID = ObjectId;
185
+ export type ID = ObjectId;
186
+
64
187
  /**
65
188
  * Test if object or relation is an instance of a schema.
66
189
  * @example
67
190
  * ```ts
68
191
  * const john = Obj.make(Person, { name: 'John' });
69
- * const johnIsPerson = Obj.instanceOf(Person)(john);
70
- *
71
192
  * const isPerson = Obj.instanceOf(Person);
72
- * if(isPerson(john)) {
193
+ * if (isPerson(john)) {
73
194
  * // john is Person
74
195
  * }
75
196
  * ```
76
197
  */
77
198
  export const instanceOf: {
78
- <S extends Type.Relation.Any | Type.Obj.Any>(schema: S): (value: unknown) => value is Schema.Schema.Type<S>;
79
- <S extends Type.Relation.Any | Type.Obj.Any>(schema: S, value: unknown): value is Schema.Schema.Type<S>;
80
- } = ((
81
- ...args: [schema: Type.Relation.Any | Type.Obj.Any, value: unknown] | [schema: Type.Relation.Any | Type.Obj.Any]
82
- ) => {
199
+ <S extends Type.Entity.Any>(schema: S): (value: unknown) => value is Schema.Schema.Type<S>;
200
+ <S extends Type.Entity.Any>(schema: S, value: unknown): value is Schema.Schema.Type<S>;
201
+ } = ((...args: [schema: Type.Entity.Any, value: unknown] | [schema: Type.Entity.Any]) => {
83
202
  if (args.length === 1) {
84
- return (obj: unknown) => EchoSchema.isInstanceOf(args[0], obj);
203
+ return (entity: unknown) => isInstanceOf(args[0], entity);
85
204
  }
86
205
 
87
- return EchoSchema.isInstanceOf(args[0], args[1]);
206
+ return isInstanceOf(args[0], args[1]);
88
207
  }) as any;
89
208
 
90
- export const getSchema = EchoSchema.getSchema;
91
-
92
209
  // TODO(dmaretskyi): Allow returning undefined.
93
- export const getDXN = (obj: Any): DXN => {
94
- assertArgument(!Schema.isSchema(obj), 'Object should not be a schema.');
95
- const dxn = EchoSchema.getObjectDXN(obj);
210
+ export const getDXN = (entity: Entity.Unknown): DXN => {
211
+ assertArgument(!Schema.isSchema(entity), 'obj', 'Object should not be a schema.');
212
+ const dxn = getObjectDXN(entity);
96
213
  invariant(dxn != null, 'Invalid object.');
97
214
  return dxn;
98
215
  };
99
216
 
100
217
  /**
101
218
  * @returns The DXN of the object's type.
102
- * @example dxn:example.com/type/Contact:1.0.0
219
+ * @example dxn:example.com/type/Person:1.0.0
220
+ */
221
+ // TODO(burdon): Must define and return type for expando.
222
+ export const getTypeDXN = getTypeDXN$;
223
+
224
+ /**
225
+ * Get the schema of the object.
103
226
  */
104
- // TODO(burdon): Expando does not have a type.
105
- export const getTypeDXN = EchoSchema.getType;
227
+ export const getSchema = getSchema$;
106
228
 
107
229
  /**
108
230
  * @returns The typename of the object's type.
109
- * @example `example.com/type/Contact`
231
+ * @example `example.com/type/Person`
110
232
  */
111
- export const getTypename = (obj: Any): string | undefined => {
112
- const schema = getSchema(obj);
233
+ export const getTypename = (entity: Entity.Unknown): string | undefined => {
234
+ const schema = getSchema$(entity);
113
235
  if (schema == null) {
114
236
  // Try to extract typename from DXN.
115
- return EchoSchema.getType(obj)?.asTypeDXN()?.type;
237
+ return getTypeDXN$(entity)?.asTypeDXN()?.type;
116
238
  }
117
239
 
118
- return EchoSchema.getSchemaTypename(schema);
240
+ return getSchemaTypename(schema);
119
241
  };
120
242
 
243
+ //
244
+ // Meta
245
+ //
246
+
247
+ export const Meta: unique symbol = MetaId as any;
248
+
249
+ // TODO(burdon): Narrow type.
121
250
  // TODO(dmaretskyi): Allow returning undefined.
122
- export const getMeta = (obj: Any): EchoSchema.ObjectMeta => {
123
- const meta = EchoSchema.getMeta(obj);
251
+ export const getMeta = (entity: AnyProperties): ObjectMeta => {
252
+ const meta = getMeta$(entity);
124
253
  invariant(meta != null, 'Invalid object.');
125
254
  return meta;
126
255
  };
127
256
 
257
+ /**
258
+ * @returns Foreign keys for the object from the specified source.
259
+ */
260
+ export const getKeys: {
261
+ (entity: Entity.Unknown, source: string): ForeignKey[];
262
+ (source: string): (entity: Entity.Unknown) => ForeignKey[];
263
+ } = Function.dual(2, (entity: Entity.Unknown, source?: string): ForeignKey[] => {
264
+ const meta = getMeta(entity);
265
+ invariant(meta != null, 'Invalid object.');
266
+ return meta.keys.filter((key) => key.source === source);
267
+ });
268
+
269
+ /**
270
+ * Delete all keys from the object for the specified source.
271
+ * @param entity
272
+ * @param source
273
+ */
274
+ export const deleteKeys = (entity: Entity.Unknown, source: string) => {
275
+ const meta = getMeta(entity);
276
+ for (let i = 0; i < meta.keys.length; i++) {
277
+ if (meta.keys[i].source === source) {
278
+ meta.keys.splice(i, 1);
279
+ i--;
280
+ }
281
+ }
282
+ };
283
+
284
+ export const addTag = (entity: Entity.Unknown, tag: string) => {
285
+ const meta = getMeta(entity);
286
+ meta.tags ??= [];
287
+ meta.tags.push(tag);
288
+ };
289
+
290
+ export const removeTag = (entity: Entity.Unknown, tag: string) => {
291
+ const meta = getMeta(entity);
292
+ if (!meta.tags) {
293
+ return;
294
+ }
295
+ for (let i = 0; i < meta.tags.length; i++) {
296
+ if (meta.tags[i] === tag) {
297
+ meta.tags.splice(i, 1);
298
+ i--;
299
+ }
300
+ }
301
+ };
302
+
128
303
  // TODO(dmaretskyi): Default to `false`.
129
- export const isDeleted = (obj: Any): boolean => {
130
- const deleted = EchoSchema.isDeleted(obj);
304
+ export const isDeleted = (entity: Entity.Unknown): boolean => {
305
+ const deleted = isDeleted$(entity);
131
306
  invariant(typeof deleted === 'boolean', 'Invalid object.');
132
307
  return deleted;
133
308
  };
134
309
 
135
- // TODO(burdon): Rename "label"
136
- export const getLabel = (obj: Any): string | undefined => {
137
- const schema = getSchema(obj);
310
+ //
311
+ // Annotations
312
+ //
313
+
314
+ export const getLabel = (entity: Entity.Unknown): string | undefined => {
315
+ const schema = getSchema$(entity);
138
316
  if (schema != null) {
139
- return EchoSchema.getLabel(schema, obj);
317
+ return getLabel$(schema, entity);
140
318
  }
141
319
  };
142
320
 
321
+ export const setLabel = (entity: Entity.Unknown, label: string) => {
322
+ const schema = getSchema$(entity);
323
+ if (schema != null) {
324
+ setLabel$(schema, entity, label);
325
+ }
326
+ };
327
+
328
+ export const getDescription = (entity: Entity.Unknown): string | undefined => {
329
+ const schema = getSchema$(entity);
330
+ if (schema != null) {
331
+ return getDescription$(schema, entity);
332
+ }
333
+ };
334
+
335
+ export const setDescription = (entity: Entity.Unknown, description: string) => {
336
+ const schema = getSchema$(entity);
337
+ if (schema != null) {
338
+ setDescription$(schema, entity, description);
339
+ }
340
+ };
341
+
342
+ //
343
+ // JSON
344
+ //
345
+
143
346
  /**
144
347
  * JSON representation of an object.
145
348
  */
146
- export type JSON = EchoSchema.ObjectJSON;
349
+ export type JSON = ObjectJSON;
147
350
 
148
351
  /**
149
352
  * Converts object to its JSON representation.
150
353
  *
151
354
  * The same algorithm is used when calling the standard `JSON.stringify(obj)` function.
152
355
  */
153
- export const toJSON = (obj: Any | Relation.Any): JSON => EchoSchema.objectToJSON(obj);
356
+ export const toJSON = (entity: Entity.Unknown): JSON => objectToJSON(entity);
154
357
 
155
358
  /**
156
359
  * Creates an object from its json representation, performing schema validation.
157
360
  * References and schemas will be resolvable if the `refResolver` is provided.
158
361
  *
159
- * The function need to be async to support resolving the schema as well as the relation endpoints.
362
+ * The function must be async to support resolving the schema as well as the relation endpoints.
363
+ *
364
+ * @param options.refResolver - Resolver for references. Produces hydrated references that can be resolved.
365
+ * @param options.dxn - Override object DXN. Changes the result of `Obj.getDXN`.
366
+ */
367
+ export const fromJSON: (json: unknown, options?: { refResolver?: Ref.Resolver; dxn?: DXN }) => Promise<Any> =
368
+ objectFromJSON as any;
369
+
370
+ //
371
+ // Sorting
372
+ //
373
+
374
+ const compare = (a?: string, b?: string) => {
375
+ if (a == null) {
376
+ return b == null ? 0 : 1;
377
+ }
378
+
379
+ if (b == null) {
380
+ return -1;
381
+ }
382
+
383
+ return a.localeCompare(b);
384
+ };
385
+
386
+ export type Comparator = (a: Entity.Unknown, b: Entity.Unknown) => number;
387
+
388
+ export const sortByLabel: Comparator = (a: Entity.Unknown, b: Entity.Unknown) => compare(getLabel(a), getLabel(b));
389
+ export const sortByTypename: Comparator = (a: Entity.Unknown, b: Entity.Unknown) =>
390
+ compare(getTypename(a), getTypename(b));
391
+ export const sort = (...comparators: Comparator[]): Comparator => {
392
+ return (a: Entity.Unknown, b: Entity.Unknown) => {
393
+ for (const comparator of comparators) {
394
+ const result = comparator(a, b);
395
+ if (result !== 0) {
396
+ return result;
397
+ }
398
+ }
399
+
400
+ return 0;
401
+ };
402
+ };
403
+
404
+ //
405
+ // Version
406
+ //
407
+
408
+ /**
409
+ * Unique symbol for version type identification.
410
+ */
411
+ export { VersionTypeId };
412
+
413
+ /**
414
+ * Represent object version.
415
+ * May be backed by Automerge.
416
+ * Objects with no history are not versioned.
160
417
  */
161
- export const fromJSON: (json: unknown, options?: { refResolver?: Ref.Resolver }) => Promise<Any> =
162
- EchoSchema.objectFromJSON as any;
418
+ export interface Version {
419
+ [VersionTypeId]: {};
420
+
421
+ /**
422
+ * Whether the object is versioned.
423
+ */
424
+ versioned: boolean;
425
+
426
+ /**
427
+ * Automerge heads.
428
+ */
429
+ automergeHeads?: string[];
430
+ }
431
+
432
+ const unversioned: Version = {
433
+ [VersionTypeId]: {},
434
+ versioned: false,
435
+ };
436
+
437
+ /**
438
+ * Checks that `obj` is a version object.
439
+ */
440
+ export const isVersion = (entity: unknown): entity is Version => {
441
+ return entity != null && typeof entity === 'object' && VersionTypeId in entity;
442
+ };
443
+
444
+ /**
445
+ * Returns the version of the object.
446
+ */
447
+ export const version = (entity: Entity.Unknown): Version => {
448
+ const version = (entity as any)[ObjectVersionId];
449
+ if (version === undefined) {
450
+ return unversioned;
451
+ }
452
+
453
+ return version;
454
+ };
455
+
456
+ /**
457
+ * Checks that `version` is a valid version object.
458
+ */
459
+ export const versionValid = (version: Version): boolean => {
460
+ assertArgument(isVersion(version), 'version', 'Invalid version object');
461
+ return !!version.versioned;
462
+ };
463
+
464
+ export type VersionCompareResult = 'unversioned' | 'equal' | 'different';
465
+
466
+ /**
467
+ * Compares two versions.
468
+ * @param version1
469
+ * @param version2
470
+ * @returns 'unversioned' if either object is unversioned, 'equal' if the versions are equal, 'different' if the versions are different.
471
+ */
472
+ export const compareVersions = (version1: Version, version2: Version): VersionCompareResult => {
473
+ assertArgument(isVersion(version1), 'version1', 'Invalid version object');
474
+ assertArgument(isVersion(version2), 'version2', 'Invalid version object');
475
+
476
+ if (!versionValid(version1) || !versionValid(version2)) {
477
+ return 'unversioned';
478
+ }
479
+
480
+ if (version1.automergeHeads?.length !== version2.automergeHeads?.length) {
481
+ return 'different';
482
+ }
483
+ if (version1.automergeHeads?.some((head) => !version2.automergeHeads?.includes(head))) {
484
+ return 'different';
485
+ }
486
+
487
+ return 'equal';
488
+ };
489
+
490
+ export const encodeVersion = (version: Version): string => {
491
+ return JSON.stringify(version);
492
+ };
493
+
494
+ export const decodeVersion = (version: string): Version => {
495
+ const parsed = JSON.parse(version);
496
+ parsed[VersionTypeId] = {};
497
+ return parsed;
498
+ };
package/src/Ref.ts CHANGED
@@ -2,28 +2,27 @@
2
2
  // Copyright 2025 DXOS.org
3
3
  //
4
4
 
5
- import * as EchoSchema from '@dxos/echo-schema';
6
-
5
+ import { Ref as Ref$, RefArray, type RefResolver } from './internal';
7
6
  import type * as Obj from './Obj';
8
7
 
9
- export type Ref<T> = EchoSchema.Ref<T>;
10
- export type Any = EchoSchema.Ref<Obj.Any>;
8
+ export type Ref<T> = Ref$<T>;
9
+ export type Any = Ref$<Obj.Any>;
11
10
 
12
- export const Array = EchoSchema.RefArray;
11
+ export const Array = RefArray;
13
12
 
14
13
  /**
15
14
  * Extract reference target.
16
15
  */
17
- export type Target<R extends Any> = R extends EchoSchema.Ref<infer T> ? T : never;
16
+ export type Target<R extends Any> = R extends Ref$<infer T> ? T : never;
18
17
 
19
18
  /**
20
19
  * Reference resolver.
21
20
  */
22
- export type Resolver = EchoSchema.RefResolver;
21
+ export type Resolver = RefResolver;
23
22
 
24
- export const isRef: (value: unknown) => value is Any = EchoSchema.Ref.isRef;
23
+ export const isRef: (value: unknown) => value is Any = Ref$.isRef;
25
24
 
26
- export const make = EchoSchema.Ref.make;
25
+ export const make = Ref$.make;
27
26
 
28
27
  // TODO(dmaretskyi): Consider just allowing `make` to accept DXN.
29
- export const fromDXN = EchoSchema.Ref.fromDXN;
28
+ export const fromDXN = Ref$.fromDXN;