@dxos/echo 0.8.4-main.c85a9c8dae → 0.8.4-main.d05539e30a

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 (558) 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 -11
  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 -14
  10. package/dist/lib/neutral/Filter.mjs +23 -10
  11. package/dist/lib/neutral/Format.mjs +3 -3
  12. package/dist/lib/neutral/JsonSchema.mjs +8 -7
  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 -10
  17. package/dist/lib/neutral/Order.mjs +1 -1
  18. package/dist/lib/neutral/Query.mjs +19 -13
  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 -11
  22. package/dist/lib/neutral/SchemaRegistry.mjs +1 -1
  23. package/dist/lib/neutral/Tag.mjs +14 -10
  24. package/dist/lib/neutral/Type.mjs +10 -7
  25. package/dist/lib/neutral/{chunk-DZQSL6RW.mjs → chunk-44HT3MEC.mjs} +2 -2
  26. package/dist/lib/neutral/{chunk-DZQSL6RW.mjs.map → chunk-44HT3MEC.mjs.map} +1 -1
  27. package/dist/lib/neutral/{chunk-OVUBTQLT.mjs → chunk-5SL5LDLD.mjs} +4 -2
  28. package/dist/lib/neutral/chunk-5SL5LDLD.mjs.map +7 -0
  29. package/dist/lib/neutral/{chunk-ANHVGJI4.mjs → chunk-7RVZT53K.mjs} +1 -1
  30. package/dist/lib/neutral/chunk-APHSOTIX.mjs +34 -0
  31. package/dist/lib/neutral/chunk-APHSOTIX.mjs.map +7 -0
  32. package/dist/lib/neutral/{chunk-ALOGSVBP.mjs → chunk-APJKDGFL.mjs} +72 -6
  33. package/dist/lib/neutral/chunk-APJKDGFL.mjs.map +7 -0
  34. package/dist/lib/neutral/{chunk-T6JEFNP6.mjs → chunk-BICZKPQG.mjs} +1 -1
  35. package/dist/lib/neutral/{chunk-T6JEFNP6.mjs.map → chunk-BICZKPQG.mjs.map} +1 -1
  36. package/dist/lib/neutral/chunk-BMB7IHGB.mjs +346 -0
  37. package/dist/lib/neutral/chunk-BMB7IHGB.mjs.map +7 -0
  38. package/dist/lib/neutral/chunk-FIWO2FZK.mjs +36 -0
  39. package/dist/lib/neutral/chunk-FIWO2FZK.mjs.map +7 -0
  40. package/dist/lib/neutral/{chunk-PP4DNUZU.mjs → chunk-G54OX4IX.mjs} +1418 -2885
  41. package/dist/lib/neutral/chunk-G54OX4IX.mjs.map +7 -0
  42. package/dist/lib/neutral/chunk-GWFFC34K.mjs +50 -0
  43. package/dist/lib/neutral/chunk-GWFFC34K.mjs.map +7 -0
  44. package/dist/lib/neutral/{chunk-5ELDDYWE.mjs → chunk-I2DARWPX.mjs} +17 -19
  45. package/dist/lib/neutral/chunk-I2DARWPX.mjs.map +7 -0
  46. package/dist/lib/neutral/{chunk-KGV3QIRX.mjs → chunk-J54QMAKF.mjs} +125 -17
  47. package/dist/lib/neutral/chunk-J54QMAKF.mjs.map +7 -0
  48. package/dist/lib/neutral/{chunk-BRJSLACP.mjs → chunk-MGSQGHOD.mjs} +71 -17
  49. package/dist/lib/neutral/chunk-MGSQGHOD.mjs.map +7 -0
  50. package/dist/lib/neutral/chunk-MLS7U7AT.mjs +734 -0
  51. package/dist/lib/neutral/chunk-MLS7U7AT.mjs.map +7 -0
  52. package/dist/lib/neutral/{chunk-4KG6IGL4.mjs → chunk-N4B7FHQT.mjs} +4 -4
  53. package/dist/lib/neutral/{chunk-4KG6IGL4.mjs.map → chunk-N4B7FHQT.mjs.map} +2 -2
  54. package/dist/lib/neutral/{chunk-QXF3LGN2.mjs → chunk-N7VOEPSV.mjs} +8 -3
  55. package/dist/lib/neutral/chunk-N7VOEPSV.mjs.map +7 -0
  56. package/dist/lib/neutral/{chunk-T4MPQJ7X.mjs → chunk-PSZBLH53.mjs} +6 -22
  57. package/dist/lib/neutral/chunk-PSZBLH53.mjs.map +7 -0
  58. package/dist/lib/neutral/{chunk-ZWKJ4LZJ.mjs → chunk-PT37DG2F.mjs} +23 -51
  59. package/dist/lib/neutral/chunk-PT37DG2F.mjs.map +7 -0
  60. package/dist/lib/neutral/{chunk-GAWKQ5DZ.mjs → chunk-Q2KKKJSV.mjs} +5 -5
  61. package/dist/lib/neutral/chunk-Q2KKKJSV.mjs.map +7 -0
  62. package/dist/lib/neutral/{chunk-AQP4QKYP.mjs → chunk-Q7ZL2P5H.mjs} +56 -19
  63. package/dist/lib/neutral/chunk-Q7ZL2P5H.mjs.map +7 -0
  64. package/dist/lib/neutral/{chunk-O3TZFQCT.mjs → chunk-QRZ2I3ZM.mjs} +2 -2
  65. package/dist/lib/neutral/chunk-QRZ2I3ZM.mjs.map +7 -0
  66. package/dist/lib/neutral/{chunk-W7OP5HGZ.mjs → chunk-SCPFDS2E.mjs} +14 -10
  67. package/dist/lib/neutral/chunk-SCPFDS2E.mjs.map +7 -0
  68. package/dist/lib/neutral/{chunk-F7KMHDPJ.mjs → chunk-TNBK56IN.mjs} +54 -31
  69. package/dist/lib/neutral/chunk-TNBK56IN.mjs.map +7 -0
  70. package/dist/lib/neutral/{chunk-M2KVTHZM.mjs → chunk-TRPZU2HV.mjs} +10 -10
  71. package/dist/lib/neutral/chunk-TRPZU2HV.mjs.map +7 -0
  72. package/dist/lib/neutral/{chunk-ZTUBYOGB.mjs → chunk-TTCSATUD.mjs} +1 -1
  73. package/dist/lib/neutral/chunk-TTCSATUD.mjs.map +7 -0
  74. package/dist/lib/neutral/{chunk-OMUPQMLR.mjs → chunk-V72DY6LU.mjs} +1 -1
  75. package/dist/lib/neutral/chunk-ZFACXBY6.mjs +136 -0
  76. package/dist/lib/neutral/chunk-ZFACXBY6.mjs.map +7 -0
  77. package/dist/lib/neutral/{chunk-MXQJZCPY.mjs → chunk-ZISMEVKD.mjs} +1 -1
  78. package/dist/lib/neutral/chunk-ZISMEVKD.mjs.map +7 -0
  79. package/dist/lib/neutral/index.mjs +38 -25
  80. package/dist/lib/neutral/internal/index.mjs +50 -38
  81. package/dist/lib/neutral/meta.json +1 -1
  82. package/dist/lib/neutral/testing/index.mjs +196 -136
  83. package/dist/lib/neutral/testing/index.mjs.map +3 -3
  84. package/dist/types/src/Annotation.d.ts +3 -3
  85. package/dist/types/src/Annotation.d.ts.map +1 -1
  86. package/dist/types/src/Collection.d.ts.map +1 -1
  87. package/dist/types/src/Database.d.ts +16 -4
  88. package/dist/types/src/Database.d.ts.map +1 -1
  89. package/dist/types/src/Dataset.d.ts +2 -1
  90. package/dist/types/src/Dataset.d.ts.map +1 -1
  91. package/dist/types/src/Entity.d.ts +17 -11
  92. package/dist/types/src/Entity.d.ts.map +1 -1
  93. package/dist/types/src/Err.d.ts +18 -18
  94. package/dist/types/src/Err.d.ts.map +1 -1
  95. package/dist/types/src/Extension.d.ts +80 -0
  96. package/dist/types/src/Extension.d.ts.map +1 -0
  97. package/dist/types/src/Extension.test.d.ts +2 -0
  98. package/dist/types/src/Extension.test.d.ts.map +1 -0
  99. package/dist/types/src/Feed.d.ts +65 -24
  100. package/dist/types/src/Feed.d.ts.map +1 -1
  101. package/dist/types/src/Filter.d.ts +54 -4
  102. package/dist/types/src/Filter.d.ts.map +1 -1
  103. package/dist/types/src/Filter.test.d.ts +2 -0
  104. package/dist/types/src/Filter.test.d.ts.map +1 -0
  105. package/dist/types/src/Format.d.ts +1 -1
  106. package/dist/types/src/Format.d.ts.map +1 -1
  107. package/dist/types/src/Hypergraph.d.ts +9 -4
  108. package/dist/types/src/Hypergraph.d.ts.map +1 -1
  109. package/dist/types/src/Json.d.ts +33 -0
  110. package/dist/types/src/Json.d.ts.map +1 -0
  111. package/dist/types/src/Json.test.d.ts +2 -0
  112. package/dist/types/src/Json.test.d.ts.map +1 -0
  113. package/dist/types/src/JsonSchema.d.ts +5 -5
  114. package/dist/types/src/JsonSchema.d.ts.map +1 -1
  115. package/dist/types/src/Migration.d.ts +69 -0
  116. package/dist/types/src/Migration.d.ts.map +1 -0
  117. package/dist/types/src/Obj.d.ts +46 -30
  118. package/dist/types/src/Obj.d.ts.map +1 -1
  119. package/dist/types/src/Order.d.ts.map +1 -1
  120. package/dist/types/src/Query.d.ts +13 -2
  121. package/dist/types/src/Query.d.ts.map +1 -1
  122. package/dist/types/src/Ref.d.ts +12 -11
  123. package/dist/types/src/Ref.d.ts.map +1 -1
  124. package/dist/types/src/Relation.d.ts +16 -16
  125. package/dist/types/src/Relation.d.ts.map +1 -1
  126. package/dist/types/src/SchemaRegistry.d.ts +1 -1
  127. package/dist/types/src/Tag.d.ts +2 -2
  128. package/dist/types/src/Tag.d.ts.map +1 -1
  129. package/dist/types/src/Type.d.ts +10 -9
  130. package/dist/types/src/Type.d.ts.map +1 -1
  131. package/dist/types/src/View.d.ts +1 -1
  132. package/dist/types/src/View.d.ts.map +1 -1
  133. package/dist/types/src/exemplars.test.d.ts +2 -0
  134. package/dist/types/src/exemplars.test.d.ts.map +1 -0
  135. package/dist/types/src/index.d.ts +3 -0
  136. package/dist/types/src/index.d.ts.map +1 -1
  137. package/dist/types/src/internal/{annotations → Annotation}/annotations.d.ts +33 -2
  138. package/dist/types/src/internal/Annotation/annotations.d.ts.map +1 -0
  139. package/dist/types/src/internal/Annotation/annotations.test.d.ts.map +1 -0
  140. package/dist/types/src/internal/Annotation/index.d.ts +4 -0
  141. package/dist/types/src/internal/Annotation/index.d.ts.map +1 -0
  142. package/dist/types/src/internal/{api → Annotation}/sorting.d.ts +1 -1
  143. package/dist/types/src/internal/Annotation/sorting.d.ts.map +1 -0
  144. package/dist/types/src/internal/{annotations → Annotation}/util.d.ts +2 -2
  145. package/dist/types/src/internal/Annotation/util.d.ts.map +1 -0
  146. package/dist/types/src/internal/{api/entity.d.ts → Entity/api.d.ts} +2 -2
  147. package/dist/types/src/internal/Entity/api.d.ts.map +1 -0
  148. package/dist/types/src/internal/{entities → Entity}/entity.d.ts +2 -2
  149. package/dist/types/src/internal/Entity/entity.d.ts.map +1 -0
  150. package/dist/types/src/internal/{entities → Entity}/index.d.ts +2 -0
  151. package/dist/types/src/internal/Entity/index.d.ts.map +1 -0
  152. package/dist/types/src/internal/{entities → Entity}/model.d.ts +4 -26
  153. package/dist/types/src/internal/Entity/model.d.ts.map +1 -0
  154. package/dist/types/src/internal/{entities → Entity}/object.d.ts +2 -2
  155. package/dist/types/src/internal/Entity/object.d.ts.map +1 -0
  156. package/dist/types/src/internal/{entities → Entity}/relation.d.ts +3 -30
  157. package/dist/types/src/internal/Entity/relation.d.ts.map +1 -0
  158. package/dist/types/src/internal/Entity/util.d.ts.map +1 -0
  159. package/dist/types/src/internal/{api → Entity}/version.d.ts +1 -1
  160. package/dist/types/src/internal/Entity/version.d.ts.map +1 -0
  161. package/dist/types/src/internal/Format/date.d.ts.map +1 -0
  162. package/dist/types/src/internal/Format/date.test.d.ts.map +1 -0
  163. package/dist/types/src/internal/Format/format.d.ts.map +1 -0
  164. package/dist/types/src/internal/Format/format.test.d.ts.map +1 -0
  165. package/dist/types/src/internal/Format/index.d.ts.map +1 -0
  166. package/dist/types/src/internal/Format/number.d.ts.map +1 -0
  167. package/dist/types/src/internal/Format/object.d.ts.map +1 -0
  168. package/dist/types/src/internal/Format/select.d.ts.map +1 -0
  169. package/dist/types/src/internal/Format/string.d.ts.map +1 -0
  170. package/dist/types/src/internal/{formats → Format}/types.d.ts +1 -1
  171. package/dist/types/src/internal/Format/types.d.ts.map +1 -0
  172. package/dist/types/src/internal/{json-schema → JsonSchema}/annotations.d.ts +1 -1
  173. package/dist/types/src/internal/JsonSchema/annotations.d.ts.map +1 -0
  174. package/dist/types/src/internal/JsonSchema/effect-schema.test.d.ts.map +1 -0
  175. package/dist/types/src/internal/JsonSchema/index.d.ts.map +1 -0
  176. package/dist/types/src/internal/JsonSchema/json-schema-normalize.d.ts.map +1 -0
  177. package/dist/types/src/internal/{json-schema → JsonSchema}/json-schema-type.d.ts +23 -23
  178. package/dist/types/src/internal/JsonSchema/json-schema-type.d.ts.map +1 -0
  179. package/dist/types/src/internal/{json-schema → JsonSchema}/json-schema.d.ts +1 -1
  180. package/dist/types/src/internal/JsonSchema/json-schema.d.ts.map +1 -0
  181. package/dist/types/src/internal/JsonSchema/json-schema.test.d.ts.map +1 -0
  182. package/dist/types/src/internal/Obj/clone.d.ts.map +1 -0
  183. package/dist/types/src/internal/Obj/common.d.ts.map +1 -0
  184. package/dist/types/src/internal/{object → Obj}/create-object.d.ts +2 -2
  185. package/dist/types/src/internal/Obj/create-object.d.ts.map +1 -0
  186. package/dist/types/src/internal/Obj/create-object.test.d.ts.map +1 -0
  187. package/dist/types/src/internal/{object → Obj}/deleted.d.ts +1 -1
  188. package/dist/types/src/internal/Obj/deleted.d.ts.map +1 -0
  189. package/dist/types/src/internal/{object → Obj}/ids.d.ts +1 -1
  190. package/dist/types/src/internal/Obj/ids.d.ts.map +1 -0
  191. package/dist/types/src/internal/Obj/index.d.ts.map +1 -0
  192. package/dist/types/src/internal/Obj/inspect.d.ts.map +1 -0
  193. package/dist/types/src/internal/{object → Obj}/json-serializer.d.ts +13 -5
  194. package/dist/types/src/internal/Obj/json-serializer.d.ts.map +1 -0
  195. package/dist/types/src/internal/Obj/json-serializer.test.d.ts.map +1 -0
  196. package/dist/types/src/internal/Obj/schema-validator.d.ts +2 -0
  197. package/dist/types/src/internal/Obj/schema-validator.d.ts.map +1 -0
  198. package/dist/types/src/internal/Obj/schema-validator.test.d.ts.map +1 -0
  199. package/dist/types/src/internal/{object → Obj}/set-value.d.ts +2 -2
  200. package/dist/types/src/internal/Obj/set-value.d.ts.map +1 -0
  201. package/dist/types/src/internal/Obj/set-value.test.d.ts.map +1 -0
  202. package/dist/types/src/internal/Obj/snapshot.d.ts.map +1 -0
  203. package/dist/types/src/internal/{object → Obj}/typed-object.d.ts +2 -2
  204. package/dist/types/src/internal/Obj/typed-object.d.ts.map +1 -0
  205. package/dist/types/src/internal/Obj/typed-object.test.d.ts.map +1 -0
  206. package/dist/types/src/internal/Query.d.ts +10 -0
  207. package/dist/types/src/internal/Query.d.ts.map +1 -0
  208. package/dist/types/src/internal/Ref/index.d.ts.map +1 -0
  209. package/dist/types/src/internal/{ref → Ref}/ref-array.d.ts +1 -1
  210. package/dist/types/src/internal/Ref/ref-array.d.ts.map +1 -0
  211. package/dist/types/src/internal/{ref → Ref}/ref.d.ts +15 -2
  212. package/dist/types/src/internal/Ref/ref.d.ts.map +1 -0
  213. package/dist/types/src/internal/Ref/ref.test.d.ts.map +1 -0
  214. package/dist/types/src/internal/{schema → Type}/compose.d.ts +1 -1
  215. package/dist/types/src/internal/Type/compose.d.ts.map +1 -0
  216. package/dist/types/src/internal/Type/compose.test.d.ts.map +1 -0
  217. package/dist/types/src/internal/{schema → Type}/echo-schema.d.ts +6 -6
  218. package/dist/types/src/internal/Type/echo-schema.d.ts.map +1 -0
  219. package/dist/types/src/internal/Type/index.d.ts.map +1 -0
  220. package/dist/types/src/internal/Type/manipulation.d.ts.map +1 -0
  221. package/dist/types/src/internal/{schema → Type}/persistent-schema.d.ts +2 -2
  222. package/dist/types/src/internal/Type/persistent-schema.d.ts.map +1 -0
  223. package/dist/types/src/internal/{api → common/api}/index.d.ts +0 -4
  224. package/dist/types/src/internal/common/api/index.d.ts.map +1 -0
  225. package/dist/types/src/internal/{api → common/api}/meta.d.ts +3 -3
  226. package/dist/types/src/internal/common/api/meta.d.ts.map +1 -0
  227. package/dist/types/src/internal/common/index.d.ts +4 -0
  228. package/dist/types/src/internal/common/index.d.ts.map +1 -0
  229. package/dist/types/src/internal/{proxy → common/proxy}/change-context.d.ts +1 -1
  230. package/dist/types/src/internal/common/proxy/change-context.d.ts.map +1 -0
  231. package/dist/types/src/internal/common/proxy/change.test.d.ts.map +1 -0
  232. package/dist/types/src/internal/common/proxy/define-hidden-property.d.ts.map +1 -0
  233. package/dist/types/src/internal/{proxy → common/proxy}/errors.d.ts +1 -1
  234. package/dist/types/src/internal/common/proxy/errors.d.ts.map +1 -0
  235. package/dist/types/src/internal/common/proxy/event-batch.d.ts.map +1 -0
  236. package/dist/types/src/internal/common/proxy/handler.test.d.ts.map +1 -0
  237. package/dist/types/src/internal/common/proxy/index.d.ts.map +1 -0
  238. package/dist/types/src/internal/common/proxy/json-serializer.d.ts.map +1 -0
  239. package/dist/types/src/internal/common/proxy/make-object.d.ts.map +1 -0
  240. package/dist/types/src/internal/common/proxy/ownership.d.ts.map +1 -0
  241. package/dist/types/src/internal/common/proxy/proxy-types.d.ts.map +1 -0
  242. package/dist/types/src/internal/common/proxy/proxy-utils.d.ts.map +1 -0
  243. package/dist/types/src/internal/{proxy → common/proxy}/reactive-array.d.ts +1 -1
  244. package/dist/types/src/internal/common/proxy/reactive-array.d.ts.map +1 -0
  245. package/dist/types/src/internal/{proxy → common/proxy}/reactive.d.ts +2 -2
  246. package/dist/types/src/internal/{proxy → common/proxy}/reactive.d.ts.map +1 -1
  247. package/dist/types/src/internal/common/proxy/reactive.test.d.ts +2 -0
  248. package/dist/types/src/internal/common/proxy/reactive.test.d.ts.map +1 -0
  249. package/dist/types/src/internal/common/proxy/schema-validator.d.ts.map +1 -0
  250. package/dist/types/src/internal/common/proxy/schema.test.d.ts.map +1 -0
  251. package/dist/types/src/internal/common/proxy/symbols.d.ts.map +1 -0
  252. package/dist/types/src/internal/common/proxy/typed-handler.d.ts.map +1 -0
  253. package/dist/types/src/internal/common/proxy/typed-handler.test.d.ts.map +1 -0
  254. package/dist/types/src/internal/common/proxy/typed-object.test.d.ts.map +1 -0
  255. package/dist/types/src/internal/common/types/base.d.ts.map +1 -0
  256. package/dist/types/src/internal/{types → common/types}/entity.d.ts +4 -4
  257. package/dist/types/src/internal/common/types/entity.d.ts.map +1 -0
  258. package/dist/types/src/internal/{types → common/types}/index.d.ts +1 -0
  259. package/dist/types/src/internal/common/types/index.d.ts.map +1 -0
  260. package/dist/types/src/internal/{types → common/types}/meta.d.ts +13 -3
  261. package/dist/types/src/internal/common/types/meta.d.ts.map +1 -0
  262. package/dist/types/src/internal/common/types/model-symbols.d.ts +54 -0
  263. package/dist/types/src/internal/common/types/model-symbols.d.ts.map +1 -0
  264. package/dist/types/src/internal/common/types/typename.d.ts.map +1 -0
  265. package/dist/types/src/internal/{types → common/types}/version.d.ts +1 -1
  266. package/dist/types/src/internal/common/types/version.d.ts.map +1 -0
  267. package/dist/types/src/internal/index.d.ts +9 -10
  268. package/dist/types/src/internal/index.d.ts.map +1 -1
  269. package/dist/types/src/testing/test-data.d.ts +8 -8
  270. package/dist/types/src/testing/test-data.d.ts.map +1 -1
  271. package/dist/types/src/testing/test-schema.d.ts +53 -53
  272. package/dist/types/src/testing/test-schema.d.ts.map +1 -1
  273. package/dist/types/src/testing/util.d.ts.map +1 -1
  274. package/dist/types/tsconfig.tsbuildinfo +1 -1
  275. package/package.json +25 -15
  276. package/src/Annotation.ts +6 -3
  277. package/src/Collection.ts +8 -1
  278. package/src/Database.ts +55 -18
  279. package/src/Dataset.ts +2 -0
  280. package/src/Entity.ts +20 -11
  281. package/src/Err.ts +2 -0
  282. package/src/Extension.test.ts +235 -0
  283. package/src/Extension.ts +122 -0
  284. package/src/Feed.ts +118 -37
  285. package/src/Filter.test.ts +90 -0
  286. package/src/Filter.ts +99 -3
  287. package/src/Format.ts +1 -1
  288. package/src/Hypergraph.ts +10 -5
  289. package/src/Json.test.ts +175 -0
  290. package/src/Json.ts +102 -0
  291. package/src/JsonSchema.ts +7 -5
  292. package/src/Migration.ts +106 -0
  293. package/src/Obj.test.ts +107 -15
  294. package/src/Obj.ts +167 -40
  295. package/src/Order.ts +2 -0
  296. package/src/Query.test.ts +222 -32
  297. package/src/Query.ts +62 -10
  298. package/src/Ref.ts +16 -12
  299. package/src/Relation.test.ts +2 -2
  300. package/src/Relation.ts +28 -21
  301. package/src/SchemaRegistry.ts +1 -1
  302. package/src/Tag.ts +3 -1
  303. package/src/Type.ts +14 -11
  304. package/src/View.ts +9 -2
  305. package/src/exemplars.test.ts +21 -0
  306. package/src/index.ts +4 -0
  307. package/src/internal/{annotations → Annotation}/annotations.test.ts +56 -7
  308. package/src/internal/{annotations → Annotation}/annotations.ts +84 -20
  309. package/src/internal/{annotations → Annotation}/index.ts +1 -0
  310. package/src/internal/{api → Annotation}/sorting.ts +2 -4
  311. package/src/internal/{annotations → Annotation}/util.ts +1 -1
  312. package/src/internal/{api/entity.ts → Entity/api.ts} +3 -2
  313. package/src/internal/{entities → Entity}/entity.ts +2 -2
  314. package/src/internal/{entities → Entity}/index.ts +2 -0
  315. package/src/internal/{entities → Entity}/model.ts +15 -42
  316. package/src/internal/{entities → Entity}/object.ts +2 -3
  317. package/src/internal/{entities → Entity}/relation.ts +19 -36
  318. package/src/internal/{api → Entity}/version.ts +2 -2
  319. package/src/internal/{formats → Format}/date.test.ts +1 -2
  320. package/src/internal/{formats → Format}/format.test.ts +1 -2
  321. package/src/internal/{formats → Format}/types.ts +2 -2
  322. package/src/internal/{json-schema → JsonSchema}/annotations.ts +3 -3
  323. package/src/internal/{json-schema → JsonSchema}/json-schema-type.ts +4 -4
  324. package/src/internal/{json-schema → JsonSchema}/json-schema.test.ts +48 -49
  325. package/src/internal/{json-schema → JsonSchema}/json-schema.ts +3 -4
  326. package/src/internal/{object → Obj}/clone.ts +3 -3
  327. package/src/internal/{object → Obj}/common.ts +2 -2
  328. package/src/internal/{object → Obj}/create-object.test.ts +5 -7
  329. package/src/internal/{object → Obj}/create-object.ts +5 -6
  330. package/src/internal/{object → Obj}/deleted.ts +2 -2
  331. package/src/internal/{object → Obj}/ids.ts +1 -1
  332. package/src/internal/{object → Obj}/inspect.ts +2 -2
  333. package/src/internal/{object → Obj}/json-serializer.test.ts +8 -9
  334. package/src/internal/{object → Obj}/json-serializer.ts +44 -26
  335. package/src/internal/{object → Obj}/schema-validator.ts +1 -1
  336. package/src/internal/{object → Obj}/set-value.test.ts +24 -24
  337. package/src/internal/{object → Obj}/set-value.ts +3 -3
  338. package/src/internal/{object → Obj}/snapshot.ts +14 -5
  339. package/src/internal/{object → Obj}/typed-object.test.ts +3 -3
  340. package/src/internal/{object → Obj}/typed-object.ts +2 -2
  341. package/src/internal/Query.ts +156 -0
  342. package/src/internal/{ref → Ref}/ref-array.ts +1 -2
  343. package/src/internal/{ref → Ref}/ref.test.ts +4 -5
  344. package/src/internal/{ref → Ref}/ref.ts +20 -3
  345. package/src/internal/{schema → Type}/compose.test.ts +5 -6
  346. package/src/internal/{schema → Type}/compose.ts +1 -1
  347. package/src/internal/{schema → Type}/echo-schema.ts +7 -8
  348. package/src/internal/{schema → Type}/manipulation.ts +1 -1
  349. package/src/internal/{schema → Type}/persistent-schema.ts +7 -7
  350. package/src/internal/{README.md → common/README.md} +1 -1
  351. package/src/internal/{api → common/api}/index.ts +0 -4
  352. package/src/internal/{api → common/api}/meta.ts +3 -3
  353. package/src/internal/common/index.ts +7 -0
  354. package/src/internal/{proxy → common/proxy}/change-context.ts +1 -1
  355. package/src/internal/{proxy → common/proxy}/change.test.ts +97 -97
  356. package/src/internal/{proxy → common/proxy}/errors.ts +2 -2
  357. package/src/internal/{proxy → common/proxy}/handler.test.ts +2 -4
  358. package/src/internal/{proxy → common/proxy}/json-serializer.ts +6 -3
  359. package/src/internal/{proxy → common/proxy}/make-object.ts +1 -2
  360. package/src/internal/{proxy → common/proxy}/ownership.ts +0 -1
  361. package/src/internal/{proxy → common/proxy}/reactive-array.ts +1 -1
  362. package/src/internal/common/proxy/reactive.test.ts +54 -0
  363. package/src/internal/{proxy → common/proxy}/reactive.ts +12 -4
  364. package/src/internal/{proxy → common/proxy}/schema.test.ts +10 -10
  365. package/src/internal/{proxy → common/proxy}/typed-handler.test.ts +6 -7
  366. package/src/internal/{proxy → common/proxy}/typed-handler.ts +10 -13
  367. package/src/internal/{proxy → common/proxy}/typed-object.test.ts +5 -6
  368. package/src/internal/{types → common/types}/entity.ts +1 -1
  369. package/src/internal/{types → common/types}/index.ts +1 -0
  370. package/src/internal/{types → common/types}/meta.ts +13 -2
  371. package/src/internal/common/types/model-symbols.ts +69 -0
  372. package/src/internal/index.ts +9 -31
  373. package/src/testing/api.test.ts +2 -3
  374. package/src/testing/test-data.ts +157 -98
  375. package/src/testing/test-schema.ts +9 -9
  376. package/dist/lib/neutral/chunk-5ELDDYWE.mjs.map +0 -7
  377. package/dist/lib/neutral/chunk-ALOGSVBP.mjs.map +0 -7
  378. package/dist/lib/neutral/chunk-AQP4QKYP.mjs.map +0 -7
  379. package/dist/lib/neutral/chunk-BRJSLACP.mjs.map +0 -7
  380. package/dist/lib/neutral/chunk-F7KMHDPJ.mjs.map +0 -7
  381. package/dist/lib/neutral/chunk-GAWKQ5DZ.mjs.map +0 -7
  382. package/dist/lib/neutral/chunk-GFNCSK7F.mjs +0 -101
  383. package/dist/lib/neutral/chunk-GFNCSK7F.mjs.map +0 -7
  384. package/dist/lib/neutral/chunk-KGV3QIRX.mjs.map +0 -7
  385. package/dist/lib/neutral/chunk-M2KVTHZM.mjs.map +0 -7
  386. package/dist/lib/neutral/chunk-MXQJZCPY.mjs.map +0 -7
  387. package/dist/lib/neutral/chunk-O3TZFQCT.mjs.map +0 -7
  388. package/dist/lib/neutral/chunk-OVUBTQLT.mjs.map +0 -7
  389. package/dist/lib/neutral/chunk-PP4DNUZU.mjs.map +0 -7
  390. package/dist/lib/neutral/chunk-QXF3LGN2.mjs.map +0 -7
  391. package/dist/lib/neutral/chunk-T4MPQJ7X.mjs.map +0 -7
  392. package/dist/lib/neutral/chunk-W7OP5HGZ.mjs.map +0 -7
  393. package/dist/lib/neutral/chunk-ZTUBYOGB.mjs.map +0 -7
  394. package/dist/lib/neutral/chunk-ZWKJ4LZJ.mjs.map +0 -7
  395. package/dist/types/src/internal/annotations/annotations.d.ts.map +0 -1
  396. package/dist/types/src/internal/annotations/annotations.test.d.ts.map +0 -1
  397. package/dist/types/src/internal/annotations/index.d.ts +0 -3
  398. package/dist/types/src/internal/annotations/index.d.ts.map +0 -1
  399. package/dist/types/src/internal/annotations/util.d.ts.map +0 -1
  400. package/dist/types/src/internal/api/annotations.d.ts +0 -23
  401. package/dist/types/src/internal/api/annotations.d.ts.map +0 -1
  402. package/dist/types/src/internal/api/entity.d.ts.map +0 -1
  403. package/dist/types/src/internal/api/index.d.ts.map +0 -1
  404. package/dist/types/src/internal/api/meta.d.ts.map +0 -1
  405. package/dist/types/src/internal/api/sorting.d.ts.map +0 -1
  406. package/dist/types/src/internal/api/version.d.ts.map +0 -1
  407. package/dist/types/src/internal/entities/entity.d.ts.map +0 -1
  408. package/dist/types/src/internal/entities/index.d.ts.map +0 -1
  409. package/dist/types/src/internal/entities/model.d.ts.map +0 -1
  410. package/dist/types/src/internal/entities/object.d.ts.map +0 -1
  411. package/dist/types/src/internal/entities/relation.d.ts.map +0 -1
  412. package/dist/types/src/internal/entities/util.d.ts.map +0 -1
  413. package/dist/types/src/internal/formats/date.d.ts.map +0 -1
  414. package/dist/types/src/internal/formats/date.test.d.ts.map +0 -1
  415. package/dist/types/src/internal/formats/format.d.ts.map +0 -1
  416. package/dist/types/src/internal/formats/format.test.d.ts.map +0 -1
  417. package/dist/types/src/internal/formats/index.d.ts.map +0 -1
  418. package/dist/types/src/internal/formats/number.d.ts.map +0 -1
  419. package/dist/types/src/internal/formats/object.d.ts.map +0 -1
  420. package/dist/types/src/internal/formats/select.d.ts.map +0 -1
  421. package/dist/types/src/internal/formats/string.d.ts.map +0 -1
  422. package/dist/types/src/internal/formats/types.d.ts.map +0 -1
  423. package/dist/types/src/internal/json-schema/annotations.d.ts.map +0 -1
  424. package/dist/types/src/internal/json-schema/effect-schema.test.d.ts.map +0 -1
  425. package/dist/types/src/internal/json-schema/index.d.ts.map +0 -1
  426. package/dist/types/src/internal/json-schema/json-schema-normalize.d.ts.map +0 -1
  427. package/dist/types/src/internal/json-schema/json-schema-type.d.ts.map +0 -1
  428. package/dist/types/src/internal/json-schema/json-schema.d.ts.map +0 -1
  429. package/dist/types/src/internal/json-schema/json-schema.test.d.ts.map +0 -1
  430. package/dist/types/src/internal/object/clone.d.ts.map +0 -1
  431. package/dist/types/src/internal/object/common.d.ts.map +0 -1
  432. package/dist/types/src/internal/object/create-object.d.ts.map +0 -1
  433. package/dist/types/src/internal/object/create-object.test.d.ts.map +0 -1
  434. package/dist/types/src/internal/object/deleted.d.ts.map +0 -1
  435. package/dist/types/src/internal/object/ids.d.ts.map +0 -1
  436. package/dist/types/src/internal/object/index.d.ts.map +0 -1
  437. package/dist/types/src/internal/object/inspect.d.ts.map +0 -1
  438. package/dist/types/src/internal/object/json-serializer.d.ts.map +0 -1
  439. package/dist/types/src/internal/object/json-serializer.test.d.ts.map +0 -1
  440. package/dist/types/src/internal/object/schema-validator.d.ts +0 -2
  441. package/dist/types/src/internal/object/schema-validator.d.ts.map +0 -1
  442. package/dist/types/src/internal/object/schema-validator.test.d.ts.map +0 -1
  443. package/dist/types/src/internal/object/set-value.d.ts.map +0 -1
  444. package/dist/types/src/internal/object/set-value.test.d.ts.map +0 -1
  445. package/dist/types/src/internal/object/snapshot.d.ts.map +0 -1
  446. package/dist/types/src/internal/object/typed-object.d.ts.map +0 -1
  447. package/dist/types/src/internal/object/typed-object.test.d.ts.map +0 -1
  448. package/dist/types/src/internal/proxy/change-context.d.ts.map +0 -1
  449. package/dist/types/src/internal/proxy/change.test.d.ts.map +0 -1
  450. package/dist/types/src/internal/proxy/define-hidden-property.d.ts.map +0 -1
  451. package/dist/types/src/internal/proxy/errors.d.ts.map +0 -1
  452. package/dist/types/src/internal/proxy/event-batch.d.ts.map +0 -1
  453. package/dist/types/src/internal/proxy/handler.test.d.ts.map +0 -1
  454. package/dist/types/src/internal/proxy/index.d.ts.map +0 -1
  455. package/dist/types/src/internal/proxy/json-serializer.d.ts.map +0 -1
  456. package/dist/types/src/internal/proxy/make-object.d.ts.map +0 -1
  457. package/dist/types/src/internal/proxy/ownership.d.ts.map +0 -1
  458. package/dist/types/src/internal/proxy/proxy-types.d.ts.map +0 -1
  459. package/dist/types/src/internal/proxy/proxy-utils.d.ts.map +0 -1
  460. package/dist/types/src/internal/proxy/reactive-array.d.ts.map +0 -1
  461. package/dist/types/src/internal/proxy/schema-validator.d.ts.map +0 -1
  462. package/dist/types/src/internal/proxy/schema.test.d.ts.map +0 -1
  463. package/dist/types/src/internal/proxy/symbols.d.ts.map +0 -1
  464. package/dist/types/src/internal/proxy/typed-handler.d.ts.map +0 -1
  465. package/dist/types/src/internal/proxy/typed-handler.test.d.ts.map +0 -1
  466. package/dist/types/src/internal/proxy/typed-object.test.d.ts.map +0 -1
  467. package/dist/types/src/internal/ref/index.d.ts.map +0 -1
  468. package/dist/types/src/internal/ref/ref-array.d.ts.map +0 -1
  469. package/dist/types/src/internal/ref/ref.d.ts.map +0 -1
  470. package/dist/types/src/internal/ref/ref.test.d.ts.map +0 -1
  471. package/dist/types/src/internal/schema/compose.d.ts.map +0 -1
  472. package/dist/types/src/internal/schema/compose.test.d.ts.map +0 -1
  473. package/dist/types/src/internal/schema/echo-schema.d.ts.map +0 -1
  474. package/dist/types/src/internal/schema/index.d.ts.map +0 -1
  475. package/dist/types/src/internal/schema/manipulation.d.ts.map +0 -1
  476. package/dist/types/src/internal/schema/persistent-schema.d.ts.map +0 -1
  477. package/dist/types/src/internal/types/base.d.ts.map +0 -1
  478. package/dist/types/src/internal/types/entity.d.ts.map +0 -1
  479. package/dist/types/src/internal/types/index.d.ts.map +0 -1
  480. package/dist/types/src/internal/types/meta.d.ts.map +0 -1
  481. package/dist/types/src/internal/types/typename.d.ts.map +0 -1
  482. package/dist/types/src/internal/types/version.d.ts.map +0 -1
  483. package/src/internal/api/annotations.ts +0 -60
  484. /package/dist/lib/neutral/{chunk-ANHVGJI4.mjs.map → chunk-7RVZT53K.mjs.map} +0 -0
  485. /package/dist/lib/neutral/{chunk-OMUPQMLR.mjs.map → chunk-V72DY6LU.mjs.map} +0 -0
  486. /package/dist/types/src/internal/{annotations → Annotation}/annotations.test.d.ts +0 -0
  487. /package/dist/types/src/internal/{entities → Entity}/util.d.ts +0 -0
  488. /package/dist/types/src/internal/{formats → Format}/date.d.ts +0 -0
  489. /package/dist/types/src/internal/{formats → Format}/date.test.d.ts +0 -0
  490. /package/dist/types/src/internal/{formats → Format}/format.d.ts +0 -0
  491. /package/dist/types/src/internal/{formats → Format}/format.test.d.ts +0 -0
  492. /package/dist/types/src/internal/{formats → Format}/index.d.ts +0 -0
  493. /package/dist/types/src/internal/{formats → Format}/number.d.ts +0 -0
  494. /package/dist/types/src/internal/{formats → Format}/object.d.ts +0 -0
  495. /package/dist/types/src/internal/{formats → Format}/select.d.ts +0 -0
  496. /package/dist/types/src/internal/{formats → Format}/string.d.ts +0 -0
  497. /package/dist/types/src/internal/{json-schema → JsonSchema}/effect-schema.test.d.ts +0 -0
  498. /package/dist/types/src/internal/{json-schema → JsonSchema}/index.d.ts +0 -0
  499. /package/dist/types/src/internal/{json-schema → JsonSchema}/json-schema-normalize.d.ts +0 -0
  500. /package/dist/types/src/internal/{json-schema → JsonSchema}/json-schema.test.d.ts +0 -0
  501. /package/dist/types/src/internal/{object → Obj}/clone.d.ts +0 -0
  502. /package/dist/types/src/internal/{object → Obj}/common.d.ts +0 -0
  503. /package/dist/types/src/internal/{object → Obj}/create-object.test.d.ts +0 -0
  504. /package/dist/types/src/internal/{object → Obj}/index.d.ts +0 -0
  505. /package/dist/types/src/internal/{object → Obj}/inspect.d.ts +0 -0
  506. /package/dist/types/src/internal/{object → Obj}/json-serializer.test.d.ts +0 -0
  507. /package/dist/types/src/internal/{object → Obj}/schema-validator.test.d.ts +0 -0
  508. /package/dist/types/src/internal/{object → Obj}/set-value.test.d.ts +0 -0
  509. /package/dist/types/src/internal/{object → Obj}/snapshot.d.ts +0 -0
  510. /package/dist/types/src/internal/{object → Obj}/typed-object.test.d.ts +0 -0
  511. /package/dist/types/src/internal/{ref → Ref}/index.d.ts +0 -0
  512. /package/dist/types/src/internal/{ref → Ref}/ref.test.d.ts +0 -0
  513. /package/dist/types/src/internal/{schema → Type}/compose.test.d.ts +0 -0
  514. /package/dist/types/src/internal/{schema → Type}/index.d.ts +0 -0
  515. /package/dist/types/src/internal/{schema → Type}/manipulation.d.ts +0 -0
  516. /package/dist/types/src/internal/{proxy → common/proxy}/change.test.d.ts +0 -0
  517. /package/dist/types/src/internal/{proxy → common/proxy}/define-hidden-property.d.ts +0 -0
  518. /package/dist/types/src/internal/{proxy → common/proxy}/event-batch.d.ts +0 -0
  519. /package/dist/types/src/internal/{proxy → common/proxy}/handler.test.d.ts +0 -0
  520. /package/dist/types/src/internal/{proxy → common/proxy}/index.d.ts +0 -0
  521. /package/dist/types/src/internal/{proxy → common/proxy}/json-serializer.d.ts +0 -0
  522. /package/dist/types/src/internal/{proxy → common/proxy}/make-object.d.ts +0 -0
  523. /package/dist/types/src/internal/{proxy → common/proxy}/ownership.d.ts +0 -0
  524. /package/dist/types/src/internal/{proxy → common/proxy}/proxy-types.d.ts +0 -0
  525. /package/dist/types/src/internal/{proxy → common/proxy}/proxy-utils.d.ts +0 -0
  526. /package/dist/types/src/internal/{proxy → common/proxy}/schema-validator.d.ts +0 -0
  527. /package/dist/types/src/internal/{proxy → common/proxy}/schema.test.d.ts +0 -0
  528. /package/dist/types/src/internal/{proxy → common/proxy}/symbols.d.ts +0 -0
  529. /package/dist/types/src/internal/{proxy → common/proxy}/typed-handler.d.ts +0 -0
  530. /package/dist/types/src/internal/{proxy → common/proxy}/typed-handler.test.d.ts +0 -0
  531. /package/dist/types/src/internal/{proxy → common/proxy}/typed-object.test.d.ts +0 -0
  532. /package/dist/types/src/internal/{types → common/types}/base.d.ts +0 -0
  533. /package/dist/types/src/internal/{types → common/types}/typename.d.ts +0 -0
  534. /package/src/internal/{entities → Entity}/util.ts +0 -0
  535. /package/src/internal/{formats → Format}/date.ts +0 -0
  536. /package/src/internal/{formats → Format}/format.ts +0 -0
  537. /package/src/internal/{formats → Format}/index.ts +0 -0
  538. /package/src/internal/{formats → Format}/number.ts +0 -0
  539. /package/src/internal/{formats → Format}/object.ts +0 -0
  540. /package/src/internal/{formats → Format}/select.ts +0 -0
  541. /package/src/internal/{formats → Format}/string.ts +0 -0
  542. /package/src/internal/{json-schema → JsonSchema}/effect-schema.test.ts +0 -0
  543. /package/src/internal/{json-schema → JsonSchema}/index.ts +0 -0
  544. /package/src/internal/{json-schema → JsonSchema}/json-schema-normalize.ts +0 -0
  545. /package/src/internal/{object → Obj}/index.ts +0 -0
  546. /package/src/internal/{object → Obj}/schema-validator.test.ts +0 -0
  547. /package/src/internal/{ref → Ref}/index.ts +0 -0
  548. /package/src/internal/{schema → Type}/index.ts +0 -0
  549. /package/src/internal/{proxy → common/proxy}/define-hidden-property.ts +0 -0
  550. /package/src/internal/{proxy → common/proxy}/event-batch.ts +0 -0
  551. /package/src/internal/{proxy → common/proxy}/index.ts +0 -0
  552. /package/src/internal/{proxy → common/proxy}/proxy-types.ts +0 -0
  553. /package/src/internal/{proxy → common/proxy}/proxy-utils.ts +0 -0
  554. /package/src/internal/{proxy → common/proxy}/schema-validator.ts +0 -0
  555. /package/src/internal/{proxy → common/proxy}/symbols.ts +0 -0
  556. /package/src/internal/{types → common/types}/base.ts +0 -0
  557. /package/src/internal/{types → common/types}/typename.ts +0 -0
  558. /package/src/internal/{types → common/types}/version.ts +0 -0
