@fragno-dev/db 0.3.0 → 0.4.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (516) hide show
  1. package/.turbo/turbo-build.log +327 -160
  2. package/CHANGELOG.md +74 -0
  3. package/README.md +24 -0
  4. package/dist/adapters/adapters.d.ts +1 -1
  5. package/dist/adapters/adapters.d.ts.map +1 -1
  6. package/dist/adapters/adapters.js.map +1 -1
  7. package/dist/adapters/generic-sql/generic-sql-adapter.d.ts +0 -3
  8. package/dist/adapters/generic-sql/generic-sql-adapter.d.ts.map +1 -1
  9. package/dist/adapters/generic-sql/generic-sql-adapter.js +11 -12
  10. package/dist/adapters/generic-sql/generic-sql-adapter.js.map +1 -1
  11. package/dist/adapters/generic-sql/generic-sql-uow-executor.js +46 -6
  12. package/dist/adapters/generic-sql/generic-sql-uow-executor.js.map +1 -1
  13. package/dist/adapters/generic-sql/migration/cold-kysely.js.map +1 -1
  14. package/dist/adapters/generic-sql/migration/dialect/mysql.js +1 -1
  15. package/dist/adapters/generic-sql/migration/dialect/mysql.js.map +1 -1
  16. package/dist/adapters/generic-sql/migration/dialect/postgres.js +1 -1
  17. package/dist/adapters/generic-sql/migration/dialect/postgres.js.map +1 -1
  18. package/dist/adapters/generic-sql/migration/dialect/sqlite.js +185 -19
  19. package/dist/adapters/generic-sql/migration/dialect/sqlite.js.map +1 -1
  20. package/dist/adapters/generic-sql/migration/executor.d.ts.map +1 -1
  21. package/dist/adapters/generic-sql/migration/executor.js +30 -3
  22. package/dist/adapters/generic-sql/migration/executor.js.map +1 -1
  23. package/dist/adapters/generic-sql/migration/prepared-migrations.d.ts.map +1 -1
  24. package/dist/adapters/generic-sql/migration/prepared-migrations.js +3 -3
  25. package/dist/adapters/generic-sql/migration/prepared-migrations.js.map +1 -1
  26. package/dist/adapters/generic-sql/migration/sql-generator.js +1 -1
  27. package/dist/adapters/generic-sql/migration/sql-generator.js.map +1 -1
  28. package/dist/adapters/generic-sql/query/create-sql-query-compiler.js +1 -1
  29. package/dist/adapters/generic-sql/query/create-sql-query-compiler.js.map +1 -1
  30. package/dist/adapters/generic-sql/query/cursor-utils.js.map +1 -1
  31. package/dist/adapters/generic-sql/query/db-now-sql.js +27 -0
  32. package/dist/adapters/generic-sql/query/db-now-sql.js.map +1 -0
  33. package/dist/adapters/generic-sql/query/generic-sql-uow-operation-compiler.js +9 -6
  34. package/dist/adapters/generic-sql/query/generic-sql-uow-operation-compiler.js.map +1 -1
  35. package/dist/adapters/generic-sql/query/select-builder.js.map +1 -1
  36. package/dist/adapters/generic-sql/query/sql-query-compiler.js +37 -9
  37. package/dist/adapters/generic-sql/query/sql-query-compiler.js.map +1 -1
  38. package/dist/adapters/generic-sql/query/where-builder.js +24 -20
  39. package/dist/adapters/generic-sql/query/where-builder.js.map +1 -1
  40. package/dist/adapters/generic-sql/uow-decoder.js +1 -1
  41. package/dist/adapters/generic-sql/uow-decoder.js.map +1 -1
  42. package/dist/adapters/generic-sql/uow-encoder.js +8 -9
  43. package/dist/adapters/generic-sql/uow-encoder.js.map +1 -1
  44. package/dist/adapters/in-memory/condition-evaluator.js +10 -6
  45. package/dist/adapters/in-memory/condition-evaluator.js.map +1 -1
  46. package/dist/adapters/in-memory/in-memory-adapter.d.ts.map +1 -1
  47. package/dist/adapters/in-memory/in-memory-adapter.js +45 -25
  48. package/dist/adapters/in-memory/in-memory-adapter.js.map +1 -1
  49. package/dist/adapters/in-memory/in-memory-uow.js +236 -13
  50. package/dist/adapters/in-memory/in-memory-uow.js.map +1 -1
  51. package/dist/adapters/in-memory/options.d.ts +2 -0
  52. package/dist/adapters/in-memory/options.d.ts.map +1 -1
  53. package/dist/adapters/in-memory/options.js +3 -2
  54. package/dist/adapters/in-memory/options.js.map +1 -1
  55. package/dist/adapters/in-memory/reference-resolution.js.map +1 -1
  56. package/dist/adapters/in-memory/store.js +1 -1
  57. package/dist/adapters/in-memory/store.js.map +1 -1
  58. package/dist/adapters/shared/from-unit-of-work-compiler.js +51 -24
  59. package/dist/adapters/shared/from-unit-of-work-compiler.js.map +1 -1
  60. package/dist/adapters/shared/uow-operation-compiler.js.map +1 -1
  61. package/dist/browser/adapters/adapters.d.ts +61 -0
  62. package/dist/browser/adapters/adapters.d.ts.map +1 -0
  63. package/dist/browser/adapters/generic-sql/migration/executor.d.ts +15 -0
  64. package/dist/browser/adapters/generic-sql/migration/executor.d.ts.map +1 -0
  65. package/dist/browser/adapters/generic-sql/migration/prepared-migrations.d.ts +66 -0
  66. package/dist/browser/adapters/generic-sql/migration/prepared-migrations.d.ts.map +1 -0
  67. package/dist/browser/adapters/generic-sql/sqlite-storage.d.ts +11 -0
  68. package/dist/browser/adapters/generic-sql/sqlite-storage.d.ts.map +1 -0
  69. package/dist/browser/adapters/in-memory/in-memory-adapter.d.ts +5 -0
  70. package/dist/browser/adapters/in-memory/index.d.ts +2 -0
  71. package/dist/browser/adapters/in-memory/options.d.ts +1 -0
  72. package/dist/browser/db-fragment-definition-builder.d.ts +237 -0
  73. package/dist/browser/db-fragment-definition-builder.d.ts.map +1 -0
  74. package/dist/browser/durable-hooks.d.ts +3 -0
  75. package/dist/browser/fragments/internal-fragment.d.ts +317 -0
  76. package/dist/browser/fragments/internal-fragment.d.ts.map +1 -0
  77. package/dist/browser/fragments/internal-fragment.schema.d.ts +1 -0
  78. package/dist/browser/hooks/durable-hooks-logger.d.ts +10 -0
  79. package/dist/browser/hooks/durable-hooks-logger.d.ts.map +1 -0
  80. package/dist/browser/hooks/hooks.d.ts +146 -0
  81. package/dist/browser/hooks/hooks.d.ts.map +1 -0
  82. package/dist/browser/id.js +1 -0
  83. package/dist/browser/internal/adapter-registry.d.ts +4 -0
  84. package/dist/browser/internal/outbox-state.d.ts +2 -0
  85. package/dist/browser/mod.d.ts +15 -0
  86. package/dist/browser/mod.d.ts.map +1 -0
  87. package/dist/browser/mod.js +17 -0
  88. package/dist/browser/mod.js.map +1 -0
  89. package/dist/browser/mod2.d.ts +48 -0
  90. package/dist/browser/mod2.d.ts.map +1 -0
  91. package/dist/browser/naming/sql-naming.d.ts +19 -0
  92. package/dist/browser/naming/sql-naming.d.ts.map +1 -0
  93. package/dist/browser/outbox/outbox.d.ts +21 -0
  94. package/dist/browser/outbox/outbox.d.ts.map +1 -0
  95. package/dist/browser/query/column-defaults.js +1 -0
  96. package/dist/browser/query/condition-builder.d.ts +44 -0
  97. package/dist/browser/query/condition-builder.d.ts.map +1 -0
  98. package/dist/browser/query/condition-builder.js +97 -0
  99. package/dist/browser/query/condition-builder.js.map +1 -0
  100. package/dist/browser/query/cursor.d.ts +105 -0
  101. package/dist/browser/query/cursor.d.ts.map +1 -0
  102. package/dist/browser/query/cursor.js +150 -0
  103. package/dist/browser/query/cursor.js.map +1 -0
  104. package/dist/browser/query/db-now.d.ts +22 -0
  105. package/dist/browser/query/db-now.d.ts.map +1 -0
  106. package/dist/browser/query/db-now.js +33 -0
  107. package/dist/browser/query/db-now.js.map +1 -0
  108. package/dist/browser/query/orm/orm.d.ts +18 -0
  109. package/dist/browser/query/orm/orm.d.ts.map +1 -0
  110. package/dist/browser/query/simple-query-interface.d.ts +108 -0
  111. package/dist/browser/query/simple-query-interface.d.ts.map +1 -0
  112. package/dist/browser/query/unit-of-work/execute-unit-of-work.d.ts +423 -0
  113. package/dist/browser/query/unit-of-work/execute-unit-of-work.d.ts.map +1 -0
  114. package/dist/browser/query/unit-of-work/execute-unit-of-work.js +507 -0
  115. package/dist/browser/query/unit-of-work/execute-unit-of-work.js.map +1 -0
  116. package/dist/browser/query/unit-of-work/retry-policy.d.ts +23 -0
  117. package/dist/browser/query/unit-of-work/retry-policy.d.ts.map +1 -0
  118. package/dist/browser/query/unit-of-work/retry-policy.js +40 -0
  119. package/dist/browser/query/unit-of-work/retry-policy.js.map +1 -0
  120. package/dist/browser/query/unit-of-work/unit-of-work.d.ts +703 -0
  121. package/dist/browser/query/unit-of-work/unit-of-work.d.ts.map +1 -0
  122. package/dist/browser/query/unit-of-work/unit-of-work.js +1206 -0
  123. package/dist/browser/query/unit-of-work/unit-of-work.js.map +1 -0
  124. package/dist/browser/query/value-encoding.js +38 -0
  125. package/dist/browser/query/value-encoding.js.map +1 -0
  126. package/dist/browser/schema/create.d.ts +326 -0
  127. package/dist/browser/schema/create.d.ts.map +1 -0
  128. package/dist/browser/schema/create.js +89 -0
  129. package/dist/browser/schema/create.js.map +1 -0
  130. package/dist/browser/schema/generate-id.js +28 -0
  131. package/dist/browser/schema/generate-id.js.map +1 -0
  132. package/dist/browser/shared/providers.d.ts +6 -0
  133. package/dist/browser/shared/providers.d.ts.map +1 -0
  134. package/dist/browser/sql-driver/connection/connection-provider.d.ts +13 -0
  135. package/dist/browser/sql-driver/connection/connection-provider.d.ts.map +1 -0
  136. package/dist/browser/sql-driver/dialect-adapter/dialect-adapter.d.ts +7 -0
  137. package/dist/browser/sql-driver/dialect-adapter/dialect-adapter.d.ts.map +1 -0
  138. package/dist/browser/sql-driver/driver/runtime-driver.d.ts +23 -0
  139. package/dist/browser/sql-driver/driver/runtime-driver.d.ts.map +1 -0
  140. package/dist/browser/sql-driver/query-executor/plugin.d.ts +17 -0
  141. package/dist/browser/sql-driver/query-executor/plugin.d.ts.map +1 -0
  142. package/dist/browser/sql-driver/query-executor/query-executor.d.ts +36 -0
  143. package/dist/browser/sql-driver/query-executor/query-executor.d.ts.map +1 -0
  144. package/dist/browser/sql-driver/sql-driver-adapter.d.ts +29 -0
  145. package/dist/browser/sql-driver/sql-driver-adapter.d.ts.map +1 -0
  146. package/dist/browser/sql-driver/sql-driver.d.ts +38 -0
  147. package/dist/browser/sql-driver/sql-driver.d.ts.map +1 -0
  148. package/dist/browser/sync/commands.d.ts +15 -0
  149. package/dist/browser/sync/commands.d.ts.map +1 -0
  150. package/dist/browser/sync/commands.js +27 -0
  151. package/dist/browser/sync/commands.js.map +1 -0
  152. package/dist/browser/sync/types.d.ts +63 -0
  153. package/dist/browser/sync/types.d.ts.map +1 -0
  154. package/dist/browser/util/types.d.ts +8 -0
  155. package/dist/browser/util/types.d.ts.map +1 -0
  156. package/dist/browser/with-database.d.ts +29 -0
  157. package/dist/browser/with-database.d.ts.map +1 -0
  158. package/dist/client.d.ts +4 -0
  159. package/dist/client.js +5 -0
  160. package/dist/db-fragment-definition-builder.d.ts +85 -28
  161. package/dist/db-fragment-definition-builder.d.ts.map +1 -1
  162. package/dist/db-fragment-definition-builder.js +374 -46
  163. package/dist/db-fragment-definition-builder.js.map +1 -1
  164. package/dist/dispatchers/cloudflare-do/dispatcher.d.ts +20 -0
  165. package/dist/dispatchers/cloudflare-do/dispatcher.d.ts.map +1 -0
  166. package/dist/dispatchers/cloudflare-do/dispatcher.js +147 -0
  167. package/dist/dispatchers/cloudflare-do/dispatcher.js.map +1 -0
  168. package/dist/dispatchers/cloudflare-do/index.d.ts +5 -20
  169. package/dist/dispatchers/cloudflare-do/index.d.ts.map +1 -1
  170. package/dist/dispatchers/cloudflare-do/index.js +23 -55
  171. package/dist/dispatchers/cloudflare-do/index.js.map +1 -1
  172. package/dist/dispatchers/node/dispatcher.d.ts +14 -0
  173. package/dist/dispatchers/node/dispatcher.d.ts.map +1 -0
  174. package/dist/dispatchers/node/dispatcher.js +80 -0
  175. package/dist/dispatchers/node/dispatcher.js.map +1 -0
  176. package/dist/dispatchers/node/index.d.ts +5 -10
  177. package/dist/dispatchers/node/index.d.ts.map +1 -1
  178. package/dist/dispatchers/node/index.js +21 -53
  179. package/dist/dispatchers/node/index.js.map +1 -1
  180. package/dist/durable-hooks.d.ts +31 -0
  181. package/dist/durable-hooks.d.ts.map +1 -0
  182. package/dist/durable-hooks.js +23 -0
  183. package/dist/durable-hooks.js.map +1 -0
  184. package/dist/fragments/internal-fragment.d.ts +128 -27
  185. package/dist/fragments/internal-fragment.d.ts.map +1 -1
  186. package/dist/fragments/internal-fragment.js +125 -78
  187. package/dist/fragments/internal-fragment.js.map +1 -1
  188. package/dist/fragments/internal-fragment.routes.js +138 -3
  189. package/dist/fragments/internal-fragment.routes.js.map +1 -1
  190. package/dist/fragments/internal-fragment.schema.d.ts +7 -1
  191. package/dist/fragments/internal-fragment.schema.d.ts.map +1 -1
  192. package/dist/fragments/internal-fragment.schema.js +18 -1
  193. package/dist/fragments/internal-fragment.schema.js.map +1 -1
  194. package/dist/hooks/durable-hooks-logger.d.ts +10 -0
  195. package/dist/hooks/durable-hooks-logger.d.ts.map +1 -0
  196. package/dist/hooks/durable-hooks-logger.js +75 -0
  197. package/dist/hooks/durable-hooks-logger.js.map +1 -0
  198. package/dist/hooks/durable-hooks-processor.d.ts +1 -14
  199. package/dist/hooks/durable-hooks-processor.js +58 -10
  200. package/dist/hooks/durable-hooks-processor.js.map +1 -1
  201. package/dist/hooks/durable-hooks-runtime.js +44 -0
  202. package/dist/hooks/durable-hooks-runtime.js.map +1 -0
  203. package/dist/hooks/hooks.d.ts +60 -2
  204. package/dist/hooks/hooks.d.ts.map +1 -1
  205. package/dist/hooks/hooks.js +214 -53
  206. package/dist/hooks/hooks.js.map +1 -1
  207. package/dist/id.d.ts +2 -2
  208. package/dist/id.js +2 -2
  209. package/dist/internal/adapter-registry.d.ts +11 -0
  210. package/dist/internal/adapter-registry.d.ts.map +1 -0
  211. package/dist/internal/adapter-registry.js +135 -0
  212. package/dist/internal/adapter-registry.js.map +1 -0
  213. package/dist/internal/outbox-state.d.ts +2 -0
  214. package/dist/internal/outbox-state.js +26 -0
  215. package/dist/internal/outbox-state.js.map +1 -0
  216. package/dist/migration-engine/auto-from-schema.d.ts +33 -0
  217. package/dist/migration-engine/auto-from-schema.d.ts.map +1 -0
  218. package/dist/migration-engine/auto-from-schema.js +210 -27
  219. package/dist/migration-engine/auto-from-schema.js.map +1 -1
  220. package/dist/migration-engine/generation-engine.d.ts.map +1 -1
  221. package/dist/migration-engine/generation-engine.js +17 -5
  222. package/dist/migration-engine/generation-engine.js.map +1 -1
  223. package/dist/migration-engine/shared.d.ts +113 -0
  224. package/dist/migration-engine/shared.d.ts.map +1 -0
  225. package/dist/migration-engine/shared.js.map +1 -1
  226. package/dist/mod.d.ts +12 -11
  227. package/dist/mod.d.ts.map +1 -1
  228. package/dist/mod.js +10 -10
  229. package/dist/mod.js.map +1 -1
  230. package/dist/naming/sql-naming.d.ts.map +1 -1
  231. package/dist/naming/sql-naming.js.map +1 -1
  232. package/dist/outbox/outbox-builder.js.map +1 -1
  233. package/dist/outbox/outbox.d.ts +3 -1
  234. package/dist/outbox/outbox.d.ts.map +1 -1
  235. package/dist/outbox/outbox.js.map +1 -1
  236. package/dist/query/column-defaults.js.map +1 -1
  237. package/dist/query/condition-builder.d.ts +7 -1
  238. package/dist/query/condition-builder.d.ts.map +1 -1
  239. package/dist/query/condition-builder.js +5 -1
  240. package/dist/query/condition-builder.js.map +1 -1
  241. package/dist/query/cursor-client.d.ts +105 -0
  242. package/dist/query/cursor-client.d.ts.map +1 -0
  243. package/dist/query/cursor-client.js +165 -0
  244. package/dist/query/cursor-client.js.map +1 -0
  245. package/dist/query/cursor.d.ts.map +1 -1
  246. package/dist/query/cursor.js +7 -1
  247. package/dist/query/cursor.js.map +1 -1
  248. package/dist/query/db-now.d.ts +15 -1
  249. package/dist/query/db-now.d.ts.map +1 -1
  250. package/dist/query/db-now.js +30 -2
  251. package/dist/query/db-now.js.map +1 -1
  252. package/dist/query/orm/orm.js.map +1 -1
  253. package/dist/query/serialize/create-sql-serializer.js +2 -2
  254. package/dist/query/serialize/create-sql-serializer.js.map +1 -1
  255. package/dist/query/serialize/dialect/mysql-serializer.js.map +1 -1
  256. package/dist/query/serialize/dialect/postgres-serializer.js.map +1 -1
  257. package/dist/query/serialize/dialect/sqlite-serializer.js +6 -2
  258. package/dist/query/serialize/dialect/sqlite-serializer.js.map +1 -1
  259. package/dist/query/simple-query-interface.d.ts +7 -3
  260. package/dist/query/simple-query-interface.d.ts.map +1 -1
  261. package/dist/query/unit-of-work/execute-unit-of-work.d.ts +37 -2
  262. package/dist/query/unit-of-work/execute-unit-of-work.d.ts.map +1 -1
  263. package/dist/query/unit-of-work/execute-unit-of-work.js +39 -18
  264. package/dist/query/unit-of-work/execute-unit-of-work.js.map +1 -1
  265. package/dist/query/unit-of-work/unit-of-work.d.ts +42 -16
  266. package/dist/query/unit-of-work/unit-of-work.d.ts.map +1 -1
  267. package/dist/query/unit-of-work/unit-of-work.js +50 -6
  268. package/dist/query/unit-of-work/unit-of-work.js.map +1 -1
  269. package/dist/query/value-decoding.js +8 -1
  270. package/dist/query/value-decoding.js.map +1 -1
  271. package/dist/query/value-encoding.js.map +1 -1
  272. package/dist/schema/create.d.ts +69 -25
  273. package/dist/schema/create.d.ts.map +1 -1
  274. package/dist/schema/create.js +91 -16
  275. package/dist/schema/create.js.map +1 -1
  276. package/dist/schema/type-conversion/create-sql-type-mapper.js +1 -1
  277. package/dist/schema/type-conversion/create-sql-type-mapper.js.map +1 -1
  278. package/dist/schema/type-conversion/dialect/sqlite.js.map +1 -1
  279. package/dist/schema/validator.d.ts.map +1 -1
  280. package/dist/schema/validator.js.map +1 -1
  281. package/dist/schema-output/drizzle.d.ts.map +1 -1
  282. package/dist/schema-output/drizzle.js +8 -6
  283. package/dist/schema-output/drizzle.js.map +1 -1
  284. package/dist/schema-output/prisma.d.ts.map +1 -1
  285. package/dist/schema-output/prisma.js +21 -10
  286. package/dist/schema-output/prisma.js.map +1 -1
  287. package/dist/sql-driver/dialects/durable-object-dialect.js +3 -9
  288. package/dist/sql-driver/dialects/durable-object-dialect.js.map +1 -1
  289. package/dist/sql-driver/query-executor/default-query-executor.js.map +1 -1
  290. package/dist/sql-driver/query-executor/query-executor-base.js.map +1 -1
  291. package/dist/sql-driver/sql-driver-adapter.js.map +1 -1
  292. package/dist/sql-driver/sql.js.map +1 -1
  293. package/dist/sync/commands.d.ts +15 -0
  294. package/dist/sync/commands.d.ts.map +1 -0
  295. package/dist/sync/commands.js +27 -0
  296. package/dist/sync/commands.js.map +1 -0
  297. package/dist/sync/index.d.ts +4 -0
  298. package/dist/sync/index.js +4 -0
  299. package/dist/sync/read-tracking.d.ts +25 -0
  300. package/dist/sync/read-tracking.d.ts.map +1 -0
  301. package/dist/sync/read-tracking.js +148 -0
  302. package/dist/sync/read-tracking.js.map +1 -0
  303. package/dist/sync/submit.js +213 -0
  304. package/dist/sync/submit.js.map +1 -0
  305. package/dist/sync/types.d.ts +63 -0
  306. package/dist/sync/types.d.ts.map +1 -0
  307. package/dist/util/default-database-adapter.js +6 -1
  308. package/dist/util/default-database-adapter.js.map +1 -1
  309. package/dist/with-database.d.ts +3 -6
  310. package/dist/with-database.d.ts.map +1 -1
  311. package/dist/with-database.js +7 -15
  312. package/dist/with-database.js.map +1 -1
  313. package/package.json +33 -41
  314. package/src/adapters/adapters.ts +5 -4
  315. package/src/adapters/drizzle/migrate-drizzle.test.ts +46 -9
  316. package/src/adapters/drizzle/migration-parity-drizzle-kit.test.ts +5 -3
  317. package/src/adapters/drizzle/test-utils.ts +2 -1
  318. package/src/adapters/generic-sql/generic-sql-adapter.test.ts +5 -3
  319. package/src/adapters/generic-sql/generic-sql-adapter.ts +21 -24
  320. package/src/adapters/generic-sql/generic-sql-uow-executor.test.ts +1 -0
  321. package/src/adapters/generic-sql/generic-sql-uow-executor.ts +81 -15
  322. package/src/adapters/generic-sql/migration/adapter-migration-parity.test.ts +4 -2
  323. package/src/adapters/generic-sql/migration/cold-kysely.ts +1 -0
  324. package/src/adapters/generic-sql/migration/dialect/mysql.test.ts +3 -2
  325. package/src/adapters/generic-sql/migration/dialect/mysql.ts +1 -0
  326. package/src/adapters/generic-sql/migration/dialect/postgres.test.ts +5 -4
  327. package/src/adapters/generic-sql/migration/dialect/postgres.ts +2 -1
  328. package/src/adapters/generic-sql/migration/dialect/sqlite.test.ts +795 -3
  329. package/src/adapters/generic-sql/migration/dialect/sqlite.ts +385 -57
  330. package/src/adapters/generic-sql/migration/executor.test.ts +52 -0
  331. package/src/adapters/generic-sql/migration/executor.ts +47 -4
  332. package/src/adapters/generic-sql/migration/prepared-migrations.test.ts +117 -14
  333. package/src/adapters/generic-sql/migration/prepared-migrations.ts +9 -8
  334. package/src/adapters/generic-sql/migration/sql-generator.ts +5 -3
  335. package/src/adapters/generic-sql/query/create-sql-query-compiler.ts +3 -3
  336. package/src/adapters/generic-sql/query/cursor-utils.test.ts +3 -2
  337. package/src/adapters/generic-sql/query/cursor-utils.ts +1 -1
  338. package/src/adapters/generic-sql/query/db-now-sql.ts +49 -0
  339. package/src/adapters/generic-sql/query/generic-sql-uow-operation-compiler.test.ts +144 -8
  340. package/src/adapters/generic-sql/query/generic-sql-uow-operation-compiler.ts +16 -17
  341. package/src/adapters/generic-sql/query/select-builder.test.ts +1 -0
  342. package/src/adapters/generic-sql/query/select-builder.ts +2 -2
  343. package/src/adapters/generic-sql/query/sql-query-compiler.test.ts +24 -5
  344. package/src/adapters/generic-sql/query/sql-query-compiler.ts +83 -13
  345. package/src/adapters/generic-sql/query/where-builder.test.ts +7 -5
  346. package/src/adapters/generic-sql/query/where-builder.ts +48 -29
  347. package/src/adapters/generic-sql/sql-adapter-pglite-migrations.test.ts +6 -15
  348. package/src/adapters/generic-sql/sql-adapter-pglite-pagination.test.ts +52 -7
  349. package/src/adapters/generic-sql/sql-adapter-pglite-queries.test.ts +9 -6
  350. package/src/adapters/generic-sql/sql-adapter-sqlite3-driver.test.ts +273 -5
  351. package/src/adapters/generic-sql/sql-adapter-sqlite3-uow.test.ts +123 -6
  352. package/src/adapters/generic-sql/sql-adapter-sqlocal.test.ts +4 -2
  353. package/src/adapters/generic-sql/uow-decoder.test.ts +4 -3
  354. package/src/adapters/generic-sql/uow-decoder.ts +3 -3
  355. package/src/adapters/generic-sql/uow-encoder.test.ts +4 -2
  356. package/src/adapters/generic-sql/uow-encoder.ts +14 -18
  357. package/src/adapters/in-memory/condition-evaluator.test.ts +2 -1
  358. package/src/adapters/in-memory/condition-evaluator.ts +9 -4
  359. package/src/adapters/in-memory/in-memory-adapter.ts +155 -44
  360. package/src/adapters/in-memory/in-memory-uow.mutations.test.ts +50 -2
  361. package/src/adapters/in-memory/in-memory-uow.retrieval.test.ts +158 -3
  362. package/src/adapters/in-memory/in-memory-uow.ts +402 -26
  363. package/src/adapters/in-memory/options.test.ts +1 -0
  364. package/src/adapters/in-memory/options.ts +5 -1
  365. package/src/adapters/in-memory/outbox.test.ts +361 -0
  366. package/src/adapters/in-memory/reference-resolution.test.ts +3 -2
  367. package/src/adapters/in-memory/reference-resolution.ts +2 -2
  368. package/src/adapters/in-memory/sorted-array-index.test.ts +1 -0
  369. package/src/adapters/in-memory/store.test.ts +1 -0
  370. package/src/adapters/in-memory/store.ts +3 -3
  371. package/src/adapters/in-memory/value-normalization.test.ts +1 -0
  372. package/src/adapters/prisma/prisma-adapter-sqlite3.test.ts +51 -7
  373. package/src/adapters/shared/from-unit-of-work-compiler.ts +156 -46
  374. package/src/adapters/shared/uow-operation-compiler.ts +3 -3
  375. package/src/browser/mod.ts +64 -0
  376. package/src/client.ts +19 -0
  377. package/src/db-fragment-definition-builder.test.ts +821 -47
  378. package/src/db-fragment-definition-builder.ts +857 -110
  379. package/src/db-fragment-instantiator.test.ts +114 -90
  380. package/src/db-fragment-integration.test.ts +9 -6
  381. package/src/dispatchers/cloudflare-do/dispatcher.ts +204 -0
  382. package/src/dispatchers/cloudflare-do/index.test.ts +145 -12
  383. package/src/dispatchers/cloudflare-do/index.ts +49 -90
  384. package/src/dispatchers/node/dispatcher.ts +112 -0
  385. package/src/dispatchers/node/index.test.ts +43 -14
  386. package/src/dispatchers/node/index.ts +38 -75
  387. package/src/durable-hooks.test.ts +80 -0
  388. package/src/durable-hooks.ts +67 -0
  389. package/src/fragments/internal-fragment.routes.test.ts +570 -0
  390. package/src/fragments/internal-fragment.routes.ts +297 -5
  391. package/src/fragments/internal-fragment.schema.ts +45 -1
  392. package/src/fragments/internal-fragment.test.ts +223 -251
  393. package/src/fragments/internal-fragment.ts +278 -154
  394. package/src/hooks/durable-hooks-logger.ts +126 -0
  395. package/src/hooks/durable-hooks-processor.pglite.test.ts +87 -0
  396. package/src/hooks/durable-hooks-processor.test.ts +179 -14
  397. package/src/hooks/durable-hooks-processor.ts +120 -14
  398. package/src/hooks/durable-hooks-runtime.test.ts +65 -0
  399. package/src/hooks/durable-hooks-runtime.ts +81 -0
  400. package/src/hooks/hooks.test.ts +314 -53
  401. package/src/hooks/hooks.ts +360 -81
  402. package/src/id.test.ts +34 -0
  403. package/src/id.ts +1 -3
  404. package/src/internal/adapter-registry.test.ts +93 -0
  405. package/src/internal/adapter-registry.ts +239 -0
  406. package/src/internal/outbox-state.ts +43 -0
  407. package/src/migration-engine/auto-from-schema.test.ts +93 -0
  408. package/src/migration-engine/auto-from-schema.ts +360 -42
  409. package/src/migration-engine/create.test.ts +2 -1
  410. package/src/migration-engine/create.ts +1 -1
  411. package/src/migration-engine/generation-engine.test.ts +66 -9
  412. package/src/migration-engine/generation-engine.ts +31 -10
  413. package/src/migration-engine/shared.ts +13 -0
  414. package/src/mod.ts +45 -27
  415. package/src/naming/sql-naming.ts +1 -0
  416. package/src/outbox/outbox-builder.ts +2 -2
  417. package/src/outbox/outbox.test.ts +216 -45
  418. package/src/outbox/outbox.ts +3 -1
  419. package/src/query/column-defaults.ts +1 -1
  420. package/src/query/condition-builder.test.ts +15 -0
  421. package/src/query/condition-builder.ts +7 -0
  422. package/src/query/cursor-client.test.ts +70 -0
  423. package/src/query/cursor-client.ts +263 -0
  424. package/src/query/cursor.test.ts +3 -2
  425. package/src/query/cursor.ts +15 -3
  426. package/src/query/db-now.ts +69 -2
  427. package/src/query/orm/orm.ts +2 -2
  428. package/src/query/query-type.test.ts +2 -1
  429. package/src/query/serialize/create-sql-serializer.ts +3 -3
  430. package/src/query/serialize/dialect/mysql-serializer.ts +1 -1
  431. package/src/query/serialize/dialect/postgres-serializer.ts +1 -1
  432. package/src/query/serialize/dialect/sqlite-serializer.test.ts +39 -2
  433. package/src/query/serialize/dialect/sqlite-serializer.ts +18 -5
  434. package/src/query/simple-query-interface.ts +10 -4
  435. package/src/query/unit-of-work/execute-unit-of-work.test.ts +347 -9
  436. package/src/query/unit-of-work/execute-unit-of-work.ts +63 -20
  437. package/src/query/unit-of-work/retry-policy.test.ts +1 -0
  438. package/src/query/unit-of-work/tx-builder.test.ts +73 -1
  439. package/src/query/unit-of-work/unit-of-work-coordinator.test.ts +5 -4
  440. package/src/query/unit-of-work/unit-of-work-types.test.ts +41 -11
  441. package/src/query/unit-of-work/unit-of-work.test.ts +28 -2
  442. package/src/query/unit-of-work/unit-of-work.ts +105 -19
  443. package/src/query/value-decoding.test.ts +50 -2
  444. package/src/query/value-decoding.ts +17 -4
  445. package/src/query/value-encoding.test.ts +1 -0
  446. package/src/query/value-encoding.ts +1 -1
  447. package/src/schema/create.test.ts +164 -5
  448. package/src/schema/create.ts +222 -24
  449. package/src/schema/generate-id.test.ts +1 -0
  450. package/src/schema/serialize.test.ts +4 -3
  451. package/src/schema/type-conversion/create-sql-type-mapper.ts +1 -1
  452. package/src/schema/type-conversion/dialect/sqlite.ts +2 -2
  453. package/src/schema/type-conversion/type-mapping.test.ts +2 -1
  454. package/src/schema/validator.test.ts +4 -2
  455. package/src/schema/validator.ts +1 -0
  456. package/src/schema-output/drizzle.test.ts +72 -19
  457. package/src/schema-output/drizzle.ts +24 -18
  458. package/src/schema-output/prisma.test.ts +172 -14
  459. package/src/schema-output/prisma.ts +34 -14
  460. package/src/sql-driver/better-sqlite3.test.ts +5 -3
  461. package/src/sql-driver/dialects/durable-object-dialect.ts +3 -8
  462. package/src/sql-driver/query-executor/default-query-executor.ts +1 -1
  463. package/src/sql-driver/query-executor/query-executor-base.ts +1 -1
  464. package/src/sql-driver/query-executor/query-executor.ts +1 -1
  465. package/src/sql-driver/sql-driver-adapter.ts +2 -2
  466. package/src/sql-driver/sql.ts +2 -1
  467. package/src/sql-driver/sqlocal.test.ts +4 -2
  468. package/src/sync/commands.test.ts +39 -0
  469. package/src/sync/commands.ts +51 -0
  470. package/src/sync/conflict-checker.test.ts +450 -0
  471. package/src/sync/conflict-checker.ts +248 -0
  472. package/src/sync/index.ts +14 -0
  473. package/src/sync/plan.ts +9 -0
  474. package/src/sync/read-tracking.test.ts +177 -0
  475. package/src/sync/read-tracking.ts +287 -0
  476. package/src/sync/submit.test.ts +205 -0
  477. package/src/sync/submit.ts +328 -0
  478. package/src/sync/types.ts +80 -0
  479. package/src/util/default-database-adapter.ts +15 -2
  480. package/src/with-database.ts +20 -50
  481. package/tsconfig.json +1 -1
  482. package/tsdown.config.ts +38 -26
  483. package/vitest.config.ts +1 -0
  484. package/dist/hooks/durable-hooks-processor.d.ts.map +0 -1
  485. package/dist/node_modules/.pnpm/rou3@0.7.12/node_modules/rou3/dist/index.js +0 -168
  486. package/dist/node_modules/.pnpm/rou3@0.7.12/node_modules/rou3/dist/index.js.map +0 -1
  487. package/dist/packages/fragno/dist/api/bind-services.js +0 -20
  488. package/dist/packages/fragno/dist/api/bind-services.js.map +0 -1
  489. package/dist/packages/fragno/dist/api/error.js +0 -48
  490. package/dist/packages/fragno/dist/api/error.js.map +0 -1
  491. package/dist/packages/fragno/dist/api/fragment-definition-builder.js +0 -321
  492. package/dist/packages/fragno/dist/api/fragment-definition-builder.js.map +0 -1
  493. package/dist/packages/fragno/dist/api/fragment-instantiator.js +0 -669
  494. package/dist/packages/fragno/dist/api/fragment-instantiator.js.map +0 -1
  495. package/dist/packages/fragno/dist/api/fragno-response.js +0 -73
  496. package/dist/packages/fragno/dist/api/fragno-response.js.map +0 -1
  497. package/dist/packages/fragno/dist/api/internal/response-stream.js +0 -81
  498. package/dist/packages/fragno/dist/api/internal/response-stream.js.map +0 -1
  499. package/dist/packages/fragno/dist/api/internal/route.js +0 -10
  500. package/dist/packages/fragno/dist/api/internal/route.js.map +0 -1
  501. package/dist/packages/fragno/dist/api/mutable-request-state.js +0 -97
  502. package/dist/packages/fragno/dist/api/mutable-request-state.js.map +0 -1
  503. package/dist/packages/fragno/dist/api/request-context-storage.js +0 -43
  504. package/dist/packages/fragno/dist/api/request-context-storage.js.map +0 -1
  505. package/dist/packages/fragno/dist/api/request-input-context.js +0 -185
  506. package/dist/packages/fragno/dist/api/request-input-context.js.map +0 -1
  507. package/dist/packages/fragno/dist/api/request-middleware.js +0 -83
  508. package/dist/packages/fragno/dist/api/request-middleware.js.map +0 -1
  509. package/dist/packages/fragno/dist/api/request-output-context.js +0 -119
  510. package/dist/packages/fragno/dist/api/request-output-context.js.map +0 -1
  511. package/dist/packages/fragno/dist/api/route.js +0 -30
  512. package/dist/packages/fragno/dist/api/route.js.map +0 -1
  513. package/dist/packages/fragno/dist/internal/symbols.js +0 -10
  514. package/dist/packages/fragno/dist/internal/symbols.js.map +0 -1
  515. package/dist/packages/fragno/dist/internal/trace-context.js +0 -12
  516. package/dist/packages/fragno/dist/internal/trace-context.js.map +0 -1
