@fragno-dev/db 0.2.2 → 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 (355) hide show
  1. package/.turbo/turbo-build.log +202 -140
  2. package/CHANGELOG.md +35 -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 +39 -29
  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 +18 -7
  80. package/dist/db-fragment-definition-builder.d.ts.map +1 -1
  81. package/dist/db-fragment-definition-builder.js +116 -54
  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 +79 -2
  92. package/dist/fragments/internal-fragment.d.ts.map +1 -1
  93. package/dist/fragments/internal-fragment.js +150 -32
  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 +42 -1
  106. package/dist/hooks/hooks.d.ts.map +1 -1
  107. package/dist/hooks/hooks.js +72 -6
  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 +15 -8
  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.map +1 -1
  165. package/dist/query/unit-of-work/execute-unit-of-work.js +11 -6
  166. package/dist/query/unit-of-work/execute-unit-of-work.js.map +1 -1
  167. package/dist/query/unit-of-work/unit-of-work.d.ts +50 -14
  168. package/dist/query/unit-of-work/unit-of-work.d.ts.map +1 -1
  169. package/dist/query/unit-of-work/unit-of-work.js +86 -5
  170. package/dist/query/unit-of-work/unit-of-work.js.map +1 -1
  171. package/dist/query/value-decoding.js +9 -6
  172. package/dist/query/value-decoding.js.map +1 -1
  173. package/dist/query/value-encoding.js +29 -9
  174. package/dist/query/value-encoding.js.map +1 -1
  175. package/dist/schema/create.d.ts +38 -14
  176. package/dist/schema/create.d.ts.map +1 -1
  177. package/dist/schema/create.js +81 -42
  178. package/dist/schema/create.js.map +1 -1
  179. package/dist/schema/generate-id.js +2 -2
  180. package/dist/schema/generate-id.js.map +1 -1
  181. package/dist/schema/type-conversion/create-sql-type-mapper.js +3 -2
  182. package/dist/schema/type-conversion/create-sql-type-mapper.js.map +1 -1
  183. package/dist/schema/type-conversion/dialect/sqlite.js +9 -0
  184. package/dist/schema/type-conversion/dialect/sqlite.js.map +1 -1
  185. package/dist/schema/validator.d.ts +10 -0
  186. package/dist/schema/validator.d.ts.map +1 -0
  187. package/dist/schema/validator.js +123 -0
  188. package/dist/schema/validator.js.map +1 -0
  189. package/dist/schema-output/drizzle.d.ts +30 -0
  190. package/dist/schema-output/drizzle.d.ts.map +1 -0
  191. package/dist/{adapters/drizzle/generate.js → schema-output/drizzle.js} +82 -56
  192. package/dist/schema-output/drizzle.js.map +1 -0
  193. package/dist/schema-output/prisma.d.ts +17 -0
  194. package/dist/schema-output/prisma.d.ts.map +1 -0
  195. package/dist/schema-output/prisma.js +296 -0
  196. package/dist/schema-output/prisma.js.map +1 -0
  197. package/dist/util/default-database-adapter.js +61 -0
  198. package/dist/util/default-database-adapter.js.map +1 -0
  199. package/dist/with-database.d.ts +1 -1
  200. package/dist/with-database.d.ts.map +1 -1
  201. package/dist/with-database.js +12 -3
  202. package/dist/with-database.js.map +1 -1
  203. package/package.json +43 -28
  204. package/src/adapters/adapters.ts +30 -24
  205. package/src/adapters/drizzle/migrate-drizzle.test.ts +54 -33
  206. package/src/adapters/drizzle/migration-parity-drizzle-kit.test.ts +599 -0
  207. package/src/adapters/drizzle/test-utils.ts +12 -8
  208. package/src/adapters/generic-sql/driver-config.ts +38 -0
  209. package/src/adapters/generic-sql/generic-sql-adapter.test.ts +5 -5
  210. package/src/adapters/generic-sql/generic-sql-adapter.ts +110 -24
  211. package/src/adapters/generic-sql/generic-sql-uow-executor.test.ts +54 -0
  212. package/src/adapters/generic-sql/generic-sql-uow-executor.ts +231 -3
  213. package/src/adapters/generic-sql/migration/adapter-migration-parity.test.ts +118 -0
  214. package/src/adapters/generic-sql/migration/dialect/mysql.test.ts +26 -8
  215. package/src/adapters/generic-sql/migration/dialect/mysql.ts +46 -8
  216. package/src/adapters/generic-sql/migration/dialect/postgres.test.ts +25 -7
  217. package/src/adapters/generic-sql/migration/dialect/postgres.ts +8 -4
  218. package/src/adapters/generic-sql/migration/dialect/sqlite.test.ts +47 -8
  219. package/src/adapters/generic-sql/migration/dialect/sqlite.ts +27 -12
  220. package/src/adapters/generic-sql/migration/prepared-migrations.test.ts +128 -39
  221. package/src/adapters/generic-sql/migration/prepared-migrations.ts +15 -8
  222. package/src/adapters/generic-sql/migration/sql-generator.ts +142 -65
  223. package/src/adapters/generic-sql/query/create-sql-query-compiler.ts +9 -6
  224. package/src/adapters/generic-sql/query/cursor-utils.test.ts +271 -0
  225. package/src/adapters/generic-sql/query/cursor-utils.ts +41 -6
  226. package/src/adapters/generic-sql/query/generic-sql-uow-operation-compiler.test.ts +27 -27
  227. package/src/adapters/generic-sql/query/generic-sql-uow-operation-compiler.ts +38 -24
  228. package/src/adapters/generic-sql/query/select-builder.test.ts +15 -11
  229. package/src/adapters/generic-sql/query/select-builder.ts +6 -2
  230. package/src/adapters/generic-sql/query/sql-query-compiler.test.ts +52 -2
  231. package/src/adapters/generic-sql/query/sql-query-compiler.ts +50 -15
  232. package/src/adapters/generic-sql/query/where-builder.test.ts +91 -17
  233. package/src/adapters/generic-sql/query/where-builder.ts +90 -38
  234. package/src/adapters/{kysely/kysely-adapter-pglite.test.ts → generic-sql/sql-adapter-pglite-migrations.test.ts} +6 -6
  235. package/src/adapters/generic-sql/sql-adapter-pglite-pagination.test.ts +806 -0
  236. package/src/adapters/{drizzle/drizzle-adapter-pglite.test.ts → generic-sql/sql-adapter-pglite-queries.test.ts} +11 -11
  237. package/src/adapters/generic-sql/{test/generic-drizzle-adapter-sqlite3.test.ts → sql-adapter-sqlite3-driver.test.ts} +10 -10
  238. package/src/adapters/{drizzle/drizzle-adapter-sqlite3.test.ts → generic-sql/sql-adapter-sqlite3-uow.test.ts} +7 -7
  239. package/src/adapters/{kysely/kysely-adapter-sqlocal.test.ts → generic-sql/sql-adapter-sqlocal.test.ts} +6 -6
  240. package/src/adapters/generic-sql/sqlite-storage.ts +20 -0
  241. package/src/adapters/generic-sql/uow-decoder.test.ts +1 -1
  242. package/src/adapters/generic-sql/uow-decoder.ts +21 -3
  243. package/src/adapters/generic-sql/uow-encoder.test.ts +33 -2
  244. package/src/adapters/generic-sql/uow-encoder.ts +50 -11
  245. package/src/adapters/in-memory/condition-evaluator.test.ts +193 -0
  246. package/src/adapters/in-memory/condition-evaluator.ts +275 -0
  247. package/src/adapters/in-memory/errors.ts +20 -0
  248. package/src/adapters/in-memory/in-memory-adapter.ts +277 -0
  249. package/src/adapters/in-memory/in-memory-uow.mutations.test.ts +296 -0
  250. package/src/adapters/in-memory/in-memory-uow.retrieval.test.ts +100 -0
  251. package/src/adapters/in-memory/in-memory-uow.ts +1348 -0
  252. package/src/adapters/in-memory/index.ts +3 -0
  253. package/src/adapters/in-memory/options.test.ts +41 -0
  254. package/src/adapters/in-memory/options.ts +87 -0
  255. package/src/adapters/in-memory/reference-resolution.test.ts +50 -0
  256. package/src/adapters/in-memory/reference-resolution.ts +67 -0
  257. package/src/adapters/in-memory/sorted-array-index.test.ts +123 -0
  258. package/src/adapters/in-memory/sorted-array-index.ts +228 -0
  259. package/src/adapters/in-memory/store.test.ts +68 -0
  260. package/src/adapters/in-memory/store.ts +145 -0
  261. package/src/adapters/in-memory/value-comparison.ts +53 -0
  262. package/src/adapters/in-memory/value-normalization.test.ts +57 -0
  263. package/src/adapters/prisma/prisma-adapter-sqlite3.test.ts +1163 -0
  264. package/src/adapters/shared/from-unit-of-work-compiler.ts +3 -1
  265. package/src/adapters/shared/uow-operation-compiler.ts +26 -16
  266. package/src/adapters/sql/index.ts +12 -0
  267. package/src/db-fragment-definition-builder.test.ts +30 -12
  268. package/src/db-fragment-definition-builder.ts +142 -73
  269. package/src/db-fragment-instantiator.test.ts +105 -13
  270. package/src/db-fragment-integration.test.ts +9 -7
  271. package/src/dispatchers/cloudflare-do/index.test.ts +73 -0
  272. package/src/dispatchers/cloudflare-do/index.ts +104 -0
  273. package/src/dispatchers/node/index.test.ts +91 -0
  274. package/src/dispatchers/node/index.ts +87 -0
  275. package/src/fragments/internal-fragment.routes.ts +42 -0
  276. package/src/fragments/internal-fragment.schema.ts +51 -0
  277. package/src/fragments/internal-fragment.test.ts +458 -8
  278. package/src/fragments/internal-fragment.ts +322 -63
  279. package/src/hooks/durable-hooks-processor.test.ts +117 -0
  280. package/src/hooks/durable-hooks-processor.ts +67 -0
  281. package/src/hooks/hooks.test.ts +165 -5
  282. package/src/hooks/hooks.ts +197 -9
  283. package/src/migration-engine/auto-from-schema.test.ts +14 -14
  284. package/src/migration-engine/auto-from-schema.ts +5 -2
  285. package/src/migration-engine/create.test.ts +2 -2
  286. package/src/migration-engine/generation-engine.test.ts +229 -104
  287. package/src/migration-engine/generation-engine.ts +94 -64
  288. package/src/migration-engine/shared.ts +1 -0
  289. package/src/mod.ts +64 -26
  290. package/src/naming/sql-naming.ts +180 -0
  291. package/src/outbox/outbox-builder.ts +241 -0
  292. package/src/outbox/outbox.test.ts +253 -0
  293. package/src/outbox/outbox.ts +137 -0
  294. package/src/query/column-defaults.ts +41 -3
  295. package/src/query/condition-builder.test.ts +3 -3
  296. package/src/query/cursor.test.ts +116 -18
  297. package/src/query/cursor.ts +75 -26
  298. package/src/query/db-now.ts +6 -0
  299. package/src/query/query-type.test.ts +2 -2
  300. package/src/query/serialize/create-sql-serializer.ts +7 -2
  301. package/src/query/serialize/dialect/mysql-serializer.ts +12 -4
  302. package/src/query/serialize/dialect/postgres-serializer.ts +34 -4
  303. package/src/query/serialize/dialect/sqlite-serializer.test.ts +51 -1
  304. package/src/query/serialize/dialect/sqlite-serializer.ts +92 -9
  305. package/src/query/serialize/sql-serializer.ts +4 -4
  306. package/src/query/simple-query-interface.ts +5 -0
  307. package/src/query/unit-of-work/execute-unit-of-work.test.ts +25 -1
  308. package/src/query/unit-of-work/execute-unit-of-work.ts +25 -8
  309. package/src/query/unit-of-work/unit-of-work-coordinator.test.ts +12 -12
  310. package/src/query/unit-of-work/unit-of-work-types.test.ts +1 -1
  311. package/src/query/unit-of-work/unit-of-work.test.ts +168 -37
  312. package/src/query/unit-of-work/unit-of-work.ts +203 -18
  313. package/src/query/value-decoding.test.ts +13 -2
  314. package/src/query/value-decoding.ts +17 -4
  315. package/src/query/value-encoding.test.ts +85 -2
  316. package/src/query/value-encoding.ts +56 -6
  317. package/src/schema/create.test.ts +129 -42
  318. package/src/schema/create.ts +185 -47
  319. package/src/schema/generate-id.test.ts +2 -2
  320. package/src/schema/generate-id.ts +2 -2
  321. package/src/schema/serialize.test.ts +14 -2
  322. package/src/schema/type-conversion/create-sql-type-mapper.ts +7 -2
  323. package/src/schema/type-conversion/dialect/sqlite.ts +18 -0
  324. package/src/schema/type-conversion/type-mapping.test.ts +25 -1
  325. package/src/schema/validator.test.ts +197 -0
  326. package/src/schema/validator.ts +231 -0
  327. package/src/{adapters/drizzle/generate.test.ts → schema-output/drizzle.test.ts} +179 -129
  328. package/src/{adapters/drizzle/generate.ts → schema-output/drizzle.ts} +143 -93
  329. package/src/schema-output/prisma.test.ts +536 -0
  330. package/src/schema-output/prisma.ts +573 -0
  331. package/src/util/default-database-adapter.ts +106 -0
  332. package/src/with-database.ts +22 -3
  333. package/tsdown.config.ts +6 -4
  334. package/dist/adapters/drizzle/drizzle-adapter.d.ts +0 -20
  335. package/dist/adapters/drizzle/drizzle-adapter.d.ts.map +0 -1
  336. package/dist/adapters/drizzle/drizzle-adapter.js +0 -27
  337. package/dist/adapters/drizzle/drizzle-adapter.js.map +0 -1
  338. package/dist/adapters/drizzle/generate.d.ts +0 -30
  339. package/dist/adapters/drizzle/generate.d.ts.map +0 -1
  340. package/dist/adapters/drizzle/generate.js.map +0 -1
  341. package/dist/adapters/kysely/kysely-adapter.d.ts +0 -19
  342. package/dist/adapters/kysely/kysely-adapter.d.ts.map +0 -1
  343. package/dist/adapters/kysely/kysely-adapter.js +0 -17
  344. package/dist/adapters/kysely/kysely-adapter.js.map +0 -1
  345. package/dist/adapters/shared/table-name-mapper.d.ts +0 -12
  346. package/dist/adapters/shared/table-name-mapper.d.ts.map +0 -1
  347. package/dist/adapters/shared/table-name-mapper.js +0 -43
  348. package/dist/adapters/shared/table-name-mapper.js.map +0 -1
  349. package/dist/node_modules/.pnpm/rou3@0.7.10/node_modules/rou3/dist/index.js.map +0 -1
  350. package/dist/schema-generator/schema-generator.d.ts +0 -15
  351. package/dist/schema-generator/schema-generator.d.ts.map +0 -1
  352. package/src/adapters/drizzle/drizzle-adapter.ts +0 -39
  353. package/src/adapters/kysely/kysely-adapter.ts +0 -27
  354. package/src/adapters/shared/table-name-mapper.ts +0 -50
  355. package/src/schema-generator/schema-generator.ts +0 -12
