@dxos/echo 0.8.4-main.8360d9e660 → 0.8.4-main.8baae0fced

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 (292) hide show
  1. package/LICENSE +102 -5
  2. package/README.md +3 -3
  3. package/dist/lib/neutral/Annotation.mjs +5 -3
  4. package/dist/lib/neutral/Database.mjs +8 -4
  5. package/dist/lib/neutral/Entity.mjs +16 -14
  6. package/dist/lib/neutral/Err.mjs +1 -1
  7. package/dist/lib/neutral/Extension.mjs +18 -0
  8. package/dist/lib/neutral/Extension.mjs.map +7 -0
  9. package/dist/lib/neutral/Feed.mjs +23 -18
  10. package/dist/lib/neutral/Filter.mjs +23 -13
  11. package/dist/lib/neutral/Format.mjs +3 -3
  12. package/dist/lib/neutral/JsonSchema.mjs +8 -8
  13. package/dist/lib/neutral/Key.mjs +1 -1
  14. package/dist/lib/neutral/Migration.mjs +17 -0
  15. package/dist/lib/neutral/Migration.mjs.map +7 -0
  16. package/dist/lib/neutral/Obj.mjs +16 -13
  17. package/dist/lib/neutral/Order.mjs +1 -1
  18. package/dist/lib/neutral/Query.mjs +19 -17
  19. package/dist/lib/neutral/QueryResult.mjs +1 -1
  20. package/dist/lib/neutral/Ref.mjs +9 -7
  21. package/dist/lib/neutral/Relation.mjs +15 -14
  22. package/dist/lib/neutral/SchemaRegistry.mjs +1 -1
  23. package/dist/lib/neutral/Tag.mjs +14 -13
  24. package/dist/lib/neutral/Type.mjs +10 -10
  25. package/dist/lib/neutral/{chunk-FZO7LQO7.mjs → chunk-44HT3MEC.mjs} +2 -2
  26. package/dist/lib/neutral/{chunk-ROAGDPV7.mjs → chunk-4A2GS5LQ.mjs} +12 -8
  27. package/dist/lib/neutral/chunk-4A2GS5LQ.mjs.map +7 -0
  28. package/dist/lib/neutral/{chunk-OENWMTE6.mjs → chunk-5SL5LDLD.mjs} +4 -2
  29. package/dist/lib/neutral/chunk-5SL5LDLD.mjs.map +7 -0
  30. package/dist/lib/neutral/{chunk-ANHVGJI4.mjs → chunk-7RVZT53K.mjs} +1 -1
  31. package/dist/lib/neutral/{chunk-YWXWXIE5.mjs → chunk-APHSOTIX.mjs} +2 -2
  32. package/dist/lib/neutral/{chunk-YWXWXIE5.mjs.map → chunk-APHSOTIX.mjs.map} +2 -2
  33. package/dist/lib/neutral/{chunk-6DNYDXCV.mjs → chunk-B2P7IVG3.mjs} +61 -14
  34. package/dist/lib/neutral/chunk-B2P7IVG3.mjs.map +7 -0
  35. package/dist/lib/neutral/{chunk-BNCCGLJN.mjs → chunk-BICZKPQG.mjs} +1 -1
  36. package/dist/lib/neutral/{chunk-43Y5DOS6.mjs → chunk-BMB7IHGB.mjs} +16 -66
  37. package/dist/lib/neutral/chunk-BMB7IHGB.mjs.map +7 -0
  38. package/dist/lib/neutral/{chunk-6K2MVI2O.mjs → chunk-BUBEC474.mjs} +4 -4
  39. package/dist/lib/neutral/{chunk-YMNSMKKW.mjs → chunk-C4PSESGN.mjs} +6 -22
  40. package/dist/lib/neutral/{chunk-YMNSMKKW.mjs.map → chunk-C4PSESGN.mjs.map} +3 -3
  41. package/dist/lib/neutral/chunk-FIWO2FZK.mjs +36 -0
  42. package/dist/lib/neutral/chunk-FIWO2FZK.mjs.map +7 -0
  43. package/dist/lib/neutral/chunk-GWFFC34K.mjs +50 -0
  44. package/dist/lib/neutral/chunk-GWFFC34K.mjs.map +7 -0
  45. package/dist/lib/neutral/{chunk-NXMFBIT5.mjs → chunk-HKETO4L4.mjs} +72 -6
  46. package/dist/lib/neutral/chunk-HKETO4L4.mjs.map +7 -0
  47. package/dist/lib/neutral/{chunk-NEGC54NE.mjs → chunk-I2DARWPX.mjs} +17 -19
  48. package/dist/lib/neutral/chunk-I2DARWPX.mjs.map +7 -0
  49. package/dist/lib/neutral/{chunk-7VNVH63N.mjs → chunk-IVSI7QO6.mjs} +50 -20
  50. package/dist/lib/neutral/chunk-IVSI7QO6.mjs.map +7 -0
  51. package/dist/lib/neutral/{chunk-BOZZPUXE.mjs → chunk-MLS7U7AT.mjs} +12 -65
  52. package/dist/lib/neutral/chunk-MLS7U7AT.mjs.map +7 -0
  53. package/dist/lib/neutral/{chunk-WYOKA6AE.mjs → chunk-N4B7FHQT.mjs} +2 -2
  54. package/dist/lib/neutral/{chunk-WYOKA6AE.mjs.map → chunk-N4B7FHQT.mjs.map} +1 -1
  55. package/dist/lib/neutral/{chunk-FXEG7EOK.mjs → chunk-N7VOEPSV.mjs} +6 -3
  56. package/dist/lib/neutral/{chunk-FXEG7EOK.mjs.map → chunk-N7VOEPSV.mjs.map} +3 -3
  57. package/dist/lib/neutral/{chunk-UTBRYVQC.mjs → chunk-QRZ2I3ZM.mjs} +2 -2
  58. package/dist/lib/neutral/{chunk-SEMVAGBM.mjs → chunk-TNBK56IN.mjs} +19 -24
  59. package/dist/lib/neutral/chunk-TNBK56IN.mjs.map +7 -0
  60. package/dist/lib/neutral/{chunk-B5OXLWZL.mjs → chunk-TRPZU2HV.mjs} +2 -2
  61. package/dist/lib/neutral/{chunk-UI6MWK5W.mjs → chunk-TTCSATUD.mjs} +1 -1
  62. package/dist/lib/neutral/{chunk-OMUPQMLR.mjs → chunk-V72DY6LU.mjs} +1 -1
  63. package/dist/lib/neutral/{chunk-6GPU7XC3.mjs → chunk-VW42HESL.mjs} +54 -17
  64. package/dist/lib/neutral/chunk-VW42HESL.mjs.map +7 -0
  65. package/dist/lib/neutral/{chunk-4JRI2ZJI.mjs → chunk-X3356HPV.mjs} +120 -12
  66. package/dist/lib/neutral/chunk-X3356HPV.mjs.map +7 -0
  67. package/dist/lib/neutral/{chunk-W47JKR3X.mjs → chunk-XEXM5HWQ.mjs} +18 -46
  68. package/dist/lib/neutral/chunk-XEXM5HWQ.mjs.map +7 -0
  69. package/dist/lib/neutral/{chunk-C4JZK4J7.mjs → chunk-Z5GKP74O.mjs} +231 -479
  70. package/dist/lib/neutral/chunk-Z5GKP74O.mjs.map +7 -0
  71. package/dist/lib/neutral/{chunk-UBEZSGXY.mjs → chunk-ZISMEVKD.mjs} +1 -1
  72. package/dist/lib/neutral/{chunk-UBEZSGXY.mjs.map → chunk-ZISMEVKD.mjs.map} +2 -2
  73. package/dist/lib/neutral/index.mjs +38 -28
  74. package/dist/lib/neutral/internal/index.mjs +15 -9
  75. package/dist/lib/neutral/meta.json +1 -1
  76. package/dist/lib/neutral/testing/index.mjs +185 -129
  77. package/dist/lib/neutral/testing/index.mjs.map +3 -3
  78. package/dist/types/src/Annotation.d.ts +1 -1
  79. package/dist/types/src/Annotation.d.ts.map +1 -1
  80. package/dist/types/src/Collection.d.ts.map +1 -1
  81. package/dist/types/src/Database.d.ts +14 -2
  82. package/dist/types/src/Database.d.ts.map +1 -1
  83. package/dist/types/src/Dataset.d.ts +2 -1
  84. package/dist/types/src/Dataset.d.ts.map +1 -1
  85. package/dist/types/src/Entity.d.ts +17 -11
  86. package/dist/types/src/Entity.d.ts.map +1 -1
  87. package/dist/types/src/Err.d.ts +18 -18
  88. package/dist/types/src/Err.d.ts.map +1 -1
  89. package/dist/types/src/Extension.d.ts +80 -0
  90. package/dist/types/src/Extension.d.ts.map +1 -0
  91. package/dist/types/src/Extension.test.d.ts +2 -0
  92. package/dist/types/src/Extension.test.d.ts.map +1 -0
  93. package/dist/types/src/Feed.d.ts +62 -21
  94. package/dist/types/src/Feed.d.ts.map +1 -1
  95. package/dist/types/src/Filter.d.ts +54 -4
  96. package/dist/types/src/Filter.d.ts.map +1 -1
  97. package/dist/types/src/Filter.test.d.ts +2 -0
  98. package/dist/types/src/Filter.test.d.ts.map +1 -0
  99. package/dist/types/src/Hypergraph.d.ts +3 -3
  100. package/dist/types/src/Hypergraph.d.ts.map +1 -1
  101. package/dist/types/src/Json.d.ts +33 -0
  102. package/dist/types/src/Json.d.ts.map +1 -0
  103. package/dist/types/src/Json.test.d.ts +2 -0
  104. package/dist/types/src/Json.test.d.ts.map +1 -0
  105. package/dist/types/src/JsonSchema.d.ts +1 -1
  106. package/dist/types/src/Migration.d.ts +69 -0
  107. package/dist/types/src/Migration.d.ts.map +1 -0
  108. package/dist/types/src/Obj.d.ts +42 -28
  109. package/dist/types/src/Obj.d.ts.map +1 -1
  110. package/dist/types/src/Order.d.ts.map +1 -1
  111. package/dist/types/src/Query.d.ts +13 -2
  112. package/dist/types/src/Query.d.ts.map +1 -1
  113. package/dist/types/src/Ref.d.ts +1 -0
  114. package/dist/types/src/Ref.d.ts.map +1 -1
  115. package/dist/types/src/Relation.d.ts +17 -18
  116. package/dist/types/src/Relation.d.ts.map +1 -1
  117. package/dist/types/src/Tag.d.ts +2 -2
  118. package/dist/types/src/Tag.d.ts.map +1 -1
  119. package/dist/types/src/Type.d.ts +3 -3
  120. package/dist/types/src/Type.d.ts.map +1 -1
  121. package/dist/types/src/View.d.ts +1 -1
  122. package/dist/types/src/View.d.ts.map +1 -1
  123. package/dist/types/src/exemplars.test.d.ts +2 -0
  124. package/dist/types/src/exemplars.test.d.ts.map +1 -0
  125. package/dist/types/src/index.d.ts +3 -0
  126. package/dist/types/src/index.d.ts.map +1 -1
  127. package/dist/types/src/internal/Annotation/annotations.d.ts +12 -2
  128. package/dist/types/src/internal/Annotation/annotations.d.ts.map +1 -1
  129. package/dist/types/src/internal/Annotation/sorting.d.ts.map +1 -1
  130. package/dist/types/src/internal/Annotation/util.d.ts +1 -1
  131. package/dist/types/src/internal/Annotation/util.d.ts.map +1 -1
  132. package/dist/types/src/internal/Entity/api.d.ts.map +1 -1
  133. package/dist/types/src/internal/Entity/model.d.ts +2 -0
  134. package/dist/types/src/internal/Entity/model.d.ts.map +1 -1
  135. package/dist/types/src/internal/Entity/object.d.ts.map +1 -1
  136. package/dist/types/src/internal/Entity/relation.d.ts.map +1 -1
  137. package/dist/types/src/internal/Entity/version.d.ts.map +1 -1
  138. package/dist/types/src/internal/Format/date.d.ts.map +1 -1
  139. package/dist/types/src/internal/Format/format.d.ts.map +1 -1
  140. package/dist/types/src/internal/Format/number.d.ts.map +1 -1
  141. package/dist/types/src/internal/Format/object.d.ts.map +1 -1
  142. package/dist/types/src/internal/Format/types.d.ts.map +1 -1
  143. package/dist/types/src/internal/JsonSchema/json-schema-normalize.d.ts.map +1 -1
  144. package/dist/types/src/internal/JsonSchema/json-schema-type.d.ts +28 -28
  145. package/dist/types/src/internal/JsonSchema/json-schema-type.d.ts.map +1 -1
  146. package/dist/types/src/internal/JsonSchema/json-schema.d.ts +1 -1
  147. package/dist/types/src/internal/JsonSchema/json-schema.d.ts.map +1 -1
  148. package/dist/types/src/internal/Obj/clone.d.ts.map +1 -1
  149. package/dist/types/src/internal/Obj/common.d.ts.map +1 -1
  150. package/dist/types/src/internal/Obj/create-object.d.ts.map +1 -1
  151. package/dist/types/src/internal/Obj/deleted.d.ts.map +1 -1
  152. package/dist/types/src/internal/Obj/ids.d.ts +1 -1
  153. package/dist/types/src/internal/Obj/ids.d.ts.map +1 -1
  154. package/dist/types/src/internal/Obj/json-serializer.d.ts +4 -3
  155. package/dist/types/src/internal/Obj/json-serializer.d.ts.map +1 -1
  156. package/dist/types/src/internal/Obj/set-value.d.ts +1 -1
  157. package/dist/types/src/internal/Obj/set-value.d.ts.map +1 -1
  158. package/dist/types/src/internal/Obj/snapshot.d.ts.map +1 -1
  159. package/dist/types/src/internal/Query.d.ts +10 -0
  160. package/dist/types/src/internal/Query.d.ts.map +1 -0
  161. package/dist/types/src/internal/Ref/ref-array.d.ts.map +1 -1
  162. package/dist/types/src/internal/Ref/ref.d.ts +14 -1
  163. package/dist/types/src/internal/Ref/ref.d.ts.map +1 -1
  164. package/dist/types/src/internal/Type/compose.d.ts.map +1 -1
  165. package/dist/types/src/internal/Type/echo-schema.d.ts +2 -2
  166. package/dist/types/src/internal/Type/echo-schema.d.ts.map +1 -1
  167. package/dist/types/src/internal/Type/manipulation.d.ts.map +1 -1
  168. package/dist/types/src/internal/common/api/meta.d.ts +3 -3
  169. package/dist/types/src/internal/common/api/meta.d.ts.map +1 -1
  170. package/dist/types/src/internal/common/proxy/change-context.d.ts +1 -1
  171. package/dist/types/src/internal/common/proxy/change-context.d.ts.map +1 -1
  172. package/dist/types/src/internal/common/proxy/define-hidden-property.d.ts.map +1 -1
  173. package/dist/types/src/internal/common/proxy/errors.d.ts +1 -1
  174. package/dist/types/src/internal/common/proxy/errors.d.ts.map +1 -1
  175. package/dist/types/src/internal/common/proxy/event-batch.d.ts.map +1 -1
  176. package/dist/types/src/internal/common/proxy/json-serializer.d.ts.map +1 -1
  177. package/dist/types/src/internal/common/proxy/make-object.d.ts.map +1 -1
  178. package/dist/types/src/internal/common/proxy/ownership.d.ts.map +1 -1
  179. package/dist/types/src/internal/common/proxy/proxy-utils.d.ts.map +1 -1
  180. package/dist/types/src/internal/common/proxy/reactive-array.d.ts +1 -1
  181. package/dist/types/src/internal/common/proxy/reactive.d.ts +1 -1
  182. package/dist/types/src/internal/common/proxy/reactive.d.ts.map +1 -1
  183. package/dist/types/src/internal/common/proxy/reactive.test.d.ts +2 -0
  184. package/dist/types/src/internal/common/proxy/reactive.test.d.ts.map +1 -0
  185. package/dist/types/src/internal/common/proxy/schema-validator.d.ts.map +1 -1
  186. package/dist/types/src/internal/common/proxy/typed-handler.d.ts.map +1 -1
  187. package/dist/types/src/internal/common/types/base.d.ts.map +1 -1
  188. package/dist/types/src/internal/common/types/entity.d.ts +4 -4
  189. package/dist/types/src/internal/common/types/entity.d.ts.map +1 -1
  190. package/dist/types/src/internal/common/types/meta.d.ts +10 -0
  191. package/dist/types/src/internal/common/types/meta.d.ts.map +1 -1
  192. package/dist/types/src/internal/common/types/version.d.ts +1 -1
  193. package/dist/types/src/internal/index.d.ts +1 -0
  194. package/dist/types/src/internal/index.d.ts.map +1 -1
  195. package/dist/types/src/testing/test-data.d.ts +8 -8
  196. package/dist/types/src/testing/test-data.d.ts.map +1 -1
  197. package/dist/types/src/testing/test-schema.d.ts +53 -53
  198. package/dist/types/src/testing/test-schema.d.ts.map +1 -1
  199. package/dist/types/src/testing/util.d.ts.map +1 -1
  200. package/dist/types/tsconfig.tsbuildinfo +1 -1
  201. package/package.json +25 -15
  202. package/src/Annotation.ts +1 -0
  203. package/src/Collection.ts +2 -2
  204. package/src/Database.ts +50 -15
  205. package/src/Entity.ts +18 -12
  206. package/src/Extension.test.ts +235 -0
  207. package/src/Extension.ts +122 -0
  208. package/src/Feed.ts +107 -34
  209. package/src/Filter.test.ts +90 -0
  210. package/src/Filter.ts +97 -3
  211. package/src/Hypergraph.ts +3 -3
  212. package/src/Json.test.ts +175 -0
  213. package/src/Json.ts +102 -0
  214. package/src/Migration.ts +106 -0
  215. package/src/Obj.test.ts +105 -13
  216. package/src/Obj.ts +154 -33
  217. package/src/Query.test.ts +199 -9
  218. package/src/Query.ts +58 -8
  219. package/src/Ref.ts +2 -0
  220. package/src/Relation.ts +24 -20
  221. package/src/Type.ts +1 -1
  222. package/src/View.ts +1 -1
  223. package/src/exemplars.test.ts +21 -0
  224. package/src/index.ts +4 -0
  225. package/src/internal/Annotation/annotations.test.ts +51 -2
  226. package/src/internal/Annotation/annotations.ts +33 -14
  227. package/src/internal/Annotation/sorting.ts +0 -1
  228. package/src/internal/Entity/api.ts +0 -1
  229. package/src/internal/Entity/model.ts +2 -0
  230. package/src/internal/Entity/object.ts +0 -1
  231. package/src/internal/Entity/version.ts +0 -1
  232. package/src/internal/Format/date.test.ts +0 -1
  233. package/src/internal/Format/format.test.ts +0 -1
  234. package/src/internal/JsonSchema/json-schema-type.ts +1 -1
  235. package/src/internal/JsonSchema/json-schema.test.ts +1 -2
  236. package/src/internal/JsonSchema/json-schema.ts +1 -2
  237. package/src/internal/Obj/clone.ts +1 -1
  238. package/src/internal/Obj/create-object.test.ts +2 -4
  239. package/src/internal/Obj/create-object.ts +2 -3
  240. package/src/internal/Obj/deleted.ts +1 -1
  241. package/src/internal/Obj/ids.ts +1 -1
  242. package/src/internal/Obj/json-serializer.test.ts +49 -5
  243. package/src/internal/Obj/json-serializer.ts +47 -25
  244. package/src/internal/Obj/set-value.test.ts +24 -24
  245. package/src/internal/Obj/set-value.ts +1 -1
  246. package/src/internal/Query.ts +156 -0
  247. package/src/internal/Ref/ref-array.ts +0 -1
  248. package/src/internal/Ref/ref.test.ts +0 -1
  249. package/src/internal/Ref/ref.ts +18 -1
  250. package/src/internal/Type/compose.test.ts +0 -1
  251. package/src/internal/Type/echo-schema.ts +3 -4
  252. package/src/internal/common/README.md +1 -1
  253. package/src/internal/common/api/meta.ts +3 -3
  254. package/src/internal/common/proxy/change-context.ts +1 -1
  255. package/src/internal/common/proxy/change.test.ts +94 -94
  256. package/src/internal/common/proxy/errors.ts +2 -2
  257. package/src/internal/common/proxy/handler.test.ts +0 -2
  258. package/src/internal/common/proxy/json-serializer.ts +4 -1
  259. package/src/internal/common/proxy/make-object.ts +0 -1
  260. package/src/internal/common/proxy/ownership.ts +0 -1
  261. package/src/internal/common/proxy/reactive-array.ts +1 -1
  262. package/src/internal/common/proxy/reactive.test.ts +54 -0
  263. package/src/internal/common/proxy/reactive.ts +11 -3
  264. package/src/internal/common/proxy/typed-handler.test.ts +0 -1
  265. package/src/internal/common/proxy/typed-handler.ts +8 -10
  266. package/src/internal/common/proxy/typed-object.test.ts +2 -3
  267. package/src/internal/common/types/entity.ts +1 -1
  268. package/src/internal/common/types/meta.ts +12 -1
  269. package/src/internal/index.ts +1 -0
  270. package/src/testing/api.test.ts +0 -1
  271. package/src/testing/test-data.ts +157 -98
  272. package/dist/lib/neutral/chunk-43Y5DOS6.mjs.map +0 -7
  273. package/dist/lib/neutral/chunk-4JRI2ZJI.mjs.map +0 -7
  274. package/dist/lib/neutral/chunk-6DNYDXCV.mjs.map +0 -7
  275. package/dist/lib/neutral/chunk-6GPU7XC3.mjs.map +0 -7
  276. package/dist/lib/neutral/chunk-7VNVH63N.mjs.map +0 -7
  277. package/dist/lib/neutral/chunk-BOZZPUXE.mjs.map +0 -7
  278. package/dist/lib/neutral/chunk-C4JZK4J7.mjs.map +0 -7
  279. package/dist/lib/neutral/chunk-NEGC54NE.mjs.map +0 -7
  280. package/dist/lib/neutral/chunk-NXMFBIT5.mjs.map +0 -7
  281. package/dist/lib/neutral/chunk-OENWMTE6.mjs.map +0 -7
  282. package/dist/lib/neutral/chunk-ROAGDPV7.mjs.map +0 -7
  283. package/dist/lib/neutral/chunk-SEMVAGBM.mjs.map +0 -7
  284. package/dist/lib/neutral/chunk-W47JKR3X.mjs.map +0 -7
  285. /package/dist/lib/neutral/{chunk-FZO7LQO7.mjs.map → chunk-44HT3MEC.mjs.map} +0 -0
  286. /package/dist/lib/neutral/{chunk-ANHVGJI4.mjs.map → chunk-7RVZT53K.mjs.map} +0 -0
  287. /package/dist/lib/neutral/{chunk-BNCCGLJN.mjs.map → chunk-BICZKPQG.mjs.map} +0 -0
  288. /package/dist/lib/neutral/{chunk-6K2MVI2O.mjs.map → chunk-BUBEC474.mjs.map} +0 -0
  289. /package/dist/lib/neutral/{chunk-UTBRYVQC.mjs.map → chunk-QRZ2I3ZM.mjs.map} +0 -0
  290. /package/dist/lib/neutral/{chunk-B5OXLWZL.mjs.map → chunk-TRPZU2HV.mjs.map} +0 -0
  291. /package/dist/lib/neutral/{chunk-UI6MWK5W.mjs.map → chunk-TTCSATUD.mjs.map} +0 -0
  292. /package/dist/lib/neutral/{chunk-OMUPQMLR.mjs.map → chunk-V72DY6LU.mjs.map} +0 -0