@@ -0,0 +1,361 @@
1
+ import { describe, expect, it } from "vitest";
2
+
3
+ import superjson, { type SuperJSONResult } from "superjson";
4
+
5
+ import { defineFragment, instantiate } from "@fragno-dev/core";
6
+
7
+ import type { InternalFragmentInstance } from "../../fragments/internal-fragment";
8
+ import { internalSchema } from "../../fragments/internal-fragment";
9
+ import { getInternalFragment } from "../../internal/adapter-registry";
10
+ import type { AnyFragnoInstantiatedDatabaseFragment, DatabaseRequestContext } from "../../mod";
11
+ import type { OutboxEntry, OutboxPayload } from "../../outbox/outbox";
12
+ import { schema, idColumn, column, referenceColumn, FragnoReference } from "../../schema/create";
13
+ import { withDatabase } from "../../with-database";
14
+ import { InMemoryAdapter } from "./in-memory-adapter";
15
+
16
+ const buildOutboxSchema = () =>
17
+ schema("outbox", (s) => {
18
+ return s
19
+ .addTable("users", (t) => {
20
+ return t
21
+ .addColumn("id", idColumn())
22
+ .addColumn("email", column("string"))
23
+ .createIndex("idx_users_email", ["email"], { unique: true });
24
+ })
25
+ .addTable("posts", (t) => {
26
+ return t
27
+ .addColumn("id", idColumn())
28
+ .addColumn("authorId", referenceColumn())
29
+ .addColumn("title", column("string"));
30
+ })
31
+ .addReference("author", {
32
+ type: "one",
33
+ from: { table: "posts", column: "authorId" },
34
+ to: { table: "users", column: "id" },
35
+ });
36
+ });
37
+
38
+ const outboxSchema = buildOutboxSchema();
39
+
40
+ const outboxFragmentName = "outbox-in-memory-test";
41
+ const outboxFragmentDef = defineFragment(outboxFragmentName)
42
+ .extend(withDatabase(outboxSchema))
43
+ .build();
44
+
45
+ type OutboxTestContext = {
46
+ fragment: AnyFragnoInstantiatedDatabaseFragment<typeof outboxSchema>;
47
+ internalFragment: InternalFragmentInstance;
48
+ cleanup: () => Promise<void>;
49
+ };
50
+
51
+ async function buildOutboxTest(options: { outboxEnabled?: boolean }): Promise<OutboxTestContext> {
52
+ const adapter = new InMemoryAdapter({
53
+ idSeed: "outbox-seed",
54
+ });
55
+
56
+ const fragment = instantiate(outboxFragmentDef)
57
+ .withConfig({})
58
+ .withRoutes([])
59
+ .withOptions({
60
+ databaseAdapter: adapter,
61
+ outbox: options.outboxEnabled ? { enabled: true } : undefined,
62
+ })
63
+ .build();
64
+
65
+ return {
66
+ fragment,
67
+ internalFragment: getInternalFragment(adapter),
68
+ cleanup: async () => {
69
+ await adapter.close();
70
+ },
71
+ };
72
+ }
73
+
74
+ async function listOutbox(
75
+ internalFragment: InternalFragmentInstance,
76
+ options?: { afterVersionstamp?: string; limit?: number },
77
+ ): Promise<OutboxEntry[]> {
78
+ return internalFragment.inContext(async function (this: DatabaseRequestContext) {
79
+ return (await this.handlerTx()
80
+ .withServiceCalls(() => [internalFragment.services.outboxService.list(options)] as const)
81
+ .transform(({ serviceResult: [result] }) => result)
82
+ .execute()) as OutboxEntry[];
83
+ });
84
+ }
85
+
86
+ async function listOutboxMutations(internalFragment: InternalFragmentInstance): Promise<
87
+ Array<{
88
+ entryVersionstamp: string;
89
+ mutationVersionstamp: string;
90
+ uowId: string;
91
+ schema: string;
92
+ table: string;
93
+ externalId: string;
94
+ op: string;
95
+ }>
96
+ > {
97
+ return internalFragment.inContext(async function (this: DatabaseRequestContext) {
98
+ return await this.handlerTx()
99
+ .retrieve(({ forSchema }) =>
100
+ forSchema(internalSchema).find("fragno_db_outbox_mutations", (b) =>
101
+ b
102
+ .whereIndex("idx_outbox_mutations_entry")
103
+ .orderByIndex("idx_outbox_mutations_entry", "asc"),
104
+ ),
105
+ )
106
+ .transformRetrieve(([result]) => result)
107
+ .execute();
108
+ });
109
+ }
110
+
111
+ async function createUser(
112
+ fragment: AnyFragnoInstantiatedDatabaseFragment<typeof outboxSchema>,
113
+ email: string,
114
+ ) {
115
+ return fragment.inContext(async function (this: DatabaseRequestContext) {
116
+ await this.handlerTx()
117
+ .mutate(({ forSchema }) => forSchema(outboxSchema).create("users", { email }))
118
+ .execute();
119
+
120
+ const user = await this.handlerTx()
121
+ .retrieve(({ forSchema }) =>
122
+ forSchema(outboxSchema).findFirst("users", (b) =>
123
+ b.whereIndex("idx_users_email", (eb) => eb("email", "=", email)),
124
+ ),
125
+ )
126
+ .transformRetrieve(([result]) => result)
127
+ .execute();
128
+
129
+ if (!user) {
130
+ throw new Error("Expected user to be created.");
131
+ }
132
+
133
+ return user.id;
134
+ });
135
+ }
136
+
137
+ async function createPost(
138
+ fragment: AnyFragnoInstantiatedDatabaseFragment<typeof outboxSchema>,
139
+ title: string,
140
+ authorId: FragnoReference,
141
+ ) {
142
+ return fragment.inContext(async function (this: DatabaseRequestContext) {
143
+ return await this.handlerTx()
144
+ .mutate(({ forSchema }) => forSchema(outboxSchema).create("posts", { title, authorId }))
145
+ .transform(({ mutateResult }) => mutateResult)
146
+ .execute();
147
+ });
148
+ }
149
+
150
+ describe("in-memory outbox", () => {
151
+ it("does not write outbox entries when disabled", async () => {
152
+ const { fragment, internalFragment, cleanup } = await buildOutboxTest({
153
+ outboxEnabled: false,
154
+ });
155
+
156
+ await createUser(fragment, "disabled@example.com");
157
+
158
+ const entries = await listOutbox(internalFragment);
159
+ expect(entries).toHaveLength(0);
160
+ const mutations = await listOutboxMutations(internalFragment);
161
+ expect(mutations).toHaveLength(0);
162
+
163
+ await cleanup();
164
+ });
165
+
166
+ it("stores refMap placeholders and lists entries in order", async () => {
167
+ const { fragment, internalFragment, cleanup } = await buildOutboxTest({ outboxEnabled: true });
168
+
169
+ const userId = await createUser(fragment, "alpha@example.com");
170
+ expect(userId.internalId).toBeDefined();
171
+
172
+ await createPost(fragment, "Hello", FragnoReference.fromInternal(userId.internalId!));
173
+
174
+ const entries = await listOutbox(internalFragment);
175
+ expect(entries).toHaveLength(2);
176
+ expect(entries[0].versionstamp < entries[1].versionstamp).toBe(true);
177
+
178
+ const payload = superjson.deserialize(entries[1].payload as SuperJSONResult) as OutboxPayload;
179
+ expect(payload.version).toBe(1);
180
+ expect(payload.mutations).toHaveLength(1);
181
+ const [mutation] = payload.mutations;
182
+ if (mutation.op !== "create") {
183
+ throw new Error("Expected create mutation in outbox payload.");
184
+ }
185
+ expect(mutation.values).toMatchObject({
186
+ authorId: { __fragno_ref: "0.authorId" },
187
+ });
188
+ expect(entries[1].refMap).toEqual({
189
+ "0.authorId": userId.externalId,
190
+ });
191
+
192
+ await cleanup();
193
+ });
194
+
195
+ it("writes mutation log rows for each outbox entry", async () => {
196
+ const { fragment, internalFragment, cleanup } = await buildOutboxTest({
197
+ outboxEnabled: true,
198
+ });
199
+
200
+ const userId = await createUser(fragment, "log-alpha@example.com");
201
+ expect(userId.internalId).toBeDefined();
202
+
203
+ await createPost(fragment, "Log", FragnoReference.fromInternal(userId.internalId!));
204
+
205
+ const entries = await listOutbox(internalFragment);
206
+ const mutations = await listOutboxMutations(internalFragment);
207
+
208
+ expect(entries).toHaveLength(2);
209
+ expect(mutations).toHaveLength(2);
210
+
211
+ const entryByVersion = new Map(entries.map((entry) => [entry.versionstamp, entry]));
212
+
213
+ for (const entry of entries) {
214
+ const payload = superjson.deserialize(entry.payload as SuperJSONResult) as OutboxPayload;
215
+ expect(payload.mutations).toHaveLength(1);
216
+ }
217
+
218
+ for (const mutationRow of mutations) {
219
+ const entry = entryByVersion.get(mutationRow.entryVersionstamp);
220
+ expect(entry).toBeDefined();
221
+ const payload = superjson.deserialize(entry!.payload as SuperJSONResult) as OutboxPayload;
222
+ const mutation = payload.mutations[0];
223
+ expect(mutationRow.mutationVersionstamp).toBe(mutation.versionstamp);
224
+ expect(mutationRow.uowId).toBe(entry!.uowId);
225
+ expect(mutationRow.schema).toBe(mutation.schema);
226
+ expect(mutationRow.table).toBe(mutation.table);
227
+ expect(mutationRow.externalId).toBe(mutation.externalId);
228
+ expect(mutationRow.op).toBe(mutation.op);
229
+ }
230
+
231
+ await cleanup();
232
+ });
233
+
234
+ it("resolves refMap lookups within the correct namespace", async () => {
235
+ const adapter = new InMemoryAdapter({
236
+ idSeed: "outbox-seed",
237
+ });
238
+
239
+ try {
240
+ const schemaA = buildOutboxSchema();
241
+ const schemaB = buildOutboxSchema();
242
+
243
+ const fragmentDefA = defineFragment(`${outboxFragmentName}-a`)
244
+ .extend(withDatabase(schemaA))
245
+ .build();
246
+ const fragmentDefB = defineFragment(`${outboxFragmentName}-b`)
247
+ .extend(withDatabase(schemaB))
248
+ .build();
249
+
250
+ const fragmentA = instantiate(fragmentDefA)
251
+ .withConfig({})
252
+ .withRoutes([])
253
+ .withOptions({
254
+ databaseAdapter: adapter,
255
+ databaseNamespace: "tenant-a",
256
+ outbox: { enabled: true },
257
+ })
258
+ .build();
259
+ const fragmentB = instantiate(fragmentDefB)
260
+ .withConfig({})
261
+ .withRoutes([])
262
+ .withOptions({
263
+ databaseAdapter: adapter,
264
+ databaseNamespace: "tenant-b",
265
+ outbox: { enabled: true },
266
+ })
267
+ .build();
268
+
269
+ const userA = await fragmentA.inContext(async function (this: DatabaseRequestContext) {
270
+ await this.handlerTx()
271
+ .mutate(({ forSchema }) =>
272
+ forSchema(schemaA).create("users", { email: "tenant-a@example.com" }),
273
+ )
274
+ .execute();
275
+
276
+ const user = await this.handlerTx()
277
+ .retrieve(({ forSchema }) =>
278
+ forSchema(schemaA).findFirst("users", (b) =>
279
+ b.whereIndex("idx_users_email", (eb) => eb("email", "=", "tenant-a@example.com")),
280
+ ),
281
+ )
282
+ .transformRetrieve(([result]) => result)
283
+ .execute();
284
+
285
+ if (!user) {
286
+ throw new Error("Expected user to be created.");
287
+ }
288
+
289
+ return user.id;
290
+ });
291
+
292
+ const userB = await fragmentB.inContext(async function (this: DatabaseRequestContext) {
293
+ await this.handlerTx()
294
+ .mutate(({ forSchema }) =>
295
+ forSchema(schemaB).create("users", { email: "tenant-b@example.com" }),
296
+ )
297
+ .execute();
298
+
299
+ const user = await this.handlerTx()
300
+ .retrieve(({ forSchema }) =>
301
+ forSchema(schemaB).findFirst("users", (b) =>
302
+ b.whereIndex("idx_users_email", (eb) => eb("email", "=", "tenant-b@example.com")),
303
+ ),
304
+ )
305
+ .transformRetrieve(([result]) => result)
306
+ .execute();
307
+
308
+ if (!user) {
309
+ throw new Error("Expected user to be created.");
310
+ }
311
+
312
+ return user.id;
313
+ });
314
+
315
+ expect(userA.internalId).toBeDefined();
316
+ expect(userB.internalId).toBeDefined();
317
+ expect(userA.internalId).toBe(userB.internalId);
318
+
319
+ await fragmentB.inContext(async function (this: DatabaseRequestContext) {
320
+ await this.handlerTx()
321
+ .mutate(({ forSchema }) =>
322
+ forSchema(schemaB).create("posts", {
323
+ title: "Hello",
324
+ authorId: FragnoReference.fromInternal(userB.internalId!),
325
+ }),
326
+ )
327
+ .execute();
328
+ });
329
+
330
+ const entries = await listOutbox(getInternalFragment(adapter));
331
+ let postEntry: OutboxEntry | undefined;
332
+ let postPayload: OutboxPayload | undefined;
333
+
334
+ for (const entry of entries) {
335
+ const payload = superjson.deserialize(entry.payload as SuperJSONResult) as OutboxPayload;
336
+ if (
337
+ payload.mutations.some(
338
+ (mutation) =>
339
+ mutation.op === "create" &&
340
+ mutation.table === "posts" &&
341
+ mutation.namespace === "tenant-b",
342
+ )
343
+ ) {
344
+ postEntry = entry;
345
+ postPayload = payload;
346
+ break;
347
+ }
348
+ }
349
+
350
+ if (!postEntry || !postPayload) {
351
+ throw new Error("Expected outbox entry for tenant-b posts.");
352
+ }
353
+
354
+ expect(postEntry.refMap).toEqual({
355
+ "0.authorId": userB.externalId,
356
+ });
357
+ } finally {
358
+ await adapter.close();
359
+ }
360
+ });
361
+ });
@@ -1,8 +1,9 @@
1
1
  import { describe, expect, it } from "vitest";
