@fragno-dev/db 0.3.0 → 0.4.1

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 (516) hide show
  1. package/.turbo/turbo-build.log +327 -160
  2. package/CHANGELOG.md +74 -0
  3. package/README.md +24 -0
  4. package/dist/adapters/adapters.d.ts +1 -1
  5. package/dist/adapters/adapters.d.ts.map +1 -1
  6. package/dist/adapters/adapters.js.map +1 -1
  7. package/dist/adapters/generic-sql/generic-sql-adapter.d.ts +0 -3
  8. package/dist/adapters/generic-sql/generic-sql-adapter.d.ts.map +1 -1
  9. package/dist/adapters/generic-sql/generic-sql-adapter.js +11 -12
  10. package/dist/adapters/generic-sql/generic-sql-adapter.js.map +1 -1
  11. package/dist/adapters/generic-sql/generic-sql-uow-executor.js +46 -6
  12. package/dist/adapters/generic-sql/generic-sql-uow-executor.js.map +1 -1
  13. package/dist/adapters/generic-sql/migration/cold-kysely.js.map +1 -1
  14. package/dist/adapters/generic-sql/migration/dialect/mysql.js +1 -1
  15. package/dist/adapters/generic-sql/migration/dialect/mysql.js.map +1 -1
  16. package/dist/adapters/generic-sql/migration/dialect/postgres.js +1 -1
  17. package/dist/adapters/generic-sql/migration/dialect/postgres.js.map +1 -1
  18. package/dist/adapters/generic-sql/migration/dialect/sqlite.js +185 -19
  19. package/dist/adapters/generic-sql/migration/dialect/sqlite.js.map +1 -1
  20. package/dist/adapters/generic-sql/migration/executor.d.ts.map +1 -1
  21. package/dist/adapters/generic-sql/migration/executor.js +30 -3
  22. package/dist/adapters/generic-sql/migration/executor.js.map +1 -1
  23. package/dist/adapters/generic-sql/migration/prepared-migrations.d.ts.map +1 -1
  24. package/dist/adapters/generic-sql/migration/prepared-migrations.js +3 -3
  25. package/dist/adapters/generic-sql/migration/prepared-migrations.js.map +1 -1
  26. package/dist/adapters/generic-sql/migration/sql-generator.js +1 -1
  27. package/dist/adapters/generic-sql/migration/sql-generator.js.map +1 -1
  28. package/dist/adapters/generic-sql/query/create-sql-query-compiler.js +1 -1
  29. package/dist/adapters/generic-sql/query/create-sql-query-compiler.js.map +1 -1
  30. package/dist/adapters/generic-sql/query/cursor-utils.js.map +1 -1
  31. package/dist/adapters/generic-sql/query/db-now-sql.js +27 -0
  32. package/dist/adapters/generic-sql/query/db-now-sql.js.map +1 -0
  33. package/dist/adapters/generic-sql/query/generic-sql-uow-operation-compiler.js +9 -6
  34. package/dist/adapters/generic-sql/query/generic-sql-uow-operation-compiler.js.map +1 -1
  35. package/dist/adapters/generic-sql/query/select-builder.js.map +1 -1
  36. package/dist/adapters/generic-sql/query/sql-query-compiler.js +37 -9
  37. package/dist/adapters/generic-sql/query/sql-query-compiler.js.map +1 -1
  38. package/dist/adapters/generic-sql/query/where-builder.js +24 -20
  39. package/dist/adapters/generic-sql/query/where-builder.js.map +1 -1
  40. package/dist/adapters/generic-sql/uow-decoder.js +1 -1
  41. package/dist/adapters/generic-sql/uow-decoder.js.map +1 -1
  42. package/dist/adapters/generic-sql/uow-encoder.js +8 -9
  43. package/dist/adapters/generic-sql/uow-encoder.js.map +1 -1
  44. package/dist/adapters/in-memory/condition-evaluator.js +10 -6
  45. package/dist/adapters/in-memory/condition-evaluator.js.map +1 -1
  46. package/dist/adapters/in-memory/in-memory-adapter.d.ts.map +1 -1
  47. package/dist/adapters/in-memory/in-memory-adapter.js +45 -25
  48. package/dist/adapters/in-memory/in-memory-adapter.js.map +1 -1
  49. package/dist/adapters/in-memory/in-memory-uow.js +236 -13
  50. package/dist/adapters/in-memory/in-memory-uow.js.map +1 -1
  51. package/dist/adapters/in-memory/options.d.ts +2 -0
  52. package/dist/adapters/in-memory/options.d.ts.map +1 -1
  53. package/dist/adapters/in-memory/options.js +3 -2
  54. package/dist/adapters/in-memory/options.js.map +1 -1
  55. package/dist/adapters/in-memory/reference-resolution.js.map +1 -1
  56. package/dist/adapters/in-memory/store.js +1 -1
  57. package/dist/adapters/in-memory/store.js.map +1 -1
  58. package/dist/adapters/shared/from-unit-of-work-compiler.js +51 -24
  59. package/dist/adapters/shared/from-unit-of-work-compiler.js.map +1 -1
  60. package/dist/adapters/shared/uow-operation-compiler.js.map +1 -1
  61. package/dist/browser/adapters/adapters.d.ts +61 -0
  62. package/dist/browser/adapters/adapters.d.ts.map +1 -0
  63. package/dist/browser/adapters/generic-sql/migration/executor.d.ts +15 -0
  64. package/dist/browser/adapters/generic-sql/migration/executor.d.ts.map +1 -0
  65. package/dist/browser/adapters/generic-sql/migration/prepared-migrations.d.ts +66 -0
  66. package/dist/browser/adapters/generic-sql/migration/prepared-migrations.d.ts.map +1 -0
  67. package/dist/browser/adapters/generic-sql/sqlite-storage.d.ts +11 -0
  68. package/dist/browser/adapters/generic-sql/sqlite-storage.d.ts.map +1 -0
  69. package/dist/browser/adapters/in-memory/in-memory-adapter.d.ts +5 -0
  70. package/dist/browser/adapters/in-memory/index.d.ts +2 -0
  71. package/dist/browser/adapters/in-memory/options.d.ts +1 -0
  72. package/dist/browser/db-fragment-definition-builder.d.ts +237 -0
  73. package/dist/browser/db-fragment-definition-builder.d.ts.map +1 -0
  74. package/dist/browser/durable-hooks.d.ts +3 -0
  75. package/dist/browser/fragments/internal-fragment.d.ts +317 -0
  76. package/dist/browser/fragments/internal-fragment.d.ts.map +1 -0
  77. package/dist/browser/fragments/internal-fragment.schema.d.ts +1 -0
  78. package/dist/browser/hooks/durable-hooks-logger.d.ts +10 -0
  79. package/dist/browser/hooks/durable-hooks-logger.d.ts.map +1 -0
  80. package/dist/browser/hooks/hooks.d.ts +146 -0
  81. package/dist/browser/hooks/hooks.d.ts.map +1 -0
  82. package/dist/browser/id.js +1 -0
  83. package/dist/browser/internal/adapter-registry.d.ts +4 -0
  84. package/dist/browser/internal/outbox-state.d.ts +2 -0
  85. package/dist/browser/mod.d.ts +15 -0
  86. package/dist/browser/mod.d.ts.map +1 -0
  87. package/dist/browser/mod.js +17 -0
  88. package/dist/browser/mod.js.map +1 -0
  89. package/dist/browser/mod2.d.ts +48 -0
  90. package/dist/browser/mod2.d.ts.map +1 -0
  91. package/dist/browser/naming/sql-naming.d.ts +19 -0
  92. package/dist/browser/naming/sql-naming.d.ts.map +1 -0
  93. package/dist/browser/outbox/outbox.d.ts +21 -0
  94. package/dist/browser/outbox/outbox.d.ts.map +1 -0
  95. package/dist/browser/query/column-defaults.js +1 -0
  96. package/dist/browser/query/condition-builder.d.ts +44 -0
  97. package/dist/browser/query/condition-builder.d.ts.map +1 -0
  98. package/dist/browser/query/condition-builder.js +97 -0
  99. package/dist/browser/query/condition-builder.js.map +1 -0
  100. package/dist/browser/query/cursor.d.ts +105 -0
  101. package/dist/browser/query/cursor.d.ts.map +1 -0
  102. package/dist/browser/query/cursor.js +150 -0
  103. package/dist/browser/query/cursor.js.map +1 -0
  104. package/dist/browser/query/db-now.d.ts +22 -0
  105. package/dist/browser/query/db-now.d.ts.map +1 -0
  106. package/dist/browser/query/db-now.js +33 -0
  107. package/dist/browser/query/db-now.js.map +1 -0
  108. package/dist/browser/query/orm/orm.d.ts +18 -0
  109. package/dist/browser/query/orm/orm.d.ts.map +1 -0
  110. package/dist/browser/query/simple-query-interface.d.ts +108 -0
  111. package/dist/browser/query/simple-query-interface.d.ts.map +1 -0
  112. package/dist/browser/query/unit-of-work/execute-unit-of-work.d.ts +423 -0
  113. package/dist/browser/query/unit-of-work/execute-unit-of-work.d.ts.map +1 -0
  114. package/dist/browser/query/unit-of-work/execute-unit-of-work.js +507 -0
  115. package/dist/browser/query/unit-of-work/execute-unit-of-work.js.map +1 -0
  116. package/dist/browser/query/unit-of-work/retry-policy.d.ts +23 -0
  117. package/dist/browser/query/unit-of-work/retry-policy.d.ts.map +1 -0
  118. package/dist/browser/query/unit-of-work/retry-policy.js +40 -0
  119. package/dist/browser/query/unit-of-work/retry-policy.js.map +1 -0
  120. package/dist/browser/query/unit-of-work/unit-of-work.d.ts +703 -0
  121. package/dist/browser/query/unit-of-work/unit-of-work.d.ts.map +1 -0
  122. package/dist/browser/query/unit-of-work/unit-of-work.js +1206 -0
  123. package/dist/browser/query/unit-of-work/unit-of-work.js.map +1 -0
  124. package/dist/browser/query/value-encoding.js +38 -0
  125. package/dist/browser/query/value-encoding.js.map +1 -0
  126. package/dist/browser/schema/create.d.ts +326 -0
  127. package/dist/browser/schema/create.d.ts.map +1 -0
  128. package/dist/browser/schema/create.js +89 -0
  129. package/dist/browser/schema/create.js.map +1 -0
  130. package/dist/browser/schema/generate-id.js +28 -0
  131. package/dist/browser/schema/generate-id.js.map +1 -0
  132. package/dist/browser/shared/providers.d.ts +6 -0
  133. package/dist/browser/shared/providers.d.ts.map +1 -0
  134. package/dist/browser/sql-driver/connection/connection-provider.d.ts +13 -0
  135. package/dist/browser/sql-driver/connection/connection-provider.d.ts.map +1 -0
  136. package/dist/browser/sql-driver/dialect-adapter/dialect-adapter.d.ts +7 -0
  137. package/dist/browser/sql-driver/dialect-adapter/dialect-adapter.d.ts.map +1 -0
  138. package/dist/browser/sql-driver/driver/runtime-driver.d.ts +23 -0
  139. package/dist/browser/sql-driver/driver/runtime-driver.d.ts.map +1 -0
  140. package/dist/browser/sql-driver/query-executor/plugin.d.ts +17 -0
  141. package/dist/browser/sql-driver/query-executor/plugin.d.ts.map +1 -0
  142. package/dist/browser/sql-driver/query-executor/query-executor.d.ts +36 -0
  143. package/dist/browser/sql-driver/query-executor/query-executor.d.ts.map +1 -0
  144. package/dist/browser/sql-driver/sql-driver-adapter.d.ts +29 -0
  145. package/dist/browser/sql-driver/sql-driver-adapter.d.ts.map +1 -0
  146. package/dist/browser/sql-driver/sql-driver.d.ts +38 -0
  147. package/dist/browser/sql-driver/sql-driver.d.ts.map +1 -0
  148. package/dist/browser/sync/commands.d.ts +15 -0
  149. package/dist/browser/sync/commands.d.ts.map +1 -0
  150. package/dist/browser/sync/commands.js +27 -0
  151. package/dist/browser/sync/commands.js.map +1 -0
  152. package/dist/browser/sync/types.d.ts +63 -0
  153. package/dist/browser/sync/types.d.ts.map +1 -0
  154. package/dist/browser/util/types.d.ts +8 -0
  155. package/dist/browser/util/types.d.ts.map +1 -0
  156. package/dist/browser/with-database.d.ts +29 -0
  157. package/dist/browser/with-database.d.ts.map +1 -0
  158. package/dist/client.d.ts +4 -0
  159. package/dist/client.js +5 -0
  160. package/dist/db-fragment-definition-builder.d.ts +85 -28
  161. package/dist/db-fragment-definition-builder.d.ts.map +1 -1
  162. package/dist/db-fragment-definition-builder.js +374 -46
  163. package/dist/db-fragment-definition-builder.js.map +1 -1
  164. package/dist/dispatchers/cloudflare-do/dispatcher.d.ts +20 -0
  165. package/dist/dispatchers/cloudflare-do/dispatcher.d.ts.map +1 -0
  166. package/dist/dispatchers/cloudflare-do/dispatcher.js +147 -0
  167. package/dist/dispatchers/cloudflare-do/dispatcher.js.map +1 -0
  168. package/dist/dispatchers/cloudflare-do/index.d.ts +5 -20
  169. package/dist/dispatchers/cloudflare-do/index.d.ts.map +1 -1
  170. package/dist/dispatchers/cloudflare-do/index.js +23 -55
  171. package/dist/dispatchers/cloudflare-do/index.js.map +1 -1
  172. package/dist/dispatchers/node/dispatcher.d.ts +14 -0
  173. package/dist/dispatchers/node/dispatcher.d.ts.map +1 -0
  174. package/dist/dispatchers/node/dispatcher.js +80 -0
  175. package/dist/dispatchers/node/dispatcher.js.map +1 -0
  176. package/dist/dispatchers/node/index.d.ts +5 -10
  177. package/dist/dispatchers/node/index.d.ts.map +1 -1
  178. package/dist/dispatchers/node/index.js +21 -53
  179. package/dist/dispatchers/node/index.js.map +1 -1
  180. package/dist/durable-hooks.d.ts +31 -0
  181. package/dist/durable-hooks.d.ts.map +1 -0
  182. package/dist/durable-hooks.js +23 -0
  183. package/dist/durable-hooks.js.map +1 -0
  184. package/dist/fragments/internal-fragment.d.ts +128 -27
  185. package/dist/fragments/internal-fragment.d.ts.map +1 -1
  186. package/dist/fragments/internal-fragment.js +125 -78
  187. package/dist/fragments/internal-fragment.js.map +1 -1
  188. package/dist/fragments/internal-fragment.routes.js +138 -3
  189. package/dist/fragments/internal-fragment.routes.js.map +1 -1
  190. package/dist/fragments/internal-fragment.schema.d.ts +7 -1
  191. package/dist/fragments/internal-fragment.schema.d.ts.map +1 -1
  192. package/dist/fragments/internal-fragment.schema.js +18 -1
  193. package/dist/fragments/internal-fragment.schema.js.map +1 -1
  194. package/dist/hooks/durable-hooks-logger.d.ts +10 -0
  195. package/dist/hooks/durable-hooks-logger.d.ts.map +1 -0
  196. package/dist/hooks/durable-hooks-logger.js +75 -0
  197. package/dist/hooks/durable-hooks-logger.js.map +1 -0
  198. package/dist/hooks/durable-hooks-processor.d.ts +1 -14
  199. package/dist/hooks/durable-hooks-processor.js +58 -10
  200. package/dist/hooks/durable-hooks-processor.js.map +1 -1
  201. package/dist/hooks/durable-hooks-runtime.js +44 -0
  202. package/dist/hooks/durable-hooks-runtime.js.map +1 -0
  203. package/dist/hooks/hooks.d.ts +60 -2
  204. package/dist/hooks/hooks.d.ts.map +1 -1
  205. package/dist/hooks/hooks.js +214 -53
  206. package/dist/hooks/hooks.js.map +1 -1
  207. package/dist/id.d.ts +2 -2
  208. package/dist/id.js +2 -2
  209. package/dist/internal/adapter-registry.d.ts +11 -0
  210. package/dist/internal/adapter-registry.d.ts.map +1 -0
  211. package/dist/internal/adapter-registry.js +135 -0
  212. package/dist/internal/adapter-registry.js.map +1 -0
  213. package/dist/internal/outbox-state.d.ts +2 -0
  214. package/dist/internal/outbox-state.js +26 -0
  215. package/dist/internal/outbox-state.js.map +1 -0
  216. package/dist/migration-engine/auto-from-schema.d.ts +33 -0
  217. package/dist/migration-engine/auto-from-schema.d.ts.map +1 -0
  218. package/dist/migration-engine/auto-from-schema.js +210 -27
  219. package/dist/migration-engine/auto-from-schema.js.map +1 -1
  220. package/dist/migration-engine/generation-engine.d.ts.map +1 -1
  221. package/dist/migration-engine/generation-engine.js +17 -5
  222. package/dist/migration-engine/generation-engine.js.map +1 -1
  223. package/dist/migration-engine/shared.d.ts +113 -0
  224. package/dist/migration-engine/shared.d.ts.map +1 -0
  225. package/dist/migration-engine/shared.js.map +1 -1
  226. package/dist/mod.d.ts +12 -11
  227. package/dist/mod.d.ts.map +1 -1
  228. package/dist/mod.js +10 -10
  229. package/dist/mod.js.map +1 -1
  230. package/dist/naming/sql-naming.d.ts.map +1 -1
  231. package/dist/naming/sql-naming.js.map +1 -1
  232. package/dist/outbox/outbox-builder.js.map +1 -1
  233. package/dist/outbox/outbox.d.ts +3 -1
  234. package/dist/outbox/outbox.d.ts.map +1 -1
  235. package/dist/outbox/outbox.js.map +1 -1
  236. package/dist/query/column-defaults.js.map +1 -1
  237. package/dist/query/condition-builder.d.ts +7 -1
  238. package/dist/query/condition-builder.d.ts.map +1 -1
  239. package/dist/query/condition-builder.js +5 -1
  240. package/dist/query/condition-builder.js.map +1 -1
  241. package/dist/query/cursor-client.d.ts +105 -0
  242. package/dist/query/cursor-client.d.ts.map +1 -0
  243. package/dist/query/cursor-client.js +165 -0
  244. package/dist/query/cursor-client.js.map +1 -0
  245. package/dist/query/cursor.d.ts.map +1 -1
  246. package/dist/query/cursor.js +7 -1
  247. package/dist/query/cursor.js.map +1 -1
  248. package/dist/query/db-now.d.ts +15 -1
  249. package/dist/query/db-now.d.ts.map +1 -1
  250. package/dist/query/db-now.js +30 -2
  251. package/dist/query/db-now.js.map +1 -1
  252. package/dist/query/orm/orm.js.map +1 -1
  253. package/dist/query/serialize/create-sql-serializer.js +2 -2
  254. package/dist/query/serialize/create-sql-serializer.js.map +1 -1
  255. package/dist/query/serialize/dialect/mysql-serializer.js.map +1 -1
  256. package/dist/query/serialize/dialect/postgres-serializer.js.map +1 -1
  257. package/dist/query/serialize/dialect/sqlite-serializer.js +6 -2
  258. package/dist/query/serialize/dialect/sqlite-serializer.js.map +1 -1
  259. package/dist/query/simple-query-interface.d.ts +7 -3
  260. package/dist/query/simple-query-interface.d.ts.map +1 -1
  261. package/dist/query/unit-of-work/execute-unit-of-work.d.ts +37 -2
  262. package/dist/query/unit-of-work/execute-unit-of-work.d.ts.map +1 -1
  263. package/dist/query/unit-of-work/execute-unit-of-work.js +39 -18
  264. package/dist/query/unit-of-work/execute-unit-of-work.js.map +1 -1
  265. package/dist/query/unit-of-work/unit-of-work.d.ts +42 -16
  266. package/dist/query/unit-of-work/unit-of-work.d.ts.map +1 -1
  267. package/dist/query/unit-of-work/unit-of-work.js +50 -6
  268. package/dist/query/unit-of-work/unit-of-work.js.map +1 -1
  269. package/dist/query/value-decoding.js +8 -1
  270. package/dist/query/value-decoding.js.map +1 -1
  271. package/dist/query/value-encoding.js.map +1 -1
  272. package/dist/schema/create.d.ts +69 -25
  273. package/dist/schema/create.d.ts.map +1 -1
  274. package/dist/schema/create.js +91 -16
  275. package/dist/schema/create.js.map +1 -1
  276. package/dist/schema/type-conversion/create-sql-type-mapper.js +1 -1
  277. package/dist/schema/type-conversion/create-sql-type-mapper.js.map +1 -1
  278. package/dist/schema/type-conversion/dialect/sqlite.js.map +1 -1
  279. package/dist/schema/validator.d.ts.map +1 -1
  280. package/dist/schema/validator.js.map +1 -1
  281. package/dist/schema-output/drizzle.d.ts.map +1 -1
  282. package/dist/schema-output/drizzle.js +8 -6
  283. package/dist/schema-output/drizzle.js.map +1 -1
  284. package/dist/schema-output/prisma.d.ts.map +1 -1
  285. package/dist/schema-output/prisma.js +21 -10
  286. package/dist/schema-output/prisma.js.map +1 -1
  287. package/dist/sql-driver/dialects/durable-object-dialect.js +3 -9
  288. package/dist/sql-driver/dialects/durable-object-dialect.js.map +1 -1
  289. package/dist/sql-driver/query-executor/default-query-executor.js.map +1 -1
  290. package/dist/sql-driver/query-executor/query-executor-base.js.map +1 -1
  291. package/dist/sql-driver/sql-driver-adapter.js.map +1 -1
  292. package/dist/sql-driver/sql.js.map +1 -1
  293. package/dist/sync/commands.d.ts +15 -0
  294. package/dist/sync/commands.d.ts.map +1 -0
  295. package/dist/sync/commands.js +27 -0
  296. package/dist/sync/commands.js.map +1 -0
  297. package/dist/sync/index.d.ts +4 -0
  298. package/dist/sync/index.js +4 -0
  299. package/dist/sync/read-tracking.d.ts +25 -0
  300. package/dist/sync/read-tracking.d.ts.map +1 -0
  301. package/dist/sync/read-tracking.js +148 -0
  302. package/dist/sync/read-tracking.js.map +1 -0
  303. package/dist/sync/submit.js +213 -0
  304. package/dist/sync/submit.js.map +1 -0
  305. package/dist/sync/types.d.ts +63 -0
  306. package/dist/sync/types.d.ts.map +1 -0
  307. package/dist/util/default-database-adapter.js +6 -1
  308. package/dist/util/default-database-adapter.js.map +1 -1
  309. package/dist/with-database.d.ts +3 -6
  310. package/dist/with-database.d.ts.map +1 -1
  311. package/dist/with-database.js +7 -15
  312. package/dist/with-database.js.map +1 -1
  313. package/package.json +33 -41
  314. package/src/adapters/adapters.ts +5 -4
  315. package/src/adapters/drizzle/migrate-drizzle.test.ts +46 -9
  316. package/src/adapters/drizzle/migration-parity-drizzle-kit.test.ts +5 -3
  317. package/src/adapters/drizzle/test-utils.ts +2 -1
  318. package/src/adapters/generic-sql/generic-sql-adapter.test.ts +5 -3
  319. package/src/adapters/generic-sql/generic-sql-adapter.ts +21 -24
  320. package/src/adapters/generic-sql/generic-sql-uow-executor.test.ts +1 -0
  321. package/src/adapters/generic-sql/generic-sql-uow-executor.ts +81 -15
  322. package/src/adapters/generic-sql/migration/adapter-migration-parity.test.ts +4 -2
  323. package/src/adapters/generic-sql/migration/cold-kysely.ts +1 -0
  324. package/src/adapters/generic-sql/migration/dialect/mysql.test.ts +3 -2
  325. package/src/adapters/generic-sql/migration/dialect/mysql.ts +1 -0
  326. package/src/adapters/generic-sql/migration/dialect/postgres.test.ts +5 -4
  327. package/src/adapters/generic-sql/migration/dialect/postgres.ts +2 -1
  328. package/src/adapters/generic-sql/migration/dialect/sqlite.test.ts +795 -3
  329. package/src/adapters/generic-sql/migration/dialect/sqlite.ts +385 -57
  330. package/src/adapters/generic-sql/migration/executor.test.ts +52 -0
  331. package/src/adapters/generic-sql/migration/executor.ts +47 -4
  332. package/src/adapters/generic-sql/migration/prepared-migrations.test.ts +117 -14
  333. package/src/adapters/generic-sql/migration/prepared-migrations.ts +9 -8
  334. package/src/adapters/generic-sql/migration/sql-generator.ts +5 -3
  335. package/src/adapters/generic-sql/query/create-sql-query-compiler.ts +3 -3
  336. package/src/adapters/generic-sql/query/cursor-utils.test.ts +3 -2
  337. package/src/adapters/generic-sql/query/cursor-utils.ts +1 -1
  338. package/src/adapters/generic-sql/query/db-now-sql.ts +49 -0
  339. package/src/adapters/generic-sql/query/generic-sql-uow-operation-compiler.test.ts +144 -8
  340. package/src/adapters/generic-sql/query/generic-sql-uow-operation-compiler.ts +16 -17
  341. package/src/adapters/generic-sql/query/select-builder.test.ts +1 -0
  342. package/src/adapters/generic-sql/query/select-builder.ts +2 -2
  343. package/src/adapters/generic-sql/query/sql-query-compiler.test.ts +24 -5
  344. package/src/adapters/generic-sql/query/sql-query-compiler.ts +83 -13
  345. package/src/adapters/generic-sql/query/where-builder.test.ts +7 -5
  346. package/src/adapters/generic-sql/query/where-builder.ts +48 -29
  347. package/src/adapters/generic-sql/sql-adapter-pglite-migrations.test.ts +6 -15
  348. package/src/adapters/generic-sql/sql-adapter-pglite-pagination.test.ts +52 -7
  349. package/src/adapters/generic-sql/sql-adapter-pglite-queries.test.ts +9 -6
  350. package/src/adapters/generic-sql/sql-adapter-sqlite3-driver.test.ts +273 -5
  351. package/src/adapters/generic-sql/sql-adapter-sqlite3-uow.test.ts +123 -6
  352. package/src/adapters/generic-sql/sql-adapter-sqlocal.test.ts +4 -2
  353. package/src/adapters/generic-sql/uow-decoder.test.ts +4 -3
  354. package/src/adapters/generic-sql/uow-decoder.ts +3 -3
  355. package/src/adapters/generic-sql/uow-encoder.test.ts +4 -2
  356. package/src/adapters/generic-sql/uow-encoder.ts +14 -18
  357. package/src/adapters/in-memory/condition-evaluator.test.ts +2 -1
  358. package/src/adapters/in-memory/condition-evaluator.ts +9 -4
  359. package/src/adapters/in-memory/in-memory-adapter.ts +155 -44
  360. package/src/adapters/in-memory/in-memory-uow.mutations.test.ts +50 -2
  361. package/src/adapters/in-memory/in-memory-uow.retrieval.test.ts +158 -3
  362. package/src/adapters/in-memory/in-memory-uow.ts +402 -26
  363. package/src/adapters/in-memory/options.test.ts +1 -0
  364. package/src/adapters/in-memory/options.ts +5 -1
  365. package/src/adapters/in-memory/outbox.test.ts +361 -0
  366. package/src/adapters/in-memory/reference-resolution.test.ts +3 -2
  367. package/src/adapters/in-memory/reference-resolution.ts +2 -2
  368. package/src/adapters/in-memory/sorted-array-index.test.ts +1 -0
  369. package/src/adapters/in-memory/store.test.ts +1 -0
  370. package/src/adapters/in-memory/store.ts +3 -3
  371. package/src/adapters/in-memory/value-normalization.test.ts +1 -0
  372. package/src/adapters/prisma/prisma-adapter-sqlite3.test.ts +51 -7
  373. package/src/adapters/shared/from-unit-of-work-compiler.ts +156 -46
  374. package/src/adapters/shared/uow-operation-compiler.ts +3 -3
  375. package/src/browser/mod.ts +64 -0
  376. package/src/client.ts +19 -0
  377. package/src/db-fragment-definition-builder.test.ts +821 -47
  378. package/src/db-fragment-definition-builder.ts +857 -110
  379. package/src/db-fragment-instantiator.test.ts +114 -90
  380. package/src/db-fragment-integration.test.ts +9 -6
  381. package/src/dispatchers/cloudflare-do/dispatcher.ts +204 -0
  382. package/src/dispatchers/cloudflare-do/index.test.ts +145 -12
  383. package/src/dispatchers/cloudflare-do/index.ts +49 -90
  384. package/src/dispatchers/node/dispatcher.ts +112 -0
  385. package/src/dispatchers/node/index.test.ts +43 -14
  386. package/src/dispatchers/node/index.ts +38 -75
  387. package/src/durable-hooks.test.ts +80 -0
  388. package/src/durable-hooks.ts +67 -0
  389. package/src/fragments/internal-fragment.routes.test.ts +570 -0
  390. package/src/fragments/internal-fragment.routes.ts +297 -5
  391. package/src/fragments/internal-fragment.schema.ts +45 -1
  392. package/src/fragments/internal-fragment.test.ts +223 -251
  393. package/src/fragments/internal-fragment.ts +278 -154
  394. package/src/hooks/durable-hooks-logger.ts +126 -0
  395. package/src/hooks/durable-hooks-processor.pglite.test.ts +87 -0
  396. package/src/hooks/durable-hooks-processor.test.ts +179 -14
  397. package/src/hooks/durable-hooks-processor.ts +120 -14
  398. package/src/hooks/durable-hooks-runtime.test.ts +65 -0
  399. package/src/hooks/durable-hooks-runtime.ts +81 -0
  400. package/src/hooks/hooks.test.ts +314 -53
  401. package/src/hooks/hooks.ts +360 -81
  402. package/src/id.test.ts +34 -0
  403. package/src/id.ts +1 -3
  404. package/src/internal/adapter-registry.test.ts +93 -0
  405. package/src/internal/adapter-registry.ts +239 -0
  406. package/src/internal/outbox-state.ts +43 -0
  407. package/src/migration-engine/auto-from-schema.test.ts +93 -0
  408. package/src/migration-engine/auto-from-schema.ts +360 -42
  409. package/src/migration-engine/create.test.ts +2 -1
  410. package/src/migration-engine/create.ts +1 -1
  411. package/src/migration-engine/generation-engine.test.ts +66 -9
  412. package/src/migration-engine/generation-engine.ts +31 -10
  413. package/src/migration-engine/shared.ts +13 -0
  414. package/src/mod.ts +45 -27
  415. package/src/naming/sql-naming.ts +1 -0
  416. package/src/outbox/outbox-builder.ts +2 -2
  417. package/src/outbox/outbox.test.ts +216 -45
  418. package/src/outbox/outbox.ts +3 -1
  419. package/src/query/column-defaults.ts +1 -1
  420. package/src/query/condition-builder.test.ts +15 -0
  421. package/src/query/condition-builder.ts +7 -0
  422. package/src/query/cursor-client.test.ts +70 -0
  423. package/src/query/cursor-client.ts +263 -0
  424. package/src/query/cursor.test.ts +3 -2
  425. package/src/query/cursor.ts +15 -3
  426. package/src/query/db-now.ts +69 -2
  427. package/src/query/orm/orm.ts +2 -2
  428. package/src/query/query-type.test.ts +2 -1
  429. package/src/query/serialize/create-sql-serializer.ts +3 -3
  430. package/src/query/serialize/dialect/mysql-serializer.ts +1 -1
  431. package/src/query/serialize/dialect/postgres-serializer.ts +1 -1
  432. package/src/query/serialize/dialect/sqlite-serializer.test.ts +39 -2
  433. package/src/query/serialize/dialect/sqlite-serializer.ts +18 -5
  434. package/src/query/simple-query-interface.ts +10 -4
  435. package/src/query/unit-of-work/execute-unit-of-work.test.ts +347 -9
  436. package/src/query/unit-of-work/execute-unit-of-work.ts +63 -20
  437. package/src/query/unit-of-work/retry-policy.test.ts +1 -0
  438. package/src/query/unit-of-work/tx-builder.test.ts +73 -1
  439. package/src/query/unit-of-work/unit-of-work-coordinator.test.ts +5 -4
  440. package/src/query/unit-of-work/unit-of-work-types.test.ts +41 -11
  441. package/src/query/unit-of-work/unit-of-work.test.ts +28 -2
  442. package/src/query/unit-of-work/unit-of-work.ts +105 -19
  443. package/src/query/value-decoding.test.ts +50 -2
  444. package/src/query/value-decoding.ts +17 -4
  445. package/src/query/value-encoding.test.ts +1 -0
  446. package/src/query/value-encoding.ts +1 -1
  447. package/src/schema/create.test.ts +164 -5
  448. package/src/schema/create.ts +222 -24
  449. package/src/schema/generate-id.test.ts +1 -0
  450. package/src/schema/serialize.test.ts +4 -3
  451. package/src/schema/type-conversion/create-sql-type-mapper.ts +1 -1
  452. package/src/schema/type-conversion/dialect/sqlite.ts +2 -2
  453. package/src/schema/type-conversion/type-mapping.test.ts +2 -1
  454. package/src/schema/validator.test.ts +4 -2
  455. package/src/schema/validator.ts +1 -0
  456. package/src/schema-output/drizzle.test.ts +72 -19
  457. package/src/schema-output/drizzle.ts +24 -18
  458. package/src/schema-output/prisma.test.ts +172 -14
  459. package/src/schema-output/prisma.ts +34 -14
  460. package/src/sql-driver/better-sqlite3.test.ts +5 -3
  461. package/src/sql-driver/dialects/durable-object-dialect.ts +3 -8
  462. package/src/sql-driver/query-executor/default-query-executor.ts +1 -1
  463. package/src/sql-driver/query-executor/query-executor-base.ts +1 -1
  464. package/src/sql-driver/query-executor/query-executor.ts +1 -1
  465. package/src/sql-driver/sql-driver-adapter.ts +2 -2
  466. package/src/sql-driver/sql.ts +2 -1
  467. package/src/sql-driver/sqlocal.test.ts +4 -2
  468. package/src/sync/commands.test.ts +39 -0
  469. package/src/sync/commands.ts +51 -0
  470. package/src/sync/conflict-checker.test.ts +450 -0
  471. package/src/sync/conflict-checker.ts +248 -0
  472. package/src/sync/index.ts +14 -0
  473. package/src/sync/plan.ts +9 -0
  474. package/src/sync/read-tracking.test.ts +177 -0
  475. package/src/sync/read-tracking.ts +287 -0
  476. package/src/sync/submit.test.ts +205 -0
  477. package/src/sync/submit.ts +328 -0
  478. package/src/sync/types.ts +80 -0
  479. package/src/util/default-database-adapter.ts +15 -2
  480. package/src/with-database.ts +20 -50
  481. package/tsconfig.json +1 -1
  482. package/tsdown.config.ts +38 -26
  483. package/vitest.config.ts +1 -0
  484. package/dist/hooks/durable-hooks-processor.d.ts.map +0 -1
  485. package/dist/node_modules/.pnpm/rou3@0.7.12/node_modules/rou3/dist/index.js +0 -168
  486. package/dist/node_modules/.pnpm/rou3@0.7.12/node_modules/rou3/dist/index.js.map +0 -1
  487. package/dist/packages/fragno/dist/api/bind-services.js +0 -20
  488. package/dist/packages/fragno/dist/api/bind-services.js.map +0 -1
  489. package/dist/packages/fragno/dist/api/error.js +0 -48
  490. package/dist/packages/fragno/dist/api/error.js.map +0 -1
  491. package/dist/packages/fragno/dist/api/fragment-definition-builder.js +0 -321
  492. package/dist/packages/fragno/dist/api/fragment-definition-builder.js.map +0 -1
  493. package/dist/packages/fragno/dist/api/fragment-instantiator.js +0 -669
  494. package/dist/packages/fragno/dist/api/fragment-instantiator.js.map +0 -1
  495. package/dist/packages/fragno/dist/api/fragno-response.js +0 -73
  496. package/dist/packages/fragno/dist/api/fragno-response.js.map +0 -1
  497. package/dist/packages/fragno/dist/api/internal/response-stream.js +0 -81
  498. package/dist/packages/fragno/dist/api/internal/response-stream.js.map +0 -1
  499. package/dist/packages/fragno/dist/api/internal/route.js +0 -10
  500. package/dist/packages/fragno/dist/api/internal/route.js.map +0 -1
  501. package/dist/packages/fragno/dist/api/mutable-request-state.js +0 -97
  502. package/dist/packages/fragno/dist/api/mutable-request-state.js.map +0 -1
  503. package/dist/packages/fragno/dist/api/request-context-storage.js +0 -43
  504. package/dist/packages/fragno/dist/api/request-context-storage.js.map +0 -1
  505. package/dist/packages/fragno/dist/api/request-input-context.js +0 -185
  506. package/dist/packages/fragno/dist/api/request-input-context.js.map +0 -1
  507. package/dist/packages/fragno/dist/api/request-middleware.js +0 -83
  508. package/dist/packages/fragno/dist/api/request-middleware.js.map +0 -1
  509. package/dist/packages/fragno/dist/api/request-output-context.js +0 -119
  510. package/dist/packages/fragno/dist/api/request-output-context.js.map +0 -1
  511. package/dist/packages/fragno/dist/api/route.js +0 -30
  512. package/dist/packages/fragno/dist/api/route.js.map +0 -1
  513. package/dist/packages/fragno/dist/internal/symbols.js +0 -10
  514. package/dist/packages/fragno/dist/internal/symbols.js.map +0 -1
  515. package/dist/packages/fragno/dist/internal/trace-context.js +0 -12
  516. package/dist/packages/fragno/dist/internal/trace-context.js.map +0 -1