@@ -0,0 +1,122 @@
1
+ // @import-as-namespace
2
+ //
3
+ // Copyright 2026 DXOS.org
4
+ //
5
+
6
+ import * as Function from 'effect/Function';
7
+ import * as Option from 'effect/Option';
8
+ import * as Schema from 'effect/Schema';
9
+ import type * as Types from 'effect/Types';
10
+
11
+ // @import-as-namespace
12
+
13
+ /**
14
+ * Extensions allow objects to contain typed properties that are not part of the schema.
15
+ */
16
+
17
+ export const TypeId = '~@dxos/echo/Extension' as const;
18
+ export type TypeId = typeof TypeId;
19
+
20
+ export interface Extension<T> extends Record<
21
+ TypeId,
22
+ {
23
+ _Type: T;
24
+ }
25
+ > {
26
+ readonly [TypeId]: {
27
+ _Type: T;
28
+ };
29
+
30
+ readonly key: Key;
31
+ readonly valueSchema: Schema.Schema<T>;
32
+ }
33
+
34
+ /**
35
+ * Create a new typed extension.
36
+ *
37
+ * ```ts
38
+ * const ColorExtension = Extension.make('color', Schema.String);
39
+ *
40
+ * const obj = Obj.make(Person, {
41
+ * [Obj.Meta]: { keys: [{ source: 'external', id: '123' }] },
42
+ * name: 'John',
43
+ * email: 'john@example.com',
44
+ * });
45
+ *
46
+ * Obj.update(obj, (obj) => {
47
+ * Extension.set(obj.extensions, ColorExtension, 'red');
48
+ * });
49
+ *
50
+ * console.log(Extension.get(obj.extensions, ColorExtension)); // 'red'
51
+ * ```
52
+ */
53
+ export const make = <S extends Schema.Schema.AnyNoContext>(
54
+ key: string,
55
+ valueSchema: S,
56
+ ): Extension<Schema.Schema.Type<S>> => {
57
+ return {
58
+ [TypeId]: {} as any,
59
+ key: Key.make(key),
60
+ valueSchema,
61
+ };
62
+ };
63
+
64
+ /**
65
+ * Unique identifier for an extension.
66
+ */
67
+ // TODO(dmaretskyi): filter to be fully qualified: (e.g., org.dxos.extension.color)
68
+ export const Key = Schema.String.pipe(Schema.brand('~@dxos/echo/ExtensionKey'));
69
+ export type Key = Schema.Schema.Type<typeof Key>;
70
+
71
+ /**
72
+ * Set of extension values.
73
+ *
74
+ * Can be used inside an object/relation schema:
75
+ *
76
+ * ```ts
77
+ * const Person = Schema.Struct({
78
+ * name: Schema.String,
79
+ * extensions: Extension.Values,
80
+ * });
81
+ * ```
82
+ */
83
+ export const Values = Schema.Record({ key: Key, value: Schema.Unknown });
84
+ export interface Values extends Schema.Schema.Type<typeof Values> {}
85
+
86
+ /**
87
+ * Get the value of an extension from a set of values.
88
+ */
89
+ export const get: {
90
+ <T>(extension: Extension<T>): (values: Values) => Option.Option<T>;
91
+ <T>(values: Values, extension: Extension<T>): Option.Option<T>;
92
+ } = Function.dual<
93
+ <T>(extension: Extension<T>) => (values: Values) => Option.Option<T>,
94
+ <T>(values: Values, extension: Extension<T>) => Option.Option<T>
95
+ >(2, (values, extension) => {
96
+ if (!(extension.key in values)) {
97
+ return Option.none();
98
+ }
99
+
100
+ return Function.pipe(values[extension.key], Schema.decodeUnknownSync(extension.valueSchema), Option.some);
101
+ });
102
+
103
+ /**
104
+ * Set the value of an extension in a set of values.
105
+ *
106
+ * Can also be used within Obj.update callback:
107
+ *
108
+ * ```ts
109
+ * Obj.update(obj, (obj) => {
110
+ * Extension.set(obj.extensions, ColorExtension, 'red');
111
+ * });
112
+ * ```
113
+ */
114
+ export const set: {
115
+ <T>(extension: Extension<T>, value: T): (values: Values) => void;
116
+ <T>(values: Types.Mutable<Values>, extension: Extension<T>, value: T): void;
117
+ } = Function.dual<
118
+ <T>(extension: Extension<T>, value: T) => (values: Values) => void,
119
+ <T>(values: Types.Mutable<Values>, extension: Extension<T>, value: T) => void
120
+ >(3, (values, extension, value) => {
121
+ values[extension.key] = Schema.encodeSync(extension.valueSchema)(value);
122
+ });
package/src/Feed.ts CHANGED
@@ -2,6 +2,8 @@
2
2
  // Copyright 2025 DXOS.org