2
- import { column, idColumn, referenceColumn, schema } from "../../schema/create";
2
+
3
3
  import { ReferenceSubquery } from "../../query/value-encoding";
4
- import { createInMemoryStore, ensureNamespaceStore } from "./store";
4
+ import { column, idColumn, referenceColumn, schema } from "../../schema/create";
5
5
  import { resolveReferenceSubqueries } from "./reference-resolution";
6
+ import { createInMemoryStore, ensureNamespaceStore } from "./store";
6
7
 
7
8
  const testSchema = schema("test", (s) =>
8
9
  s
@@ -1,7 +1,7 @@
1
- import type { AnyTable } from "../../schema/create";
1
+ import type { NamingResolver } from "../../naming/sql-naming";
2
2
  import { ReferenceSubquery } from "../../query/value-encoding";
3
+ import type { AnyTable } from "../../schema/create";
3
4
  import type { InMemoryNamespaceStore, InMemoryTableStore } from "./store";
4
- import type { NamingResolver } from "../../naming/sql-naming";
5
5
 
6
6
  const getTableStore = (
7
7
  namespaceStore: InMemoryNamespaceStore,
@@ -1,4 +1,5 @@
1
1
  import { describe, expect, it } from "vitest";
2
+
2
3
  import { UniqueConstraintError } from "./errors";
3
4
  import { SortedArrayIndex } from "./sorted-array-index";
4
5
 
@@ -1,4 +1,5 @@
1
1
  import { describe, expect, it } from "vitest";
2
+
2
3
  import { column, idColumn, schema } from "../../schema/create";
3
4
  import { createInMemoryStore, ensureNamespaceStore } from "./store";
4
5
 
@@ -1,9 +1,9 @@
1
+ import type { NamingResolver } from "../../naming/sql-naming";
2
+ import { createSQLSerializer } from "../../query/serialize/create-sql-serializer";
1
3
  import type { AnyColumn, AnySchema, AnyTable } from "../../schema/create";
2
- import { SortedArrayIndex } from "./sorted-array-index";
3
4
  import { SQLocalDriverConfig } from "../generic-sql/driver-config";
4
- import { createSQLSerializer } from "../../query/serialize/create-sql-serializer";
5
+ import { SortedArrayIndex } from "./sorted-array-index";
5
6
  import { compareNormalizedValues } from "./value-comparison";
6
- import type { NamingResolver } from "../../naming/sql-naming";
7
7
 
8
8
  export type InMemoryRow = Record<string, unknown>;
9
9
 
@@ -1,4 +1,5 @@
1
1
  import { describe, expect, it } from "vitest";
2
+
2
3
  import { column, idColumn, referenceColumn, schema } from "../../schema/create";
3
4
  import { buildIndexKey } from "./store";
4
5
 
@@ -1,19 +1,21 @@
1
+ import { beforeAll, describe, expect, expectTypeOf, it } from "vitest";
2
+
1
3
  import SQLite from "better-sqlite3";
2
4
  import { SqliteDialect } from "kysely";
3
- import { beforeAll, describe, expect, expectTypeOf, it } from "vitest";
4
- import { SqlAdapter } from "../generic-sql/generic-sql-adapter";
5
- import { column, idColumn, referenceColumn, schema, type FragnoId } from "../../schema/create";
5
+
6
+ import { internalSchema } from "../../fragments/internal-fragment";
7
+ import { generateSchemaArtifacts } from "../../migration-engine/generation-engine";
8
+ import { FragnoDatabase } from "../../mod";
6
9
  import { Cursor } from "../../query/cursor";
7
10
  import {
8
11
  createHandlerTxBuilder,
9
12
  createServiceTxBuilder,
10
13
  } from "../../query/unit-of-work/execute-unit-of-work";
11
14
  import { ExponentialBackoffRetryPolicy } from "../../query/unit-of-work/retry-policy";
15
+ import { column, idColumn, referenceColumn, schema, type FragnoId } from "../../schema/create";
12
16
  import { BetterSQLite3DriverConfig } from "../generic-sql/driver-config";
13
- import { internalSchema } from "../../fragments/internal-fragment";
17
+ import { SqlAdapter } from "../generic-sql/generic-sql-adapter";
14
18
  import { sqliteStorageDefault, sqliteStoragePrisma } from "../generic-sql/sqlite-storage";
15
- import { FragnoDatabase } from "../../mod";
16
- import { generateSchemaArtifacts } from "../../migration-engine/generation-engine";
17
19
 
18
20
  describe("SqlAdapter SQLite (prisma profile)", () => {
19
21
  const testSchema = schema("test", (s) => {
@@ -23,7 +25,8 @@ describe("SqlAdapter SQLite (prisma profile)", () => {
23
25
  .addColumn("id", idColumn())
24
26
  .addColumn("name", column("string"))
25
27
  .addColumn("age", column("integer").nullable())
26
- .createIndex("name_idx", ["name"]);
28
+ .createIndex("name_idx", ["name"])
29
+ .createIndex("name_id_idx", ["name", "id"]);
27
30
  })
28
31
  .addTable("emails", (t) => {
29
32
  return t
@@ -384,6 +387,47 @@ describe("SqlAdapter SQLite (prisma profile)", () => {
384
387
  expect(secondPage.map((u) => u.name)).toEqual([`${prefix} C`, `${prefix} D`]);
385
388
  });
386
389
 
390
+ it("should support composite cursor pagination", async () => {
391
+ const queryEngine = adapter.createQueryEngine(testSchema, "namespace");
392
+ const nameValue = "Prisma SQLite Composite Cursor";
393
+
394
+ const createUow = queryEngine.createUnitOfWork("create-composite-cursor-users");
395
+ for (let i = 0; i < 5; i += 1) {
396
+ createUow.create("users", { name: nameValue, age: 30 + i });
397
+ }
398
+ await createUow.executeMutations();
399
+
400
+ const all = await queryEngine.find("users", (b) =>
401
+ b
402
+ .whereIndex("name_id_idx", (eb) => eb("name", "=", nameValue))
403
+ .orderByIndex("name_id_idx", "asc"),
404
+ );
405
+
406
+ const firstPage = await queryEngine.findWithCursor("users", (b) =>
407
+ b
408
+ .whereIndex("name_id_idx", (eb) => eb("name", "=", nameValue))
409
+ .orderByIndex("name_id_idx", "asc")
410
+ .pageSize(2),
411
+ );
412
+
413
+ const secondPage = await queryEngine.findWithCursor("users", (b) =>
414
+ b
415
+ .whereIndex("name_id_idx", (eb) => eb("name", "=", nameValue))
416
+ .after(firstPage.cursor!)
417
+ .orderByIndex("name_id_idx", "asc")
418
+ .pageSize(2),
419
+ );
420
+
421
+ const ids = (rows: { id: FragnoId }[]) => rows.map((row) => row.id.externalId);
422
+
423
+ expect(all.length).toBeGreaterThanOrEqual(5);
424
+ expect(firstPage.items).toHaveLength(2);
425
+ expect(firstPage.cursor).toBeInstanceOf(Cursor);
426
+ expect(firstPage.hasNextPage).toBe(true);
427
+ expect(ids(firstPage.items)).toEqual(ids(all).slice(0, 2));
428
+ expect(ids([...firstPage.items, ...secondPage.items])).toEqual(ids(all).slice(0, 4));
429
+ });
430
+
387
431
  it("should verify hasNextPage in cursor pagination", async () => {
388
432
  const queryEngine = adapter.createQueryEngine(testSchema, "namespace");
389
433
  const prefix = "Prisma SQLite HasNextPage";