@@ -1,27 +1,38 @@
1
+ import { beforeAll, describe, expect, it } from "vitest";
2
+
1
3
  import SQLite from "better-sqlite3";
2
4
  import { SqliteDialect } from "kysely";
3
- import { beforeAll, describe, expect, it } from "vitest";
5
+
4
6
  import { instantiate } from "@fragno-dev/core";
7
+
8
+ import { BetterSQLite3DriverConfig } from "../adapters/generic-sql/driver-config";
9
+ import { SqlAdapter } from "../adapters/generic-sql/generic-sql-adapter";
10
+ import type { FragnoPublicConfigWithDatabase } from "../db-fragment-definition-builder";
11
+ import { getRegistryForAdapterSync } from "../internal/adapter-registry";
12
+ import { dbNow } from "../query/db-now";
13
+ import { ExponentialBackoffRetryPolicy, NoRetryPolicy } from "../query/unit-of-work/retry-policy";
14
+ import type { FragnoId } from "../schema/create";
5
15
  import {
6
16
  internalFragmentDef,
7
17
  internalSchema,
8
18
  SETTINGS_NAMESPACE,
9
19
  getSchemaVersionFromDatabase,
10
20
  } from "./internal-fragment";
11
- import type { FragnoPublicConfigWithDatabase } from "../db-fragment-definition-builder";
12
- import { SqlAdapter } from "../adapters/generic-sql/generic-sql-adapter";
13
- import { BetterSQLite3DriverConfig } from "../adapters/generic-sql/driver-config";
14
- import { ExponentialBackoffRetryPolicy, NoRetryPolicy } from "../query/unit-of-work/retry-policy";
15
- import { ConcurrencyConflictError } from "../query/unit-of-work/execute-unit-of-work";
16
- import type { FragnoId } from "../schema/create";
21
+
22
+ type OptionsWithAdapter = FragnoPublicConfigWithDatabase & {
23
+ databaseAdapter: SqlAdapter;
24
+ };
17
25
 
