@fragno-dev/db 0.2.1 → 0.3.0

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 (362) hide show
  1. package/.turbo/turbo-build.log +206 -140
  2. package/CHANGELOG.md +67 -0
  3. package/README.md +30 -9
  4. package/dist/adapters/adapters.d.ts +23 -21
  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/driver-config.d.ts +16 -1
  8. package/dist/adapters/generic-sql/driver-config.d.ts.map +1 -1
  9. package/dist/adapters/generic-sql/driver-config.js +23 -1
  10. package/dist/adapters/generic-sql/driver-config.js.map +1 -1
  11. package/dist/adapters/generic-sql/generic-sql-adapter.d.ts +27 -9
  12. package/dist/adapters/generic-sql/generic-sql-adapter.d.ts.map +1 -1
  13. package/dist/adapters/generic-sql/generic-sql-adapter.js +55 -16
  14. package/dist/adapters/generic-sql/generic-sql-adapter.js.map +1 -1
  15. package/dist/adapters/generic-sql/generic-sql-uow-executor.js +129 -3
  16. package/dist/adapters/generic-sql/generic-sql-uow-executor.js.map +1 -1
  17. package/dist/adapters/generic-sql/migration/dialect/mysql.js +24 -5
  18. package/dist/adapters/generic-sql/migration/dialect/mysql.js.map +1 -1
  19. package/dist/adapters/generic-sql/migration/dialect/postgres.js +6 -5
  20. package/dist/adapters/generic-sql/migration/dialect/postgres.js.map +1 -1
  21. package/dist/adapters/generic-sql/migration/dialect/sqlite.js +21 -10
  22. package/dist/adapters/generic-sql/migration/dialect/sqlite.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 +8 -8
  25. package/dist/adapters/generic-sql/migration/prepared-migrations.js.map +1 -1
  26. package/dist/adapters/generic-sql/migration/sql-generator.js +74 -51
  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 +6 -5
  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 +42 -4
  31. package/dist/adapters/generic-sql/query/cursor-utils.js.map +1 -1
  32. package/dist/adapters/generic-sql/query/generic-sql-uow-operation-compiler.js +25 -17
  33. package/dist/adapters/generic-sql/query/generic-sql-uow-operation-compiler.js.map +1 -1
  34. package/dist/adapters/generic-sql/query/select-builder.js +5 -3
  35. package/dist/adapters/generic-sql/query/select-builder.js.map +1 -1
  36. package/dist/adapters/generic-sql/query/sql-query-compiler.js +15 -12
  37. package/dist/adapters/generic-sql/query/sql-query-compiler.js.map +1 -1
  38. package/dist/adapters/generic-sql/query/where-builder.js +38 -28
  39. package/dist/adapters/generic-sql/query/where-builder.js.map +1 -1
  40. package/dist/adapters/generic-sql/sqlite-storage.d.ts +13 -0
  41. package/dist/adapters/generic-sql/sqlite-storage.d.ts.map +1 -0
  42. package/dist/adapters/generic-sql/sqlite-storage.js +15 -0
  43. package/dist/adapters/generic-sql/sqlite-storage.js.map +1 -0
  44. package/dist/adapters/generic-sql/uow-decoder.js +7 -3
  45. package/dist/adapters/generic-sql/uow-decoder.js.map +1 -1
  46. package/dist/adapters/generic-sql/uow-encoder.js +28 -8
  47. package/dist/adapters/generic-sql/uow-encoder.js.map +1 -1
  48. package/dist/adapters/in-memory/condition-evaluator.js +131 -0
  49. package/dist/adapters/in-memory/condition-evaluator.js.map +1 -0
  50. package/dist/adapters/in-memory/errors.d.ts +13 -0
  51. package/dist/adapters/in-memory/errors.d.ts.map +1 -0
  52. package/dist/adapters/in-memory/errors.js +23 -0
  53. package/dist/adapters/in-memory/errors.js.map +1 -0
  54. package/dist/adapters/in-memory/in-memory-adapter.d.ts +27 -0
  55. package/dist/adapters/in-memory/in-memory-adapter.d.ts.map +1 -0
  56. package/dist/adapters/in-memory/in-memory-adapter.js +176 -0
  57. package/dist/adapters/in-memory/in-memory-adapter.js.map +1 -0
  58. package/dist/adapters/in-memory/in-memory-uow.js +648 -0
  59. package/dist/adapters/in-memory/in-memory-uow.js.map +1 -0
  60. package/dist/adapters/in-memory/index.d.ts +4 -0
  61. package/dist/adapters/in-memory/index.js +4 -0
  62. package/dist/adapters/in-memory/options.d.ts +28 -0
  63. package/dist/adapters/in-memory/options.d.ts.map +1 -0
  64. package/dist/adapters/in-memory/options.js +61 -0
  65. package/dist/adapters/in-memory/options.js.map +1 -0
  66. package/dist/adapters/in-memory/reference-resolution.js +26 -0
  67. package/dist/adapters/in-memory/reference-resolution.js.map +1 -0
  68. package/dist/adapters/in-memory/sorted-array-index.js +129 -0
  69. package/dist/adapters/in-memory/sorted-array-index.js.map +1 -0
  70. package/dist/adapters/in-memory/store.js +71 -0
  71. package/dist/adapters/in-memory/store.js.map +1 -0
  72. package/dist/adapters/in-memory/value-comparison.js +28 -0
  73. package/dist/adapters/in-memory/value-comparison.js.map +1 -0
  74. package/dist/adapters/shared/from-unit-of-work-compiler.js.map +1 -1
  75. package/dist/adapters/shared/uow-operation-compiler.js +11 -11
  76. package/dist/adapters/shared/uow-operation-compiler.js.map +1 -1
  77. package/dist/adapters/sql/index.d.ts +5 -0
  78. package/dist/adapters/sql/index.js +4 -0
  79. package/dist/db-fragment-definition-builder.d.ts +45 -96
  80. package/dist/db-fragment-definition-builder.d.ts.map +1 -1
  81. package/dist/db-fragment-definition-builder.js +121 -99
  82. package/dist/db-fragment-definition-builder.js.map +1 -1
  83. package/dist/dispatchers/cloudflare-do/index.d.ts +26 -0
  84. package/dist/dispatchers/cloudflare-do/index.d.ts.map +1 -0
  85. package/dist/dispatchers/cloudflare-do/index.js +63 -0
  86. package/dist/dispatchers/cloudflare-do/index.js.map +1 -0
  87. package/dist/dispatchers/node/index.d.ts +17 -0
  88. package/dist/dispatchers/node/index.d.ts.map +1 -0
  89. package/dist/dispatchers/node/index.js +59 -0
  90. package/dist/dispatchers/node/index.js.map +1 -0
  91. package/dist/fragments/internal-fragment.d.ts +172 -9
  92. package/dist/fragments/internal-fragment.d.ts.map +1 -1
  93. package/dist/fragments/internal-fragment.js +193 -74
  94. package/dist/fragments/internal-fragment.js.map +1 -1
  95. package/dist/fragments/internal-fragment.routes.js +29 -0
  96. package/dist/fragments/internal-fragment.routes.js.map +1 -0
  97. package/dist/fragments/internal-fragment.schema.d.ts +9 -0
  98. package/dist/fragments/internal-fragment.schema.d.ts.map +1 -0
  99. package/dist/fragments/internal-fragment.schema.js +22 -0
  100. package/dist/fragments/internal-fragment.schema.js.map +1 -0
  101. package/dist/hooks/durable-hooks-processor.d.ts +14 -0
  102. package/dist/hooks/durable-hooks-processor.d.ts.map +1 -0
  103. package/dist/hooks/durable-hooks-processor.js +32 -0
  104. package/dist/hooks/durable-hooks-processor.js.map +1 -0
  105. package/dist/hooks/hooks.d.ts +47 -4
  106. package/dist/hooks/hooks.d.ts.map +1 -1
  107. package/dist/hooks/hooks.js +106 -39
  108. package/dist/hooks/hooks.js.map +1 -1
  109. package/dist/migration-engine/auto-from-schema.js +14 -11
  110. package/dist/migration-engine/auto-from-schema.js.map +1 -1
  111. package/dist/migration-engine/generation-engine.d.ts +16 -10
  112. package/dist/migration-engine/generation-engine.d.ts.map +1 -1
  113. package/dist/migration-engine/generation-engine.js +72 -33
  114. package/dist/migration-engine/generation-engine.js.map +1 -1
  115. package/dist/migration-engine/shared.js.map +1 -1
  116. package/dist/mod.d.ts +17 -10
  117. package/dist/mod.d.ts.map +1 -1
  118. package/dist/mod.js +14 -8
  119. package/dist/mod.js.map +1 -1
  120. package/dist/naming/sql-naming.d.ts +19 -0
  121. package/dist/naming/sql-naming.d.ts.map +1 -0
  122. package/dist/naming/sql-naming.js +116 -0
  123. package/dist/naming/sql-naming.js.map +1 -0
  124. package/dist/node_modules/.pnpm/{rou3@0.7.10 → rou3@0.7.12}/node_modules/rou3/dist/index.js +8 -5
  125. package/dist/node_modules/.pnpm/rou3@0.7.12/node_modules/rou3/dist/index.js.map +1 -0
  126. package/dist/outbox/outbox-builder.js +156 -0
  127. package/dist/outbox/outbox-builder.js.map +1 -0
  128. package/dist/outbox/outbox.d.ts +52 -0
  129. package/dist/outbox/outbox.d.ts.map +1 -0
  130. package/dist/outbox/outbox.js +37 -0
  131. package/dist/outbox/outbox.js.map +1 -0
  132. package/dist/packages/fragno/dist/api/fragment-definition-builder.js +3 -2
  133. package/dist/packages/fragno/dist/api/fragment-definition-builder.js.map +1 -1
  134. package/dist/packages/fragno/dist/api/fragment-instantiator.js +164 -20
  135. package/dist/packages/fragno/dist/api/fragment-instantiator.js.map +1 -1
  136. package/dist/packages/fragno/dist/api/request-input-context.js +67 -0
  137. package/dist/packages/fragno/dist/api/request-input-context.js.map +1 -1
  138. package/dist/packages/fragno/dist/api/route.js +14 -1
  139. package/dist/packages/fragno/dist/api/route.js.map +1 -1
  140. package/dist/packages/fragno/dist/internal/trace-context.js +12 -0
  141. package/dist/packages/fragno/dist/internal/trace-context.js.map +1 -0
  142. package/dist/query/column-defaults.js +20 -4
  143. package/dist/query/column-defaults.js.map +1 -1
  144. package/dist/query/cursor.d.ts +3 -1
  145. package/dist/query/cursor.d.ts.map +1 -1
  146. package/dist/query/cursor.js +45 -14
  147. package/dist/query/cursor.js.map +1 -1
  148. package/dist/query/db-now.d.ts +8 -0
  149. package/dist/query/db-now.d.ts.map +1 -0
  150. package/dist/query/db-now.js +7 -0
  151. package/dist/query/db-now.js.map +1 -0
  152. package/dist/query/serialize/create-sql-serializer.js +3 -2
  153. package/dist/query/serialize/create-sql-serializer.js.map +1 -1
  154. package/dist/query/serialize/dialect/mysql-serializer.js +12 -6
  155. package/dist/query/serialize/dialect/mysql-serializer.js.map +1 -1
  156. package/dist/query/serialize/dialect/postgres-serializer.js +25 -7
  157. package/dist/query/serialize/dialect/postgres-serializer.js.map +1 -1
  158. package/dist/query/serialize/dialect/sqlite-serializer.js +55 -11
  159. package/dist/query/serialize/dialect/sqlite-serializer.js.map +1 -1
  160. package/dist/query/serialize/sql-serializer.js +2 -2
  161. package/dist/query/serialize/sql-serializer.js.map +1 -1
  162. package/dist/query/simple-query-interface.d.ts +6 -1
  163. package/dist/query/simple-query-interface.d.ts.map +1 -1
  164. package/dist/query/unit-of-work/execute-unit-of-work.d.ts +351 -100
  165. package/dist/query/unit-of-work/execute-unit-of-work.d.ts.map +1 -1
  166. package/dist/query/unit-of-work/execute-unit-of-work.js +440 -267
  167. package/dist/query/unit-of-work/execute-unit-of-work.js.map +1 -1
  168. package/dist/query/unit-of-work/unit-of-work.d.ts +67 -22
  169. package/dist/query/unit-of-work/unit-of-work.d.ts.map +1 -1
  170. package/dist/query/unit-of-work/unit-of-work.js +110 -13
  171. package/dist/query/unit-of-work/unit-of-work.js.map +1 -1
  172. package/dist/query/value-decoding.js +8 -5
  173. package/dist/query/value-decoding.js.map +1 -1
  174. package/dist/query/value-encoding.js +29 -9
  175. package/dist/query/value-encoding.js.map +1 -1
  176. package/dist/schema/create.d.ts +40 -14
  177. package/dist/schema/create.d.ts.map +1 -1
  178. package/dist/schema/create.js +82 -42
  179. package/dist/schema/create.js.map +1 -1
  180. package/dist/schema/generate-id.d.ts +20 -0
  181. package/dist/schema/generate-id.d.ts.map +1 -0
  182. package/dist/schema/generate-id.js +28 -0
  183. package/dist/schema/generate-id.js.map +1 -0
  184. package/dist/schema/type-conversion/create-sql-type-mapper.js +3 -2
  185. package/dist/schema/type-conversion/create-sql-type-mapper.js.map +1 -1
  186. package/dist/schema/type-conversion/dialect/sqlite.js +9 -0
  187. package/dist/schema/type-conversion/dialect/sqlite.js.map +1 -1
  188. package/dist/schema/validator.d.ts +10 -0
  189. package/dist/schema/validator.d.ts.map +1 -0
  190. package/dist/schema/validator.js +123 -0
  191. package/dist/schema/validator.js.map +1 -0
  192. package/dist/schema-output/drizzle.d.ts +30 -0
  193. package/dist/schema-output/drizzle.d.ts.map +1 -0
  194. package/dist/{adapters/drizzle/generate.js → schema-output/drizzle.js} +82 -56
  195. package/dist/schema-output/drizzle.js.map +1 -0
  196. package/dist/schema-output/prisma.d.ts +17 -0
  197. package/dist/schema-output/prisma.d.ts.map +1 -0
  198. package/dist/schema-output/prisma.js +296 -0
  199. package/dist/schema-output/prisma.js.map +1 -0
  200. package/dist/util/default-database-adapter.js +61 -0
  201. package/dist/util/default-database-adapter.js.map +1 -0
  202. package/dist/with-database.d.ts +1 -1
  203. package/dist/with-database.d.ts.map +1 -1
  204. package/dist/with-database.js +12 -3
  205. package/dist/with-database.js.map +1 -1
  206. package/package.json +43 -28
  207. package/src/adapters/adapters.ts +30 -24
  208. package/src/adapters/drizzle/migrate-drizzle.test.ts +54 -33
  209. package/src/adapters/drizzle/migration-parity-drizzle-kit.test.ts +599 -0
  210. package/src/adapters/drizzle/test-utils.ts +12 -8
  211. package/src/adapters/generic-sql/driver-config.ts +38 -0
  212. package/src/adapters/generic-sql/generic-sql-adapter.test.ts +5 -5
  213. package/src/adapters/generic-sql/generic-sql-adapter.ts +110 -24
  214. package/src/adapters/generic-sql/generic-sql-uow-executor.test.ts +54 -0
  215. package/src/adapters/generic-sql/generic-sql-uow-executor.ts +231 -3
  216. package/src/adapters/generic-sql/migration/adapter-migration-parity.test.ts +118 -0
  217. package/src/adapters/generic-sql/migration/dialect/mysql.test.ts +26 -8
  218. package/src/adapters/generic-sql/migration/dialect/mysql.ts +46 -8
  219. package/src/adapters/generic-sql/migration/dialect/postgres.test.ts +25 -7
  220. package/src/adapters/generic-sql/migration/dialect/postgres.ts +8 -4
  221. package/src/adapters/generic-sql/migration/dialect/sqlite.test.ts +47 -8
  222. package/src/adapters/generic-sql/migration/dialect/sqlite.ts +27 -12
  223. package/src/adapters/generic-sql/migration/prepared-migrations.test.ts +128 -39
  224. package/src/adapters/generic-sql/migration/prepared-migrations.ts +15 -8
  225. package/src/adapters/generic-sql/migration/sql-generator.ts +142 -65
  226. package/src/adapters/generic-sql/query/create-sql-query-compiler.ts +9 -6
  227. package/src/adapters/generic-sql/query/cursor-utils.test.ts +271 -0
  228. package/src/adapters/generic-sql/query/cursor-utils.ts +41 -6
  229. package/src/adapters/generic-sql/query/generic-sql-uow-operation-compiler.test.ts +27 -27
  230. package/src/adapters/generic-sql/query/generic-sql-uow-operation-compiler.ts +38 -24
  231. package/src/adapters/generic-sql/query/select-builder.test.ts +15 -11
  232. package/src/adapters/generic-sql/query/select-builder.ts +6 -2
  233. package/src/adapters/generic-sql/query/sql-query-compiler.test.ts +52 -2
  234. package/src/adapters/generic-sql/query/sql-query-compiler.ts +50 -15
  235. package/src/adapters/generic-sql/query/where-builder.test.ts +91 -17
  236. package/src/adapters/generic-sql/query/where-builder.ts +90 -38
  237. package/src/adapters/{kysely/kysely-adapter-pglite.test.ts → generic-sql/sql-adapter-pglite-migrations.test.ts} +6 -6
  238. package/src/adapters/generic-sql/sql-adapter-pglite-pagination.test.ts +806 -0
  239. package/src/adapters/{drizzle/drizzle-adapter-pglite.test.ts → generic-sql/sql-adapter-pglite-queries.test.ts} +11 -11
  240. package/src/adapters/generic-sql/{test/generic-drizzle-adapter-sqlite3.test.ts → sql-adapter-sqlite3-driver.test.ts} +49 -35
  241. package/src/adapters/{drizzle/drizzle-adapter-sqlite3.test.ts → generic-sql/sql-adapter-sqlite3-uow.test.ts} +48 -32
  242. package/src/adapters/{kysely/kysely-adapter-sqlocal.test.ts → generic-sql/sql-adapter-sqlocal.test.ts} +6 -6
  243. package/src/adapters/generic-sql/sqlite-storage.ts +20 -0
  244. package/src/adapters/generic-sql/uow-decoder.test.ts +1 -1
  245. package/src/adapters/generic-sql/uow-decoder.ts +21 -3
  246. package/src/adapters/generic-sql/uow-encoder.test.ts +33 -2
  247. package/src/adapters/generic-sql/uow-encoder.ts +50 -11
  248. package/src/adapters/in-memory/condition-evaluator.test.ts +193 -0
  249. package/src/adapters/in-memory/condition-evaluator.ts +275 -0
  250. package/src/adapters/in-memory/errors.ts +20 -0
  251. package/src/adapters/in-memory/in-memory-adapter.ts +277 -0
  252. package/src/adapters/in-memory/in-memory-uow.mutations.test.ts +296 -0
  253. package/src/adapters/in-memory/in-memory-uow.retrieval.test.ts +100 -0
  254. package/src/adapters/in-memory/in-memory-uow.ts +1348 -0
  255. package/src/adapters/in-memory/index.ts +3 -0
  256. package/src/adapters/in-memory/options.test.ts +41 -0
  257. package/src/adapters/in-memory/options.ts +87 -0
  258. package/src/adapters/in-memory/reference-resolution.test.ts +50 -0
  259. package/src/adapters/in-memory/reference-resolution.ts +67 -0
  260. package/src/adapters/in-memory/sorted-array-index.test.ts +123 -0
  261. package/src/adapters/in-memory/sorted-array-index.ts +228 -0
  262. package/src/adapters/in-memory/store.test.ts +68 -0
  263. package/src/adapters/in-memory/store.ts +145 -0
  264. package/src/adapters/in-memory/value-comparison.ts +53 -0
  265. package/src/adapters/in-memory/value-normalization.test.ts +57 -0
  266. package/src/adapters/prisma/prisma-adapter-sqlite3.test.ts +1163 -0
  267. package/src/adapters/shared/from-unit-of-work-compiler.ts +3 -1
  268. package/src/adapters/shared/uow-operation-compiler.ts +26 -16
  269. package/src/adapters/sql/index.ts +12 -0
  270. package/src/db-fragment-definition-builder.test.ts +88 -54
  271. package/src/db-fragment-definition-builder.ts +201 -322
  272. package/src/db-fragment-instantiator.test.ts +169 -101
  273. package/src/db-fragment-integration.test.ts +301 -149
  274. package/src/dispatchers/cloudflare-do/index.test.ts +73 -0
  275. package/src/dispatchers/cloudflare-do/index.ts +104 -0
  276. package/src/dispatchers/node/index.test.ts +91 -0
  277. package/src/dispatchers/node/index.ts +87 -0
  278. package/src/fragments/internal-fragment.routes.ts +42 -0
  279. package/src/fragments/internal-fragment.schema.ts +51 -0
  280. package/src/fragments/internal-fragment.test.ts +730 -274
  281. package/src/fragments/internal-fragment.ts +447 -154
  282. package/src/hooks/durable-hooks-processor.test.ts +117 -0
  283. package/src/hooks/durable-hooks-processor.ts +67 -0
  284. package/src/hooks/hooks.test.ts +411 -259
  285. package/src/hooks/hooks.ts +265 -66
  286. package/src/migration-engine/auto-from-schema.test.ts +14 -14
  287. package/src/migration-engine/auto-from-schema.ts +5 -2
  288. package/src/migration-engine/create.test.ts +2 -2
  289. package/src/migration-engine/generation-engine.test.ts +229 -104
  290. package/src/migration-engine/generation-engine.ts +94 -64
  291. package/src/migration-engine/shared.ts +1 -0
  292. package/src/mod.ts +78 -30
  293. package/src/naming/sql-naming.ts +180 -0
  294. package/src/outbox/outbox-builder.ts +241 -0
  295. package/src/outbox/outbox.test.ts +253 -0
  296. package/src/outbox/outbox.ts +137 -0
  297. package/src/query/column-defaults.ts +41 -3
  298. package/src/query/condition-builder.test.ts +3 -3
  299. package/src/query/cursor.test.ts +116 -18
  300. package/src/query/cursor.ts +75 -26
  301. package/src/query/db-now.ts +6 -0
  302. package/src/query/query-type.test.ts +2 -2
  303. package/src/query/serialize/create-sql-serializer.ts +7 -2
  304. package/src/query/serialize/dialect/mysql-serializer.ts +12 -4
  305. package/src/query/serialize/dialect/postgres-serializer.ts +34 -4
  306. package/src/query/serialize/dialect/sqlite-serializer.test.ts +51 -1
  307. package/src/query/serialize/dialect/sqlite-serializer.ts +92 -9
  308. package/src/query/serialize/sql-serializer.ts +4 -4
  309. package/src/query/simple-query-interface.ts +5 -0
  310. package/src/query/unit-of-work/execute-unit-of-work.test.ts +1512 -1458
  311. package/src/query/unit-of-work/execute-unit-of-work.ts +1708 -596
  312. package/src/query/unit-of-work/tx-builder.test.ts +1041 -0
  313. package/src/query/unit-of-work/unit-of-work-coordinator.test.ts +32 -32
  314. package/src/query/unit-of-work/unit-of-work-types.test.ts +1 -1
  315. package/src/query/unit-of-work/unit-of-work.test.ts +231 -36
  316. package/src/query/unit-of-work/unit-of-work.ts +229 -31
  317. package/src/query/value-decoding.test.ts +13 -2
  318. package/src/query/value-decoding.ts +17 -4
  319. package/src/query/value-encoding.test.ts +85 -2
  320. package/src/query/value-encoding.ts +56 -6
  321. package/src/schema/create.test.ts +129 -42
  322. package/src/schema/create.ts +187 -47
  323. package/src/schema/generate-id.test.ts +57 -0
  324. package/src/schema/generate-id.ts +38 -0
  325. package/src/schema/serialize.test.ts +14 -2
  326. package/src/schema/type-conversion/create-sql-type-mapper.ts +7 -2
  327. package/src/schema/type-conversion/dialect/sqlite.ts +18 -0
  328. package/src/schema/type-conversion/type-mapping.test.ts +25 -1
  329. package/src/schema/validator.test.ts +197 -0
  330. package/src/schema/validator.ts +231 -0
  331. package/src/{adapters/drizzle/generate.test.ts → schema-output/drizzle.test.ts} +179 -129
  332. package/src/{adapters/drizzle/generate.ts → schema-output/drizzle.ts} +143 -93
  333. package/src/schema-output/prisma.test.ts +536 -0
  334. package/src/schema-output/prisma.ts +573 -0
  335. package/src/util/default-database-adapter.ts +106 -0
  336. package/src/with-database.ts +22 -3
  337. package/tsdown.config.ts +6 -4
  338. package/dist/adapters/drizzle/drizzle-adapter.d.ts +0 -20
  339. package/dist/adapters/drizzle/drizzle-adapter.d.ts.map +0 -1
  340. package/dist/adapters/drizzle/drizzle-adapter.js +0 -27
  341. package/dist/adapters/drizzle/drizzle-adapter.js.map +0 -1
  342. package/dist/adapters/drizzle/generate.d.ts +0 -30
  343. package/dist/adapters/drizzle/generate.d.ts.map +0 -1
  344. package/dist/adapters/drizzle/generate.js.map +0 -1
  345. package/dist/adapters/kysely/kysely-adapter.d.ts +0 -19
  346. package/dist/adapters/kysely/kysely-adapter.d.ts.map +0 -1
  347. package/dist/adapters/kysely/kysely-adapter.js +0 -17
  348. package/dist/adapters/kysely/kysely-adapter.js.map +0 -1
  349. package/dist/adapters/shared/table-name-mapper.d.ts +0 -12
  350. package/dist/adapters/shared/table-name-mapper.d.ts.map +0 -1
  351. package/dist/adapters/shared/table-name-mapper.js +0 -43
  352. package/dist/adapters/shared/table-name-mapper.js.map +0 -1
  353. package/dist/node_modules/.pnpm/rou3@0.7.10/node_modules/rou3/dist/index.js.map +0 -1
  354. package/dist/schema-generator/schema-generator.d.ts +0 -15
  355. package/dist/schema-generator/schema-generator.d.ts.map +0 -1
  356. package/src/adapters/drizzle/drizzle-adapter.ts +0 -39
  357. package/src/adapters/kysely/kysely-adapter.ts +0 -27
  358. package/src/adapters/shared/table-name-mapper.ts +0 -50
  359. package/src/schema-generator/schema-generator.ts +0 -12
  360. package/src/shared/config.ts +0 -10
  361. package/src/shared/connection-pool.ts +0 -24
  362. package/src/shared/prisma.ts +0 -45