@@ -2,9 +2,8 @@
2
2
  // Copyright 2025 DXOS.org
3
3
  //
4
4
 
5
- import { inspect } from 'util';
6
-
7
5
  import * as Schema from 'effect/Schema';
6
+ import { inspect } from 'util';
8
7
  import { describe, expect, test } from 'vitest';
9
8
 
10
9
  import { DXN } from '@dxos/keys';
@@ -12,9 +11,8 @@ import { DXN } from '@dxos/keys';
12
11
  import { Relation } from '../../index';
13
12
  import { TestSchema } from '../../testing';
14
13
  import { getSchemaDXN, getTypeDXN, isInstanceOf } from '../Annotation';
15
- import { ATTR_RELATION_SOURCE, ATTR_RELATION_TARGET } from '../Entity';
16
14
  import { ATTR_META, ATTR_TYPE, getSchema } from '../common/types';
17
-
15
+ import { ATTR_RELATION_SOURCE, ATTR_RELATION_TARGET } from '../Entity';
18
16
  import { createObject } from './create-object';
19
17
  import { objectToJSON } from './json-serializer';
20
18
 
@@ -9,6 +9,8 @@ import { assertArgument, failedInvariant } from '@dxos/invariant';
9
9
  import { ObjectId } from '@dxos/keys';