18
26
  describe("Internal Fragment", () => {
19
27
  let sqliteDatabase: SQLite.Database;
20
28
  let adapter: SqlAdapter;
21
29
  let fragment: ReturnType<typeof instantiateFragment>;
22
30
 
23
- function instantiateFragment(options: FragnoPublicConfigWithDatabase) {
24
- return instantiate(internalFragmentDef).withConfig({}).withOptions(options).build();
31
+ function instantiateFragment(options: OptionsWithAdapter) {
32
+ return instantiate(internalFragmentDef)
33
+ .withConfig({ registry: getRegistryForAdapterSync(options.databaseAdapter) })
34
+ .withOptions(options)
35
+ .build();
25
36
  }
26
37
 
27
38
  beforeAll(async () => {
@@ -42,7 +53,7 @@ describe("Internal Fragment", () => {
42
53
  }
43
54
 
44
55
  // Instantiate fragment with shared database adapter
45
- const options: FragnoPublicConfigWithDatabase = {
56
+ const options: OptionsWithAdapter = {
46
57
  databaseAdapter: adapter,
47
58
  databaseNamespace: null,
48
59
  };
@@ -161,8 +172,11 @@ describe("Hook Service", () => {
161
172
  let adapter: SqlAdapter;
162
173
  let fragment: ReturnType<typeof instantiateFragment>;
163
174
 
164
- function instantiateFragment(options: FragnoPublicConfigWithDatabase) {
165
- return instantiate(internalFragmentDef).withConfig({}).withOptions(options).build();
175
+ function instantiateFragment(options: OptionsWithAdapter) {
176
+ return instantiate(internalFragmentDef)
177
+ .withConfig({ registry: getRegistryForAdapterSync(options.databaseAdapter) })
178
+ .withOptions(options)
179
+ .build();
166
180
  }
167
181
 
168
182
  beforeAll(async () => {
@@ -182,7 +196,7 @@ describe("Hook Service", () => {
182
196
  await migrations.executeWithDriver(adapter.driver, 0);
183
197
  }
184
198
 
185
- const options: FragnoPublicConfigWithDatabase = {
199
+ const options: OptionsWithAdapter = {
186
200
  databaseAdapter: adapter,
187
201
  databaseNamespace: null,
188
202
  };
@@ -232,107 +246,119 @@ describe("Hook Service", () => {
232
246
  const events = await fragment.inContext(async function () {
233
247
  return await this.handlerTx()
234
248
  .withServiceCalls(
235
- () => [fragment.services.hookService.getPendingHookEvents("test-namespace")] as const,
249
+ () => [fragment.services.hookService.getHooksByNamespace("test-namespace")] as const,
236
250
  )
237
251
  .transform(({ serviceResult: [result] }) => result)
238
252
  .execute();
239
253
  });
240
254
 
241
- expect(events).toHaveLength(1);
242
- expect(events[0]).toMatchObject({
255
+ const event = events.find((entry) => entry.hookName === "onTest" && entry.status === "pending");
256
+ expect(event).toBeDefined();
257
+ expect(event).toMatchObject({
243
258
  hookName: "onTest",
244
259
  payload: { test: "data" },
245
260
  attempts: 0,
246
261
  maxAttempts: 5,
247
- idempotencyKey: nonce,
262
+ status: "pending",
263
+ nonce,
248
264
  });
249
265
  });
250
266
 
251
- it("should mark a hook event as completed", async () => {
252
- const nonce = "test-nonce-2";
253
- let eventId: FragnoId;
267
+ it("should paginate hooks by namespace with a cursor", async () => {
268
+ const namespace = "page-cursor";
269
+ const createdAt1 = new Date("2024-01-01T00:00:00.000Z");
270
+ const createdAt2 = new Date("2024-01-02T00:00:00.000Z");
271
+ const createdAt3 = new Date("2024-01-03T00:00:00.000Z");
272
+
273
+ let id1!: FragnoId;
274
+ let id2!: FragnoId;
275
+ let id3!: FragnoId;
254
276
 
255
277
  await fragment.inContext(async function () {
256
278
  await this.handlerTx()
257
279
  .mutate(({ forSchema }) => {
258
280
  const uow = forSchema(internalSchema);
259
- eventId = uow.create("fragno_hooks", {
260
- namespace: "test-namespace",
261
- hookName: "onComplete",
262
- payload: { test: "data" },
281
+ id1 = uow.create("fragno_hooks", {
282
+ namespace,
283
+ hookName: "onPage1",
284
+ payload: { test: "page-1" },
263
285
  status: "pending",
264
286
  attempts: 0,
265
287
  maxAttempts: 5,
266
288
  lastAttemptAt: null,
267
289
  nextRetryAt: null,
268
290
  error: null,
269
- nonce,
291
+ nonce: "test-nonce-page-1",
292
+ createdAt: createdAt1,
270
293
  });
271
- })
272
- .execute();
273
- });
274
-
275
- await fragment.inContext(async function () {
276
- await this.handlerTx()
277
- .withServiceCalls(() => [fragment.services.hookService.markHookCompleted(eventId)] as const)
278
- .execute();
279
- });
280
-
281
- const result = await fragment.inContext(async function () {
282
- return await this.handlerTx()
283
- .withServiceCalls(() => [fragment.services.hookService.getHookById(eventId)] as const)
284
- .transform(({ serviceResult: [event] }) => event)
285
- .execute();
286
- });
287
-
288
- expect(result).toBeDefined();
289
- expect(result?.status).toBe("completed");
290
- expect(result?.lastAttemptAt).toBeInstanceOf(Date);
291
- });
292
-
293
- it("should reject marking completed with a stale id", async () => {
294
- const namespace = "complete-stale";
295
- const nonce = "test-nonce-complete-stale";
296
- let staleId!: FragnoId;
297
-
298
- await fragment.inContext(async function () {
299
- const createdId = await this.handlerTx()
300
- .mutate(({ forSchema }) => {
301
- const uow = forSchema(internalSchema);
302
- return uow.create("fragno_hooks", {
294
+ id2 = uow.create("fragno_hooks", {
303
295
  namespace,
304
- hookName: "onCompleteStale",
305
- payload: { test: "data" },
296
+ hookName: "onPage2",
297
+ payload: { test: "page-2" },
306
298
  status: "pending",
307
299
  attempts: 0,
308
300
  maxAttempts: 5,
309
301
  lastAttemptAt: null,
310
302
  nextRetryAt: null,
311
303
  error: null,
312
- nonce,
304
+ nonce: "test-nonce-page-2",
305
+ createdAt: createdAt2,
306
+ });
307
+ id3 = uow.create("fragno_hooks", {
308
+ namespace,
309
+ hookName: "onPage3",
310
+ payload: { test: "page-3" },
311
+ status: "pending",
312
+ attempts: 0,
313
+ maxAttempts: 5,
314
+ lastAttemptAt: null,
315
+ nextRetryAt: null,
316
+ error: null,
317
+ nonce: "test-nonce-page-3",
318
+ createdAt: createdAt3,
313
319
  });
314
320
  })
315
321
  .execute();
316
- staleId = createdId;
317
322
  });
318
323
 
319
- await fragment.inContext(async function () {
320
- await this.handlerTx()
324
+ const firstPage = await fragment.inContext(async function () {
325
+ return await this.handlerTx()
321
326
  .withServiceCalls(
322
- () => [fragment.services.hookService.claimPendingHookEvents(namespace)] as const,
327
+ () =>
328
+ [
329
+ fragment.services.hookService.getHooksByNamespacePage(namespace, { pageSize: 2 }),
330
+ ] as const,
323
331
  )
332
+ .transform(({ serviceResult: [result] }) => result)
324
333
  .execute();
325
334
  });
326
335
 
327
- await expect(
328
- fragment.inContext(async function () {
329
- await this.handlerTx()
330
- .withServiceCalls(
331
- () => [fragment.services.hookService.markHookCompleted(staleId)] as const,
332
- )
333
- .execute();
334
- }),
335
- ).rejects.toThrow(ConcurrencyConflictError);
336
+ expect(firstPage.items).toHaveLength(2);
337
+ expect(firstPage.hasNextPage).toBe(true);
338
+ expect(firstPage.cursor).toBeDefined();
339
+ expect(firstPage.items.map((event) => event.id.externalId)).toEqual([
340
+ id3.externalId,
341
+ id2.externalId,
342
+ ]);
343
+
344
+ const secondPage = await fragment.inContext(async function () {
345
+ return await this.handlerTx()
346
+ .withServiceCalls(
347
+ () =>
348
+ [
349
+ fragment.services.hookService.getHooksByNamespacePage(namespace, {
350
+ cursor: firstPage.cursor ?? undefined,
351
+ pageSize: 2,
352
+ }),
353
+ ] as const,
354
+ )
355
+ .transform(({ serviceResult: [result] }) => result)
356
+ .execute();
357
+ });
358
+
359
+ expect(secondPage.items).toHaveLength(1);
360
+ expect(secondPage.hasNextPage).toBe(false);
361
+ expect(secondPage.items[0]?.id.externalId).toBe(id1.externalId);
336
362
  });
337
363
 
338
364
  it("should mark a hook event as processing", async () => {
@@ -488,167 +514,6 @@ describe("Hook Service", () => {
488
514
  expect(result?.error).toBe("Max attempts reached");
489
515
  });
490
516
 
491
- it("should retrieve stale events ready for retry", async () => {
492
- const nonce = "test-nonce-6";
493
- let eventId: FragnoId;
494
-
495
- const pastTime = new Date(Date.now() - 10000);
496
-
497
- await fragment.inContext(async function () {
498
- const createdId = await this.handlerTx()
499
- .mutate(({ forSchema }) => {
500
- const uow = forSchema(internalSchema);
501
- return uow.create("fragno_hooks", {
502
- namespace: "test-namespace",
503
- hookName: "onStale",
504
- payload: { test: "stale" },
505
- status: "pending",
506
- attempts: 1,
507
- maxAttempts: 5,
508
- lastAttemptAt: pastTime,
509
- nextRetryAt: pastTime,
510
- error: "Previous error",
511
- nonce,
512
- });
513
- })
514
- .execute();
515
- eventId = createdId;
516
- });
517
-
518
- const events = await fragment.inContext(async function () {
519
- return await this.handlerTx()
520
- .withServiceCalls(
521
- () => [fragment.services.hookService.getPendingHookEvents("test-namespace")] as const,
522
- )
523
- .transform(({ serviceResult: [result] }) => result)
524
- .execute();
525
- });
526
-
527
- const staleEvent = events.find((e) => e.id.externalId === eventId.externalId);
528
- expect(staleEvent).toBeDefined();
529
- expect(staleEvent?.hookName).toBe("onStale");
530
- expect(staleEvent?.attempts).toBe(1);
531
- });
532
-
533
- it("should detect conflicts when requeueing after another update in the same transaction", async () => {
534
- const namespace = "requeue-conflict";
535
- const nonce = "test-nonce-requeue-conflict";
536
- let eventId!: FragnoId;
537
-
538
- const staleBefore = new Date(Date.now() - 60_000);
539
- const lastAttemptAt = new Date(Date.now() - 120_000);
540
-
541
- await fragment.inContext(async function () {
542
- eventId = await this.handlerTx()
543
- .mutate(({ forSchema }) => {
544
- const uow = forSchema(internalSchema);
545
- return uow.create("fragno_hooks", {
546
- namespace,
547
- hookName: "onRequeueConflict",
548
- payload: { test: "requeue" },
549
- status: "processing",
550
- attempts: 0,
551
- maxAttempts: 5,
552
- lastAttemptAt,
553
- nextRetryAt: null,
554
- error: null,
555
- nonce,
556
- });
557
- })
558
- .execute();
559
- });
560
-
561
- await expect(
562
- fragment.inContext(async function () {
563
- await this.handlerTx({ retryPolicy: new NoRetryPolicy() })
564
- .withServiceCalls(
565
- () =>
566
- [
567
- fragment.services.hookService.markHookProcessing(eventId),
568
- fragment.services.hookService.requeueStuckProcessingHooks(namespace, staleBefore),
569
- ] as const,
570
- )
571
- .execute();
572
- }),
573
- ).rejects.toThrow(ConcurrencyConflictError);
574
- });
575
-
576
- it("should not retrieve events from different namespace", async () => {
577
- const nonce = "test-nonce-7";
578
-
579
- await fragment.inContext(async function () {
580
- await this.handlerTx()
581
- .mutate(({ forSchema }) => {
582
- const uow = forSchema(internalSchema);
583
- uow.create("fragno_hooks", {
584
- namespace: "other-namespace",
585
- hookName: "onOther",
586
- payload: { test: "other" },
587
- status: "pending",
588
- attempts: 0,
589
- maxAttempts: 5,
590
- lastAttemptAt: null,
591
- nextRetryAt: null,
592
- error: null,
593
- nonce,
594
- });
595
- })
596
- .execute();
597
- });
598
-
599
- const events = await fragment.inContext(async function () {
600
- return await this.handlerTx()
601
- .withServiceCalls(
602
- () => [fragment.services.hookService.getPendingHookEvents("test-namespace")] as const,
603
- )
604
- .transform(({ serviceResult: [result] }) => result)
605
- .execute();
606
- });
607
-
608
- const otherEvent = events.find((e) => e.hookName === "onOther");
609
- expect(otherEvent).toBeUndefined();
610
- });
611
-
612
- it("should not retrieve events not yet ready for retry", async () => {
613
- const nonce = "test-nonce-8";
614
- let eventId: FragnoId;
615
-
616
- const futureTime = new Date(Date.now() + 60000);
617
-
618
- await fragment.inContext(async function () {
619
- const createdId = await this.handlerTx()
620
- .mutate(({ forSchema }) => {
621
- const uow = forSchema(internalSchema);
622
- return uow.create("fragno_hooks", {
623
- namespace: "test-namespace",
624
- hookName: "onFuture",
625
- payload: { test: "future" },
626
- status: "pending",
627
- attempts: 1,
628
- maxAttempts: 5,
629
- lastAttemptAt: new Date(),
630
- nextRetryAt: futureTime,
631
- error: "Previous error",
632
- nonce,
633
- });
634
- })
635
- .execute();
636
- eventId = createdId;
637
- });
638
-
639
- const events = await fragment.inContext(async function () {
640
- return await this.handlerTx()
641
- .withServiceCalls(
642
- () => [fragment.services.hookService.getPendingHookEvents("test-namespace")] as const,
643
- )
644
- .transform(({ serviceResult: [result] }) => result)
645
- .execute();
646
- });
647
-
648
- const futureEvent = events.find((e) => e.id.externalId === eventId.externalId);
649
- expect(futureEvent).toBeUndefined();
650
- });
651
-
652
517
  it("should claim only ready pending events and mark them processing", async () => {
653
518
  const namespace = "claim-ready";
654
519
  const pastTime = new Date(Date.now() - 10000);
@@ -657,6 +522,7 @@ describe("Hook Service", () => {
657
522
  let nullRetryId!: FragnoId;
658
523
  let pastRetryId!: FragnoId;
659
524
  let futureRetryId!: FragnoId;
525
+ let otherNamespaceId!: FragnoId;
660
526
 
661
527
  await fragment.inContext(async function () {
662
528
  await this.handlerTx()
@@ -698,6 +564,18 @@ describe("Hook Service", () => {
698
564
  error: "Previous error",
699
565
  nonce: "test-nonce-claim-future",
700
566
  });
567
+ otherNamespaceId = uow.create("fragno_hooks", {
568
+ namespace: "other-namespace",
569
+ hookName: "onOtherNamespace",
570
+ payload: { test: "other" },
571
+ status: "pending",
572
+ attempts: 0,
573
+ maxAttempts: 5,
574
+ lastAttemptAt: null,
575
+ nextRetryAt: null,
576
+ error: null,
577
+ nonce: "test-nonce-claim-other",
578
+ });
701
579
  })
702
580
  .execute();
703
581
  });
@@ -716,30 +594,36 @@ describe("Hook Service", () => {
716
594
  expect(claimedIds.has(nullRetryId.externalId)).toBe(true);
717
595
  expect(claimedIds.has(pastRetryId.externalId)).toBe(true);
718
596
  expect(claimedIds.has(futureRetryId.externalId)).toBe(false);
597
+ expect(claimedIds.has(otherNamespaceId.externalId)).toBe(false);
719
598
 
720
- const [nullEvent, pastEvent, futureEvent] = await fragment.inContext(async function () {
721
- return await this.handlerTx()
722
- .withServiceCalls(
723
- () =>
724
- [
725
- fragment.services.hookService.getHookById(nullRetryId),
726
- fragment.services.hookService.getHookById(pastRetryId),
727
- fragment.services.hookService.getHookById(futureRetryId),
728
- ] as const,
729
- )
730
- .transform(({ serviceResult: [nullResult, pastResult, futureResult] }) => [
731
- nullResult,
732
- pastResult,
733
- futureResult,
734
- ])
735
- .execute();
736
- });
599
+ const [nullEvent, pastEvent, futureEvent, otherEvent] = await fragment.inContext(
600
+ async function () {
601
+ return await this.handlerTx()
602
+ .withServiceCalls(
603
+ () =>
604
+ [
605
+ fragment.services.hookService.getHookById(nullRetryId),
606
+ fragment.services.hookService.getHookById(pastRetryId),
607
+ fragment.services.hookService.getHookById(futureRetryId),
608
+ fragment.services.hookService.getHookById(otherNamespaceId),
609
+ ] as const,
610
+ )
611
+ .transform(({ serviceResult: [nullResult, pastResult, futureResult, otherResult] }) => [
612
+ nullResult,
613
+ pastResult,
614
+ futureResult,
615
+ otherResult,
616
+ ])
617
+ .execute();
618
+ },
619
+ );
737
620
 
738
621
  expect(nullEvent?.status).toBe("processing");
739
622
  expect(nullEvent?.lastAttemptAt).toBeInstanceOf(Date);
740
623
  expect(pastEvent?.status).toBe("processing");
741
624
  expect(pastEvent?.lastAttemptAt).toBeInstanceOf(Date);
742
625
  expect(futureEvent?.status).toBe("pending");
626
+ expect(otherEvent?.status).toBe("pending");
743
627
  });
744
628
 
745
629
  it("should return claimed ids with incremented versions", async () => {
@@ -781,6 +665,94 @@ describe("Hook Service", () => {
781
665
  expect(claimed[0]?.id.version).toBe(createdId.version + 1);
782
666
  });
783
667
 
668
+ it("should claim stale processing events and return stuck metadata", async () => {
669
+ const namespace = "claim-stuck";
670
+ const staleBefore = dbNow().plus({ minutes: -1 });
671
+ const staleLastAttemptAt = new Date(Date.now() - 5 * 60_000);
672
+ const freshLastAttemptAt = new Date(Date.now() - 10_000);
673
+ const staleNextRetryAt = new Date(Date.now() + 60_000);
674
+
675
+ let staleId!: FragnoId;
676
+ let freshId!: FragnoId;
677
+
678
+ await fragment.inContext(async function () {
679
+ await this.handlerTx()
680
+ .mutate(({ forSchema }) => {
681
+ const uow = forSchema(internalSchema);
682
+ staleId = uow.create("fragno_hooks", {
683
+ namespace,
684
+ hookName: "onStuck",
685
+ payload: { test: "stuck" },
686
+ status: "processing",
687
+ attempts: 1,
688
+ maxAttempts: 5,
689
+ lastAttemptAt: staleLastAttemptAt,
690
+ nextRetryAt: staleNextRetryAt,
691
+ error: null,
692
+ nonce: "test-nonce-stuck",
693
+ });
694
+ freshId = uow.create("fragno_hooks", {
695
+ namespace,
696
+ hookName: "onFresh",
697
+ payload: { test: "fresh" },
698
+ status: "processing",
699
+ attempts: 0,
700
+ maxAttempts: 5,
701
+ lastAttemptAt: freshLastAttemptAt,
702
+ nextRetryAt: null,
703
+ error: null,
704
+ nonce: "test-nonce-fresh",
705
+ });
706
+ })
707
+ .execute();
708
+ });
709
+
710
+ const claimResult = await fragment.inContext(async function () {
711
+ return await this.handlerTx()
712
+ .withServiceCalls(
713
+ () =>
714
+ [
715
+ fragment.services.hookService.claimStuckProcessingHookEvents(namespace, staleBefore),
716
+ ] as const,
717
+ )
718
+ .transform(({ serviceResult: [result] }) => result)
719
+ .execute();
720
+ });
721
+
722
+ expect(claimResult.events).toHaveLength(1);
723
+ expect(claimResult.events[0]?.hookName).toBe("onStuck");
724
+ expect(claimResult.events[0]?.id.version).toBe(staleId.version + 1);
725
+
726
+ expect(claimResult.stuckEvents).toHaveLength(1);
727
+ expect(claimResult.stuckEvents[0]?.hookName).toBe("onStuck");
728
+ expect(claimResult.stuckEvents[0]?.id.externalId).toBe(staleId.externalId);
729
+ expect(claimResult.stuckEvents[0]?.id.version).toBe(staleId.version);
730
+ expect(claimResult.stuckEvents[0]?.lastAttemptAt?.getTime()).toBe(staleLastAttemptAt.getTime());
731
+ expect(claimResult.stuckEvents[0]?.nextRetryAt?.getTime()).toBe(staleNextRetryAt.getTime());
732
+
733
+ const [staleEvent, freshEvent] = await fragment.inContext(async function () {
734
+ return await this.handlerTx()
735
+ .withServiceCalls(
736
+ () =>
737
+ [
738
+ fragment.services.hookService.getHookById(staleId),
739
+ fragment.services.hookService.getHookById(freshId),
740
+ ] as const,
741
+ )
742
+ .transform(({ serviceResult: [staleResult, freshResult] }) => [staleResult, freshResult])
743
+ .execute();
744
+ });
745
+
746
+ expect(staleEvent?.status).toBe("processing");
747
+ expect(staleEvent?.nextRetryAt).toBeNull();
748
+ expect(staleEvent?.lastAttemptAt).toBeInstanceOf(Date);
749
+ expect(staleEvent?.lastAttemptAt?.getTime()).toBeGreaterThan(staleLastAttemptAt.getTime());
750
+ expect(freshEvent?.lastAttemptAt).toBeInstanceOf(Date);
751
+ expect(
752
+ Math.abs((freshEvent?.lastAttemptAt?.getTime() ?? 0) - freshLastAttemptAt.getTime()),
753
+ ).toBeLessThan(2000);
754
+ });
755
+
784
756
  it("should return now when pending hooks have no nextRetryAt", async () => {
785
757
  const namespace = "wake-now";
786
758