@@ -1,35 +1,37 @@
1
- import "./query/unit-of-work/unit-of-work.js";
2
- import { executeRestrictedUnitOfWork, executeServiceTx, executeTxArray, executeTxCallbacks } from "./query/unit-of-work/execute-unit-of-work.js";
3
- import { prepareHookMutations, processHooks } from "./hooks/hooks.js";
1
+ import { createHandlerTxBuilder, createServiceTxBuilder } from "./query/unit-of-work/execute-unit-of-work.js";
2
+ import { createHookScheduler, prepareHookMutations } from "./hooks/hooks.js";
3
+ import { sanitizeNamespace } from "./naming/sql-naming.js";
4
+ import { resolveDatabaseAdapter } from "./util/default-database-adapter.js";
4
5
 
5
6
  //#region src/db-fragment-definition-builder.ts
6
7
  /**
7
8
  * Create database context from options.
8
9
  * This extracts the database adapter and creates the ORM instance.
9
10
  */
10
- function createDatabaseContext(options, schema, namespace) {
11
- const databaseAdapter = options.databaseAdapter;
12
- if (!databaseAdapter) throw new Error("Database fragment requires a database adapter to be provided in options.databaseAdapter");
11
+ function createDatabaseContext(options, schema) {
12
+ const databaseAdapter = resolveDatabaseAdapter(options, schema);
13
+ const namespace = resolveDatabaseNamespace(options, schema);
13
14
  return {
14
15
  databaseAdapter,
15
16
  db: databaseAdapter.createQueryEngine(schema, namespace)
16
17
  };
17
18
  }
19
+ function resolveDatabaseNamespace(options, schema) {
20
+ return options.databaseNamespace !== void 0 ? options.databaseNamespace ?? null : sanitizeNamespace(schema.name);
21
+ }
18
22
  /**
19
23
  * Builder for database fragments that wraps the core fragment builder
20
24
  * and provides database-specific functionality.
21
25
  *
22
- * Database fragments always require FragnoPublicConfigWithDatabase (which includes databaseAdapter).
26
+ * Database fragments use FragnoPublicConfigWithDatabase and default the adapter when possible.
23
27
  */
24
28
  var DatabaseFragmentDefinitionBuilder = class DatabaseFragmentDefinitionBuilder {
25
29
  #baseBuilder;
26
30
  #schema;
27
- #namespace;
28
31
  #hooksFactory;
29
- constructor(baseBuilder, schema, namespace, hooksFactory) {
32
+ constructor(baseBuilder, schema, hooksFactory) {
30
33
  this.#baseBuilder = baseBuilder;
31
34
  this.#schema = schema;
32
- this.#namespace = namespace ?? baseBuilder.name;
33
35
  this.#hooksFactory = hooksFactory;
34
36
  }
35
37
  /**
@@ -38,7 +40,8 @@ var DatabaseFragmentDefinitionBuilder = class DatabaseFragmentDefinitionBuilder
38
40
  */
39
41
  withDependencies(fn) {
40
42
  const wrappedFn = (context) => {
41
- const dbContext = createDatabaseContext(context.options, this.#schema, this.#namespace);
43
+ const dbContext = createDatabaseContext(context.options, this.#schema);
44
+ const namespace = resolveDatabaseNamespace(context.options, this.#schema);
42
45
  const userDeps = fn({
43
46
  config: context.config,
44
47
  options: context.options,
@@ -49,7 +52,7 @@ var DatabaseFragmentDefinitionBuilder = class DatabaseFragmentDefinitionBuilder
49
52
  const implicitDeps = {
50
53
  db: dbContext.db,
51
54
  schema: this.#schema,
52
- namespace: this.#namespace,
55
+ namespace,
53
56
  createUnitOfWork: createUow
54
57
  };
55
58
  return {
@@ -57,13 +60,13 @@ var DatabaseFragmentDefinitionBuilder = class DatabaseFragmentDefinitionBuilder
57
60
  ...implicitDeps
58
61
  };
59
62
  };
60
- return new DatabaseFragmentDefinitionBuilder(this.#baseBuilder.withDependencies(wrappedFn), this.#schema, this.#namespace, this.#hooksFactory);
63
+ return new DatabaseFragmentDefinitionBuilder(this.#baseBuilder.withDependencies(wrappedFn), this.#schema, this.#hooksFactory);
61
64
  }
62
65
  providesBaseService(fn) {
63
- return new DatabaseFragmentDefinitionBuilder(this.#baseBuilder.providesBaseService(fn), this.#schema, this.#namespace, this.#hooksFactory);
66
+ return new DatabaseFragmentDefinitionBuilder(this.#baseBuilder.providesBaseService(fn), this.#schema, this.#hooksFactory);
64
67
  }
65
68
  providesService(serviceName, fn) {
66
- return new DatabaseFragmentDefinitionBuilder(this.#baseBuilder.providesService(serviceName, fn), this.#schema, this.#namespace, this.#hooksFactory);
69
+ return new DatabaseFragmentDefinitionBuilder(this.#baseBuilder.providesService(serviceName, fn), this.#schema, this.#hooksFactory);
67
70
  }
68
71
  /**
69
72
  * Provide a private service that is only accessible to the fragment author.
@@ -73,7 +76,7 @@ var DatabaseFragmentDefinitionBuilder = class DatabaseFragmentDefinitionBuilder
73
76
  * to later ones.
74
77
  */
75
78
  providesPrivateService(serviceName, fn) {
76
- return new DatabaseFragmentDefinitionBuilder(this.#baseBuilder.providesPrivateService(serviceName, fn), this.#schema, this.#namespace, this.#hooksFactory);
79
+ return new DatabaseFragmentDefinitionBuilder(this.#baseBuilder.providesPrivateService(serviceName, fn), this.#schema, this.#hooksFactory);
77
80
  }
78
81
  /**
79
82
  * Define durable hooks for this fragment.
@@ -86,7 +89,7 @@ var DatabaseFragmentDefinitionBuilder = class DatabaseFragmentDefinitionBuilder
86
89
  * ```ts
87
90
  * .provideHooks(({ defineHook, config }) => ({
88
91
  * onSubscribe: defineHook(async function (payload: { email: string }) {
89
- * // 'this' context available (HookServiceContext with nonce)
92
+ * // 'this' context available (HookServiceContext with idempotencyKey)
90
93
  * await config.onSubscribe?.(payload.email);
91
94
  * }),
92
95
  * }))
@@ -103,7 +106,7 @@ var DatabaseFragmentDefinitionBuilder = class DatabaseFragmentDefinitionBuilder
103
106
  defineHook
104
107
  });
105
108
  };
106
- const newBuilder = new DatabaseFragmentDefinitionBuilder(this.#baseBuilder, this.#schema, this.#namespace);
109
+ const newBuilder = new DatabaseFragmentDefinitionBuilder(this.#baseBuilder, this.#schema);
107
110
  newBuilder.#hooksFactory = hooksFactory;
108
111
  return newBuilder;
109
112
  }
@@ -112,14 +115,14 @@ var DatabaseFragmentDefinitionBuilder = class DatabaseFragmentDefinitionBuilder
112
115
  * Delegates to the base builder.
113
116
  */
114
117
  usesService(serviceName) {
115
- return new DatabaseFragmentDefinitionBuilder(this.#baseBuilder.usesService(serviceName), this.#schema, this.#namespace, this.#hooksFactory);
118
+ return new DatabaseFragmentDefinitionBuilder(this.#baseBuilder.usesService(serviceName), this.#schema, this.#hooksFactory);
116
119
  }
117
120
  /**
118
121
  * Declare that this fragment uses an optional service provided by the runtime.
119
122
  * Delegates to the base builder.
120
123
  */
121
124
  usesOptionalService(serviceName) {
122
- return new DatabaseFragmentDefinitionBuilder(this.#baseBuilder.usesOptionalService(serviceName), this.#schema, this.#namespace, this.#hooksFactory);
125
+ return new DatabaseFragmentDefinitionBuilder(this.#baseBuilder.usesOptionalService(serviceName), this.#schema, this.#hooksFactory);
123
126
  }
124
127
  /**
125
128
  * Build the final database fragment definition.
@@ -138,11 +141,12 @@ var DatabaseFragmentDefinitionBuilder = class DatabaseFragmentDefinitionBuilder
138
141
  userDeps = {};
139
142
  } else throw error;
140
143
  }
141
- const { db } = createDatabaseContext(context.options, this.#schema, this.#namespace);
144
+ const { db } = createDatabaseContext(context.options, this.#schema);
145
+ const namespace = resolveDatabaseNamespace(context.options, this.#schema);
142
146
  const implicitDeps = {
143
147
  db,
144
148
  schema: this.#schema,
145
- namespace: this.#namespace,
149
+ namespace,
146
150
  createUnitOfWork: () => db.createUnitOfWork()
147
151
  };
148
152
  return {
@@ -151,96 +155,114 @@ var DatabaseFragmentDefinitionBuilder = class DatabaseFragmentDefinitionBuilder
151
155
  };
152
156
  };
153
157
  const builderWithStorage = this.#baseBuilder.withExternalRequestStorage(({ options }) => {
154
- return createDatabaseContext(options, this.#schema, this.#namespace).databaseAdapter.contextStorage;
158
+ return createDatabaseContext(options, this.#schema).databaseAdapter.contextStorage;
155
159
  }).withRequestStorage(({ options }) => {
156
- return { uow: createDatabaseContext(options, this.#schema, this.#namespace).db.createUnitOfWork() };
160
+ return { uow: createDatabaseContext(options, this.#schema).db.createUnitOfWork() };
157
161
  });
158
162
  const internalFragmentFactory = baseDef.linkedFragments?.["_fragno_internal"];
159
- return {
160
- ...builderWithStorage.withThisContext(({ storage, config, options }) => {
161
- const hooksConfig = this.#hooksFactory ? {
162
- hooks: this.#hooksFactory({
163
- config,
164
- options
165
- }),
166
- namespace: this.#namespace,
167
- internalFragment: internalFragmentFactory({
168
- config,
169
- options
170
- })
171
- } : void 0;
172
- function forSchema(schema) {
173
- const uow$1 = storage.getStore()?.uow;
174
- if (!uow$1) throw new Error("No UnitOfWork in context. Service must be called within a route handler OR using `withUnitOfWork`.");
175
- return uow$1.restrict().forSchema(schema);
176
- }
177
- const serviceTx = async (schema, callbacks) => {
178
- const uow$1 = storage.getStore()?.uow;
179
- if (!uow$1) throw new Error("No UnitOfWork in context. Service must be called within a route handler OR using `withUnitOfWork`.");
180
- return executeServiceTx(schema, callbacks, uow$1);
181
- };
182
- const serviceContext = {
183
- uow: forSchema,
184
- tx: serviceTx
185
- };
186
- async function uow(callback, execOptions) {
187
- const currentStorage = storage.getStore();
188
- if (!currentStorage) throw new Error("No storage in context. Handler must be called within a request context.");
189
- const wrappedCallback = async (context) => {
190
- return callback(context);
191
- };
163
+ const hooksConfigCache = /* @__PURE__ */ new WeakMap();
164
+ const createHooksConfig = (context) => {
165
+ if (!this.#hooksFactory) return;
166
+ const depsKey = typeof context.deps === "object" && context.deps !== null ? context.deps : void 0;
167
+ const cachedHooksConfig = depsKey ? hooksConfigCache.get(depsKey) : void 0;
168
+ if (cachedHooksConfig) return cachedHooksConfig;
169
+ const namespaceKey = resolveDatabaseNamespace(context.options, this.#schema) ?? this.#schema.name;
170
+ const durableHooksOptions = context.options.durableHooks;
171
+ const baseAdapter = resolveDatabaseAdapter(context.options, this.#schema);
172
+ const hookAdapter = baseAdapter.getHookProcessingAdapter?.() ?? baseAdapter;
173
+ const hookOptions = hookAdapter === baseAdapter ? context.options : {
174
+ ...context.options,
175
+ databaseAdapter: hookAdapter
176
+ };
177
+ const dbContextForHooks = createDatabaseContext(hookOptions, this.#schema);
178
+ const hooksConfig = {
179
+ hooks: this.#hooksFactory(context),
180
+ namespace: namespaceKey,
181
+ internalFragment: internalFragmentFactory({
182
+ config: context.config,
183
+ options: hookOptions
184
+ }),
185
+ handlerTx: (execOptions) => {
192
186
  const userOnBeforeMutate = execOptions?.onBeforeMutate;
193
- const userOnSuccess = execOptions?.onSuccess;
194
- return executeRestrictedUnitOfWork(wrappedCallback, {
187
+ const userOnAfterMutate = execOptions?.onAfterMutate;
188
+ return createHandlerTxBuilder({
195
189
  ...execOptions,
196
190
  createUnitOfWork: () => {
197
- currentStorage.uow.reset();
198
- if (hooksConfig) currentStorage.uow.registerSchema(hooksConfig.internalFragment.$internal.deps.schema, hooksConfig.internalFragment.$internal.deps.namespace);
199
- return currentStorage.uow;
191
+ const uow = dbContextForHooks.db.createUnitOfWork();
192
+ uow.registerSchema(hooksConfig.internalFragment.$internal.deps.schema, hooksConfig.internalFragment.$internal.deps.namespace);
193
+ return uow;
200
194
  },
201
- onBeforeMutate: (uow$1) => {
202
- if (hooksConfig) prepareHookMutations(uow$1, hooksConfig);
203
- if (userOnBeforeMutate) userOnBeforeMutate(uow$1);
195
+ onBeforeMutate: (uow) => {
196
+ prepareHookMutations(uow, hooksConfig);
197
+ if (userOnBeforeMutate) userOnBeforeMutate(uow);
204
198
  },
205
- onSuccess: async (uow$1) => {
206
- if (hooksConfig) await processHooks(hooksConfig);
207
- if (userOnSuccess) await userOnSuccess(uow$1);
199
+ onAfterMutate: async (uow) => {
200
+ hooksConfig.scheduler?.schedule().catch((error) => {
201
+ console.error("Durable hooks processing failed", error);
202
+ });
203
+ if (userOnAfterMutate) await userOnAfterMutate(uow);
208
204
  }
209
205
  });
210
- }
211
- async function handlerTx(factoryOrCallbacks, execOptions) {
212
- const currentStorage = storage.getStore();
213
- if (!currentStorage) throw new Error("No storage in context. Handler must be called within a request context.");
214
- const userOnBeforeMutate = execOptions?.onBeforeMutate;
215
- const userOnSuccess = execOptions?.onSuccess;
216
- const createUow = () => {
206
+ },
207
+ stuckProcessingTimeoutMinutes: durableHooksOptions?.stuckProcessingTimeoutMinutes,
208
+ onStuckProcessingHooks: durableHooksOptions?.onStuckProcessingHooks
209
+ };
210
+ hooksConfig.scheduler = createHookScheduler(hooksConfig);
211
+ if (depsKey) hooksConfigCache.set(depsKey, hooksConfig);
212
+ return hooksConfig;
213
+ };
214
+ const finalDef = builderWithStorage.withThisContext(({ storage, config, options, deps }) => {
215
+ const hooksConfig = createHooksConfig({
216
+ config,
217
+ options,
218
+ deps
219
+ });
220
+ const internalFragment = hooksConfig?.internalFragment ?? (internalFragmentFactory ? internalFragmentFactory({
221
+ config,
222
+ options
223
+ }) : void 0);
224
+ function serviceTx(schema) {
225
+ const uow = storage.getStore()?.uow;
226
+ if (!uow) throw new Error("No UnitOfWork in context. Service must be called within a route handler OR using `withUnitOfWork`.");
227
+ return createServiceTxBuilder(schema, uow, hooksConfig?.hooks);
228
+ }
229
+ const serviceContext = { serviceTx };
230
+ function handlerTx(execOptions) {
231
+ const currentStorage = storage.getStore();
232
+ if (!currentStorage) throw new Error("No storage in context. Handler must be called within a request context.");
233
+ const userOnBeforeMutate = execOptions?.onBeforeMutate;
234
+ const userOnAfterMutate = execOptions?.onAfterMutate;
235
+ return createHandlerTxBuilder({
236
+ ...execOptions,
237
+ createUnitOfWork: () => {
217
238
  currentStorage.uow.reset();
218
- if (hooksConfig) currentStorage.uow.registerSchema(hooksConfig.internalFragment.$internal.deps.schema, hooksConfig.internalFragment.$internal.deps.namespace);
239
+ if (internalFragment) currentStorage.uow.registerSchema(internalFragment.$internal.deps.schema, internalFragment.$internal.deps.namespace);
219
240
  return currentStorage.uow;
220
- };
221
- const options$1 = {
222
- ...execOptions,
223
- createUnitOfWork: createUow,
224
- onBeforeMutate: (uow$1) => {
225
- if (hooksConfig) prepareHookMutations(uow$1, hooksConfig);
226
- if (userOnBeforeMutate) userOnBeforeMutate(uow$1);
227
- },
228
- onSuccess: async (uow$1) => {
229
- if (hooksConfig) await processHooks(hooksConfig);
230
- if (userOnSuccess) await userOnSuccess(uow$1);
231
- }
232
- };
233
- if (typeof factoryOrCallbacks === "function") return executeTxArray(factoryOrCallbacks, options$1);
234
- else return executeTxCallbacks(factoryOrCallbacks, options$1);
235
- }
236
- return {
237
- serviceContext,
238
- handlerContext: {
239
- uow,
240
- tx: handlerTx
241
+ },
242
+ onBeforeMutate: (uow) => {
243
+ if (hooksConfig) prepareHookMutations(uow, hooksConfig);
244
+ if (userOnBeforeMutate) userOnBeforeMutate(uow);
245
+ },
246
+ onAfterMutate: async (uow) => {
247
+ if (hooksConfig?.scheduler) hooksConfig.scheduler.schedule().catch((error) => {
248
+ console.error("Durable hooks processing failed", error);
249
+ });
250
+ if (userOnAfterMutate) await userOnAfterMutate(uow);
241
251
  }
242
- };
243
- }).build(),
252
+ });
253
+ }
254
+ return {
255
+ serviceContext,
256
+ handlerContext: { handlerTx }
257
+ };
258
+ }).build();
259
+ if (this.#hooksFactory) finalDef.internalDataFactory = ({ config, options, deps }) => ({ durableHooks: createHooksConfig({
260
+ config,
261
+ options,
262
+ deps
263
+ }) });
264
+ return {
265
+ ...finalDef,
244
266
  dependencies
245
267
  };
246
268
  }
@@ -1 +1 @@
1
- {"version":3,"file":"db-fragment-definition-builder.js","names":["#baseBuilder","#schema","#namespace","#hooksFactory","implicitDeps: ImplicitDatabaseDependencies<TSchema>","uow","serviceContext: DatabaseServiceContext<THooks>","options: ExecuteRestrictedUnitOfWorkOptions","options"],"sources":["../src/db-fragment-definition-builder.ts"],"sourcesContent":["import type { AnySchema } from \"./schema/create\";\nimport type { SimpleQueryInterface } from \"./query/simple-query-interface\";\nimport type { DatabaseAdapter } from \"./adapters/adapters\";\nimport type { IUnitOfWork } from \"./query/unit-of-work/unit-of-work\";\nimport { TypedUnitOfWork, UnitOfWork } from \"./query/unit-of-work/unit-of-work\";\nimport type {\n RequestThisContext,\n FragnoPublicConfig,\n AnyFragnoInstantiatedFragment,\n} from \"@fragno-dev/core\";\nimport {\n FragmentDefinitionBuilder,\n type FragmentDefinition,\n type ServiceConstructorFn,\n} from \"@fragno-dev/core\";\nimport {\n executeRestrictedUnitOfWork,\n executeTxArray,\n executeTxCallbacks,\n executeServiceTx,\n type AwaitedPromisesInObject,\n type ExecuteRestrictedUnitOfWorkOptions,\n type HandlerTxCallbacks,\n type ServiceTxCallbacks,\n} from \"./query/unit-of-work/execute-unit-of-work\";\nimport {\n prepareHookMutations,\n processHooks,\n type HooksMap,\n type HookFn,\n type HookContext,\n} from \"./hooks/hooks\";\nimport type { InternalFragmentInstance } from \"./fragments/internal-fragment\";\n\n/**\n * Extended FragnoPublicConfig that includes a database adapter.\n * Use this type when creating fragments with database support.\n */\nexport type FragnoPublicConfigWithDatabase = FragnoPublicConfig & {\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n databaseAdapter: DatabaseAdapter<any>;\n};\n\n/**\n * Implicit dependencies that database fragments get automatically.\n * These are injected without requiring explicit configuration.\n */\nexport type ImplicitDatabaseDependencies<TSchema extends AnySchema> = {\n /**\n * Database query engine for the fragment's schema.\n */\n db: SimpleQueryInterface<TSchema>;\n /**\n * The schema definition for this fragment.\n */\n schema: TSchema;\n /**\n * The database namespace for this fragment.\n */\n namespace: string;\n /**\n * Create a new Unit of Work for database operations.\n */\n createUnitOfWork: () => IUnitOfWork;\n};\n\n/**\n * Service context for database fragments - provides restricted UOW access without execute methods.\n */\nexport type DatabaseServiceContext<THooks extends HooksMap> = RequestThisContext & {\n /**\n * Get a typed, restricted Unit of Work for the given schema.\n * @param schema - Schema to get a typed view for\n * @returns TypedUnitOfWork (restricted version without execute methods)\n */\n uow<TSchema extends AnySchema>(schema: TSchema): TypedUnitOfWork<TSchema, [], unknown, THooks>;\n\n /**\n * Execute a transaction with two-phase callbacks (retrieve + mutate).\n *\n * @param schema - Schema to use for the transaction\n * @param callbacks - Object containing retrieve and mutate callbacks\n * @returns Promise resolving to the mutation result with promises awaited 1 level deep\n *\n * @example\n * ```ts\n * return this.tx(schema, {\n * retrieve: (uow) => uow.findFirst(\"users\", ...),\n * mutate: async (uow, [user]) => {\n * await validateUser(user);\n * uow.update(\"users\", user.id, ...).check();\n * return { ok: true };\n * }\n * });\n * ```\n */\n tx<TSchema extends AnySchema, TRetrievalResults extends unknown[], TMutationResult = void>(\n schema: TSchema,\n callbacks: ServiceTxCallbacks<TSchema, TRetrievalResults, TMutationResult, THooks>,\n ): Promise<AwaitedPromisesInObject<TMutationResult>>;\n};\n\n/**\n * Handler context for database fragments - provides UOW execution with automatic retry support.\n */\nexport type DatabaseHandlerContext<THooks extends HooksMap = {}> = RequestThisContext & {\n /**\n * Execute a Unit of Work with explicit phase control and automatic retry support.\n * This method provides an API where users call forSchema to create a schema-specific\n * UOW, then call executeRetrieve() and executeMutate() to execute the phases. The entire\n * callback is re-executed on optimistic concurrency conflicts, ensuring retries work properly.\n * Automatically provides the UOW factory from context.\n * Promises in the returned object are awaited 1 level deep.\n *\n * @param callback - Async function that receives a context with forSchema, executeRetrieve, executeMutate, nonce, and currentAttempt\n * @param options - Optional configuration for retry policy and abort signal\n * @returns Promise resolving to the callback's return value with promises awaited 1 level deep\n * @throws Error if retries are exhausted or callback throws an error\n *\n * @example\n * ```ts\n * const result = await this.uow(async ({ forSchema, executeRetrieve, executeMutate, nonce, currentAttempt }) => {\n * const uow = forSchema(schema);\n * const userId = uow.create(\"users\", { name: \"John\" });\n *\n * // Execute retrieval phase\n * await executeRetrieve();\n *\n * const profileId = uow.create(\"profiles\", { userId });\n *\n * // Execute mutation phase\n * await executeMutate();\n *\n * return { userId, profileId };\n * });\n * ```\n */\n uow<TResult>(\n callback: (context: {\n forSchema: <TSchema extends AnySchema, H extends HooksMap = THooks>(\n schema: TSchema,\n hooks?: H,\n ) => TypedUnitOfWork<TSchema, [], unknown, H>;\n executeRetrieve: () => Promise<void>;\n executeMutate: () => Promise<void>;\n nonce: string;\n currentAttempt: number;\n }) => Promise<TResult> | TResult,\n options?: Omit<ExecuteRestrictedUnitOfWorkOptions, \"createUnitOfWork\">,\n ): Promise<AwaitedPromisesInObject<TResult>>;\n\n /**\n * Execute a transaction with automatic retry support.\n * Provides two overloads: array syntax (common case) and callback syntax (advanced).\n *\n * Array syntax - pass a factory function that returns service calls:\n * ```ts\n * const [transfer, notify] = await this.tx(\n * () => [\n * services.transfer({ from, to, amount }),\n * services.notify({ userId, message })\n * ]\n * );\n * ```\n *\n * Callback syntax - for handlers that need direct UOW access:\n * ```ts\n * const result = await this.tx({\n * retrieve: ({ forSchema }) => {\n * const uow = forSchema(schema);\n * uow.find(\"users\", ...);\n * return services.transfer({ ... });\n * },\n * mutate: ({ forSchema }, transferPromise) => {\n * const uow = forSchema(schema);\n * uow.create(\"auditLog\", { ... });\n * return transferPromise;\n * }\n * });\n * ```\n *\n * Note: Handler callbacks are synchronous only to prevent accidentally awaiting services in wrong place.\n */\n // Overload 1: Array syntax (common case) - accepts a factory\n tx<T extends readonly unknown[]>(\n servicesFactory: () => readonly [...{ [K in keyof T]: Promise<T[K]> }],\n options?: Omit<ExecuteRestrictedUnitOfWorkOptions, \"createUnitOfWork\">,\n ): Promise<{ [K in keyof T]: T[K] }>;\n // Overload 2: Callback syntax (advanced, sync callbacks only)\n tx<TRetrieveResult, TMutationResult>(\n callbacks: HandlerTxCallbacks<TRetrieveResult, TMutationResult, THooks>,\n options?: Omit<ExecuteRestrictedUnitOfWorkOptions, \"createUnitOfWork\">,\n ): Promise<AwaitedPromisesInObject<TMutationResult>>;\n};\n\n/**\n * Database fragment context provided to user callbacks.\n */\nexport type DatabaseFragmentContext<TSchema extends AnySchema> = {\n /**\n * Database adapter instance.\n */\n databaseAdapter: DatabaseAdapter<any>; // eslint-disable-line @typescript-eslint/no-explicit-any\n /**\n * ORM query engine for the fragment's schema.\n */\n db: SimpleQueryInterface<TSchema>;\n};\n\n/**\n * Create database context from options.\n * This extracts the database adapter and creates the ORM instance.\n */\nfunction createDatabaseContext<TSchema extends AnySchema>(\n options: FragnoPublicConfigWithDatabase,\n schema: TSchema,\n namespace: string,\n): DatabaseFragmentContext<TSchema> {\n const databaseAdapter = options.databaseAdapter;\n\n if (!databaseAdapter) {\n throw new Error(\n \"Database fragment requires a database adapter to be provided in options.databaseAdapter\",\n );\n }\n\n const db = databaseAdapter.createQueryEngine(schema, namespace);\n\n return { databaseAdapter, db };\n}\n\n/**\n * Storage type for database fragments - stores the Unit of Work.\n */\nexport type DatabaseRequestStorage = {\n uow: IUnitOfWork;\n};\n\n/**\n * Builder for database fragments that wraps the core fragment builder\n * and provides database-specific functionality.\n *\n * Database fragments always require FragnoPublicConfigWithDatabase (which includes databaseAdapter).\n */\nexport class DatabaseFragmentDefinitionBuilder<\n TSchema extends AnySchema,\n TConfig,\n TDeps,\n TBaseServices,\n TServices,\n TServiceDependencies,\n TPrivateServices,\n THooks extends HooksMap = {},\n TServiceThisContext extends RequestThisContext = DatabaseHandlerContext,\n THandlerThisContext extends RequestThisContext = DatabaseHandlerContext,\n TLinkedFragments extends Record<string, AnyFragnoInstantiatedFragment> = {},\n> {\n // Store the base builder - we'll replace its storage and context setup when building\n #baseBuilder: FragmentDefinitionBuilder<\n TConfig,\n FragnoPublicConfigWithDatabase,\n TDeps,\n TBaseServices,\n TServices,\n TServiceDependencies,\n TPrivateServices,\n TServiceThisContext,\n THandlerThisContext,\n DatabaseRequestStorage,\n TLinkedFragments\n >;\n #schema: TSchema;\n #namespace: string;\n #hooksFactory?: (context: { config: TConfig; options: FragnoPublicConfigWithDatabase }) => THooks;\n\n constructor(\n baseBuilder: FragmentDefinitionBuilder<\n TConfig,\n FragnoPublicConfigWithDatabase,\n TDeps,\n TBaseServices,\n TServices,\n TServiceDependencies,\n TPrivateServices,\n TServiceThisContext,\n THandlerThisContext,\n DatabaseRequestStorage,\n TLinkedFragments\n >,\n schema: TSchema,\n namespace?: string,\n hooksFactory?: (context: {\n config: TConfig;\n options: FragnoPublicConfigWithDatabase;\n }) => THooks,\n ) {\n this.#baseBuilder = baseBuilder;\n this.#schema = schema;\n this.#namespace = namespace ?? baseBuilder.name;\n this.#hooksFactory = hooksFactory;\n }\n\n /**\n * Define dependencies for this database fragment.\n * The context includes database adapter and ORM instance.\n */\n withDependencies<TNewDeps>(\n fn: (context: {\n config: TConfig;\n options: FragnoPublicConfigWithDatabase;\n db: SimpleQueryInterface<TSchema>;\n databaseAdapter: DatabaseAdapter<any>; // eslint-disable-line @typescript-eslint/no-explicit-any\n }) => TNewDeps,\n ): DatabaseFragmentDefinitionBuilder<\n TSchema,\n TConfig,\n TNewDeps & ImplicitDatabaseDependencies<TSchema>,\n {},\n {},\n TServiceDependencies,\n {},\n THooks,\n TServiceThisContext,\n THandlerThisContext,\n TLinkedFragments\n > {\n // Wrap user function to inject DB context\n const wrappedFn = (context: { config: TConfig; options: FragnoPublicConfigWithDatabase }) => {\n const dbContext = createDatabaseContext(context.options, this.#schema, this.#namespace);\n\n // Call user function with enriched context\n const userDeps = fn({\n config: context.config,\n options: context.options,\n db: dbContext.db,\n databaseAdapter: dbContext.databaseAdapter,\n });\n\n // Create implicit dependencies\n const createUow = () => dbContext.db.createUnitOfWork();\n const implicitDeps: ImplicitDatabaseDependencies<TSchema> = {\n db: dbContext.db,\n schema: this.#schema,\n namespace: this.#namespace,\n createUnitOfWork: createUow,\n };\n\n return {\n ...userDeps,\n ...implicitDeps,\n };\n };\n\n // Create new base builder with wrapped function\n const newBaseBuilder = this.#baseBuilder.withDependencies(wrappedFn);\n\n return new DatabaseFragmentDefinitionBuilder(\n newBaseBuilder,\n this.#schema,\n this.#namespace,\n this.#hooksFactory,\n );\n }\n\n providesBaseService<TNewService>(\n fn: ServiceConstructorFn<\n TConfig,\n FragnoPublicConfigWithDatabase,\n TDeps,\n TServiceDependencies,\n TPrivateServices,\n TNewService,\n TServiceThisContext\n >,\n ): DatabaseFragmentDefinitionBuilder<\n TSchema,\n TConfig,\n TDeps,\n TNewService,\n TServices,\n TServiceDependencies,\n TPrivateServices,\n THooks,\n TServiceThisContext,\n THandlerThisContext,\n TLinkedFragments\n > {\n const newBaseBuilder = this.#baseBuilder.providesBaseService<TNewService>(fn);\n\n return new DatabaseFragmentDefinitionBuilder(\n newBaseBuilder,\n this.#schema,\n this.#namespace,\n this.#hooksFactory,\n );\n }\n\n providesService<TServiceName extends string, TService>(\n serviceName: TServiceName,\n fn: ServiceConstructorFn<\n TConfig,\n FragnoPublicConfigWithDatabase,\n TDeps,\n TServiceDependencies,\n TPrivateServices,\n TService,\n TServiceThisContext\n >,\n ): DatabaseFragmentDefinitionBuilder<\n TSchema,\n TConfig,\n TDeps,\n TBaseServices,\n TServices & { [K in TServiceName]: TService },\n TServiceDependencies,\n TPrivateServices,\n THooks,\n TServiceThisContext,\n THandlerThisContext,\n TLinkedFragments\n > {\n const newBaseBuilder = this.#baseBuilder.providesService<TServiceName, TService>(\n serviceName,\n fn,\n );\n\n return new DatabaseFragmentDefinitionBuilder(\n newBaseBuilder,\n this.#schema,\n this.#namespace,\n this.#hooksFactory,\n );\n }\n\n /**\n * Provide a private service that is only accessible to the fragment author.\n * Private services are NOT exposed on the fragment instance, but can be used\n * when defining other services (baseServices, namedServices, and other privateServices).\n * Private services are instantiated in order, so earlier private services are available\n * to later ones.\n */\n providesPrivateService<TServiceName extends string, TService>(\n serviceName: TServiceName,\n fn: ServiceConstructorFn<\n TConfig,\n FragnoPublicConfigWithDatabase,\n TDeps,\n TServiceDependencies,\n TPrivateServices,\n TService,\n TServiceThisContext\n >,\n ): DatabaseFragmentDefinitionBuilder<\n TSchema,\n TConfig,\n TDeps,\n TBaseServices,\n TServices,\n TServiceDependencies,\n TPrivateServices & { [K in TServiceName]: TService },\n THooks,\n TServiceThisContext,\n THandlerThisContext,\n TLinkedFragments\n > {\n const newBaseBuilder = this.#baseBuilder.providesPrivateService<TServiceName, TService>(\n serviceName,\n fn,\n );\n\n return new DatabaseFragmentDefinitionBuilder(\n newBaseBuilder,\n this.#schema,\n this.#namespace,\n this.#hooksFactory,\n );\n }\n\n /**\n * Define durable hooks for this fragment.\n * Hooks are automatically persisted and retried on failure.\n *\n * @param fn - Function that receives defineHook helper and returns a hooks map\n * @returns Builder with hooks type set\n *\n * @example\n * ```ts\n * .provideHooks(({ defineHook, config }) => ({\n * onSubscribe: defineHook(async function (payload: { email: string }) {\n * // 'this' context available (HookServiceContext with nonce)\n * await config.onSubscribe?.(payload.email);\n * }),\n * }))\n * ```\n */\n provideHooks<TNewHooks extends HooksMap>(\n fn: (context: {\n config: TConfig;\n options: FragnoPublicConfigWithDatabase;\n defineHook: <TPayload>(\n hook: (this: HookContext, payload: TPayload) => void | Promise<void>,\n ) => HookFn<TPayload>;\n }) => TNewHooks,\n ): DatabaseFragmentDefinitionBuilder<\n TSchema,\n TConfig,\n TDeps,\n TBaseServices,\n TServices,\n TServiceDependencies,\n TPrivateServices,\n TNewHooks,\n DatabaseServiceContext<TNewHooks>,\n THandlerThisContext,\n TLinkedFragments\n > {\n const defineHook = <TPayload>(\n hook: (this: HookContext, payload: TPayload) => void | Promise<void>,\n ): HookFn<TPayload> => {\n return hook;\n };\n\n // Store the hooks factory - it will be called in build() with config/options\n const hooksFactory = (context: {\n config: TConfig;\n options: FragnoPublicConfigWithDatabase;\n }) => {\n return fn({\n config: context.config,\n options: context.options,\n defineHook,\n });\n };\n\n // Create new builder with hooks factory stored\n // Cast is safe: we're only changing THooks and TServiceThisContext type parameters\n const newBuilder = new DatabaseFragmentDefinitionBuilder(\n this.#baseBuilder,\n this.#schema,\n this.#namespace,\n ) as unknown as DatabaseFragmentDefinitionBuilder<\n TSchema,\n TConfig,\n TDeps,\n TBaseServices,\n TServices,\n TServiceDependencies,\n TPrivateServices,\n TNewHooks,\n DatabaseServiceContext<TNewHooks>,\n THandlerThisContext,\n TLinkedFragments\n >;\n\n newBuilder.#hooksFactory = hooksFactory;\n\n return newBuilder;\n }\n\n /**\n * Declare that this fragment uses a required service provided by the runtime.\n * Delegates to the base builder.\n */\n usesService<TServiceName extends string, TService>(\n serviceName: TServiceName,\n ): DatabaseFragmentDefinitionBuilder<\n TSchema,\n TConfig,\n TDeps,\n TBaseServices,\n TServices,\n TServiceDependencies & { [K in TServiceName]: TService },\n TPrivateServices,\n THooks,\n TServiceThisContext,\n THandlerThisContext,\n TLinkedFragments\n > {\n const newBaseBuilder = this.#baseBuilder.usesService<TServiceName, TService>(serviceName);\n\n return new DatabaseFragmentDefinitionBuilder(\n newBaseBuilder,\n this.#schema,\n this.#namespace,\n this.#hooksFactory,\n );\n }\n\n /**\n * Declare that this fragment uses an optional service provided by the runtime.\n * Delegates to the base builder.\n */\n usesOptionalService<TServiceName extends string, TService>(\n serviceName: TServiceName,\n ): DatabaseFragmentDefinitionBuilder<\n TSchema,\n TConfig,\n TDeps,\n TBaseServices,\n TServices,\n TServiceDependencies & { [K in TServiceName]: TService | undefined },\n TPrivateServices,\n THooks,\n TServiceThisContext,\n THandlerThisContext,\n TLinkedFragments\n > {\n const newBaseBuilder = this.#baseBuilder.usesOptionalService<TServiceName, TService>(\n serviceName,\n );\n\n return new DatabaseFragmentDefinitionBuilder(\n newBaseBuilder,\n this.#schema,\n this.#namespace,\n this.#hooksFactory,\n );\n }\n\n /**\n * Build the final database fragment definition.\n * This includes the request context setup for UnitOfWork management.\n * Note: TDeps already includes ImplicitDatabaseDependencies from withDatabase().\n */\n build(): FragmentDefinition<\n TConfig,\n FragnoPublicConfigWithDatabase,\n TDeps,\n TBaseServices,\n TServices,\n TServiceDependencies,\n TPrivateServices,\n DatabaseServiceContext<THooks>,\n DatabaseHandlerContext<THooks>,\n DatabaseRequestStorage,\n TLinkedFragments\n > {\n const baseDef = this.#baseBuilder.build();\n\n // Ensure dependencies callback always exists for database fragments\n // If no user dependencies were defined, create a minimal one that only adds implicit deps\n const dependencies = (context: {\n config: TConfig;\n options: FragnoPublicConfigWithDatabase;\n }): TDeps => {\n // In dry run mode, allow user deps to fail gracefully.\n // This is critical for database fragments because the CLI needs access to the schema\n // even when user dependencies (like API clients) can't be initialized.\n // Without this, if user deps fail, we'd lose the implicit database dependencies\n // (including schema), and the CLI couldn't extract schema information.\n let userDeps;\n try {\n userDeps = baseDef.dependencies?.(context);\n } catch (error) {\n if (process.env[\"FRAGNO_INIT_DRY_RUN\"] === \"true\") {\n console.warn(\n \"Warning: Failed to initialize user dependencies in dry run mode:\",\n error instanceof Error ? error.message : String(error),\n );\n userDeps = {};\n } else {\n throw error;\n }\n }\n\n const { db } = createDatabaseContext(context.options, this.#schema, this.#namespace);\n\n const implicitDeps: ImplicitDatabaseDependencies<TSchema> = {\n db,\n schema: this.#schema,\n namespace: this.#namespace,\n createUnitOfWork: () => db.createUnitOfWork(),\n };\n\n return {\n ...userDeps,\n ...implicitDeps,\n } as TDeps;\n };\n\n // Use the adapter's shared context storage (all fragments using the same adapter share this storage)\n const builderWithExternalStorage = this.#baseBuilder.withExternalRequestStorage(\n ({ options }) => {\n const dbContext = createDatabaseContext(options, this.#schema, this.#namespace);\n return dbContext.databaseAdapter.contextStorage;\n },\n );\n\n // Set up request storage to initialize the Unit of Work\n const builderWithStorage = builderWithExternalStorage.withRequestStorage(\n ({ options }): DatabaseRequestStorage => {\n // Create database context - needed here to create the UOW\n const dbContextForStorage = createDatabaseContext(options, this.#schema, this.#namespace);\n\n // Create a new Unit of Work for this request\n const uow: IUnitOfWork = dbContextForStorage.db.createUnitOfWork();\n\n return { uow };\n },\n );\n\n // Get the internal fragment factory from linked fragments (added by withDatabase)\n // Cast is safe: withDatabase() guarantees this fragment exists and has the correct type\n const internalFragmentFactory = baseDef.linkedFragments?.[\"_fragno_internal\"] as (context: {\n config: TConfig;\n options: FragnoPublicConfigWithDatabase;\n }) => InternalFragmentInstance;\n\n const builderWithContext = builderWithStorage.withThisContext<\n DatabaseServiceContext<THooks>,\n DatabaseHandlerContext<THooks>\n >(({ storage, config, options }) => {\n // Create hooks config if hooks factory is defined\n const hooksConfig = this.#hooksFactory\n ? {\n hooks: this.#hooksFactory({ config, options }),\n namespace: this.#namespace,\n internalFragment: internalFragmentFactory({ config, options }),\n }\n : undefined;\n\n function forSchema<TSchema extends AnySchema>(\n schema: TSchema,\n ): TypedUnitOfWork<TSchema, [], unknown, THooks> {\n const uow = storage.getStore()?.uow;\n if (!uow) {\n throw new Error(\n \"No UnitOfWork in context. Service must be called within a route handler OR using `withUnitOfWork`.\",\n );\n }\n\n return uow.restrict().forSchema<TSchema, THooks>(schema);\n }\n\n const serviceTx = async <\n TSchema extends AnySchema,\n TRetrievalResults extends unknown[],\n TMutationResult = void,\n >(\n schema: TSchema,\n callbacks: ServiceTxCallbacks<TSchema, TRetrievalResults, TMutationResult, THooks>,\n ): Promise<AwaitedPromisesInObject<TMutationResult>> => {\n const uow = storage.getStore()?.uow;\n if (!uow) {\n throw new Error(\n \"No UnitOfWork in context. Service must be called within a route handler OR using `withUnitOfWork`.\",\n );\n }\n\n return executeServiceTx(schema, callbacks, uow);\n };\n\n const serviceContext: DatabaseServiceContext<THooks> = {\n uow: forSchema,\n tx: serviceTx,\n };\n\n async function uow<TResult>(\n callback: (context: {\n forSchema: <S extends AnySchema, H extends HooksMap = THooks>(\n schema: S,\n hooks?: H,\n ) => TypedUnitOfWork<S, [], unknown, H>;\n executeRetrieve: () => Promise<void>;\n executeMutate: () => Promise<void>;\n nonce: string;\n currentAttempt: number;\n }) => Promise<TResult> | TResult,\n execOptions?: Omit<ExecuteRestrictedUnitOfWorkOptions, \"createUnitOfWork\">,\n ): Promise<AwaitedPromisesInObject<TResult>> {\n const currentStorage = storage.getStore();\n if (!currentStorage) {\n throw new Error(\n \"No storage in context. Handler must be called within a request context.\",\n );\n }\n\n const wrappedCallback = async (context: {\n forSchema: <S extends AnySchema, H extends HooksMap = THooks>(\n schema: S,\n hooks?: H,\n ) => TypedUnitOfWork<S, [], unknown, H>;\n executeRetrieve: () => Promise<void>;\n executeMutate: () => Promise<void>;\n nonce: string;\n currentAttempt: number;\n }): Promise<TResult> => {\n return callback(context);\n };\n\n const userOnBeforeMutate = execOptions?.onBeforeMutate;\n const userOnSuccess = execOptions?.onSuccess;\n\n return executeRestrictedUnitOfWork(wrappedCallback, {\n ...execOptions,\n createUnitOfWork: () => {\n currentStorage.uow.reset();\n if (hooksConfig) {\n currentStorage.uow.registerSchema(\n hooksConfig.internalFragment.$internal.deps.schema,\n hooksConfig.internalFragment.$internal.deps.namespace,\n );\n }\n // Safe cast: currentStorage.uow is always a UnitOfWork instance\n return currentStorage.uow as UnitOfWork;\n },\n onBeforeMutate: (uow) => {\n if (hooksConfig) {\n prepareHookMutations(uow, hooksConfig);\n }\n if (userOnBeforeMutate) {\n userOnBeforeMutate(uow);\n }\n },\n onSuccess: async (uow) => {\n if (hooksConfig) {\n await processHooks(hooksConfig);\n }\n if (userOnSuccess) {\n await userOnSuccess(uow);\n }\n },\n });\n }\n\n // Handler tx method with two overloads\n function handlerTx<T extends readonly unknown[]>(\n servicesFactory: () => readonly [...{ [K in keyof T]: Promise<T[K]> }],\n execOptions?: Omit<ExecuteRestrictedUnitOfWorkOptions, \"createUnitOfWork\">,\n ): Promise<{ [K in keyof T]: T[K] }>;\n function handlerTx<TRetrieveResult, TMutationResult>(\n callbacks: HandlerTxCallbacks<TRetrieveResult, TMutationResult, THooks>,\n execOptions?: Omit<ExecuteRestrictedUnitOfWorkOptions, \"createUnitOfWork\">,\n ): Promise<AwaitedPromisesInObject<TMutationResult>>;\n async function handlerTx<T extends readonly unknown[], TRetrieveResult, TMutationResult>(\n factoryOrCallbacks:\n | (() => readonly [...{ [K in keyof T]: Promise<T[K]> }])\n | HandlerTxCallbacks<TRetrieveResult, TMutationResult, THooks>,\n execOptions?: Omit<ExecuteRestrictedUnitOfWorkOptions, \"createUnitOfWork\">,\n ): Promise<{ [K in keyof T]: T[K] } | AwaitedPromisesInObject<TMutationResult>> {\n const currentStorage = storage.getStore();\n if (!currentStorage) {\n throw new Error(\n \"No storage in context. Handler must be called within a request context.\",\n );\n }\n\n const userOnBeforeMutate = execOptions?.onBeforeMutate;\n const userOnSuccess = execOptions?.onSuccess;\n\n const createUow = () => {\n currentStorage.uow.reset();\n\n if (hooksConfig) {\n currentStorage.uow.registerSchema(\n hooksConfig.internalFragment.$internal.deps.schema,\n hooksConfig.internalFragment.$internal.deps.namespace,\n );\n }\n // Safe cast: currentStorage.uow is always a UnitOfWork instance\n return currentStorage.uow as UnitOfWork;\n };\n\n const options: ExecuteRestrictedUnitOfWorkOptions = {\n ...execOptions,\n createUnitOfWork: createUow,\n onBeforeMutate: (uow) => {\n if (hooksConfig) {\n prepareHookMutations(uow, hooksConfig);\n }\n if (userOnBeforeMutate) {\n userOnBeforeMutate(uow);\n }\n },\n onSuccess: async (uow) => {\n if (hooksConfig) {\n await processHooks(hooksConfig);\n }\n if (userOnSuccess) {\n await userOnSuccess(uow);\n }\n },\n };\n\n // Check if it's a function (array syntax factory) or callbacks object (callback syntax)\n if (typeof factoryOrCallbacks === \"function\") {\n // Array syntax - factoryOrCallbacks is a factory function\n return executeTxArray(\n factoryOrCallbacks as () => readonly [...{ [K in keyof T]: Promise<T[K]> }],\n options,\n ) as Promise<{ [K in keyof T]: T[K] }>;\n } else {\n return executeTxCallbacks(\n factoryOrCallbacks as HandlerTxCallbacks<TRetrieveResult, TMutationResult, THooks>,\n options,\n );\n }\n }\n\n const handlerContext: DatabaseHandlerContext<THooks> = {\n uow,\n tx: handlerTx,\n };\n\n return { serviceContext, handlerContext };\n });\n\n // Build the final definition\n const finalDef = builderWithContext.build();\n\n // Return the complete definition with proper typing and dependencies\n return {\n ...finalDef,\n dependencies,\n };\n }\n}\n"],"mappings":"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"}
1
+ {"version":3,"file":"db-fragment-definition-builder.js","names":["#baseBuilder","#schema","#hooksFactory","implicitDeps: ImplicitDatabaseDependencies<TSchema>","hooksConfig: HookProcessorConfig<THooks>","serviceContext: DatabaseServiceContext<THooks>"],"sources":["../src/db-fragment-definition-builder.ts"],"sourcesContent":["import type { AnySchema } from \"./schema/create\";\nimport type { SimpleQueryInterface } from \"./query/simple-query-interface\";\nimport type { DatabaseAdapter } from \"./adapters/adapters\";\nimport type { IUnitOfWork } from \"./query/unit-of-work/unit-of-work\";\nimport type {\n RequestThisContext,\n FragnoPublicConfig,\n AnyFragnoInstantiatedFragment,\n} from \"@fragno-dev/core\";\nimport {\n FragmentDefinitionBuilder,\n type FragmentDefinition,\n type ServiceConstructorFn,\n} from \"@fragno-dev/core\";\nimport {\n createServiceTxBuilder,\n createHandlerTxBuilder,\n ServiceTxBuilder,\n HandlerTxBuilder,\n type ExecuteTxOptions,\n} from \"./query/unit-of-work/execute-unit-of-work\";\nimport {\n prepareHookMutations,\n type HooksMap,\n type HookFn,\n type HookContext,\n type HookProcessorConfig,\n type DurableHooksProcessingOptions,\n createHookScheduler,\n} from \"./hooks/hooks\";\nimport type { InternalFragmentInstance } from \"./fragments/internal-fragment\";\nimport { resolveDatabaseAdapter } from \"./util/default-database-adapter\";\nimport { sanitizeNamespace } from \"./naming/sql-naming\";\n\n/**\n * Extended FragnoPublicConfig for database fragments.\n * If databaseAdapter is omitted and better-sqlite3 is available, a default SQLite adapter is used.\n */\nexport type FragnoPublicConfigWithDatabase = FragnoPublicConfig & {\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n databaseAdapter?: DatabaseAdapter<any>;\n /**\n * Optional durable hooks processing configuration.\n */\n durableHooks?: DurableHooksProcessingOptions;\n /**\n * Optional override for database namespace. If provided (including null), it is used as-is\n * without sanitization — the caller is responsible for providing a valid namespace.\n * When omitted, defaults to a sanitized version of schema.name.\n */\n databaseNamespace?: string | null;\n};\n\n/**\n * Implicit dependencies that database fragments get automatically.\n * These are injected without requiring explicit configuration.\n */\nexport type ImplicitDatabaseDependencies<TSchema extends AnySchema> = {\n /**\n * Database query engine for the fragment's schema.\n * @deprecated Prefer handlerTx/serviceTx instead of direct db usage.\n */\n db: SimpleQueryInterface<TSchema>;\n /**\n * The schema definition for this fragment.\n */\n schema: TSchema;\n /**\n * The database namespace for this fragment.\n */\n namespace: string | null;\n /**\n * Create a new Unit of Work for database operations.\n */\n createUnitOfWork: () => IUnitOfWork;\n};\n\n/**\n * Service context for database fragments - provides restricted UOW access without execute methods.\n */\nexport type DatabaseServiceContext<THooks extends HooksMap> = RequestThisContext & {\n /**\n * Create a service-level transaction builder using the fluent API.\n * Returns a builder that can be chained with withServiceCalls, retrieve,\n * transformRetrieve, mutate, transform, and build.\n *\n * @example\n * ```ts\n * return this.serviceTx(schema)\n * .withServiceCalls(() => [otherService.getData()])\n * .retrieve((uow) => uow.find(\"users\", ...))\n * .transformRetrieve(([users]) => users[0])\n * .mutate(({ uow, retrieveResult, serviceIntermediateResult }) =>\n * uow.create(\"records\", { ... })\n * )\n * .transform(({ mutateResult, serviceResult }) => ({ id: mutateResult }))\n * .build();\n * ```\n */\n serviceTx<TSchema extends AnySchema>(\n schema: TSchema,\n ): ServiceTxBuilder<\n TSchema,\n readonly [],\n [],\n [],\n unknown,\n unknown,\n false,\n false,\n false,\n false,\n THooks\n >;\n};\n\n/**\n * Handler context for database fragments - provides UOW execution with automatic retry support.\n */\nexport type DatabaseHandlerContext<THooks extends HooksMap = {}> = RequestThisContext & {\n /**\n * Create a handler-level transaction builder using the fluent API.\n * Returns a builder that can be chained with withServiceCalls, retrieve,\n * transformRetrieve, mutate, transform, and execute.\n *\n * @example\n * ```ts\n * const result = await this.handlerTx()\n * .withServiceCalls(() => [userService.getUser(id)])\n * .mutate(({ forSchema, idempotencyKey, currentAttempt, serviceIntermediateResult }) => {\n * return forSchema(ordersSchema).create(\"orders\", { ... });\n * })\n * .transform(({ mutateResult, serviceResult }) => ({ ... }))\n * .execute();\n * ```\n */\n handlerTx(\n options?: Omit<ExecuteTxOptions, \"createUnitOfWork\">,\n ): HandlerTxBuilder<readonly [], [], [], unknown, unknown, false, false, false, false, THooks>;\n};\n\n/**\n * Database fragment context provided to user callbacks.\n */\nexport type DatabaseFragmentContext<TSchema extends AnySchema> = {\n /**\n * Database adapter instance.\n */\n databaseAdapter: DatabaseAdapter<any>; // eslint-disable-line @typescript-eslint/no-explicit-any\n /**\n * ORM query engine for the fragment's schema.\n */\n db: SimpleQueryInterface<TSchema>;\n};\n\n/**\n * Create database context from options.\n * This extracts the database adapter and creates the ORM instance.\n */\nfunction createDatabaseContext<TSchema extends AnySchema>(\n options: FragnoPublicConfigWithDatabase,\n schema: TSchema,\n): DatabaseFragmentContext<TSchema> {\n const databaseAdapter = resolveDatabaseAdapter(options, schema);\n\n const namespace = resolveDatabaseNamespace(options, schema);\n const db = databaseAdapter.createQueryEngine(schema, namespace);\n\n return { databaseAdapter, db };\n}\n\nfunction resolveDatabaseNamespace<TSchema extends AnySchema>(\n options: FragnoPublicConfigWithDatabase,\n schema: TSchema,\n): string | null {\n const hasOverride = options.databaseNamespace !== undefined;\n return hasOverride ? (options.databaseNamespace ?? null) : sanitizeNamespace(schema.name);\n}\n\n/**\n * Storage type for database fragments - stores the Unit of Work.\n */\nexport type DatabaseRequestStorage = {\n uow: IUnitOfWork;\n};\n\n/**\n * Builder for database fragments that wraps the core fragment builder\n * and provides database-specific functionality.\n *\n * Database fragments use FragnoPublicConfigWithDatabase and default the adapter when possible.\n */\nexport class DatabaseFragmentDefinitionBuilder<\n TSchema extends AnySchema,\n TConfig,\n TDeps,\n TBaseServices,\n TServices,\n TServiceDependencies,\n TPrivateServices,\n THooks extends HooksMap = {},\n TServiceThisContext extends RequestThisContext = DatabaseHandlerContext,\n THandlerThisContext extends RequestThisContext = DatabaseHandlerContext,\n TLinkedFragments extends Record<string, AnyFragnoInstantiatedFragment> = {},\n> {\n // Store the base builder - we'll replace its storage and context setup when building\n #baseBuilder: FragmentDefinitionBuilder<\n TConfig,\n FragnoPublicConfigWithDatabase,\n TDeps,\n TBaseServices,\n TServices,\n TServiceDependencies,\n TPrivateServices,\n TServiceThisContext,\n THandlerThisContext,\n DatabaseRequestStorage,\n TLinkedFragments\n >;\n #schema: TSchema;\n #hooksFactory?: (context: { config: TConfig; options: FragnoPublicConfigWithDatabase }) => THooks;\n\n constructor(\n baseBuilder: FragmentDefinitionBuilder<\n TConfig,\n FragnoPublicConfigWithDatabase,\n TDeps,\n TBaseServices,\n TServices,\n TServiceDependencies,\n TPrivateServices,\n TServiceThisContext,\n THandlerThisContext,\n DatabaseRequestStorage,\n TLinkedFragments\n >,\n schema: TSchema,\n hooksFactory?: (context: {\n config: TConfig;\n options: FragnoPublicConfigWithDatabase;\n }) => THooks,\n ) {\n this.#baseBuilder = baseBuilder;\n this.#schema = schema;\n this.#hooksFactory = hooksFactory;\n }\n\n /**\n * Define dependencies for this database fragment.\n * The context includes database adapter and ORM instance.\n */\n withDependencies<TNewDeps>(\n fn: (context: {\n config: TConfig;\n options: FragnoPublicConfigWithDatabase;\n db: SimpleQueryInterface<TSchema>;\n databaseAdapter: DatabaseAdapter<any>; // eslint-disable-line @typescript-eslint/no-explicit-any\n }) => TNewDeps,\n ): DatabaseFragmentDefinitionBuilder<\n TSchema,\n TConfig,\n TNewDeps & ImplicitDatabaseDependencies<TSchema>,\n {},\n {},\n TServiceDependencies,\n {},\n THooks,\n TServiceThisContext,\n THandlerThisContext,\n TLinkedFragments\n > {\n // Wrap user function to inject DB context\n const wrappedFn = (context: { config: TConfig; options: FragnoPublicConfigWithDatabase }) => {\n const dbContext = createDatabaseContext(context.options, this.#schema);\n const namespace = resolveDatabaseNamespace(context.options, this.#schema);\n\n // Call user function with enriched context\n const userDeps = fn({\n config: context.config,\n options: context.options,\n db: dbContext.db,\n databaseAdapter: dbContext.databaseAdapter,\n });\n\n // Create implicit dependencies\n const createUow = () => dbContext.db.createUnitOfWork();\n const implicitDeps: ImplicitDatabaseDependencies<TSchema> = {\n db: dbContext.db,\n schema: this.#schema,\n namespace,\n createUnitOfWork: createUow,\n };\n\n return {\n ...userDeps,\n ...implicitDeps,\n };\n };\n\n // Create new base builder with wrapped function\n const newBaseBuilder = this.#baseBuilder.withDependencies(wrappedFn);\n\n return new DatabaseFragmentDefinitionBuilder(newBaseBuilder, this.#schema, this.#hooksFactory);\n }\n\n providesBaseService<TNewService>(\n fn: ServiceConstructorFn<\n TConfig,\n FragnoPublicConfigWithDatabase,\n TDeps,\n TServiceDependencies,\n TPrivateServices,\n TNewService,\n TServiceThisContext\n >,\n ): DatabaseFragmentDefinitionBuilder<\n TSchema,\n TConfig,\n TDeps,\n TNewService,\n TServices,\n TServiceDependencies,\n TPrivateServices,\n THooks,\n TServiceThisContext,\n THandlerThisContext,\n TLinkedFragments\n > {\n const newBaseBuilder = this.#baseBuilder.providesBaseService<TNewService>(fn);\n\n return new DatabaseFragmentDefinitionBuilder(newBaseBuilder, this.#schema, this.#hooksFactory);\n }\n\n providesService<TServiceName extends string, TService>(\n serviceName: TServiceName,\n fn: ServiceConstructorFn<\n TConfig,\n FragnoPublicConfigWithDatabase,\n TDeps,\n TServiceDependencies,\n TPrivateServices,\n TService,\n TServiceThisContext\n >,\n ): DatabaseFragmentDefinitionBuilder<\n TSchema,\n TConfig,\n TDeps,\n TBaseServices,\n TServices & { [K in TServiceName]: TService },\n TServiceDependencies,\n TPrivateServices,\n THooks,\n TServiceThisContext,\n THandlerThisContext,\n TLinkedFragments\n > {\n const newBaseBuilder = this.#baseBuilder.providesService<TServiceName, TService>(\n serviceName,\n fn,\n );\n\n return new DatabaseFragmentDefinitionBuilder(newBaseBuilder, this.#schema, this.#hooksFactory);\n }\n\n /**\n * Provide a private service that is only accessible to the fragment author.\n * Private services are NOT exposed on the fragment instance, but can be used\n * when defining other services (baseServices, namedServices, and other privateServices).\n * Private services are instantiated in order, so earlier private services are available\n * to later ones.\n */\n providesPrivateService<TServiceName extends string, TService>(\n serviceName: TServiceName,\n fn: ServiceConstructorFn<\n TConfig,\n FragnoPublicConfigWithDatabase,\n TDeps,\n TServiceDependencies,\n TPrivateServices,\n TService,\n TServiceThisContext\n >,\n ): DatabaseFragmentDefinitionBuilder<\n TSchema,\n TConfig,\n TDeps,\n TBaseServices,\n TServices,\n TServiceDependencies,\n TPrivateServices & { [K in TServiceName]: TService },\n THooks,\n TServiceThisContext,\n THandlerThisContext,\n TLinkedFragments\n > {\n const newBaseBuilder = this.#baseBuilder.providesPrivateService<TServiceName, TService>(\n serviceName,\n fn,\n );\n\n return new DatabaseFragmentDefinitionBuilder(newBaseBuilder, this.#schema, this.#hooksFactory);\n }\n\n /**\n * Define durable hooks for this fragment.\n * Hooks are automatically persisted and retried on failure.\n *\n * @param fn - Function that receives defineHook helper and returns a hooks map\n * @returns Builder with hooks type set\n *\n * @example\n * ```ts\n * .provideHooks(({ defineHook, config }) => ({\n * onSubscribe: defineHook(async function (payload: { email: string }) {\n * // 'this' context available (HookServiceContext with idempotencyKey)\n * await config.onSubscribe?.(payload.email);\n * }),\n * }))\n * ```\n */\n provideHooks<TNewHooks extends HooksMap>(\n fn: (context: {\n config: TConfig;\n options: FragnoPublicConfigWithDatabase;\n defineHook: <TPayload>(\n hook: (this: HookContext, payload: TPayload) => void | Promise<void>,\n ) => HookFn<TPayload>;\n }) => TNewHooks,\n ): DatabaseFragmentDefinitionBuilder<\n TSchema,\n TConfig,\n TDeps,\n TBaseServices,\n TServices,\n TServiceDependencies,\n TPrivateServices,\n TNewHooks,\n DatabaseServiceContext<TNewHooks>,\n THandlerThisContext,\n TLinkedFragments\n > {\n const defineHook = <TPayload>(\n hook: (this: HookContext, payload: TPayload) => void | Promise<void>,\n ): HookFn<TPayload> => {\n return hook;\n };\n\n // Store the hooks factory - it will be called in build() with config/options\n const hooksFactory = (context: {\n config: TConfig;\n options: FragnoPublicConfigWithDatabase;\n }) => {\n return fn({\n config: context.config,\n options: context.options,\n defineHook,\n });\n };\n\n // Create new builder with hooks factory stored\n // Cast is safe: we're only changing THooks and TServiceThisContext type parameters\n const newBuilder = new DatabaseFragmentDefinitionBuilder(\n this.#baseBuilder,\n this.#schema,\n ) as unknown as DatabaseFragmentDefinitionBuilder<\n TSchema,\n TConfig,\n TDeps,\n TBaseServices,\n TServices,\n TServiceDependencies,\n TPrivateServices,\n TNewHooks,\n DatabaseServiceContext<TNewHooks>,\n THandlerThisContext,\n TLinkedFragments\n >;\n\n newBuilder.#hooksFactory = hooksFactory;\n\n return newBuilder;\n }\n\n /**\n * Declare that this fragment uses a required service provided by the runtime.\n * Delegates to the base builder.\n */\n usesService<TServiceName extends string, TService>(\n serviceName: TServiceName,\n ): DatabaseFragmentDefinitionBuilder<\n TSchema,\n TConfig,\n TDeps,\n TBaseServices,\n TServices,\n TServiceDependencies & { [K in TServiceName]: TService },\n TPrivateServices,\n THooks,\n TServiceThisContext,\n THandlerThisContext,\n TLinkedFragments\n > {\n const newBaseBuilder = this.#baseBuilder.usesService<TServiceName, TService>(serviceName);\n\n return new DatabaseFragmentDefinitionBuilder(newBaseBuilder, this.#schema, this.#hooksFactory);\n }\n\n /**\n * Declare that this fragment uses an optional service provided by the runtime.\n * Delegates to the base builder.\n */\n usesOptionalService<TServiceName extends string, TService>(\n serviceName: TServiceName,\n ): DatabaseFragmentDefinitionBuilder<\n TSchema,\n TConfig,\n TDeps,\n TBaseServices,\n TServices,\n TServiceDependencies & { [K in TServiceName]: TService | undefined },\n TPrivateServices,\n THooks,\n TServiceThisContext,\n THandlerThisContext,\n TLinkedFragments\n > {\n const newBaseBuilder = this.#baseBuilder.usesOptionalService<TServiceName, TService>(\n serviceName,\n );\n\n return new DatabaseFragmentDefinitionBuilder(newBaseBuilder, this.#schema, this.#hooksFactory);\n }\n\n /**\n * Build the final database fragment definition.\n * This includes the request context setup for UnitOfWork management.\n * Note: TDeps already includes ImplicitDatabaseDependencies from withDatabase().\n */\n build(): FragmentDefinition<\n TConfig,\n FragnoPublicConfigWithDatabase,\n TDeps,\n TBaseServices,\n TServices,\n TServiceDependencies,\n TPrivateServices,\n DatabaseServiceContext<THooks>,\n DatabaseHandlerContext<THooks>,\n DatabaseRequestStorage,\n TLinkedFragments\n > {\n const baseDef = this.#baseBuilder.build();\n\n // Ensure dependencies callback always exists for database fragments\n // If no user dependencies were defined, create a minimal one that only adds implicit deps\n const dependencies = (context: {\n config: TConfig;\n options: FragnoPublicConfigWithDatabase;\n }): TDeps => {\n // In dry run mode, allow user deps to fail gracefully.\n // This is critical for database fragments because the CLI needs access to the schema\n // even when user dependencies (like API clients) can't be initialized.\n // Without this, if user deps fail, we'd lose the implicit database dependencies\n // (including schema), and the CLI couldn't extract schema information.\n let userDeps;\n try {\n userDeps = baseDef.dependencies?.(context);\n } catch (error) {\n if (process.env[\"FRAGNO_INIT_DRY_RUN\"] === \"true\") {\n console.warn(\n \"Warning: Failed to initialize user dependencies in dry run mode:\",\n error instanceof Error ? error.message : String(error),\n );\n userDeps = {};\n } else {\n throw error;\n }\n }\n\n const { db } = createDatabaseContext(context.options, this.#schema);\n const namespace = resolveDatabaseNamespace(context.options, this.#schema);\n\n const implicitDeps: ImplicitDatabaseDependencies<TSchema> = {\n db,\n schema: this.#schema,\n namespace,\n createUnitOfWork: () => db.createUnitOfWork(),\n };\n\n return {\n ...userDeps,\n ...implicitDeps,\n } as TDeps;\n };\n\n // Use the adapter's shared context storage (all fragments using the same adapter share this storage)\n const builderWithExternalStorage = this.#baseBuilder.withExternalRequestStorage(\n ({ options }) => {\n const dbContext = createDatabaseContext(options, this.#schema);\n return dbContext.databaseAdapter.contextStorage;\n },\n );\n\n // Set up request storage to initialize the Unit of Work\n const builderWithStorage = builderWithExternalStorage.withRequestStorage(\n ({ options }): DatabaseRequestStorage => {\n // Create database context - needed here to create the UOW\n const dbContextForStorage = createDatabaseContext(options, this.#schema);\n\n // Create a new Unit of Work for this request\n const uow: IUnitOfWork = dbContextForStorage.db.createUnitOfWork();\n\n return { uow };\n },\n );\n\n // Get the internal fragment factory from linked fragments (added by withDatabase)\n // Cast is safe: withDatabase() guarantees this fragment exists and has the correct type\n const internalFragmentFactory = baseDef.linkedFragments?.[\"_fragno_internal\"] as (context: {\n config: TConfig;\n options: FragnoPublicConfigWithDatabase;\n }) => InternalFragmentInstance;\n\n // Cache per instantiated fragment (deps object is unique per instantiation).\n const hooksConfigCache = new WeakMap<object, HookProcessorConfig<THooks>>();\n\n const createHooksConfig = (context: {\n config: TConfig;\n options: FragnoPublicConfigWithDatabase;\n deps: TDeps;\n }) => {\n if (!this.#hooksFactory) {\n return undefined;\n }\n const depsKey =\n typeof context.deps === \"object\" && context.deps !== null\n ? (context.deps as object)\n : undefined;\n const cachedHooksConfig = depsKey ? hooksConfigCache.get(depsKey) : undefined;\n if (cachedHooksConfig) {\n return cachedHooksConfig;\n }\n\n const namespace = resolveDatabaseNamespace(context.options, this.#schema);\n const namespaceKey = namespace ?? this.#schema.name;\n const durableHooksOptions = context.options.durableHooks;\n const baseAdapter = resolveDatabaseAdapter(context.options, this.#schema);\n const hookAdapter = baseAdapter.getHookProcessingAdapter?.() ?? baseAdapter;\n const hookOptions =\n hookAdapter === baseAdapter\n ? context.options\n : { ...context.options, databaseAdapter: hookAdapter };\n const dbContextForHooks = createDatabaseContext(hookOptions, this.#schema);\n const hooksConfig: HookProcessorConfig<THooks> = {\n hooks: this.#hooksFactory(context),\n namespace: namespaceKey,\n internalFragment: internalFragmentFactory({\n config: context.config,\n options: hookOptions,\n }),\n handlerTx: (execOptions?: Omit<ExecuteTxOptions, \"createUnitOfWork\">) => {\n const userOnBeforeMutate = execOptions?.onBeforeMutate;\n const userOnAfterMutate = execOptions?.onAfterMutate;\n return createHandlerTxBuilder<THooks>({\n ...execOptions,\n createUnitOfWork: () => {\n const uow = dbContextForHooks.db.createUnitOfWork();\n uow.registerSchema(\n hooksConfig.internalFragment.$internal.deps.schema,\n hooksConfig.internalFragment.$internal.deps.namespace,\n );\n return uow;\n },\n onBeforeMutate: (uow) => {\n prepareHookMutations(uow, hooksConfig);\n if (userOnBeforeMutate) {\n userOnBeforeMutate(uow);\n }\n },\n onAfterMutate: async (uow) => {\n void hooksConfig.scheduler?.schedule().catch((error) => {\n console.error(\"Durable hooks processing failed\", error);\n });\n if (userOnAfterMutate) {\n await userOnAfterMutate(uow);\n }\n },\n });\n },\n stuckProcessingTimeoutMinutes: durableHooksOptions?.stuckProcessingTimeoutMinutes,\n onStuckProcessingHooks: durableHooksOptions?.onStuckProcessingHooks,\n };\n hooksConfig.scheduler = createHookScheduler(hooksConfig);\n if (depsKey) {\n hooksConfigCache.set(depsKey, hooksConfig);\n }\n return hooksConfig;\n };\n\n const builderWithContext = builderWithStorage.withThisContext<\n DatabaseServiceContext<THooks>,\n DatabaseHandlerContext<THooks>\n >(({ storage, config, options, deps }) => {\n // Create hooks config if hooks factory is defined\n const hooksConfig = createHooksConfig({ config, options, deps });\n const internalFragment =\n hooksConfig?.internalFragment ??\n (internalFragmentFactory ? internalFragmentFactory({ config, options }) : undefined);\n\n // Builder API: serviceTx using createServiceTxBuilder\n function serviceTx<TSchema extends AnySchema>(schema: TSchema) {\n const uow = storage.getStore()?.uow;\n if (!uow) {\n throw new Error(\n \"No UnitOfWork in context. Service must be called within a route handler OR using `withUnitOfWork`.\",\n );\n }\n return createServiceTxBuilder<TSchema, THooks>(schema, uow, hooksConfig?.hooks);\n }\n\n const serviceContext: DatabaseServiceContext<THooks> = {\n serviceTx,\n };\n\n // Builder API: handlerTx using createHandlerTxBuilder\n function handlerTx(execOptions?: Omit<ExecuteTxOptions, \"createUnitOfWork\">) {\n const currentStorage = storage.getStore();\n if (!currentStorage) {\n throw new Error(\n \"No storage in context. Handler must be called within a request context.\",\n );\n }\n\n const userOnBeforeMutate = execOptions?.onBeforeMutate;\n const userOnAfterMutate = execOptions?.onAfterMutate;\n\n return createHandlerTxBuilder<THooks>({\n ...execOptions,\n createUnitOfWork: () => {\n currentStorage.uow.reset();\n if (internalFragment) {\n currentStorage.uow.registerSchema(\n internalFragment.$internal.deps.schema,\n internalFragment.$internal.deps.namespace,\n );\n }\n return currentStorage.uow;\n },\n onBeforeMutate: (uow) => {\n if (hooksConfig) {\n prepareHookMutations(uow, hooksConfig);\n }\n if (userOnBeforeMutate) {\n userOnBeforeMutate(uow);\n }\n },\n onAfterMutate: async (uow) => {\n if (hooksConfig?.scheduler) {\n void hooksConfig.scheduler.schedule().catch((error) => {\n console.error(\"Durable hooks processing failed\", error);\n });\n }\n if (userOnAfterMutate) {\n await userOnAfterMutate(uow);\n }\n },\n });\n }\n\n const handlerContext: DatabaseHandlerContext<THooks> = {\n handlerTx,\n };\n\n return { serviceContext, handlerContext };\n });\n\n // Build the final definition\n const finalDef = builderWithContext.build();\n if (this.#hooksFactory) {\n finalDef.internalDataFactory = ({ config, options, deps }) => ({\n durableHooks: createHooksConfig({\n config: config as TConfig,\n options: options as FragnoPublicConfigWithDatabase,\n deps: deps as TDeps,\n }),\n });\n }\n\n // Return the complete definition with proper typing and dependencies\n return {\n ...finalDef,\n dependencies,\n };\n }\n}\n"],"mappings":"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"}
@@ -0,0 +1,26 @@
1
+ import { DurableHooksProcessor } from "../../hooks/durable-hooks-processor.js";
2
+
3
+ //#region src/dispatchers/cloudflare-do/index.d.ts
4
+ type AlarmStorage = {
5
+ setAlarm?: (timestamp: number | Date) => Promise<void>;
6
+ deleteAlarm?: () => Promise<void>;
7
+ };
8
+ type DurableHooksDispatcherDurableObjectState = {
9
+ readonly storage: AlarmStorage;
10
+ };
11
+ type DurableHooksDispatcherDurableObjectHandler = {
12
+ fetch?: (request: Request) => Promise<Response>;
13
+ alarm?: () => Promise<void>;
14
+ };
15
+ type DurableHooksDispatcherDurableObjectFactory<TEnv = unknown> = (state: DurableHooksDispatcherDurableObjectState, env: TEnv) => DurableHooksDispatcherDurableObjectHandler;
16
+ type DurableHooksDispatcherDurableObjectOptions<TEnv = unknown> = {
17
+ createProcessor: (context: {
18
+ state: DurableHooksDispatcherDurableObjectState;
19
+ env: TEnv;
20
+ }) => DurableHooksProcessor;
21
+ onProcessError?: (error: unknown) => void;
22
+ };
23
+ declare function createDurableHooksDispatcherDurableObject<TEnv>(options: DurableHooksDispatcherDurableObjectOptions<TEnv>): DurableHooksDispatcherDurableObjectFactory<TEnv>;
24
+ //#endregion
25
+ export { DurableHooksDispatcherDurableObjectFactory, DurableHooksDispatcherDurableObjectHandler, DurableHooksDispatcherDurableObjectOptions, DurableHooksDispatcherDurableObjectState, createDurableHooksDispatcherDurableObject };
26
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","names":[],"sources":["../../../src/dispatchers/cloudflare-do/index.ts"],"sourcesContent":[],"mappings":";;;KAEK,YAAA;kCAC6B,SAAS;EADtC,WAAA,CAAA,EAAA,GAAY,GAEK,OAFL,CAAA,IAAA,CAAA;CACiB;AAAS,KAI/B,wCAAA,GAJ+B;EACrB,SAAA,OAAA,EAIF,YAJE;CAAO;AAGjB,KAIA,0CAAA,GAJwC;EAIxC,KAAA,CAAA,EAAA,CAAA,OAAA,EACQ,OADR,EAAA,GACoB,OADpB,CAC4B,QADc,CAAA;EAClC,KAAA,CAAA,EAAA,GAAA,GACJ,OADI,CAAA,IAAA,CAAA;CAAoB;AAAR,KAIpB,0CAJoB,CAAA,OAAA,OAAA,CAAA,GAAA,CAAA,KAAA,EAKvB,wCALuB,EAAA,GAAA,EAMzB,IANyB,EAAA,GAO3B,0CAP2B;AAChB,KAQJ,0CARI,CAAA,OAAA,OAAA,CAAA,GAAA;EAAO,eAAA,EAAA,CAAA,OAAA,EAAA;IAGX,KAAA,EAOD,wCAP2C;IAC7C,GAAA,EAOA,IAPA;EACF,CAAA,EAAA,GAOC,qBAPD;EACF,cAAA,CAAA,EAAA,CAAA,KAAA,EAAA,OAAA,EAAA,GAAA,IAAA;CAA0C;AAEnC,iBAQI,yCARsC,CAAA,IAAA,CAAA,CAAA,OAAA,EAS3C,0CAT2C,CASA,IATA,CAAA,CAAA,EAUnD,0CAVmD,CAUR,IAVQ,CAAA"}
@@ -0,0 +1,63 @@
1
+ //#region src/dispatchers/cloudflare-do/index.ts
2
+ function createDurableHooksDispatcherDurableObject(options) {
3
+ return (state, env) => {
4
+ const processor = options.createProcessor({
5
+ state,
6
+ env
7
+ });
8
+ const onProcessError = options.onProcessError ?? ((error) => {
9
+ console.error("Durable hooks dispatcher error", error);
10
+ });
11
+ const rawSetAlarm = state.storage.setAlarm;
12
+ const rawDeleteAlarm = state.storage.deleteAlarm;
13
+ if (!rawSetAlarm) throw new Error("Durable hooks dispatcher requires state.storage.setAlarm to schedule alarms.");
14
+ const setAlarm = rawSetAlarm.bind(state.storage);
15
+ const deleteAlarm = rawDeleteAlarm?.bind(state.storage);
16
+ let processing = false;
17
+ let queued = false;
18
+ let currentPromise;
19
+ const runProcess = () => {
20
+ if (processing) {
21
+ queued = true;
22
+ return currentPromise ?? Promise.resolve();
23
+ }
24
+ processing = true;
25
+ currentPromise = (async () => {
26
+ do {
27
+ queued = false;
28
+ try {
29
+ await processor.process();
30
+ } catch (error) {
31
+ onProcessError(error);
32
+ }
33
+ } while (queued);
34
+ processing = false;
35
+ })();
36
+ return currentPromise;
37
+ };
38
+ const scheduleNextAlarm = async () => {
39
+ const nextWakeAt = await processor.getNextWakeAt();
40
+ if (!nextWakeAt) {
41
+ await deleteAlarm?.();
42
+ return;
43
+ }
44
+ const now = Date.now();
45
+ await setAlarm(new Date(Math.max(nextWakeAt.getTime(), now)));
46
+ };
47
+ scheduleNextAlarm().catch((error) => {
48
+ onProcessError(error);
49
+ });
50
+ return { alarm: async () => {
51
+ try {
52
+ await runProcess();
53
+ await scheduleNextAlarm();
54
+ } catch (error) {
55
+ onProcessError(error);
56
+ }
57
+ } };
58
+ };
59
+ }
60
+
61
+ //#endregion
62
+ export { createDurableHooksDispatcherDurableObject };
63
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","names":["currentPromise: Promise<void> | undefined"],"sources":["../../../src/dispatchers/cloudflare-do/index.ts"],"sourcesContent":["import type { DurableHooksProcessor } from \"../../hooks/durable-hooks-processor\";\n\ntype AlarmStorage = {\n setAlarm?: (timestamp: number | Date) => Promise<void>;\n deleteAlarm?: () => Promise<void>;\n};\n\nexport type DurableHooksDispatcherDurableObjectState = {\n readonly storage: AlarmStorage;\n};\n\nexport type DurableHooksDispatcherDurableObjectHandler = {\n fetch?: (request: Request) => Promise<Response>;\n alarm?: () => Promise<void>;\n};\n\nexport type DurableHooksDispatcherDurableObjectFactory<TEnv = unknown> = (\n state: DurableHooksDispatcherDurableObjectState,\n env: TEnv,\n) => DurableHooksDispatcherDurableObjectHandler;\n\nexport type DurableHooksDispatcherDurableObjectOptions<TEnv = unknown> = {\n createProcessor: (context: {\n state: DurableHooksDispatcherDurableObjectState;\n env: TEnv;\n }) => DurableHooksProcessor;\n onProcessError?: (error: unknown) => void;\n};\n\nexport function createDurableHooksDispatcherDurableObject<TEnv>(\n options: DurableHooksDispatcherDurableObjectOptions<TEnv>,\n): DurableHooksDispatcherDurableObjectFactory<TEnv> {\n return (state, env) => {\n const processor = options.createProcessor({ state, env });\n const onProcessError =\n options.onProcessError ??\n ((error: unknown) => {\n console.error(\"Durable hooks dispatcher error\", error);\n });\n const rawSetAlarm = state.storage.setAlarm;\n const rawDeleteAlarm = state.storage.deleteAlarm;\n\n if (!rawSetAlarm) {\n throw new Error(\n \"Durable hooks dispatcher requires state.storage.setAlarm to schedule alarms.\",\n );\n }\n const setAlarm = rawSetAlarm.bind(state.storage);\n const deleteAlarm = rawDeleteAlarm?.bind(state.storage);\n\n let processing = false;\n let queued = false;\n let currentPromise: Promise<void> | undefined;\n\n const runProcess = () => {\n if (processing) {\n queued = true;\n return currentPromise ?? Promise.resolve();\n }\n\n processing = true;\n currentPromise = (async () => {\n do {\n queued = false;\n try {\n await processor.process();\n } catch (error) {\n onProcessError(error);\n }\n } while (queued);\n processing = false;\n })();\n\n return currentPromise;\n };\n\n const scheduleNextAlarm = async () => {\n const nextWakeAt = await processor.getNextWakeAt();\n if (!nextWakeAt) {\n await deleteAlarm?.();\n return;\n }\n\n const now = Date.now();\n const scheduledAt = new Date(Math.max(nextWakeAt.getTime(), now));\n await setAlarm(scheduledAt);\n };\n\n void scheduleNextAlarm().catch((error) => {\n onProcessError(error);\n });\n\n return {\n alarm: async () => {\n try {\n await runProcess();\n await scheduleNextAlarm();\n } catch (error) {\n onProcessError(error);\n }\n },\n };\n };\n}\n"],"mappings":";AA6BA,SAAgB,0CACd,SACkD;AAClD,SAAQ,OAAO,QAAQ;EACrB,MAAM,YAAY,QAAQ,gBAAgB;GAAE;GAAO;GAAK,CAAC;EACzD,MAAM,iBACJ,QAAQ,oBACN,UAAmB;AACnB,WAAQ,MAAM,kCAAkC,MAAM;;EAE1D,MAAM,cAAc,MAAM,QAAQ;EAClC,MAAM,iBAAiB,MAAM,QAAQ;AAErC,MAAI,CAAC,YACH,OAAM,IAAI,MACR,+EACD;EAEH,MAAM,WAAW,YAAY,KAAK,MAAM,QAAQ;EAChD,MAAM,cAAc,gBAAgB,KAAK,MAAM,QAAQ;EAEvD,IAAI,aAAa;EACjB,IAAI,SAAS;EACb,IAAIA;EAEJ,MAAM,mBAAmB;AACvB,OAAI,YAAY;AACd,aAAS;AACT,WAAO,kBAAkB,QAAQ,SAAS;;AAG5C,gBAAa;AACb,qBAAkB,YAAY;AAC5B,OAAG;AACD,cAAS;AACT,SAAI;AACF,YAAM,UAAU,SAAS;cAClB,OAAO;AACd,qBAAe,MAAM;;aAEhB;AACT,iBAAa;OACX;AAEJ,UAAO;;EAGT,MAAM,oBAAoB,YAAY;GACpC,MAAM,aAAa,MAAM,UAAU,eAAe;AAClD,OAAI,CAAC,YAAY;AACf,UAAM,eAAe;AACrB;;GAGF,MAAM,MAAM,KAAK,KAAK;AAEtB,SAAM,SADc,IAAI,KAAK,KAAK,IAAI,WAAW,SAAS,EAAE,IAAI,CAAC,CACtC;;AAG7B,EAAK,mBAAmB,CAAC,OAAO,UAAU;AACxC,kBAAe,MAAM;IACrB;AAEF,SAAO,EACL,OAAO,YAAY;AACjB,OAAI;AACF,UAAM,YAAY;AAClB,UAAM,mBAAmB;YAClB,OAAO;AACd,mBAAe,MAAM;;KAG1B"}
@@ -0,0 +1,17 @@
1
+ import { DurableHooksProcessor } from "../../hooks/durable-hooks-processor.js";
2
+
3
+ //#region src/dispatchers/node/index.d.ts
4
+ type DurableHooksDispatcher = {
5
+ wake: () => Promise<void>;
6
+ startPolling: () => void;
7
+ stopPolling: () => void;
8
+ };
9
+ type DurableHooksDispatcherOptions = {
10
+ processor: DurableHooksProcessor;
11
+ pollIntervalMs?: number;
12
+ onError?: (error: unknown) => void;
13
+ };
14
+ declare function createDurableHooksDispatcher(options: DurableHooksDispatcherOptions): DurableHooksDispatcher;
15
+ //#endregion
16
+ export { DurableHooksDispatcher, DurableHooksDispatcherOptions, createDurableHooksDispatcher };
17
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","names":[],"sources":["../../../src/dispatchers/node/index.ts"],"sourcesContent":[],"mappings":";;;KAEY,sBAAA;cACE;EADF,YAAA,EAAA,GAAA,GAAA,IAAA;EAMA,WAAA,EAAA,GAAA,GAAA,IAAA;AAMZ,CAAA;KANY,6BAAA;aACC;;;;iBAKG,4BAAA,UACL,gCACR"}