10
10
 
11
11
  import { getSchemaDXN, getTypeAnnotation, setTypename } from '../Annotation';
12
+ import { defineHiddenProperty } from '../common/proxy';
13
+ import { EntityKind, KindId, MetaId, setSchema } from '../common/types';
12
14
  import {
13
15
  RelationSourceDXNId,
14
16
  RelationSourceId,
@@ -17,9 +19,6 @@ import {
17
19
  assertObjectModel,
18
20
  getObjectDXN,
19
21
  } from '../Entity';
20
- import { defineHiddenProperty } from '../common/proxy';
21
- import { EntityKind, KindId, MetaId, setSchema } from '../common/types';
22
-
23
22
  import { attachedTypedObjectInspector } from './inspect';
24
23
  import { attachTypedJsonSerializer } from './json-serializer';
25
24
 
@@ -2,8 +2,8 @@
2
2
  // Copyright 2025 DXOS.org
3
3
  //
4
4
 
5
- import { ObjectDeletedId } from '../Entity';
6
5
  import { type AnyProperties } from '../common/types';
6
+ import { ObjectDeletedId } from '../Entity';
7
7
 
8
8
  /**
9
9
  * @returns `true` if the object has been marked as deleted.
@@ -5,7 +5,7 @@
5
5
  import { DXN, ObjectId, QueueSubspaceTags, SpaceId } from '@dxos/keys';
6
6
 
7
7
  /**
8
- * @deprecated Use `db.queues.create()`
8
+ * @deprecated Use `Feed.make(...)` + `db.add(feed)` then `Feed.getQueueDxn(feed)`.
9
9
  */
10
10
  // TODO(burdon): Move to @dxos/keys.
11
11
  export const createQueueDXN = (spaceId = SpaceId.random(), queueId = ObjectId.random()) =>
@@ -2,27 +2,28 @@
2
2
  // Copyright 2025 DXOS.org
3
3
  //
4
4
 
5
+ import * as Schema from 'effect/Schema';
5
6
  import { describe, expect, test } from 'vitest';
6
7
 
7
8
  import { DXN } from '@dxos/keys';
8
9
 
9
10
  import * as Obj from '../../Obj';
10
11
  import { TestSchema } from '../../testing';
12
+ import * as Type from '../../Type';
11
13
  import { getSchemaDXN, getSchemaTypename, getTypeDXN, getTypename } from '../Annotation';
12
14
  import { getMetaChecked } from '../common/api';
13
- import { RelationSourceId, RelationTargetId, getObjectDXN } from '../Entity';
14
15
  import { makeObject } from '../common/proxy';
15
- import { Ref, StaticRefResolver } from '../Ref';
16
16
  import { ATTR_TYPE, EntityKind, KindId, MetaId, TypeId, getSchema } from '../common/types';
17
-
17
+ import { RelationSourceId, RelationTargetId, getObjectDXN } from '../Entity';
18
+ import { Ref, StaticRefResolver } from '../Ref';
18
19
  import { createObject } from './create-object';
19
20
  import { objectFromJSON, objectToJSON } from './json-serializer';
20
21
 
21
22
  describe('Object JSON serializer', () => {
22
23
  test('should serialize and deserialize object', async () => {
23
24
  const contact = makeObject(TestSchema.Person, { name: 'Alice' });
24
- Obj.change(contact, (c) => {
25
- getMetaChecked(c).keys.push({ id: '12345', source: 'example.com' });
25
+ Obj.update(contact, (contact) => {
26
+ getMetaChecked(contact).keys.push({ id: '12345', source: 'example.com' });
26
27
  });
27
28
 
28
29
  const task = createObject(TestSchema.Task, {
@@ -118,4 +119,47 @@ describe('Object JSON serializer', () => {
118
119
  expect(expandoFromJson.message).toBe('local-only');
119
120
  expect((expandoFromJson as any)[ATTR_TYPE]).toBeUndefined();
120
121
  });
122
+
123
+ describe('Uint8Array', () => {
124
+ const Blob = Schema.Struct({
125
+ name: Schema.String,
126
+ bytes: Schema.Uint8ArrayFromSelf,
127
+ }).pipe(
128
+ Type.object({
129
+ typename: 'com.example.type.blob',
130
+ version: '0.1.0',
131
+ }),
132
+ );
133
+ interface Blob extends Schema.Schema.Type<typeof Blob> {}
134
+
135
+ test('round-trips Uint8Array field through JSON with schema', async ({ expect }) => {
136
+ const bytes = new Uint8Array([0, 1, 2, 3, 250, 251, 252, 253, 254, 255]);
137
+ const blob = Obj.make(Blob, { name: 'blob', bytes });
138
+
139
+ const blobJson = objectToJSON(blob);
140
+ // JSON must round-trip through stringify/parse without loss.
141
+ const roundTripped = JSON.parse(JSON.stringify(blobJson));
142
+
143
+ const refResolver = new StaticRefResolver().addSchema(Blob);
144
+ const blobFromJson = (await objectFromJSON(roundTripped, { refResolver })) as Blob;
145
+
146
+ expect(blobFromJson.name).toBe('blob');
147
+ expect(blobFromJson.bytes).toBeInstanceOf(Uint8Array);
148
+ expect(Array.from(blobFromJson.bytes)).toEqual(Array.from(bytes));
149
+ });
150
+
151
+ test('round-trips Uint8Array field through JSON without schema resolver', async ({ expect }) => {
152
+ const bytes = new Uint8Array([10, 20, 30, 40, 50]);
153
+ const blob = Obj.make(Blob, { name: 'blob', bytes });
154
+
155
+ const blobJson = objectToJSON(blob);
156
+ const roundTripped = JSON.parse(JSON.stringify(blobJson));
157
+
158
+ const blobFromJson = (await objectFromJSON(roundTripped)) as Blob;
159
+
160
+ expect(blobFromJson.name).toBe('blob');
161
+ expect(blobFromJson.bytes).toBeInstanceOf(Uint8Array);
162
+ expect(Array.from(blobFromJson.bytes)).toEqual(Array.from(bytes));
163
+ });
164
+ });
121
165
  });
@@ -8,11 +8,24 @@ import { raise } from '@dxos/debug';
8
8
  import { type EncodedReference, ObjectStructure, isEncodedReference } from '@dxos/echo-protocol';
9
9
  import { assertArgument, invariant } from '@dxos/invariant';
10
10
  import { DXN, ObjectId } from '@dxos/keys';
11
- import { assumeType, deepMapValues, visitValues } from '@dxos/util';
11
+ import { assumeType, decodeUint8ArrayFromJson, deepMapValues, isEncodedUint8Array, visitValues } from '@dxos/util';
12
12
 
13
13
  import type * as Database from '../../Database';
14
14
  import type * as Obj from '../../Obj';
15
15
  import { getTypeDXN, setTypename } from '../Annotation';
16
+ import { attachTypedJsonSerializer, defineHiddenProperty, typedJsonSerializer } from '../common/proxy';
17
+ import {
18
+ ATTR_META,
19
+ ATTR_PARENT,
20
+ ATTR_TYPE,
21
+ type AnyEntity,
22
+ EntityKind,
23
+ KindId,
24
+ MetaId,
25
+ ObjectMetaSchema,
26
+ ParentId,
27
+ setSchema,
28
+ } from '../common/types';
16
29
  import {
17
30
  ATTR_DELETED,
18
31
  ATTR_RELATION_SOURCE,
@@ -27,20 +40,7 @@ import {
27
40
  SelfDXNId,
28
41
  assertObjectModel,
29
42
  } from '../Entity';
30
- import { attachTypedJsonSerializer, defineHiddenProperty, typedJsonSerializer } from '../common/proxy';
31
43
  import { Ref, type RefResolver, refFromEncodedReference, setRefResolver } from '../Ref';
32
- import {
33
- ATTR_META,
34
- ATTR_PARENT,
35
- ATTR_TYPE,
36
- type AnyEntity,
37
- EntityKind,
38
- KindId,
39
- MetaId,
40
- ObjectMetaSchema,
41
- ParentId,
42
- setSchema,
43
- } from '../common/types';
44
44
 
45
45
  // Re-export for backward compatibility.
46
46
  export { attachTypedJsonSerializer };
@@ -80,7 +80,12 @@ export const objectToJSON = <T extends AnyEntity>(obj: T): SerializedObject<T> =
80
80
  */
81
81
  export const objectFromJSON = async (
82
82
  jsonData: unknown,
83
- { refResolver, dxn, database }: { refResolver?: RefResolver; dxn?: DXN; database?: Database.Database } = {},
83
+ {
84
+ refResolver,
85
+ dxn,
86
+ database,
87
+ parent,
88
+ }: { refResolver?: RefResolver; dxn?: DXN; database?: Database.Database; parent?: Obj.Unknown } = {},
84
89
  ): Promise<AnyEntity> => {
85
90
  assumeType<ObjectJSON>(jsonData);
86
91
  assertArgument(typeof jsonData === 'object' && jsonData !== null, 'jsonData', 'expect object');
@@ -90,7 +95,7 @@ export const objectFromJSON = async (
90
95
  const type = DXN.parse(jsonData[ATTR_TYPE]);
91
96
  const schema = await refResolver?.resolveSchema(type);
92
97
  invariant(schema === undefined || Schema.isSchema(schema));
93
- const decodedInput = stripInternalJsonKeys(jsonData);
98
+ const decodedInput = restoreUint8Arrays(stripInternalJsonKeys(jsonData));
94
99
 
95
100
  let obj: any;
96
101
  if (schema != null) {
@@ -111,15 +116,15 @@ export const objectFromJSON = async (
111
116
  const isRelation =
112
117
  typeof jsonData[ATTR_RELATION_SOURCE] === 'string' || typeof jsonData[ATTR_RELATION_TARGET] === 'string';
113
118
  if (isRelation) {
114
- const sourceDxn: DXN = DXN.parse(jsonData[ATTR_RELATION_SOURCE] ?? raise(new TypeError('Missing relation source')));
115
- const targetDxn: DXN = DXN.parse(jsonData[ATTR_RELATION_TARGET] ?? raise(new TypeError('Missing relation target')));
119
+ const sourceDXN: DXN = DXN.parse(jsonData[ATTR_RELATION_SOURCE] ?? raise(new TypeError('Missing relation source')));
120
+ const targetDXN: DXN = DXN.parse(jsonData[ATTR_RELATION_TARGET] ?? raise(new TypeError('Missing relation target')));
116
121
 
117
- const source = (await refResolver?.resolve(sourceDxn)) as AnyEntity | undefined;
118
- const target = (await refResolver?.resolve(targetDxn)) as AnyEntity | undefined;
122
+ const source = (await refResolver?.resolve(sourceDXN)) as AnyEntity | undefined;
123
+ const target = (await refResolver?.resolve(targetDXN)) as AnyEntity | undefined;
119
124
 
120
125
  defineHiddenProperty(obj, KindId, EntityKind.Relation);
121
- defineHiddenProperty(obj, RelationSourceDXNId, sourceDxn);
122
- defineHiddenProperty(obj, RelationTargetDXNId, targetDxn);
126
+ defineHiddenProperty(obj, RelationSourceDXNId, sourceDXN);
127
+ defineHiddenProperty(obj, RelationTargetDXNId, targetDXN);
123
128
  defineHiddenProperty(obj, RelationSourceId, source);
124
129
  defineHiddenProperty(obj, RelationTargetId, target);
125
130
  } else {
@@ -137,8 +142,10 @@ export const objectFromJSON = async (
137
142
  }
138
143
 
139
144
  if (jsonData[ATTR_PARENT]) {
140
- const parentDxn = DXN.parse(jsonData[ATTR_PARENT]);
141
- const parent = (await refResolver?.resolve(parentDxn)) as Obj.Unknown | undefined;
145
+ const parentDXN = DXN.parse(jsonData[ATTR_PARENT]);
146
+ const resolvedParent = (await refResolver?.resolve(parentDXN)) as Obj.Unknown | undefined;
147
+ defineHiddenProperty(obj, ParentId, resolvedParent);
148
+ } else if (parent) {
142
149
  defineHiddenProperty(obj, ParentId, parent);
143
150
  }
144
151
 
@@ -167,17 +174,32 @@ const decodeGeneric = (jsonData: unknown, options: { refResolver?: RefResolver }
167
174
  if (isEncodedReference(value)) {
168
175
  return refFromEncodedReference(value, options.refResolver);
169
176
  }
177
+ if (isEncodedUint8Array(value)) {
178
+ return decodeUint8ArrayFromJson(value);
179
+ }
170
180
 
171
181
  return visitor(value);
172
182
  });
173
183
  };
174
184
 
185
+ /**
186
+ * Recursively replaces encoded `Uint8Array` JSON markers with actual `Uint8Array` instances.
187
+ * Runs before schema decoding so `Schema.Uint8ArrayFromSelf` sees real bytes.
188
+ */
189
+ const restoreUint8Arrays = (data: unknown): any =>
190
+ deepMapValues(data, (value, recurse) => {
191
+ if (isEncodedUint8Array(value)) {
192
+ return decodeUint8ArrayFromJson(value);
193
+ }
194
+ return recurse(value);
195
+ });
196
+
175
197
  const stripInternalJsonKeys = (jsonData: unknown) => {
176
198
  const {
177
199
  [ATTR_TYPE]: _type,
178
200
  [ATTR_META]: _meta,
179
201
  [ATTR_DELETED]: _deleted,
180
- [ATTR_SELF_DXN]: _selfDxn,
202
+ [ATTR_SELF_DXN]: _selfDXN,
181
203
  [ATTR_RELATION_SOURCE]: _relationSource,
182
204
  [ATTR_RELATION_TARGET]: _relationTarget,
183
205
  ...props
@@ -17,7 +17,7 @@ describe('Obj.setValue', () => {
17
17
  email: 'john@example.com',
18
18
  });
19
19
 
20
- Obj.change(person, (p) => Obj.setValue(p, ['address', 'city'], 'NYC'));
20
+ Obj.update(person, (person) => Obj.setValue(person, ['address', 'city'], 'NYC'));
21
21
 
22
22
  expect(person.address).toBeDefined();
23
23
  expect(person.address?.city).toBe('NYC');
@@ -30,7 +30,7 @@ describe('Obj.setValue', () => {
30
30
  email: 'john@example.com',
31
31
  });
32
32
 
33
- Obj.change(person, (p) => Obj.setValue(p, ['fields', 0, 'label'], 'Phone'));
33
+ Obj.update(person, (person) => Obj.setValue(person, ['fields', 0, 'label'], 'Phone'));
34
34
 
35
35
  expect(Array.isArray(person.fields)).toBe(true);
36
36
  expect(person.fields?.[0].label).toBe('Phone');
@@ -45,7 +45,7 @@ describe('Obj.setValue', () => {
45
45
  email: 'john@example.com',
46
46
  });
47
47
 
48
- Obj.change(person, (p) => Obj.setValue(p, ['address', 'coordinates', 'lat'], 40.7128));
48
+ Obj.update(person, (person) => Obj.setValue(person, ['address', 'coordinates', 'lat'], 40.7128));
49
49
 
50
50
  expect(person.address).toBeDefined();
51
51
  expect(person.address?.coordinates).toBeDefined();
@@ -69,10 +69,10 @@ describe('Obj.setValue', () => {
69
69
 
70
70
  const container = Obj.make(Container, { name: 'box' });
71
71
 
72
- Obj.change(container, (c) => {
73
- Obj.setValue(c, ['items', 0, 'value'], 10);
74
- Obj.setValue(c, ['items', 1, 'value'], 20);
75
- Obj.setValue(c, ['items', 2, 'value'], 30);
72
+ Obj.update(container, (container) => {
73
+ Obj.setValue(container, ['items', 0, 'value'], 10);
74
+ Obj.setValue(container, ['items', 1, 'value'], 20);
75
+ Obj.setValue(container, ['items', 2, 'value'], 30);
76
76
  });
77
77
 
78
78
  expect(container.items).toHaveLength(3);
@@ -88,7 +88,7 @@ describe('Obj.setValue', () => {
88
88
  email: 'john@example.com',
89
89
  });
90
90
 
91
- Obj.change(person, (p) => Obj.setValue(p, ['age'], 25));
91
+ Obj.update(person, (person) => Obj.setValue(person, ['age'], 25));
92
92
 
93
93
  expect(person.age).toBe(25);
94
94
  });
@@ -100,7 +100,7 @@ describe('Obj.setValue', () => {
100
100
  email: 'john@example.com',
101
101
  });
102
102
 
103
- Obj.change(person, (p) => Obj.setValue(p, ['address', 'city'], 'NYC'));
103
+ Obj.update(person, (person) => Obj.setValue(person, ['address', 'city'], 'NYC'));
104
104
 
105
105
  expect(person.address).toBeDefined();
106
106
  expect(person.address?.city).toBe('NYC');
@@ -114,8 +114,8 @@ describe('Obj.setValue', () => {
114
114
  });
115
115
 
116
116
  let result: any;
117
- Obj.change(person, (p) => {
118
- result = Obj.setValue(p, ['age'], 30);
117
+ Obj.update(person, (person) => {
118
+ result = Obj.setValue(person, ['age'], 30);
119
119
  });
120
120
 
121
121
  expect(result).toBe(30);
@@ -129,7 +129,7 @@ describe('Obj.setValue', () => {
129
129
  age: 25,
130
130
  });
131
131
 
132
- Obj.change(person, (p) => Obj.setValue(p, ['age'], 30));
132
+ Obj.update(person, (person) => Obj.setValue(person, ['age'], 30));
133
133
 
134
134
  expect(person.age).toBe(30);
135
135
  });
@@ -142,7 +142,7 @@ describe('Obj.setValue', () => {
142
142
  address: { city: 'Boston', state: 'MA', coordinates: {} },
143
143
  });
144
144
 
145
- Obj.change(person, (p) => Obj.setValue(p, ['address', 'city'], 'NYC'));
145
+ Obj.update(person, (person) => Obj.setValue(person, ['address', 'city'], 'NYC'));
146
146
 
147
147
  expect(person.address?.city).toBe('NYC');
148
148
  expect(person.address?.state).toBe('MA');
@@ -160,11 +160,11 @@ describe('Obj.setValue', () => {
160
160
 
161
161
  const matrix = Obj.make(Matrix, {});
162
162
 
163
- Obj.change(matrix, (m) => {
164
- Obj.setValue(m, ['values', 0, 0], 1);
165
- Obj.setValue(m, ['values', 0, 1], 2);
166
- Obj.setValue(m, ['values', 1, 0], 3);
167
- Obj.setValue(m, ['values', 1, 1], 4);
163
+ Obj.update(matrix, (matrix) => {
164
+ Obj.setValue(matrix, ['values', 0, 0], 1);
165
+ Obj.setValue(matrix, ['values', 0, 1], 2);
166
+ Obj.setValue(matrix, ['values', 1, 0], 3);
167
+ Obj.setValue(matrix, ['values', 1, 1], 4);
168
168
  });
169
169
 
170
170
  expect(matrix.values?.[0][0]).toBe(1);
@@ -180,8 +180,8 @@ describe('Obj.setValue', () => {
180
180
  email: 'john@example.com',
181
181
  });
182
182
 
183
- Obj.change(person, (p) => {
184
- expect(() => Obj.setValue(p, [], 'value')).toThrow('Path must not be empty');
183
+ Obj.update(person, (person) => {
184
+ expect(() => Obj.setValue(person, [], 'value')).toThrow('Path must not be empty');
185
185
  });
186
186
  });
187
187
 
@@ -192,7 +192,7 @@ describe('Obj.setValue', () => {
192
192
  email: 'john@example.com',
193
193
  });
194
194
 
195
- Obj.change(person, (p) => Obj.setValue(p, ['age'], 30));
195
+ Obj.update(person, (person) => Obj.setValue(person, ['age'], 30));
196
196
 
197
197
  expect(person.age).toBe(30);
198
198
  });
@@ -215,7 +215,7 @@ describe('Obj.setValue', () => {
215
215
  const container = Obj.make(Container, { name: 'box' });
216
216
 
217
217
  // Using string '0' for array index.
218
- Obj.change(container, (c) => Obj.setValue(c, ['items', '0', 'value'], 42));
218
+ Obj.update(container, (container) => Obj.setValue(container, ['items', '0', 'value'], 42));
219
219
 
220
220
  expect(container.items?.[0].value).toBe(42);
221
221
  });
@@ -243,7 +243,7 @@ describe('Obj.setValue', () => {
243
243
 
244
244
  // This should work: setting a nested property on an array element.
245
245
  // The required 'id' field should be initialized with a default value.
246
- Obj.change(todoList, (t) => Obj.setValue(t, ['tasks', 0, 'title'], 'Buy groceries'));
246
+ Obj.update(todoList, (todoList) => Obj.setValue(todoList, ['tasks', 0, 'title'], 'Buy groceries'));
247
247
 
248
248
  expect(todoList.tasks?.[0].id).toBe(''); // Default value for required String
249
249
  expect(todoList.tasks?.[0].title).toBe('Buy groceries');
@@ -270,7 +270,7 @@ describe('Obj.setValue', () => {
270
270
 
271
271
  const container = Obj.make(Container, { name: 'box' });
272
272
 
273
- Obj.change(container, (c) => Obj.setValue(c, ['items', 0, 'label'], 'First Item'));
273
+ Obj.update(container, (container) => Obj.setValue(container, ['items', 0, 'label'], 'First Item'));
274
274
 
275
275
  // All required primitive fields should have default values.
276
276
  expect(container.items?.[0].id).toBe('');
@@ -21,7 +21,7 @@ import { getSchema } from '../common/types';
21
21
 
22
22
  /**
23
23
  * Set a deeply nested property on an object.
24
- * Must be called within an Obj.change or Relation.change callback.
24
+ * Must be called within an Obj.update or Relation.update callback.
25
25
  */
26
26
  export const setValue = (obj: Mutable<any>, path: readonly (string | number)[], value: any): void => {
27
27
  invariant(path.length > 0, 'Path must not be empty');
@@ -0,0 +1,156 @@
1
+ //
2
+ // Copyright 2025 DXOS.org
3
+ //
4
+
5
+ import { type QueryAST } from '@dxos/echo-protocol';
6
+
7
+ /**
8
+ * Returns a human-readable string representation of a Filter AST.
9
+ */
10
+ export const prettyFilter = (filter: QueryAST.Filter): string => {
11
+ switch (filter.type) {
12
+ case 'object': {
13
+ // A type-less object filter with only a meta-key constraint is `Filter.key(...)`.
14
+ if (
15
+ filter.typename === null &&
16
+ (filter.id === undefined || filter.id.length === 0) &&
17
+ Object.keys(filter.props).length === 0 &&
18
+ (filter.foreignKeys === undefined || filter.foreignKeys.length === 0) &&
19
+ filter.metaKey !== undefined
20
+ ) {
21
+ return filter.metaVersion !== undefined
22
+ ? `Filter.key(${JSON.stringify(filter.metaKey)}, { version: ${JSON.stringify(filter.metaVersion)} })`
23
+ : `Filter.key(${JSON.stringify(filter.metaKey)})`;
24
+ }
25
+ const parts: string[] = [];
26
+ if (filter.typename !== null) {
27
+ parts.push(String(filter.typename));
28
+ }
29
+ if (filter.id !== undefined && filter.id.length > 0) {
30
+ parts.push(`id: [${filter.id.join(', ')}]`);
31
+ }
32
+ const propEntries = Object.entries(filter.props);
33
+ if (propEntries.length > 0) {
34
+ const propsStr = propEntries.map(([k, v]) => `${k}: ${prettyFilter(v)}`).join(', ');
35
+ parts.push(`{ ${propsStr} }`);
36
+ }
37
+ if (filter.foreignKeys !== undefined && filter.foreignKeys.length > 0) {
38
+ parts.push(`foreignKeys: [${filter.foreignKeys.map((fk) => JSON.stringify(fk)).join(', ')}]`);
39
+ }
40
+ if (filter.metaKey !== undefined) {
41
+ parts.push(
42
+ filter.metaVersion !== undefined
43
+ ? `metaKey: ${JSON.stringify(filter.metaKey)} (${filter.metaVersion})`
44
+ : `metaKey: ${JSON.stringify(filter.metaKey)}`,
45
+ );
46
+ }
47
+ return parts.length > 0 ? `Filter.type(${parts.join(', ')})` : 'Filter.everything()';
48
+ }
49
+ case 'compare':
50
+ return `Filter.${filter.operator}(${JSON.stringify(filter.value)})`;
51
+ case 'in':
52
+ return `Filter.in([${filter.values.map((v) => JSON.stringify(v)).join(', ')}])`;
53
+ case 'contains':
54
+ return `Filter.contains(${JSON.stringify(filter.value)})`;
55
+ case 'tag':
56
+ return `Filter.tag(${JSON.stringify(filter.tag)})`;
57
+ case 'range':
58
+ return `Filter.range(${JSON.stringify(filter.from)}, ${JSON.stringify(filter.to)})`;
59
+ case 'text-search':
60
+ return filter.searchKind
61
+ ? `Filter.textSearch(${JSON.stringify(filter.text)}, ${JSON.stringify(filter.searchKind)})`
62
+ : `Filter.textSearch(${JSON.stringify(filter.text)})`;
63
+ case 'timestamp':
64
+ return `Filter.${filter.field}.${filter.operator}(${filter.value})`;
65
+ case 'child-of':
66
+ return `Filter.childOf([${filter.parents.map((p) => JSON.stringify(p)).join(', ')}], { transitive: ${filter.transitive} })`;
67
+ case 'not':
68
+ return `Filter.not(${prettyFilter(filter.filter)})`;
69
+ case 'and':
70
+ return `Filter.and(${filter.filters.map(prettyFilter).join(', ')})`;
71
+ case 'or':
72
+ return `Filter.or(${filter.filters.map(prettyFilter).join(', ')})`;
73
+ }
74
+ };
75
+
76
+ /**
77
+ * Returns a human-readable string representation of a Query AST.
78
+ */
79
+ export const prettyQuery = (query: QueryAST.Query): string => {
80
+ switch (query.type) {
81
+ case 'select':
82
+ return `Query.select(${prettyFilter(query.filter)})`;
83
+ case 'filter':
84
+ return `${prettyQuery(query.selection)}.select(${prettyFilter(query.filter)})`;
85
+ case 'reference-traversal':
86
+ return `${prettyQuery(query.anchor)}.reference(${JSON.stringify(query.property)})`;
87
+ case 'incoming-references': {
88
+ const args: string[] = [];
89
+ if (query.typename !== null) {
90
+ args.push(String(query.typename));
91
+ }
92
+ if (query.property !== null) {
93
+ args.push(JSON.stringify(query.property));
94
+ }
95
+ return `${prettyQuery(query.anchor)}.referencedBy(${args.join(', ')})`;
96
+ }
97
+ case 'relation': {
98
+ const method =
99
+ query.direction === 'outgoing' ? 'sourceOf' : query.direction === 'incoming' ? 'targetOf' : 'relationOf';
100
+ const filterStr = query.filter !== undefined ? prettyFilter(query.filter) : '';
101
+ return `${prettyQuery(query.anchor)}.${method}(${filterStr})`;
102
+ }
103
+ case 'relation-traversal':
104
+ return `${prettyQuery(query.anchor)}.${query.direction}()`;
105
+ case 'hierarchy-traversal':
106
+ return query.direction === 'to-parent'
107
+ ? `${prettyQuery(query.anchor)}.parent()`
108
+ : `${prettyQuery(query.anchor)}.children()`;
109
+ case 'union':
110
+ return `Query.all(${query.queries.map(prettyQuery).join(', ')})`;
111
+ case 'set-difference':
112
+ return `Query.without(${prettyQuery(query.source)}, ${prettyQuery(query.exclude)})`;
113
+ case 'order': {
114
+ const orders = query.order.map((o) => {
115
+ if (o.kind === 'natural') {
116
+ return 'Order.natural()';
117
+ } else if (o.kind === 'rank') {
118
+ return `Order.rank(${JSON.stringify(o.direction)})`;
119
+ } else {
120
+ return `Order.property(${JSON.stringify(o.property)}, ${JSON.stringify(o.direction)})`;
121
+ }
122
+ });
123
+ return `${prettyQuery(query.query)}.orderBy(${orders.join(', ')})`;
124
+ }
125
+ case 'options': {
126
+ const opts = query.options;
127
+ const parts: string[] = [];
128
+ if (opts.deleted !== undefined) {
129
+ parts.push(`deleted: ${JSON.stringify(opts.deleted)}`);
130
+ }
131
+ if (opts.debugLabel !== undefined) {
132
+ parts.push(`debugLabel: ${JSON.stringify(opts.debugLabel)}`);
133
+ }
134
+ return `${prettyQuery(query.query)}.options({ ${parts.join(', ')} })`;
135
+ }
136
+ case 'from': {
137
+ if (query.from._tag === 'scope') {
138
+ const scope = query.from.scope;
139
+ const parts: string[] = [];
140
+ if (scope.spaceIds !== undefined) {
141
+ parts.push(`spaceIds: [${scope.spaceIds.map((s) => JSON.stringify(s)).join(', ')}]`);
142
+ }
143
+ if (scope.feeds !== undefined) {
144
+ parts.push(`feeds: [${scope.feeds.map(String).join(', ')}]`);
145
+ }
146
+ if (scope.allFeedsFromSpaces !== undefined) {
147
+ parts.push(`allFeedsFromSpaces: ${scope.allFeedsFromSpaces}`);
148
+ }
149
+ return `${prettyQuery(query.query)}.from({ ${parts.join(', ')} })`;
150
+ }
151
+ return `${prettyQuery(query.query)}.from(${prettyQuery(query.from.query)})`;
152
+ }
153
+ case 'limit':
154
+ return `${prettyQuery(query.query)}.limit(${query.limit})`;
155
+ }
156
+ };
@@ -6,7 +6,6 @@ import { type ObjectId } from '@dxos/keys';
6
6
  import { isNonNullable } from '@dxos/util';
7
7
 
8
8
  import { type AnyEntity } from '../common/types';
9
-
10
9
  import { Ref } from './ref';
11
10
 
12
11
  /**
@@ -9,7 +9,6 @@ import { DXN, ObjectId } from '@dxos/keys';
9
9
 
10
10
  import { EchoObjectSchema, getObjectDXN } from '../Entity';
11
11
  import { createObject } from '../Obj';
12
-
13
12
  import { Ref, getReferenceAst } from './ref';
14
13
 
15
14
  const Task = Schema.Struct({
@@ -20,8 +20,8 @@ import { DXN, ObjectId } from '@dxos/keys';
20
20
 
21
21
  import * as Database from '../../Database';
22
22
  import { ReferenceAnnotationId, getSchemaDXN, getTypeAnnotation, getTypeIdentifierAnnotation } from '../Annotation';
23
- import { type JsonSchemaType } from '../JsonSchema';
24
23
  import type { AnyEntity, AnyProperties } from '../common/types';
24
+ import { type JsonSchemaType } from '../JsonSchema';
25
25
 
26
26
  /**
27
27
  * The `$id` and `$ref` fields for an ECHO reference schema.
@@ -164,6 +164,16 @@ export interface Ref<T> extends Pipeable.Pipeable {
164
164
 
165
165
  tryLoad(): Promise<T | undefined>;
166
166
 
167
+ /**
168
+ * Subscribe to the ref's resolution event.
169
+ * The callback fires when the target object becomes available in the working set
170
+ * (e.g. when its document is loaded after sibling-client mutation).
171
+ * Note: the resolver only schedules a notification when the target is requested
172
+ * via {@link target} while it is not yet loaded.
173
+ * @returns Function that unsubscribes the callback.
174
+ */
175
+ onResolved(callback: () => void): () => void;
176
+
167
177
  /**
168
178
  * Do not inline the target object in the reference.
169
179
  * Makes .target unavailable unless the reference is connected to a database context.
@@ -420,6 +430,13 @@ export class RefImpl<T> implements Ref<T> {
420
430
  return (await this.#resolver.resolve(this.#dxn)) as T | undefined;
421
431
  }
422
432
 
433
+ /**
434
+ * @inheritdoc
435
+ */
436
+ onResolved(callback: () => void): () => void {
437
+ return this.#resolved.on(callback);
438
+ }
439
+
423
440
  /**
424
441
  * Do not inline the target object in the reference.
425
442
  * Makes .target unavailable unless the reference is connected to a database context.
@@ -9,7 +9,6 @@ import { FieldPath } from '../Annotation';
9
9
  import { EchoObjectSchema } from '../Entity';
10
10
  import { FormatAnnotation, TypeFormat } from '../Format';
11
11
  import { ECHO_ANNOTATIONS_NS_KEY, toJsonSchema } from '../JsonSchema';
12
-
13
12
  import { composeSchema } from './compose';
14
13
 
15
14
  describe('schema composition', () => {