3
3
  //
4
4
 
5
+ // @import-as-namespace
6
+
5
7
  import * as Context from 'effect/Context';
6
8
  import * as Effect from 'effect/Effect';
7
9
  import * as Layer from 'effect/Layer';
@@ -10,6 +12,7 @@ import * as Schema from 'effect/Schema';
10
12
 
11
13
  import { DXN } from '@dxos/keys';
12
14
 
15
+ import * as Annotation from './Annotation';
13
16
  import type * as Entity from './Entity';
14
17
  import type * as Filter from './Filter';
15
18
  import * as internal from './internal';
@@ -23,7 +26,7 @@ import * as Type from './Type';
23
26
  *
24
27
  * @example
25
28
  * ```ts
26
- * const feed = Obj.make(Feed.Feed, { name: 'notifications', kind: 'dxos.org/plugin/notifications/v1' });
29
+ * const feed = Obj.make(Feed.Feed, { name: 'notifications', kind: 'org.dxos.plugin.notifications.v1' });
27
30
  * ```
28
31
  */
29
32
  export const Feed = Schema.Struct({
@@ -31,11 +34,24 @@ export const Feed = Schema.Struct({
31
34
  name: Schema.String.pipe(Schema.optional),
32
35
  /** Identifier for the feed's kind (e.g., plugin id). */
33
36
  kind: Schema.String.pipe(internal.FormInputAnnotation.set(false), Schema.optional),
37
+
38
+ /**
39
+ * Feed namespace.
40
+ * Controls how feed data is stored and replicated.
41
+ * - `data`: Data feed (default).
42
+ * - `trace`: Trace feed.
43
+ */
44
+ namespace: Schema.optional(Schema.Literal('data', 'trace')),
34
45
  }).pipe(
35
46
  Type.object({
36
- typename: 'dxos.org/type/Feed',
47
+ typename: 'org.dxos.type.feed',
37
48
  version: '0.1.0',
38
49
  }),
50
+ internal.SystemTypeAnnotation.set(true),
51
+ Annotation.IconAnnotation.set({
52
+ icon: 'ph--rows--regular',
53
+ hue: 'yellow',
54
+ }),
39
55
  );
40
56
 
41
57
  /**
@@ -47,12 +63,6 @@ export interface Feed extends Schema.Schema.Type<typeof Feed> {}
47
63
  // Types
48
64
  //
49
65
 
50
- /**
51
- * Meta key source for storing the backing DXN bound to a feed object.
52
- */
53
- // TODO(dmaretskyi): Enforce that Feed ObjectId = feed storage ID. And remove this key.
54
- export const DXN_KEY = 'dxos.org/key/feed';
55
-
56
66
  /**
57
67
  * Opaque cursor for iterating over feed items.
58
68
  */
@@ -70,6 +80,16 @@ export interface RetentionOptions {
70
80
  cursor?: string;
71
81
  }
72
82
 
83
+ /**
84
+ * Sync options for a feed.
85
+ */
86
+ export interface SyncOptions {
87
+ /** Push local changes to the server. Defaults to true. */
88
+ shouldPush?: boolean;
89
+ /** Pull remote changes from the server. Defaults to true. */
90
+ shouldPull?: boolean;
91
+ }
92
+
73
93
  //
74
94
  // Factory
75
95
  //
@@ -79,21 +99,24 @@ export interface RetentionOptions {
79
99
  *
80
100
  * @example
81
101
  * ```ts
82
- * const feed = Feed.make({ name: 'notifications', kind: 'dxos.org/plugin/notifications/v1' });
102
+ * const feed = Feed.make({ name: 'notifications', kind: 'org.dxos.plugin.notifications.v1' });
83
103
  * ```
84
104
  */
85
105
  // TODO(wittjosiah): How to control the feed namespace (data/trace)? Why do feeds have namespaces?
86
- export const make = (props: Obj.MakeProps<typeof Feed>): Feed => Obj.make(Feed, props);
106
+ export const make = (props: Obj.MakeProps<typeof Feed> = {}): Feed => Obj.make(Feed, props);
87
107
 
88
108
  /**
89
- * Reads the queue DXN from feed metadata.
109
+ * Derives the queue DXN from the feed object's DXN.
110
+ * Returns `undefined` when the feed is not stored in a space yet.
90
111
  *
91
- * @deprecated
112
+ * Used internally by the feed service layer.
92
113
  */
93
- // TODO(wittjosiah): Align backing feed dxn's with object DXN.
94
114
  export const getQueueDxn = (feed: Feed): DXN | undefined => {
95
- const keys = Obj.getKeys(feed, DXN_KEY);
96
- return keys.length === 0 ? undefined : DXN.parse(keys[0].id);
115
+ const self = Obj.getDXN(feed).asEchoDXN();
116
+ if (!self || !self.spaceId) {
117
+ return undefined;
118
+ }
119
+ return new DXN(DXN.kind.QUEUE, [feed.namespace ?? 'data', self.spaceId, self.echoId]);
97
120
  };
98
121
 
99
122
  //
@@ -105,8 +128,8 @@ export const getQueueDxn = (feed: Feed): DXN | undefined => {
105
128
  * Provides the bridge to the underlying storage implementation.
106
129
  * Must be provided by the application layer (e.g., echo-db).
107
130
  */
108
- export class Service extends Context.Tag('@dxos/echo/Feed/Service')<
109
- Service,
131
+ export class FeedService extends Context.Tag('@dxos/echo/Feed/FeedService')<
132
+ FeedService,
110
133
  {
111
134
  /**
112
135
  * Appends items to a feed.
@@ -126,24 +149,61 @@ export class Service extends Context.Tag('@dxos/echo/Feed/Service')<
126
149
  <Q extends Query.Any>(feed: Feed, query: Q): QueryResult.QueryResult<Query.Type<Q>>;
127
150
  <F extends Filter.Any>(feed: Feed, filter: F): QueryResult.QueryResult<Filter.Type<F>>;
128
151
  };
152
+
153
+ /**
154
+ * Syncs the feed with the server.
155
+ */
156
+ sync(feed: Feed, options?: SyncOptions): Promise<void>;
129
157
  }
130
158
  >() {}
131
159
 
160
+ /**
161
+ * @deprecated Use `FeedService` instead.
162
+ */
163
+ export type Service = FeedService;
164
+
165
+ /**
166
+ * @deprecated Use `FeedService` instead.
167
+ */
168
+ export const Service = FeedService;
169
+
132
170
  /**
133
171
  * Layer that provides a Feed service that throws when accessed.
134
172
  * Useful as a default layer when no feed service is available.
135
173
  */
136
- export const notAvailable: Layer.Layer<Service> = Layer.succeed(Service, {
174
+ export const notAvailable: Layer.Layer<FeedService> = Layer.succeed(FeedService, {
137
175
  append: () => {
138
- throw new Error('Feed.Service not available');
176
+ throw new Error('Feed.FeedService not available');
139
177
  },
140
178
  remove: () => {
141
- throw new Error('Feed.Service not available');
179
+ throw new Error('Feed.FeedService not available');
142
180
  },
143
181
  query: () => {
144
- throw new Error('Feed.Service not available');
182
+ throw new Error('Feed.FeedService not available');
145
183
  },
146
- } as Context.Tag.Service<Service>);
184
+ sync: () => {
185
+ throw new Error('Feed.FeedService not available');
186
+ },
187
+ } as Context.Tag.Service<FeedService>);
188
+
189
+ //
190
+ // Context (per-call) service
191
+ //
192
+
193
+ /**
194
+ * Effect service exposing a single `Feed.Feed` as the "current" feed for a call site.
195
+ *
196
+ * @deprecated Prefer threading a `Feed.Feed` explicitly through function signatures
197
+ * over hiding it behind a context service.
198
+ */
199
+ export class ContextFeedService extends Context.Tag('@dxos/echo/Feed/ContextFeedService')<
200
+ ContextFeedService,
201
+ {
202
+ readonly feed: Feed;
203
+ }
204
+ >() {
205
+ static layer = (feed: Feed): Layer.Layer<ContextFeedService> => Layer.succeed(ContextFeedService, { feed });
206
+ }
147
207
 
148
208
  //
149
209
  // Operations
@@ -157,9 +217,9 @@ export const notAvailable: Layer.Layer<Service> = Layer.succeed(Service, {
157
217
  * yield* Feed.append(feed, [Obj.make(Notification, { title: 'Hello' })]);
158
218
  * ```
159
219
  */
160
- export const append = (feed: Feed, items: Entity.Unknown[]): Effect.Effect<void, never, Service> =>
220
+ export const append = (feed: Feed, items: Entity.Unknown[]): Effect.Effect<void, never, FeedService> =>
161
221
  Effect.gen(function* () {
162
- const service = yield* Service;
222
+ const service = yield* FeedService;
163
223
  yield* Effect.promise(() => service.append(feed, items));
164
224
  });
165
225
 
@@ -172,9 +232,9 @@ export const append = (feed: Feed, items: Entity.Unknown[]): Effect.Effect<void,
172
232
  * ```
173
233
  */
174
234
  // TODO(dmaretskyi): Should we allow snapshots here? - what does it mean to remove a snapshot?
175
- export const remove = (feed: Feed, items: (Entity.Unknown | Obj.Snapshot)[]): Effect.Effect<void, never, Service> =>
235
+ export const remove = (feed: Feed, items: (Entity.Unknown | Obj.Snapshot)[]): Effect.Effect<void, never, FeedService> =>
176
236
  Effect.gen(function* () {
177
- const service = yield* Service;
237
+ const service = yield* FeedService;
178
238
  const ids = items.map((item) => item.id);
179
239
  yield* Effect.promise(() => service.remove(feed, ids));
180
240
  });
@@ -193,10 +253,16 @@ export const remove = (feed: Feed, items: (Entity.Unknown | Obj.Snapshot)[]): Ef
193
253
  // const object = yield* feed.pipe(Feed.query(Filter.type(Person))).first;
194
254
  // ... unify for Database and schema queries.
195
255
  export const query: {
196
- <Q extends Query.Any>(feed: Feed, query: Q): Effect.Effect<QueryResult.QueryResult<Query.Type<Q>>, never, Service>;
197
- <F extends Filter.Any>(feed: Feed, filter: F): Effect.Effect<QueryResult.QueryResult<Filter.Type<F>>, never, Service>;
256
+ <Q extends Query.Any>(
257
+ feed: Feed,
258
+ query: Q,
259
+ ): Effect.Effect<QueryResult.QueryResult<Query.Type<Q>>, never, FeedService>;
260
+ <F extends Filter.Any>(
261
+ feed: Feed,
262
+ filter: F,
263
+ ): Effect.Effect<QueryResult.QueryResult<Filter.Type<F>>, never, FeedService>;
198
264
  } = (feed: Feed, queryOrFilter: Query.Any | Filter.Any) =>
199
- Service.pipe(Effect.map((service) => service.query(feed, queryOrFilter as any) as QueryResult.QueryResult<any>));
265
+ FeedService.pipe(Effect.map((service) => service.query(feed, queryOrFilter as any) as QueryResult.QueryResult<any>));
200
266
 
201
267
  /**
202
268
  * Executes a feed query once and returns the results.
@@ -207,11 +273,26 @@ export const query: {
207
273
  * ```
208
274
  */
209
275
  export const runQuery: {
210
- <Q extends Query.Any>(feed: Feed, query: Q): Effect.Effect<Query.Type<Q>[], never, Service>;
211
- <F extends Filter.Any>(feed: Feed, filter: F): Effect.Effect<Filter.Type<F>[], never, Service>;
276
+ <Q extends Query.Any>(feed: Feed, query: Q): Effect.Effect<Query.Type<Q>[], never, FeedService>;
277
+ <F extends Filter.Any>(feed: Feed, filter: F): Effect.Effect<Filter.Type<F>[], never, FeedService>;
212
278
  } = (feed: Feed, queryOrFilter: Query.Any | Filter.Any) =>
213
279
  query(feed, queryOrFilter as any).pipe(Effect.flatMap((queryResult) => Effect.promise(() => queryResult.run())));
214
280
 
281
+ /**
282
+ * Syncs the feed with the server.
283
+ *
284
+ * @example
285
+ * ```ts
286
+ * yield* Feed.sync(feed);
287
+ * yield* Feed.sync(feed, { shouldPush: false });
288
+ * ```
289
+ */
290
+ export const sync = (feed: Feed, options?: SyncOptions): Effect.Effect<void, never, FeedService> =>
291
+ Effect.gen(function* () {
292
+ const service = yield* FeedService;
293
+ yield* Effect.promise(() => service.sync(feed, options));
294
+ });
295
+
215
296
  /**
216
297
  * Creates a cursor for iterating over feed items.
217
298
  * Currently stubbed — cursor operations are not yet implemented.
@@ -223,32 +304,32 @@ export const runQuery: {
223
304
  * ```
224
305
  */
225
306
  // TODO(wittjosiah): Implement cursor operations. Use Effect streams?
226
- export const cursor = <T = Obj.Snapshot>(_feed: Feed): Effect.Effect<Cursor<T>, never, Service> =>
307
+ export const cursor = <T = Obj.Snapshot>(_feed: Feed): Effect.Effect<Cursor<T>, never, FeedService> =>
227
308
  Effect.succeed({ _tag: 'Cursor' } as Cursor<T>);
228
309
 
229
310
  /**
230
311
  * Returns the next item from a feed cursor.
231
312
  * Currently stubbed — cursor operations are not yet implemented.
232
313
  */
233
- export const next = <T = Obj.Snapshot>(_cursor: Cursor<T>): Effect.Effect<T, never, Service> =>
314
+ export const next = <T = Obj.Snapshot>(_cursor: Cursor<T>): Effect.Effect<T, never, FeedService> =>
234
315
  Effect.die('Feed.next is not yet implemented');
235
316
 
236
317
  /**
237
318
  * Returns the next item from a feed cursor as an Option.
238
319
  * Currently stubbed — cursor operations are not yet implemented.
239
320
  */
240
- export const nextOption = <T = Obj.Snapshot>(_cursor: Cursor<T>): Effect.Effect<Option.Option<T>, never, Service> =>
321
+ export const nextOption = <T = Obj.Snapshot>(_cursor: Cursor<T>): Effect.Effect<Option.Option<T>, never, FeedService> =>
241
322
  Effect.die('Feed.nextOption is not yet implemented');
242
323
 
243
324
  /**
244
325
  * Sets the local retention policy for a feed.
245
- * Currently stubbed — queues do not yet support retention.
326
+ * Currently stubbed — feeds do not yet support retention.
246
327
  *
247
328
  * @example
248
329
  * ```ts
249
330
  * yield* Feed.setRetention(feed, { count: 1000 });
250
331
  * ```
251
332
  */
252
- // TODO(feed): Implement when queue retention is supported.
253
- export const setRetention = (_feed: Feed, _options: RetentionOptions): Effect.Effect<void, never, Service> =>
333
+ // TODO(dmaretskyi): Implement when feed retention is supported.
334
+ export const setRetention = (_feed: Feed, _options: RetentionOptions): Effect.Effect<void, never, FeedService> =>
254
335
  Effect.void;
@@ -0,0 +1,90 @@
1
+ //
2
+ // Copyright 2026 DXOS.org
3
+ //
4
+
5
+ import { describe, expect, test } from 'vitest';
6
+
7
+ import * as Filter from './Filter';
8
+
9
+ describe('Filter timestamp builders', () => {
10
+ test('updated({ after }) produces correct AST', () => {
11
+ const ts = 1700000000000;
12
+ const f = Filter.updated({ after: ts });
13
+ expect(f.ast).toEqual({ type: 'timestamp', field: 'updatedAt', operator: 'gte', value: ts });
14
+ });
15
+
16
+ test('updated({ before }) produces correct AST', () => {
17
+ const ts = 1700000000000;
18
+ const f = Filter.updated({ before: ts });
19
+ expect(f.ast).toEqual({ type: 'timestamp', field: 'updatedAt', operator: 'lte', value: ts });
20
+ });
21
+
22
+ test('created({ after }) produces correct AST', () => {
23
+ const ts = 1700000000000;
24
+ const f = Filter.created({ after: ts });
25
+ expect(f.ast).toEqual({ type: 'timestamp', field: 'createdAt', operator: 'gte', value: ts });
26
+ });
27
+
28
+ test('created({ before }) produces correct AST', () => {
29
+ const ts = 1700000000000;
30
+ const f = Filter.created({ before: ts });
31
+ expect(f.ast).toEqual({ type: 'timestamp', field: 'createdAt', operator: 'lte', value: ts });
32
+ });
33
+
34
+ test('updated() accepts Date objects', () => {
35
+ const date = new Date('2026-03-20T21:00:00Z');
36
+ const f = Filter.updated({ after: date });
37
+ expect(f.ast).toEqual({ type: 'timestamp', field: 'updatedAt', operator: 'gte', value: date.getTime() });
38
+ });
39
+
40
+ test('created() accepts Date objects', () => {
41
+ const date = new Date('2026-03-20T09:00:00Z');
42
+ const f = Filter.created({ before: date });
43
+ expect(f.ast).toEqual({ type: 'timestamp', field: 'createdAt', operator: 'lte', value: date.getTime() });
44
+ });
45
+
46
+ test('updated({ after, before }) produces AND of two timestamp nodes', () => {
47
+ const from = 1700000000000;
48
+ const to = 1700086400000;
49
+ const f = Filter.updated({ after: from, before: to });
50
+ expect(f.ast).toEqual({
51
+ type: 'and',
52
+ filters: [
53
+ { type: 'timestamp', field: 'updatedAt', operator: 'gte', value: from },
54
+ { type: 'timestamp', field: 'updatedAt', operator: 'lte', value: to },
55
+ ],
56
+ });
57
+ });
58
+
59
+ test('updated({ after, before }) accepts Date objects', () => {
60
+ const from = new Date('2026-03-19T00:00:00Z');
61
+ const to = new Date('2026-03-20T00:00:00Z');
62
+ const f = Filter.updated({ after: from, before: to });
63
+ expect(f.ast).toEqual({
64
+ type: 'and',
65
+ filters: [
66
+ { type: 'timestamp', field: 'updatedAt', operator: 'gte', value: from.getTime() },
67
+ { type: 'timestamp', field: 'updatedAt', operator: 'lte', value: to.getTime() },
68
+ ],
69
+ });
70
+ });
71
+
72
+ test('timestamp filters compose with and()', () => {
73
+ const typeFilter = Filter.everything();
74
+ const timeFilter = Filter.updated({ after: 1700000000000 });
75
+ const combined = Filter.and(typeFilter, timeFilter);
76
+ expect(combined.ast.type).toBe('and');
77
+ expect((combined.ast as any).filters).toHaveLength(2);
78
+ expect((combined.ast as any).filters[1]).toEqual({
79
+ type: 'timestamp',
80
+ field: 'updatedAt',
81
+ operator: 'gte',
82
+ value: 1700000000000,
83
+ });
84
+ });
85
+
86
+ test('timestamp filters pass the is() check', () => {
87
+ const f = Filter.updated({ after: Date.now() });
88
+ expect(Filter.is(f)).toBe(true);
89
+ });
90
+ });
package/src/Filter.ts CHANGED
@@ -2,6 +2,8 @@
2
2
  // Copyright 2025 DXOS.org
3
3
  //
4
4
 
5
+ // @import-as-namespace
6
+
5
7
  import * as Match from 'effect/Match';
6
8
  import * as Schema from 'effect/Schema';
7
9
  import * as SchemaAST from 'effect/SchemaAST';
@@ -12,11 +14,12 @@ import { assertArgument } from '@dxos/invariant';
12
14
  import { DXN, ObjectId } from '@dxos/keys';
13
15
 
14
16
  import * as internal from './internal';
17
+ import type * as Obj from './Obj';
15
18
  import * as Ref from './Ref';
16
19
 
17
20
  export interface Filter<T> {
18
21
  // TODO(dmaretskyi): See new effect-schema approach to variance.
19
- '~Filter': { value: Types.Contravariant<T> };
22
+ '~Filter': { value: Types.Covariant<T> };
20
23
 
21
24
  ast: QueryAST.Filter;
22
25
  }
@@ -155,6 +158,37 @@ export const tag = (tag: string): Any => {
155
158
  });
156
159
  };
157
160
 
161
+ /**
162
+ * Options for {@link key} filter.
163
+ */
164
+ export type KeyFilterOptions = {
165
+ /**
166
+ * Optional semver range expression (e.g. `^1.2.3`, `~2.0.0`, `>=1.0.0 <2.0.0`).
167
+ * Matches the object's meta `version` field against the range.
168
+ * If omitted, matches any version (including objects with no version).
169
+ */
170
+ version?: string;
171
+ };
172
+
173
+ /**
174
+ * Filter by registry key stored in object meta.
175
+ *
176
+ * @example
177
+ * ```ts
178
+ * Filter.key('org.example.type.foo');
179
+ * Filter.key('org.example.type.foo', { version: '^1.2.3' });
180
+ * ```
181
+ */
182
+ export const key = (key: string, options?: KeyFilterOptions): Any => {
183
+ return new FilterClass({
184
+ type: 'object',
185
+ typename: null,
186
+ props: {},
187
+ metaKey: key,
188
+ metaVersion: options?.version,
189
+ });
190
+ };
191
+
158
192
  /**
159
193
  * Filter by properties.
160
194
  */
@@ -276,7 +310,7 @@ export const lte = <T>(value: T): Filter<T | undefined> => {
276
310
  * Predicate for property to be in the provided array.
277
311
  * @param values - Values to check against.
278
312
  */
279
- const in$ = <T>(...values: T[]): Filter<T | undefined> => {
313
+ const in$ = <T>(...values: T[]): Filter<T> => {
280
314
  return new FilterClass({
281
315
  type: 'in',
282
316
  values,
@@ -300,7 +334,7 @@ export const contains = <T>(value: T): Filter<readonly T[] | undefined> => {
300
334
  * @param from - Start of the range (inclusive).
301
335
  * @param to - End of the range (exclusive).
302
336
  */
303
- export const between = <T>(from: T, to: T): Filter<unknown> => {
337
+ export const between = <T>(from: T, to: T): Filter<T> => {
304
338
  return new FilterClass({
305
339
  type: 'range',
306
340
  from,
@@ -308,6 +342,63 @@ export const between = <T>(from: T, to: T): Filter<unknown> => {
308
342
  });
309
343
  };
310
344
 
345
+ type TimeRange = { after?: Date | number; before?: Date | number };
346
+
347
+ const _toUnixMs = (date: Date | number): number => (typeof date === 'number' ? date : date.getTime());
348
+
349
+ const _timeRangeFilter = (field: 'updatedAt' | 'createdAt', range: TimeRange): Any => {
350
+ const filters: Any[] = [];
351
+ if (range.after != null) {
352
+ filters.push(new FilterClass({ type: 'timestamp', field, operator: 'gte', value: _toUnixMs(range.after) }));
353
+ }
354
+ if (range.before != null) {
355
+ filters.push(new FilterClass({ type: 'timestamp', field, operator: 'lte', value: _toUnixMs(range.before) }));
356
+ }
357
+ if (filters.length === 0) {
358
+ return everything();
359
+ }
360
+ return filters.length === 1 ? filters[0] : and(...filters);
361
+ };
362
+
363
+ /**
364
+ * Filter objects by updatedAt timestamp.
365
+ */
366
+ export const updated = (range: TimeRange): Any => _timeRangeFilter('updatedAt', range);
367
+
368
+ /**
369
+ * Filter objects by createdAt timestamp.
370
+ */
371
+ export const created = (range: TimeRange): Any => _timeRangeFilter('createdAt', range);
372
+
373
+ export type ChildOfOptions = {
374
+ /** Whether to match transitively (grandchildren, etc.). Defaults to true. */
375
+ transitive?: boolean;
376
+ };
377
+
378
+ /**
379
+ * Filter objects that are children of the specified parent(s).
380
+ * Accepts ECHO objects, Refs, or arrays of either.
381
+ * Refs are resolved to DXNs without loading; objects use {@link Obj.getDXN}.
382
+ * With transitive=true (default), also matches grandchildren and beyond.
383
+ */
384
+ export const childOf = (
385
+ parents: Obj.Unknown | Ref.Unknown | readonly (Obj.Unknown | Ref.Unknown)[],
386
+ options?: ChildOfOptions,
387
+ ): Any => {
388
+ const items = Array.isArray(parents) ? parents : [parents];
389
+ const dxns = items.map((item) => {
390
+ if (Ref.isRef(item)) {
391
+ return item.dxn.toString();
392
+ }
393
+ return internal.getDXN(item).toString();
394
+ });
395
+ return new FilterClass({
396
+ type: 'child-of',
397
+ parents: dxns,
398
+ transitive: options?.transitive ?? true,
399
+ });
400
+ };
401
+
311
402
  /**
312
403
  * Negate the filter.
313
404
  */
@@ -387,3 +478,8 @@ const processPredicate = (predicate: any): QueryAST.Filter => {
387
478
  Match.orElse((value) => eq(value).ast),
388
479
  );
389
480
  };
481
+
482
+ /**
483
+ * Returns a human-readable string representation of a Filter AST.
484
+ */
485
+ export const pretty = (filter: Any): string => internal.prettyFilter(filter.ast);
package/src/Format.ts CHANGED
@@ -4,6 +4,6 @@
4
4
 
5
5
  import type * as internal from './internal';
6
6
 
7
- export * from './internal/formats';
7
+ export * from './internal/Format';
8
8
 
9
9
  export type Format = internal.TypeFormat;