@@ -0,0 +1,3 @@
1
+ export { InMemoryAdapter, type InMemoryUowConfig } from "./in-memory-adapter";
2
+ export type { InMemoryAdapterOptions, ResolvedInMemoryAdapterOptions } from "./options";
3
+ export { UniqueConstraintError, ForeignKeyConstraintError, NotFoundError } from "./errors";
@@ -0,0 +1,41 @@
1
+ import { afterEach, describe, expect, it, vi } from "vitest";
2
+ import { resolveInMemoryAdapterOptions } from "./options";
3
+
4
+ describe("resolveInMemoryAdapterOptions", () => {
5
+ afterEach(() => {
6
+ vi.useRealTimers();
7
+ });
8
+
9
+ it("creates deterministic ids when seeded", () => {
10
+ vi.useFakeTimers();
11
+ vi.setSystemTime(new Date("2020-01-01T00:00:00.000Z"));
12
+
13
+ const first = resolveInMemoryAdapterOptions({ idSeed: "seed" }).idGenerator;
14
+ const second = resolveInMemoryAdapterOptions({ idSeed: "seed" }).idGenerator;
15
+
16
+ const firstIds = [first(), first(), first()];
17
+ const secondIds = [second(), second(), second()];
18
+
19
+ expect(firstIds).toEqual(secondIds);
20
+ });
21
+
22
+ it("provides a monotonic internal id generator", () => {
23
+ const generator = resolveInMemoryAdapterOptions({ idSeed: "seed" }).internalIdGenerator;
24
+
25
+ expect(generator()).toBe(1n);
26
+ expect(generator()).toBe(2n);
27
+ });
28
+
29
+ it("defaults enforceConstraints and btreeOrder", () => {
30
+ const options = resolveInMemoryAdapterOptions({ idSeed: "seed" });
31
+
32
+ expect(options.enforceConstraints).toBe(true);
33
+ expect(options.btreeOrder).toBe(32);
34
+ });
35
+
36
+ it("defaults to a clock that returns dates", () => {
37
+ const options = resolveInMemoryAdapterOptions({ idSeed: "seed" });
38
+
39
+ expect(options.clock.now()).toBeInstanceOf(Date);
40
+ });
41
+ });
@@ -0,0 +1,87 @@
1
+ import { createId, init } from "@paralleldrive/cuid2";
2
+ import type { SqlNamingStrategy } from "../../naming/sql-naming";
3
+
4
+ export type InMemoryAdapterOptions = {
5
+ clock?: { now: () => Date };
6
+ idGenerator?: () => string;
7
+ idSeed?: string;
8
+ internalIdGenerator?: () => bigint;
9
+ enforceConstraints?: boolean;
10
+ btreeOrder?: number;
11
+ namingStrategy?: SqlNamingStrategy;
12
+ };
13
+
14
+ export type ResolvedInMemoryAdapterOptions = {
15
+ clock: { now: () => Date };
16
+ idGenerator: () => string;
17
+ idSeed: string;
18
+ internalIdGenerator: () => bigint;
19
+ internalIdGeneratorProvided: boolean;
20
+ enforceConstraints: boolean;
21
+ btreeOrder: number;
22
+ };
23
+
24
+ const defaultClock = {
25
+ now: () => new Date(),
26
+ };
27
+
28
+ const defaultBtreeOrder = 32;
29
+ const defaultEnforceConstraints = true;
30
+ const initialCountMax = 476782367;
31
+ const seededRandomStep = 1831565813;
32
+
33
+ const createCounter = (start: number) => {
34
+ let value = start;
35
+ return () => value++;
36
+ };
37
+
38
+ const hashSeed = (seed: string): number => {
39
+ let hash = 2166136261;
40
+ for (let i = 0; i < seed.length; i += 1) {
41
+ hash ^= seed.charCodeAt(i);
42
+ hash = Math.imul(hash, 16777619);
43
+ }
44
+ return hash >>> 0;
45
+ };
46
+
47
+ const createSeededRandom = (seed: string) => {
48
+ let state = hashSeed(seed);
49
+ return () => {
50
+ state |= 0;
51
+ state = (state + seededRandomStep) | 0;
52
+ let t = Math.imul(state ^ (state >>> 15), 1 | state);
53
+ t = (t + Math.imul(t ^ (t >>> 7), 61 | t)) ^ t;
54
+ return ((t ^ (t >>> 14)) >>> 0) / 4294967296;
55
+ };
56
+ };
57
+
58
+ const createSeededIdGenerator = (seed: string): (() => string) => {
59
+ const random = createSeededRandom(seed);
60
+ const counter = createCounter(Math.floor(random() * initialCountMax));
61
+ return init({ random, counter });
62
+ };
63
+
64
+ const createInternalIdGenerator = (): (() => bigint) => {
65
+ let current = 0n;
66
+ return () => {
67
+ current += 1n;
68
+ return current;
69
+ };
70
+ };
71
+
72
+ export const resolveInMemoryAdapterOptions = (
73
+ options: InMemoryAdapterOptions = {},
74
+ ): ResolvedInMemoryAdapterOptions => {
75
+ const idSeed = options.idSeed ?? createId();
76
+ const internalIdGeneratorProvided = options.internalIdGenerator !== undefined;
77
+
78
+ return {
79
+ clock: options.clock ?? defaultClock,
80
+ idGenerator: options.idGenerator ?? createSeededIdGenerator(idSeed),
81
+ idSeed,
82
+ internalIdGenerator: options.internalIdGenerator ?? createInternalIdGenerator(),
83
+ internalIdGeneratorProvided,
84
+ enforceConstraints: options.enforceConstraints ?? defaultEnforceConstraints,
85
+ btreeOrder: options.btreeOrder ?? defaultBtreeOrder,
86
+ };
87
+ };
@@ -0,0 +1,50 @@
1
+ import { describe, expect, it } from "vitest";
2
+ import { column, idColumn, referenceColumn, schema } from "../../schema/create";
3
+ import { ReferenceSubquery } from "../../query/value-encoding";
4
+ import { createInMemoryStore, ensureNamespaceStore } from "./store";
5
+ import { resolveReferenceSubqueries } from "./reference-resolution";
6
+
7
+ const testSchema = schema("test", (s) =>
8
+ s
9
+ .addTable("users", (t) => t.addColumn("id", idColumn()).addColumn("name", column("string")))
10
+ .addTable("posts", (t) => t.addColumn("id", idColumn()).addColumn("userId", referenceColumn()))
11
+ .addReference("author", {
12
+ type: "one",
13
+ from: { table: "posts", column: "userId" },
14
+ to: { table: "users", column: "id" },
15
+ }),
16
+ );
17
+
18
+ describe("in-memory reference resolution", () => {
19
+ it("resolves reference subqueries to internal IDs", () => {
20
+ const store = createInMemoryStore();
21
+ const namespaceStore = ensureNamespaceStore(store, "test", testSchema);
22
+ const usersStore = namespaceStore.tables.get("users");
23
+ if (!usersStore) {
24
+ throw new Error("Missing users table store.");
25
+ }
26
+
27
+ usersStore.rows.set(1n, { id: "user-1", name: "Ada", _internalId: 1n, _version: 0 });
28
+
29
+ const values = {
30
+ userId: new ReferenceSubquery(testSchema.tables.users, "user-1"),
31
+ };
32
+
33
+ const resolved = resolveReferenceSubqueries(namespaceStore, values);
34
+
35
+ expect(resolved).toEqual({ userId: 1n });
36
+ });
37
+
38
+ it("returns null when the referenced row is missing", () => {
39
+ const store = createInMemoryStore();
40
+ const namespaceStore = ensureNamespaceStore(store, "test", testSchema);
41
+
42
+ const values = {
43
+ userId: new ReferenceSubquery(testSchema.tables.users, "missing-user"),
44
+ };
45
+
46
+ const resolved = resolveReferenceSubqueries(namespaceStore, values);
47
+
48
+ expect(resolved).toEqual({ userId: null });
49
+ });
50
+ });
@@ -0,0 +1,67 @@
1
+ import type { AnyTable } from "../../schema/create";
2
+ import { ReferenceSubquery } from "../../query/value-encoding";
3
+ import type { InMemoryNamespaceStore, InMemoryTableStore } from "./store";
4
+ import type { NamingResolver } from "../../naming/sql-naming";
5
+
6
+ const getTableStore = (
7
+ namespaceStore: InMemoryNamespaceStore,
8
+ table: AnyTable,
9
+ resolver?: NamingResolver,
10
+ ): InMemoryTableStore => {
11
+ const tableName = resolver ? resolver.getTableName(table.name) : table.name;
12
+ const store = namespaceStore.tables.get(tableName);
13
+ if (!store) {
14
+ throw new Error(`Missing in-memory table store for "${tableName}".`);
15
+ }
16
+ return store;
17
+ };
18
+
19
+ const resolveExternalIdToInternalId = (
20
+ tableStore: InMemoryTableStore,
21
+ table: AnyTable,
22
+ externalId: string,
23
+ resolver?: NamingResolver,
24
+ ): bigint | undefined => {
25
+ const idColumnName = table.getIdColumn().name;
26
+ const physicalIdColumnName = resolver
27
+ ? resolver.getColumnName(table.name, idColumnName)
28
+ : idColumnName;
29
+ for (const [internalId, row] of tableStore.rows) {
30
+ if (row[physicalIdColumnName] === externalId) {
31
+ return internalId;
32
+ }
33
+ }
34
+ return undefined;
35
+ };
36
+
37
+ export const resolveReferenceSubquery = (
38
+ namespaceStore: InMemoryNamespaceStore,
39
+ reference: ReferenceSubquery,
40
+ resolver?: NamingResolver,
41
+ ): bigint | null => {
42
+ const tableStore = getTableStore(namespaceStore, reference.referencedTable, resolver);
43
+ const resolved = resolveExternalIdToInternalId(
44
+ tableStore,
45
+ reference.referencedTable,
46
+ reference.externalIdValue,
47
+ resolver,
48
+ );
49
+ return resolved ?? null;
50
+ };
51
+
52
+ export const resolveReferenceSubqueries = (
53
+ namespaceStore: InMemoryNamespaceStore,
54
+ values: Record<string, unknown>,
55
+ resolver?: NamingResolver,
56
+ ): Record<string, unknown> => {
57
+ const resolved: Record<string, unknown> = {};
58
+
59
+ for (const [key, value] of Object.entries(values)) {
60
+ resolved[key] =
61
+ value instanceof ReferenceSubquery
62
+ ? resolveReferenceSubquery(namespaceStore, value, resolver)
63
+ : value;
64
+ }
65
+
66
+ return resolved;
67
+ };
@@ -0,0 +1,123 @@
1
+ import { describe, expect, it } from "vitest";
2
+ import { UniqueConstraintError } from "./errors";
3
+ import { SortedArrayIndex } from "./sorted-array-index";
4
+
5
+ const compareValue = (left: unknown, right: unknown): number => {
6
+ if (left === right) {
7
+ return 0;
8
+ }
9
+
10
+ if (left === null || left === undefined) {
11
+ return -1;
12
+ }
13
+
14
+ if (right === null || right === undefined) {
15
+ return 1;
16
+ }
17
+
18
+ if (typeof left === "number" && typeof right === "number") {
19
+ return left < right ? -1 : 1;
20
+ }
21
+
22
+ if (typeof left === "string" && typeof right === "string") {
23
+ return left < right ? -1 : 1;
24
+ }
25
+
26
+ if (typeof left === "bigint" && typeof right === "bigint") {
27
+ return left < right ? -1 : 1;
28
+ }
29
+
30
+ throw new Error("Unsupported comparison in test helper.");
31
+ };
32
+
33
+ const valuesFromScan = (
34
+ index: SortedArrayIndex<string>,
35
+ options?: { direction?: "asc" | "desc" },
36
+ ) => index.scan(options).map((entry) => entry.value);
37
+
38
+ describe("sorted-array index", () => {
39
+ it("keeps entries sorted by key and supports duplicates", () => {
40
+ const index = new SortedArrayIndex<string>(compareValue);
41
+
42
+ index.insert([2], "b");
43
+ index.insert([1], "a");
44
+ index.insert([2], "c");
45
+ index.insert([3], "d");
46
+
47
+ expect(valuesFromScan(index)).toEqual(["a", "b", "c", "d"]);
48
+ });
49
+
50
+ it("removes entries by key and value when duplicates exist", () => {
51
+ const index = new SortedArrayIndex<string>(compareValue);
52
+
53
+ index.insert([1], "a");
54
+ index.insert([1], "b");
55
+ index.insert([2], "c");
56
+
57
+ expect(index.remove([1], "b")).toBe(true);
58
+ expect(valuesFromScan(index)).toEqual(["a", "c"]);
59
+ expect(index.remove([3])).toBe(false);
60
+ });
61
+
62
+ it("updates entries by removing and re-inserting", () => {
63
+ const index = new SortedArrayIndex<string>(compareValue);
64
+
65
+ index.insert([1], "a");
66
+ index.insert([2], "b");
67
+
68
+ index.update([1], [3], "a");
69
+
70
+ expect(valuesFromScan(index)).toEqual(["b", "a"]);
71
+ });
72
+
73
+ it("scans ranges with inclusive/exclusive bounds and direction", () => {
74
+ const index = new SortedArrayIndex<string>(compareValue);
75
+
76
+ index.insert([1], "a");
77
+ index.insert([2], "b");
78
+ index.insert([3], "c");
79
+ index.insert([4], "d");
80
+
81
+ const inclusive = index
82
+ .scan({ start: [2], startInclusive: true, end: [3], endInclusive: true })
83
+ .map((entry) => entry.value);
84
+ expect(inclusive).toEqual(["b", "c"]);
85
+
86
+ const exclusive = index
87
+ .scan({ start: [2], startInclusive: false, end: [4], endInclusive: false })
88
+ .map((entry) => entry.value);
89
+ expect(exclusive).toEqual(["c"]);
90
+
91
+ const descending = index
92
+ .scan({ start: [2], end: [4], direction: "desc", endInclusive: true })
93
+ .map((entry) => entry.value);
94
+ expect(descending).toEqual(["d", "c", "b"]);
95
+ });
96
+
97
+ it("respects scan limits", () => {
98
+ const index = new SortedArrayIndex<string>(compareValue);
99
+
100
+ index.insert([1], "a");
101
+ index.insert([2], "b");
102
+ index.insert([3], "c");
103
+
104
+ expect(index.scan({ limit: 2 }).map((entry) => entry.value)).toEqual(["a", "b"]);
105
+ });
106
+
107
+ it("throws on duplicate keys when unique", () => {
108
+ const index = new SortedArrayIndex<string>(compareValue, { unique: true });
109
+
110
+ index.insert([1], "a");
111
+
112
+ expect(() => index.insert([1], "b")).toThrow(UniqueConstraintError);
113
+ });
114
+
115
+ it("allows duplicate keys when unique enforcement is disabled", () => {
116
+ const index = new SortedArrayIndex<string>(compareValue, { unique: true });
117
+
118
+ index.insert([1], "a", { enforceUnique: false });
119
+ index.insert([1], "b", { enforceUnique: false });
120
+
121
+ expect(valuesFromScan(index)).toEqual(["a", "b"]);
122
+ });
123
+ });
@@ -0,0 +1,228 @@
1
+ export type IndexKey = readonly unknown[];
2
+
3
+ export type SortedArrayIndexEntry<T> = {
4
+ key: IndexKey;
5
+ value: T;
6
+ };
7
+
8
+ import { UniqueConstraintError } from "./errors";
9
+
10
+ export type CompareValue = (left: unknown, right: unknown) => number;
11
+
12
+ export type SortedArrayIndexOptions = {
13
+ unique?: boolean;
14
+ };
15
+
16
+ export type UniqueEnforcementOptions = {
17
+ enforceUnique?: boolean;
18
+ };
19
+
20
+ export type IndexScanOptions = {
21
+ start?: IndexKey;
22
+ startInclusive?: boolean;
23
+ end?: IndexKey;
24
+ endInclusive?: boolean;
25
+ direction?: "asc" | "desc";
26
+ limit?: number;
27
+ };
28
+
29
+ const compareKeys = (left: IndexKey, right: IndexKey, compareValue: CompareValue): number => {
30
+ const length = Math.min(left.length, right.length);
31
+ for (let i = 0; i < length; i += 1) {
32
+ const result = compareValue(left[i], right[i]);
33
+ if (result !== 0) {
34
+ return result;
35
+ }
36
+ }
37
+
38
+ return left.length - right.length;
39
+ };
40
+
41
+ export class SortedArrayIndex<T> {
42
+ #entries: SortedArrayIndexEntry<T>[] = [];
43
+ #compareValue: CompareValue;
44
+ #unique: boolean;
45
+
46
+ constructor(compareValue: CompareValue, options: SortedArrayIndexOptions = {}) {
47
+ this.#compareValue = compareValue;
48
+ this.#unique = options.unique ?? false;
49
+ }
50
+
51
+ get size(): number {
52
+ return this.#entries.length;
53
+ }
54
+
55
+ entries(): SortedArrayIndexEntry<T>[] {
56
+ return this.#entries.slice();
57
+ }
58
+
59
+ insert(key: IndexKey, value: T, options: UniqueEnforcementOptions = {}): void {
60
+ const enforceUnique = options.enforceUnique ?? true;
61
+ if (this.#unique && enforceUnique) {
62
+ this.#assertUnique(key);
63
+ }
64
+ const entry = { key, value };
65
+ const index = this.#upperBound(key);
66
+ this.#entries.splice(index, 0, entry);
67
+ }
68
+
69
+ remove(key: IndexKey, value?: T): boolean {
70
+ const start = this.#lowerBound(key);
71
+ const end = this.#upperBound(key);
72
+ if (start === end) {
73
+ return false;
74
+ }
75
+
76
+ if (value === undefined) {
77
+ this.#entries.splice(start, 1);
78
+ return true;
79
+ }
80
+
81
+ for (let i = start; i < end; i += 1) {
82
+ if (this.#entries[i]?.value === value) {
83
+ this.#entries.splice(i, 1);
84
+ return true;
85
+ }
86
+ }
87
+
88
+ return false;
89
+ }
90
+
91
+ update(
92
+ oldKey: IndexKey,
93
+ newKey: IndexKey,
94
+ value: T,
95
+ options: UniqueEnforcementOptions = {},
96
+ ): void {
97
+ const enforceUnique = options.enforceUnique ?? true;
98
+ if (this.#unique && enforceUnique) {
99
+ const range = this.#rangeForKey(newKey);
100
+ if (range.start !== range.end) {
101
+ const sameKey = compareKeys(oldKey, newKey, this.#compareValue) === 0;
102
+ const singleEntry = range.end - range.start === 1;
103
+ const entry = this.#entries[range.start];
104
+ const isSameEntry = sameKey && singleEntry && entry?.value === value;
105
+ if (!isSameEntry) {
106
+ throw new UniqueConstraintError(this.#uniqueErrorMessage(newKey));
107
+ }
108
+ if (sameKey) {
109
+ return;
110
+ }
111
+ }
112
+ }
113
+ const removed = this.remove(oldKey, value);
114
+ if (!removed) {
115
+ throw new Error("SortedArrayIndex update failed to locate existing entry.");
116
+ }
117
+ this.insert(newKey, value, { enforceUnique });
118
+ }
119
+
120
+ scan(options: IndexScanOptions = {}): SortedArrayIndexEntry<T>[] {
121
+ const direction = options.direction ?? "asc";
122
+ const limit = options.limit ?? Number.POSITIVE_INFINITY;
123
+
124
+ const start = options.start
125
+ ? options.startInclusive === false
126
+ ? this.#upperBound(options.start)
127
+ : this.#lowerBound(options.start)
128
+ : 0;
129
+
130
+ const end = options.end
131
+ ? options.endInclusive === true
132
+ ? this.#upperBound(options.end)
133
+ : this.#lowerBound(options.end)
134
+ : this.#entries.length;
135
+
136
+ if (start >= end || limit <= 0) {
137
+ return [];
138
+ }
139
+
140
+ const results: SortedArrayIndexEntry<T>[] = [];
141
+
142
+ if (direction === "desc") {
143
+ for (let i = end - 1; i >= start && results.length < limit; i -= 1) {
144
+ results.push(this.#entries[i]);
145
+ }
146
+ return results;
147
+ }
148
+
149
+ for (let i = start; i < end && results.length < limit; i += 1) {
150
+ results.push(this.#entries[i]);
151
+ }
152
+
153
+ return results;
154
+ }
155
+
156
+ #compareEntryWithKey(entry: SortedArrayIndexEntry<T>, key: IndexKey): number {
157
+ return compareKeys(entry.key, key, this.#compareValue);
158
+ }
159
+
160
+ #rangeForKey(key: IndexKey): { start: number; end: number } {
161
+ return {
162
+ start: this.#lowerBound(key),
163
+ end: this.#upperBound(key),
164
+ };
165
+ }
166
+
167
+ #assertUnique(key: IndexKey): void {
168
+ const range = this.#rangeForKey(key);
169
+ if (range.start !== range.end) {
170
+ throw new UniqueConstraintError(this.#uniqueErrorMessage(key));
171
+ }
172
+ }
173
+
174
+ #uniqueErrorMessage(key: IndexKey): string {
175
+ return `Unique constraint violation for key ${this.#formatKey(key)}.`;
176
+ }
177
+
178
+ #formatKey(key: IndexKey): string {
179
+ const parts = key.map((value) => this.#formatKeyValue(value));
180
+ return `[${parts.join(", ")}]`;
181
+ }
182
+
183
+ #formatKeyValue(value: unknown): string {
184
+ if (value instanceof Date) {
185
+ return value.toISOString();
186
+ }
187
+ if (typeof value === "bigint") {
188
+ return `${value}n`;
189
+ }
190
+ return String(value);
191
+ }
192
+
193
+ #lowerBound(key: IndexKey): number {
194
+ let low = 0;
195
+ let high = this.#entries.length;
196
+
197
+ while (low < high) {
198
+ const mid = (low + high) >>> 1;
199
+ const entry = this.#entries[mid];
200
+ const comparison = entry ? this.#compareEntryWithKey(entry, key) : 0;
201
+ if (comparison < 0) {
202
+ low = mid + 1;
203
+ } else {
204
+ high = mid;
205
+ }
206
+ }
207
+
208
+ return low;
209
+ }
210
+
211
+ #upperBound(key: IndexKey): number {
212
+ let low = 0;
213
+ let high = this.#entries.length;
214
+
215
+ while (low < high) {
216
+ const mid = (low + high) >>> 1;
217
+ const entry = this.#entries[mid];
218
+ const comparison = entry ? this.#compareEntryWithKey(entry, key) : 0;
219
+ if (comparison <= 0) {
220
+ low = mid + 1;
221
+ } else {
222
+ high = mid;
223
+ }
224
+ }
225
+
226
+ return low;
227
+ }
228
+ }
@@ -0,0 +1,68 @@
1
+ import { describe, expect, it } from "vitest";
2
+ import { column, idColumn, schema } from "../../schema/create";
3
+ import { createInMemoryStore, ensureNamespaceStore } from "./store";
4
+
5
+ const testSchema = schema("test", (s) =>
6
+ s
7
+ .addTable("users", (t) =>
8
+ t
9
+ .addColumn("id", idColumn())
10
+ .addColumn("name", column("string"))
11
+ .createIndex("name_idx", ["name"]),
12
+ )
13
+ .addTable("posts", (t) =>
14
+ t
15
+ .addColumn("id", idColumn())
16
+ .addColumn("userId", column("string"))
17
+ .createIndex("user_idx", ["userId"], { unique: true }),
18
+ ),
19
+ );
20
+
21
+ describe("in-memory store", () => {
22
+ it("creates namespace tables with empty row maps and counters", () => {
23
+ const store = createInMemoryStore();
24
+ const namespaceStore = ensureNamespaceStore(store, "test", testSchema);
25
+
26
+ expect(namespaceStore.tables.size).toBe(2);
27
+ expect(namespaceStore.tables.has("users")).toBe(true);
28
+ expect(namespaceStore.tables.has("posts")).toBe(true);
29
+
30
+ const usersTable = namespaceStore.tables.get("users");
31
+ const postsTable = namespaceStore.tables.get("posts");
32
+
33
+ expect(usersTable?.rows.size).toBe(0);
34
+ expect(postsTable?.rows.size).toBe(0);
35
+ expect(usersTable?.nextInternalId).toBe(1n);
36
+ expect(postsTable?.nextInternalId).toBe(1n);
37
+ expect(usersTable?.indexes.size).toBe(2);
38
+ expect(postsTable?.indexes.size).toBe(2);
39
+ expect(usersTable?.indexes.has("_primary")).toBe(true);
40
+ expect(postsTable?.indexes.has("_primary")).toBe(true);
41
+
42
+ const usersPrimary = usersTable?.indexes.get("_primary");
43
+ const usersNameIdx = usersTable?.indexes.get("name_idx");
44
+ const postsUserIdx = postsTable?.indexes.get("user_idx");
45
+
46
+ expect(usersPrimary?.definition.columnNames).toEqual(["id"]);
47
+ expect(usersPrimary?.definition.unique).toBe(true);
48
+ expect(usersNameIdx?.definition.columnNames).toEqual(["name"]);
49
+ expect(usersNameIdx?.definition.unique).toBe(false);
50
+ expect(postsUserIdx?.definition.columnNames).toEqual(["userId"]);
51
+ expect(postsUserIdx?.definition.unique).toBe(true);
52
+ });
53
+
54
+ it("reuses existing namespace stores", () => {
55
+ const store = createInMemoryStore();
56
+ const first = ensureNamespaceStore(store, "test", testSchema);
57
+ const usersTable = first.tables.get("users");
58
+ if (usersTable) {
59
+ usersTable.nextInternalId = 42n;
60
+ }
61
+
62
+ const second = ensureNamespaceStore(store, "test", testSchema);
63
+ const usersTableAgain = second.tables.get("users");
64
+
65
+ expect(second).toBe(first);
66
+ expect(usersTableAgain?.nextInternalId).toBe(42n);
67
+ });
68
+ });