@effect-app/infra 4.0.0-beta.22 → 4.0.0-beta.220

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 (310) hide show
  1. package/CHANGELOG.md +1640 -0
  2. package/_check.sh +1 -1
  3. package/dist/CUPS.d.ts +7 -7
  4. package/dist/CUPS.d.ts.map +1 -1
  5. package/dist/CUPS.js +10 -12
  6. package/dist/Emailer/Sendgrid.d.ts +14 -14
  7. package/dist/Emailer/Sendgrid.d.ts.map +1 -1
  8. package/dist/Emailer/Sendgrid.js +16 -15
  9. package/dist/Emailer/fake.d.ts +1 -1
  10. package/dist/Emailer/service.d.ts +10 -4
  11. package/dist/Emailer/service.d.ts.map +1 -1
  12. package/dist/Emailer/service.js +3 -3
  13. package/dist/Emailer.d.ts +1 -1
  14. package/dist/MainFiberSet.d.ts +9 -9
  15. package/dist/MainFiberSet.d.ts.map +1 -1
  16. package/dist/MainFiberSet.js +3 -3
  17. package/dist/Model/Repository/Registry.d.ts +20 -0
  18. package/dist/Model/Repository/Registry.d.ts.map +1 -0
  19. package/dist/Model/Repository/Registry.js +17 -0
  20. package/dist/Model/Repository/ext.d.ts +33 -15
  21. package/dist/Model/Repository/ext.d.ts.map +1 -1
  22. package/dist/Model/Repository/ext.js +54 -2
  23. package/dist/Model/Repository/internal/internal.d.ts +6 -6
  24. package/dist/Model/Repository/internal/internal.d.ts.map +1 -1
  25. package/dist/Model/Repository/internal/internal.js +103 -51
  26. package/dist/Model/Repository/legacy.d.ts +1 -1
  27. package/dist/Model/Repository/makeRepo.d.ts +7 -6
  28. package/dist/Model/Repository/makeRepo.d.ts.map +1 -1
  29. package/dist/Model/Repository/makeRepo.js +5 -1
  30. package/dist/Model/Repository/service.d.ts +28 -23
  31. package/dist/Model/Repository/service.d.ts.map +1 -1
  32. package/dist/Model/Repository/validation.d.ts +46 -17
  33. package/dist/Model/Repository/validation.d.ts.map +1 -1
  34. package/dist/Model/Repository/validation.js +5 -5
  35. package/dist/Model/Repository.d.ts +2 -1
  36. package/dist/Model/Repository.d.ts.map +1 -1
  37. package/dist/Model/Repository.js +2 -1
  38. package/dist/Model/dsl.d.ts +4 -4
  39. package/dist/Model/dsl.d.ts.map +1 -1
  40. package/dist/Model/filter/filterApi.d.ts +5 -5
  41. package/dist/Model/filter/filterApi.d.ts.map +1 -1
  42. package/dist/Model/filter/types/errors.d.ts +1 -1
  43. package/dist/Model/filter/types/fields.d.ts +1 -1
  44. package/dist/Model/filter/types/path/common.d.ts +1 -1
  45. package/dist/Model/filter/types/path/eager.d.ts +1 -1
  46. package/dist/Model/filter/types/path/eager.d.ts.map +1 -1
  47. package/dist/Model/filter/types/path/eager.js +1 -1
  48. package/dist/Model/filter/types/path/index.d.ts +1 -1
  49. package/dist/Model/filter/types/utils.d.ts +1 -1
  50. package/dist/Model/filter/types/validator.d.ts +1 -1
  51. package/dist/Model/filter/types.d.ts +1 -1
  52. package/dist/Model/query/dsl.d.ts +139 -16
  53. package/dist/Model/query/dsl.d.ts.map +1 -1
  54. package/dist/Model/query/dsl.js +187 -1
  55. package/dist/Model/query/new-kid-interpreter.d.ts +76 -7
  56. package/dist/Model/query/new-kid-interpreter.d.ts.map +1 -1
  57. package/dist/Model/query/new-kid-interpreter.js +122 -6
  58. package/dist/Model/query.d.ts +1 -1
  59. package/dist/Model.d.ts +2 -1
  60. package/dist/Model.d.ts.map +1 -1
  61. package/dist/Model.js +2 -1
  62. package/dist/QueueMaker/SQLQueue.d.ts +5 -7
  63. package/dist/QueueMaker/SQLQueue.d.ts.map +1 -1
  64. package/dist/QueueMaker/SQLQueue.js +130 -116
  65. package/dist/QueueMaker/errors.d.ts +2 -2
  66. package/dist/QueueMaker/errors.d.ts.map +1 -1
  67. package/dist/QueueMaker/memQueue.d.ts +7 -4
  68. package/dist/QueueMaker/memQueue.d.ts.map +1 -1
  69. package/dist/QueueMaker/memQueue.js +75 -63
  70. package/dist/QueueMaker/sbqueue.d.ts +6 -3
  71. package/dist/QueueMaker/sbqueue.d.ts.map +1 -1
  72. package/dist/QueueMaker/sbqueue.js +52 -53
  73. package/dist/QueueMaker/service.d.ts +1 -1
  74. package/dist/RequestContext.d.ts +74 -35
  75. package/dist/RequestContext.d.ts.map +1 -1
  76. package/dist/RequestContext.js +13 -14
  77. package/dist/RequestFiberSet.d.ts +7 -7
  78. package/dist/RequestFiberSet.d.ts.map +1 -1
  79. package/dist/RequestFiberSet.js +3 -3
  80. package/dist/Store/ContextMapContainer.d.ts +19 -3
  81. package/dist/Store/ContextMapContainer.d.ts.map +1 -1
  82. package/dist/Store/ContextMapContainer.js +13 -3
  83. package/dist/Store/Cosmos/query.d.ts +5 -1
  84. package/dist/Store/Cosmos/query.d.ts.map +1 -1
  85. package/dist/Store/Cosmos/query.js +113 -34
  86. package/dist/Store/Cosmos.d.ts +1 -1
  87. package/dist/Store/Cosmos.d.ts.map +1 -1
  88. package/dist/Store/Cosmos.js +335 -243
  89. package/dist/Store/Disk.d.ts +2 -2
  90. package/dist/Store/Disk.d.ts.map +1 -1
  91. package/dist/Store/Disk.js +72 -35
  92. package/dist/Store/Memory.d.ts +6 -4
  93. package/dist/Store/Memory.d.ts.map +1 -1
  94. package/dist/Store/Memory.js +242 -58
  95. package/dist/Store/SQL/Pg.d.ts +4 -0
  96. package/dist/Store/SQL/Pg.d.ts.map +1 -0
  97. package/dist/Store/SQL/Pg.js +231 -0
  98. package/dist/Store/SQL/query.d.ts +42 -0
  99. package/dist/Store/SQL/query.d.ts.map +1 -0
  100. package/dist/Store/SQL/query.js +479 -0
  101. package/dist/Store/SQL.d.ts +20 -0
  102. package/dist/Store/SQL.d.ts.map +1 -0
  103. package/dist/Store/SQL.js +446 -0
  104. package/dist/Store/codeFilter.d.ts +1 -1
  105. package/dist/Store/codeFilter.d.ts.map +1 -1
  106. package/dist/Store/codeFilter.js +4 -2
  107. package/dist/Store/index.d.ts +5 -2
  108. package/dist/Store/index.d.ts.map +1 -1
  109. package/dist/Store/index.js +15 -3
  110. package/dist/Store/service.d.ts +22 -8
  111. package/dist/Store/service.d.ts.map +1 -1
  112. package/dist/Store/service.js +24 -6
  113. package/dist/Store/utils.d.ts +1 -1
  114. package/dist/Store/utils.d.ts.map +1 -1
  115. package/dist/Store/utils.js +3 -4
  116. package/dist/Store.d.ts +1 -1
  117. package/dist/adapters/SQL/Model.d.ts +31 -42
  118. package/dist/adapters/SQL/Model.d.ts.map +1 -1
  119. package/dist/adapters/SQL/Model.js +29 -38
  120. package/dist/adapters/SQL.d.ts +1 -1
  121. package/dist/adapters/ServiceBus.d.ts +11 -11
  122. package/dist/adapters/ServiceBus.d.ts.map +1 -1
  123. package/dist/adapters/ServiceBus.js +25 -21
  124. package/dist/adapters/cosmos-client.d.ts +3 -3
  125. package/dist/adapters/cosmos-client.d.ts.map +1 -1
  126. package/dist/adapters/cosmos-client.js +3 -3
  127. package/dist/adapters/index.d.ts +8 -2
  128. package/dist/adapters/index.d.ts.map +1 -1
  129. package/dist/adapters/index.js +8 -2
  130. package/dist/adapters/logger.d.ts +1 -1
  131. package/dist/adapters/logger.d.ts.map +1 -1
  132. package/dist/adapters/memQueue.d.ts +3 -3
  133. package/dist/adapters/memQueue.d.ts.map +1 -1
  134. package/dist/adapters/memQueue.js +3 -3
  135. package/dist/adapters/mongo-client.d.ts +3 -3
  136. package/dist/adapters/mongo-client.d.ts.map +1 -1
  137. package/dist/adapters/mongo-client.js +3 -3
  138. package/dist/adapters/redis-client.d.ts +3 -3
  139. package/dist/adapters/redis-client.d.ts.map +1 -1
  140. package/dist/adapters/redis-client.js +3 -3
  141. package/dist/api/ContextProvider.d.ts +8 -8
  142. package/dist/api/ContextProvider.d.ts.map +1 -1
  143. package/dist/api/ContextProvider.js +6 -6
  144. package/dist/api/codec.d.ts +1 -1
  145. package/dist/api/internal/RequestContextMiddleware.d.ts +2 -2
  146. package/dist/api/internal/RequestContextMiddleware.d.ts.map +1 -1
  147. package/dist/api/internal/RequestContextMiddleware.js +9 -6
  148. package/dist/api/internal/auth.d.ts +44 -6
  149. package/dist/api/internal/auth.d.ts.map +1 -1
  150. package/dist/api/internal/auth.js +160 -29
  151. package/dist/api/internal/events.d.ts +3 -3
  152. package/dist/api/internal/events.d.ts.map +1 -1
  153. package/dist/api/internal/events.js +10 -8
  154. package/dist/api/internal/health.d.ts +1 -1
  155. package/dist/api/layerUtils.d.ts +6 -6
  156. package/dist/api/layerUtils.d.ts.map +1 -1
  157. package/dist/api/layerUtils.js +5 -5
  158. package/dist/api/middlewares.d.ts +1 -1
  159. package/dist/api/reportError.d.ts +1 -1
  160. package/dist/api/routing/middleware/RouterMiddleware.d.ts +4 -4
  161. package/dist/api/routing/middleware/RouterMiddleware.d.ts.map +1 -1
  162. package/dist/api/routing/middleware/middleware.d.ts +39 -3
  163. package/dist/api/routing/middleware/middleware.d.ts.map +1 -1
  164. package/dist/api/routing/middleware/middleware.js +48 -16
  165. package/dist/api/routing/middleware.d.ts +1 -2
  166. package/dist/api/routing/middleware.d.ts.map +1 -1
  167. package/dist/api/routing/middleware.js +1 -2
  168. package/dist/api/routing/schema/jwt.d.ts +1 -1
  169. package/dist/api/routing/schema/jwt.d.ts.map +1 -1
  170. package/dist/api/routing/tsort.d.ts +1 -1
  171. package/dist/api/routing/tsort.d.ts.map +1 -1
  172. package/dist/api/routing/utils.d.ts +3 -3
  173. package/dist/api/routing/utils.d.ts.map +1 -1
  174. package/dist/api/routing.d.ts +80 -37
  175. package/dist/api/routing.d.ts.map +1 -1
  176. package/dist/api/routing.js +109 -41
  177. package/dist/api/setupRequest.d.ts +8 -5
  178. package/dist/api/setupRequest.d.ts.map +1 -1
  179. package/dist/api/setupRequest.js +12 -7
  180. package/dist/api/util.d.ts +1 -1
  181. package/dist/arbs.d.ts +1 -1
  182. package/dist/arbs.d.ts.map +1 -1
  183. package/dist/arbs.js +5 -3
  184. package/dist/errorReporter.d.ts +4 -4
  185. package/dist/errorReporter.d.ts.map +1 -1
  186. package/dist/errorReporter.js +20 -25
  187. package/dist/errors.d.ts +1 -1
  188. package/dist/fileUtil.d.ts +1 -1
  189. package/dist/fileUtil.d.ts.map +1 -1
  190. package/dist/index.d.ts +1 -1
  191. package/dist/logger/jsonLogger.d.ts +1 -1
  192. package/dist/logger/logFmtLogger.d.ts +1 -1
  193. package/dist/logger/shared.d.ts +1 -1
  194. package/dist/logger/shared.js +2 -2
  195. package/dist/logger.d.ts +1 -1
  196. package/dist/logger.d.ts.map +1 -1
  197. package/dist/otel.d.ts +75 -0
  198. package/dist/otel.d.ts.map +1 -0
  199. package/dist/otel.js +65 -0
  200. package/dist/rateLimit.d.ts +9 -3
  201. package/dist/rateLimit.d.ts.map +1 -1
  202. package/dist/rateLimit.js +5 -11
  203. package/dist/test.d.ts +2 -2
  204. package/dist/test.d.ts.map +1 -1
  205. package/dist/test.js +1 -1
  206. package/dist/vitest.d.ts +1 -1
  207. package/examples/query.ts +42 -38
  208. package/package.json +46 -37
  209. package/src/CUPS.ts +9 -11
  210. package/src/Emailer/Sendgrid.ts +17 -14
  211. package/src/Emailer/service.ts +9 -3
  212. package/src/MainFiberSet.ts +5 -6
  213. package/src/Model/Repository/Registry.ts +33 -0
  214. package/src/Model/Repository/ext.ts +96 -10
  215. package/src/Model/Repository/internal/internal.ts +218 -149
  216. package/src/Model/Repository/makeRepo.ts +12 -10
  217. package/src/Model/Repository/service.ts +31 -22
  218. package/src/Model/Repository/validation.ts +4 -4
  219. package/src/Model/Repository.ts +1 -0
  220. package/src/Model/dsl.ts +3 -3
  221. package/src/Model/filter/types/path/eager.ts +1 -2
  222. package/src/Model/query/dsl.ts +348 -18
  223. package/src/Model/query/new-kid-interpreter.ts +206 -6
  224. package/src/Model.ts +1 -0
  225. package/src/QueueMaker/SQLQueue.ts +144 -152
  226. package/src/QueueMaker/memQueue.ts +104 -103
  227. package/src/QueueMaker/sbqueue.ts +70 -86
  228. package/src/RequestContext.ts +14 -16
  229. package/src/RequestFiberSet.ts +2 -2
  230. package/src/Store/ContextMapContainer.ts +41 -2
  231. package/src/Store/Cosmos/query.ts +140 -43
  232. package/src/Store/Cosmos.ts +482 -349
  233. package/src/Store/Disk.ts +102 -65
  234. package/src/Store/Memory.ts +275 -87
  235. package/src/Store/SQL/Pg.ts +361 -0
  236. package/src/Store/SQL/query.ts +539 -0
  237. package/src/Store/SQL.ts +731 -0
  238. package/src/Store/codeFilter.ts +3 -1
  239. package/src/Store/index.ts +17 -2
  240. package/src/Store/service.ts +41 -10
  241. package/src/Store/utils.ts +23 -22
  242. package/src/adapters/SQL/Model.ts +41 -40
  243. package/src/adapters/ServiceBus.ts +125 -121
  244. package/src/adapters/cosmos-client.ts +2 -2
  245. package/src/adapters/index.ts +7 -0
  246. package/src/adapters/memQueue.ts +2 -2
  247. package/src/adapters/mongo-client.ts +2 -2
  248. package/src/adapters/redis-client.ts +2 -2
  249. package/src/api/ContextProvider.ts +12 -13
  250. package/src/api/internal/RequestContextMiddleware.ts +15 -5
  251. package/src/api/internal/auth.ts +246 -44
  252. package/src/api/internal/events.ts +13 -9
  253. package/src/api/layerUtils.ts +8 -8
  254. package/src/api/routing/middleware/RouterMiddleware.ts +4 -4
  255. package/src/api/routing/middleware/middleware.ts +55 -14
  256. package/src/api/routing/middleware.ts +0 -2
  257. package/src/api/routing.ts +296 -131
  258. package/src/api/setupRequest.ts +28 -8
  259. package/src/arbs.ts +4 -2
  260. package/src/errorReporter.ts +62 -74
  261. package/src/logger/shared.ts +1 -1
  262. package/src/otel.ts +152 -0
  263. package/src/rateLimit.ts +30 -22
  264. package/src/test.ts +1 -1
  265. package/test/auth.test.ts +101 -0
  266. package/test/contextProvider.test.ts +11 -11
  267. package/test/controller.test.ts +21 -30
  268. package/test/dist/auth.test.d.ts.map +1 -0
  269. package/test/dist/contextProvider.test.d.ts.map +1 -1
  270. package/test/dist/controller.test.d.ts.map +1 -1
  271. package/test/dist/date-query.test.d.ts.map +1 -0
  272. package/test/dist/fixtures.d.ts +26 -12
  273. package/test/dist/fixtures.d.ts.map +1 -1
  274. package/test/dist/fixtures.js +12 -10
  275. package/test/dist/query.test.d.ts.map +1 -1
  276. package/test/dist/rawQuery.test.d.ts.map +1 -1
  277. package/test/dist/repository-ext.test.d.ts.map +1 -0
  278. package/test/dist/requires.test.d.ts.map +1 -1
  279. package/test/dist/router-generator.test.d.ts.map +1 -0
  280. package/test/dist/routing-interruptibility.test.d.ts.map +1 -0
  281. package/test/dist/rpc-e2e-invalidation.test.d.ts.map +1 -0
  282. package/test/dist/rpc-multi-middleware.test.d.ts.map +1 -1
  283. package/test/dist/rpc-stream-fullstack.test.d.ts.map +1 -0
  284. package/test/dist/sql-store.test.d.ts.map +1 -0
  285. package/test/fixtures.ts +11 -9
  286. package/test/query.test.ts +813 -38
  287. package/test/rawQuery.test.ts +301 -20
  288. package/test/repository-ext.test.ts +60 -0
  289. package/test/requires.test.ts +6 -6
  290. package/test/router-generator.test.ts +183 -0
  291. package/test/routing-interruptibility.test.ts +63 -0
  292. package/test/rpc-e2e-invalidation.test.ts +251 -0
  293. package/test/rpc-multi-middleware.test.ts +78 -9
  294. package/test/rpc-stream-fullstack.test.ts +300 -0
  295. package/test/sql-store.test.ts +1592 -0
  296. package/test/validateSample.test.ts +15 -12
  297. package/tsconfig.examples.json +1 -1
  298. package/tsconfig.json +0 -1
  299. package/tsconfig.json.bak +2 -2
  300. package/tsconfig.src.json +35 -35
  301. package/tsconfig.test.json +2 -2
  302. package/dist/Operations.d.ts +0 -55
  303. package/dist/Operations.d.ts.map +0 -1
  304. package/dist/Operations.js +0 -102
  305. package/dist/OperationsRepo.d.ts +0 -41
  306. package/dist/OperationsRepo.d.ts.map +0 -1
  307. package/dist/OperationsRepo.js +0 -14
  308. package/eslint.config.mjs +0 -24
  309. package/src/Operations.ts +0 -235
  310. package/src/OperationsRepo.ts +0 -16
@@ -0,0 +1,1592 @@
1
+ /* eslint-disable @typescript-eslint/no-explicit-any */
2
+ import type Sqlite from "better-sqlite3"
3
+ import BetterSqlite from "better-sqlite3"
4
+ import { describe, expect, it } from "vitest"
5
+ import { parseRow } from "../src/Store/SQL.js"
6
+ import { buildWhereSQLQuery, pgDialect, sqliteDialect } from "../src/Store/SQL/query.js"
7
+ import { makeETag } from "../src/Store/utils.js"
8
+
9
+ const query = (db: Sqlite.Database, sql: string, params: unknown[] = []) =>
10
+ db.prepare(sql).all(...params as any[]) as any[]
11
+
12
+ // --- Query builder unit tests ---
13
+
14
+ describe("SQL query builder (SQLite dialect)", () => {
15
+ it("where eq string", () => {
16
+ const result = buildWhereSQLQuery(
17
+ sqliteDialect,
18
+ "id",
19
+ [{ t: "where", path: "name", op: "eq", value: "John" }],
20
+ "users",
21
+ {}
22
+ )
23
+ expect(result.sql).toContain("json_extract(data, '$.name') = ?")
24
+ expect(result.params).toContain("John")
25
+ })
26
+
27
+ it("where eq number", () => {
28
+ const result = buildWhereSQLQuery(
29
+ sqliteDialect,
30
+ "id",
31
+ [{ t: "where", path: "age", op: "eq", value: 25 as any }],
32
+ "users",
33
+ {}
34
+ )
35
+ expect(result.sql).toContain("json_extract(data, '$.age') = ?")
36
+ expect(result.params).toContain(25)
37
+ })
38
+
39
+ it("where gt", () => {
40
+ const result = buildWhereSQLQuery(
41
+ sqliteDialect,
42
+ "id",
43
+ [{ t: "where", path: "age", op: "gt", value: 18 as any }],
44
+ "users",
45
+ {}
46
+ )
47
+ expect(result.sql).toContain("json_extract(data, '$.age') > ?")
48
+ expect(result.params).toContain(18)
49
+ })
50
+
51
+ it("where or", () => {
52
+ const result = buildWhereSQLQuery(
53
+ sqliteDialect,
54
+ "id",
55
+ [
56
+ { t: "where", path: "name", op: "eq", value: "Alice" },
57
+ { t: "or", path: "name", op: "eq", value: "Bob" }
58
+ ],
59
+ "users",
60
+ {}
61
+ )
62
+ expect(result.sql).toContain("= ?")
63
+ expect(result.sql).toContain("OR")
64
+ expect(result.params).toEqual(expect.arrayContaining(["Alice", "Bob"]))
65
+ })
66
+
67
+ it("where and", () => {
68
+ const result = buildWhereSQLQuery(
69
+ sqliteDialect,
70
+ "id",
71
+ [
72
+ { t: "where", path: "name", op: "eq", value: "Alice" },
73
+ { t: "and", path: "age", op: "gt", value: 18 as any }
74
+ ],
75
+ "users",
76
+ {}
77
+ )
78
+ expect(result.sql).toContain("AND")
79
+ expect(result.params).toEqual(expect.arrayContaining(["Alice", 18]))
80
+ })
81
+
82
+ it("where in", () => {
83
+ const result = buildWhereSQLQuery(
84
+ sqliteDialect,
85
+ "id",
86
+ [{ t: "where", path: "id", op: "in", value: ["a", "b", "c"] as any }],
87
+ "users",
88
+ {}
89
+ )
90
+ expect(result.sql).toContain("id IN (?, ?, ?)")
91
+ expect(result.params).toEqual(expect.arrayContaining(["a", "b", "c"]))
92
+ })
93
+
94
+ it("where null", () => {
95
+ const result = buildWhereSQLQuery(
96
+ sqliteDialect,
97
+ "id",
98
+ [{ t: "where", path: "status", op: "eq", value: null as any }],
99
+ "users",
100
+ {}
101
+ )
102
+ expect(result.sql).toContain("IS NULL")
103
+ })
104
+
105
+ it("where neq null", () => {
106
+ const result = buildWhereSQLQuery(
107
+ sqliteDialect,
108
+ "id",
109
+ [{ t: "where", path: "status", op: "neq", value: null as any }],
110
+ "users",
111
+ {}
112
+ )
113
+ expect(result.sql).toContain("IS NOT NULL")
114
+ })
115
+
116
+ it("where contains", () => {
117
+ const result = buildWhereSQLQuery(
118
+ sqliteDialect,
119
+ "id",
120
+ [{ t: "where", path: "name", op: "contains", value: "oh" }],
121
+ "users",
122
+ {}
123
+ )
124
+ expect(result.sql).toContain("LIKE")
125
+ expect(result.sql).toContain("LOWER")
126
+ expect(result.params).toContain("%oh%")
127
+ })
128
+
129
+ it("where startsWith", () => {
130
+ const result = buildWhereSQLQuery(
131
+ sqliteDialect,
132
+ "id",
133
+ [{ t: "where", path: "name", op: "startsWith", value: "Jo" }],
134
+ "users",
135
+ {}
136
+ )
137
+ expect(result.sql).toContain("LIKE")
138
+ expect(result.params).toContain("Jo%")
139
+ })
140
+
141
+ it("where endsWith", () => {
142
+ const result = buildWhereSQLQuery(
143
+ sqliteDialect,
144
+ "id",
145
+ [{ t: "where", path: "name", op: "endsWith", value: "hn" }],
146
+ "users",
147
+ {}
148
+ )
149
+ expect(result.sql).toContain("LIKE")
150
+ expect(result.params).toContain("%hn")
151
+ })
152
+
153
+ it("where includes (array contains)", () => {
154
+ const result = buildWhereSQLQuery(
155
+ sqliteDialect,
156
+ "id",
157
+ [{ t: "where", path: "tags", op: "includes", value: "admin" }],
158
+ "users",
159
+ {}
160
+ )
161
+ expect(result.sql).toContain("json_each")
162
+ expect(result.sql).toContain("value = ?")
163
+ expect(result.params).toContain("admin")
164
+ })
165
+
166
+ it("where includes-any (array contains any)", () => {
167
+ const result = buildWhereSQLQuery(
168
+ sqliteDialect,
169
+ "id",
170
+ [{ t: "where", path: "tags", op: "includes-any", value: ["admin", "user"] as any }],
171
+ "users",
172
+ {}
173
+ )
174
+ expect(result.sql).toContain("json_each")
175
+ expect(result.sql).toContain("IN")
176
+ })
177
+
178
+ it("nested scopes", () => {
179
+ const result = buildWhereSQLQuery(
180
+ sqliteDialect,
181
+ "id",
182
+ [
183
+ { t: "where", path: "a", op: "eq", value: "1" },
184
+ {
185
+ t: "or-scope",
186
+ result: [
187
+ { t: "where", path: "b", op: "eq", value: "2" },
188
+ { t: "and", path: "c", op: "eq", value: "3" }
189
+ ],
190
+ relation: "some" as const
191
+ }
192
+ ],
193
+ "test",
194
+ {}
195
+ )
196
+ expect(result.sql).toContain("OR (")
197
+ expect(result.sql).toContain("AND")
198
+ expect(result.params).toEqual(expect.arrayContaining(["1", "2", "3"]))
199
+ })
200
+
201
+ it("id key maps to id column", () => {
202
+ const result = buildWhereSQLQuery(
203
+ sqliteDialect,
204
+ "myId",
205
+ [{ t: "where", path: "myId", op: "eq", value: "123" }],
206
+ "users",
207
+ {}
208
+ )
209
+ expect(result.sql).toContain("id = ?")
210
+ expect(result.sql).not.toContain("json_extract")
211
+ expect(result.params).toContain("123")
212
+ })
213
+
214
+ it("order + limit + skip", () => {
215
+ const result = buildWhereSQLQuery(
216
+ sqliteDialect,
217
+ "id",
218
+ [],
219
+ "users",
220
+ {},
221
+ undefined,
222
+ [{ key: "name", direction: "ASC" }] as any,
223
+ 5,
224
+ 10
225
+ )
226
+ expect(result.sql).toContain("ORDER BY")
227
+ expect(result.sql).toContain("ASC")
228
+ expect(result.sql).toContain("LIMIT")
229
+ expect(result.sql).toContain("OFFSET")
230
+ })
231
+
232
+ it("computed relation count projection", () => {
233
+ const result = buildWhereSQLQuery(
234
+ sqliteDialect,
235
+ "id",
236
+ [],
237
+ "users",
238
+ {},
239
+ [{
240
+ key: "pickedCount",
241
+ computed: {
242
+ _tag: "relation-count",
243
+ path: "items",
244
+ filter: [{ t: "where", path: "items.-1.description", op: "contains", value: "picked" }]
245
+ }
246
+ }]
247
+ )
248
+ expect(result.sql).toContain(`SELECT COUNT(1) FROM json_each(data, '$.items') AS _items`)
249
+ expect(result.sql).toContain(`AS "pickedCount"`)
250
+ expect(result.params).toContain("%picked%")
251
+ })
252
+
253
+ it("computed relation any projection (sqlite bool encoding)", () => {
254
+ const result = buildWhereSQLQuery(
255
+ sqliteDialect,
256
+ "id",
257
+ [],
258
+ "users",
259
+ {},
260
+ [{
261
+ key: "hasPicked",
262
+ computed: {
263
+ _tag: "relation-any",
264
+ path: "items",
265
+ filter: [{ t: "where", path: "items.-1.description", op: "contains", value: "picked" }]
266
+ }
267
+ }]
268
+ )
269
+ expect(result.sql).toContain("CASE WHEN EXISTS(")
270
+ expect(result.sql).toContain(`AS "hasPicked"`)
271
+ })
272
+
273
+ it("computed relation every projection (sqlite emits NOT EXISTS NOT)", () => {
274
+ const result = buildWhereSQLQuery(
275
+ sqliteDialect,
276
+ "id",
277
+ [],
278
+ "users",
279
+ {},
280
+ [{
281
+ key: "allPicked",
282
+ computed: {
283
+ _tag: "relation-every",
284
+ path: "items",
285
+ filter: [{ t: "where", path: "items.-1.state._tag", op: "eq", value: "picked" }]
286
+ }
287
+ }]
288
+ )
289
+ expect(result.sql).toContain(`NOT EXISTS(SELECT 1 FROM json_each(data, '$.items') AS _items WHERE NOT (`)
290
+ expect(result.sql).toContain(`AS "allPicked"`)
291
+ expect(result.params).toContain("picked")
292
+ })
293
+
294
+ it("computed relation-every with no filter degenerates to true", () => {
295
+ const result = buildWhereSQLQuery(
296
+ sqliteDialect,
297
+ "id",
298
+ [],
299
+ "users",
300
+ {},
301
+ [{
302
+ key: "allPicked",
303
+ computed: { _tag: "relation-every", path: "items", filter: [] }
304
+ }]
305
+ )
306
+ expect(result.sql).toContain("CASE WHEN 1=1")
307
+ expect(result.sql).toContain(`AS "allPicked"`)
308
+ })
309
+
310
+ it("computed relation-distinct-count projection (sqlite)", () => {
311
+ const result = buildWhereSQLQuery(
312
+ sqliteDialect,
313
+ "id",
314
+ [],
315
+ "users",
316
+ {},
317
+ [{
318
+ key: "positionCount",
319
+ computed: {
320
+ _tag: "relation-distinct-count",
321
+ path: "items",
322
+ field: "rowId",
323
+ filter: [{ t: "where", path: "items.-1.state._tag", op: "neq", value: "cancelled" }]
324
+ }
325
+ }]
326
+ )
327
+ expect(result.sql).toContain(`SELECT COUNT(DISTINCT json_extract(_items.value, '$.rowId'))`)
328
+ expect(result.sql).toContain(`AS "positionCount"`)
329
+ expect(result.params).toContain("cancelled")
330
+ })
331
+
332
+ it("computed relation-sum projection (sqlite casts to REAL)", () => {
333
+ const result = buildWhereSQLQuery(
334
+ sqliteDialect,
335
+ "id",
336
+ [],
337
+ "users",
338
+ {},
339
+ [{
340
+ key: "totalWeight",
341
+ computed: { _tag: "relation-sum", path: "items", field: "weight", filter: [] }
342
+ }]
343
+ )
344
+ expect(result.sql).toContain(`SELECT COALESCE(SUM(CAST(json_extract(_items.value, '$.weight') AS REAL)), 0)`)
345
+ expect(result.sql).toContain(`AS "totalWeight"`)
346
+ })
347
+
348
+ it("computed relation-sum-expr projection (sqlite)", () => {
349
+ const result = buildWhereSQLQuery(
350
+ sqliteDialect,
351
+ "id",
352
+ [],
353
+ "users",
354
+ {},
355
+ [{
356
+ key: "totalWeighted",
357
+ computed: {
358
+ _tag: "relation-sum-expr",
359
+ path: "items",
360
+ expression: {
361
+ _tag: "mul",
362
+ left: { _tag: "field", field: "weight" },
363
+ right: { _tag: "field", field: "tradeUnit.amount" }
364
+ },
365
+ filter: []
366
+ }
367
+ }]
368
+ )
369
+ expect(result.sql).toContain(
370
+ `COALESCE(SUM((CAST(json_extract(_items.value, '$.weight') AS REAL) * CAST(json_extract(_items.value, '$.tradeUnit.amount') AS REAL))), 0)`
371
+ )
372
+ expect(result.sql).toContain(`AS "totalWeighted"`)
373
+ })
374
+
375
+ it("computed relation-sum-expr-by projection (sqlite)", () => {
376
+ const result = buildWhereSQLQuery(
377
+ sqliteDialect,
378
+ "id",
379
+ [],
380
+ "users",
381
+ {},
382
+ [{
383
+ key: "totalsByUnit",
384
+ computed: {
385
+ _tag: "relation-sum-expr-by",
386
+ path: "items",
387
+ expression: {
388
+ _tag: "mul",
389
+ left: { _tag: "field", field: "weight" },
390
+ right: { _tag: "field", field: "tradeUnit.amount" }
391
+ },
392
+ unit: "tradeUnit.unit",
393
+ filter: []
394
+ }
395
+ }]
396
+ )
397
+ expect(result.sql).toContain(`json_group_array(json_object('unit', __unit, 'total', __total))`)
398
+ expect(result.sql).toContain(`GROUP BY json_extract(_items.value, '$.tradeUnit.unit')`)
399
+ expect(result.sql).toContain(`AS "totalsByUnit"`)
400
+ })
401
+
402
+ it("computed relation-sum-expr-normalized projection (sqlite)", () => {
403
+ const result = buildWhereSQLQuery(
404
+ sqliteDialect,
405
+ "id",
406
+ [],
407
+ "users",
408
+ {},
409
+ [{
410
+ key: "totalKg",
411
+ computed: {
412
+ _tag: "relation-sum-expr-normalized",
413
+ path: "items",
414
+ expression: {
415
+ _tag: "mul",
416
+ left: { _tag: "field", field: "weight" },
417
+ right: { _tag: "field", field: "tradeUnit.amount" }
418
+ },
419
+ unit: "tradeUnit.unit",
420
+ toBase: "kg",
421
+ factors: { g: 0.001 },
422
+ filter: []
423
+ }
424
+ }]
425
+ )
426
+ expect(result.sql).toContain(
427
+ `CASE json_extract(_items.value, '$.tradeUnit.unit') WHEN 'kg' THEN 1 WHEN 'g' THEN 0.001 ELSE NULL END`
428
+ )
429
+ expect(result.sql).toContain(`AS "totalKg"`)
430
+ })
431
+
432
+ it("computed relation-collect (non-distinct) projection (sqlite)", () => {
433
+ const result = buildWhereSQLQuery(
434
+ sqliteDialect,
435
+ "id",
436
+ [],
437
+ "users",
438
+ {},
439
+ [{
440
+ key: "tags",
441
+ computed: { _tag: "relation-collect", path: "items", field: "articleId", distinct: false, filter: [] }
442
+ }]
443
+ )
444
+ expect(result.sql).toContain(
445
+ `SELECT COALESCE(json_group_array(json_extract(_items.value, '$.articleId')), json_array())`
446
+ )
447
+ expect(result.sql).toContain(`AS "tags"`)
448
+ })
449
+
450
+ it("computed relation-collect (distinct) emits inner DISTINCT subquery (sqlite)", () => {
451
+ const result = buildWhereSQLQuery(
452
+ sqliteDialect,
453
+ "id",
454
+ [],
455
+ "users",
456
+ {},
457
+ [{
458
+ key: "tags",
459
+ computed: { _tag: "relation-collect", path: "items", field: "articleId", distinct: true, filter: [] }
460
+ }]
461
+ )
462
+ expect(result.sql).toContain(`json_group_array(__v)`)
463
+ expect(result.sql).toContain(`SELECT DISTINCT json_extract(_items.value, '$.articleId') AS __v`)
464
+ expect(result.sql).toContain(`AS "tags"`)
465
+ })
466
+ })
467
+
468
+ describe("SQL query builder (PostgreSQL dialect)", () => {
469
+ it("where eq string uses ->> operator", () => {
470
+ const result = buildWhereSQLQuery(
471
+ pgDialect,
472
+ "id",
473
+ [{ t: "where", path: "name", op: "eq", value: "John" }],
474
+ "users",
475
+ {}
476
+ )
477
+ expect(result.sql).toContain("data->>'name'")
478
+ expect(result.sql).toContain("$1")
479
+ expect(result.params).toContain("John")
480
+ })
481
+
482
+ it("where contains uses ILIKE", () => {
483
+ const result = buildWhereSQLQuery(
484
+ pgDialect,
485
+ "id",
486
+ [{ t: "where", path: "name", op: "contains", value: "oh" }],
487
+ "users",
488
+ {}
489
+ )
490
+ expect(result.sql).toContain("ILIKE")
491
+ expect(result.params).toContain("%oh%")
492
+ })
493
+
494
+ it("where in uses $N placeholders", () => {
495
+ const result = buildWhereSQLQuery(
496
+ pgDialect,
497
+ "id",
498
+ [{ t: "where", path: "status", op: "in", value: ["active", "pending"] as any }],
499
+ "users",
500
+ {}
501
+ )
502
+ expect(result.sql).toContain("$1")
503
+ expect(result.sql).toContain("$2")
504
+ expect(result.params).toEqual(expect.arrayContaining(["active", "pending"]))
505
+ })
506
+
507
+ it("where includes uses @> jsonb operator", () => {
508
+ const result = buildWhereSQLQuery(
509
+ pgDialect,
510
+ "id",
511
+ [{ t: "where", path: "tags", op: "includes", value: "admin" }],
512
+ "users",
513
+ {}
514
+ )
515
+ expect(result.sql).toContain("@>")
516
+ expect(result.sql).toContain("jsonb")
517
+ })
518
+
519
+ it("nested path uses chained -> operators", () => {
520
+ const result = buildWhereSQLQuery(
521
+ pgDialect,
522
+ "id",
523
+ [{ t: "where", path: "address.city", op: "eq", value: "NYC" }],
524
+ "users",
525
+ {}
526
+ )
527
+ expect(result.sql).toContain("data->'address'->>'city'")
528
+ })
529
+
530
+ it("computed relation any projection", () => {
531
+ const result = buildWhereSQLQuery(
532
+ pgDialect,
533
+ "id",
534
+ [],
535
+ "users",
536
+ {},
537
+ [{
538
+ key: "hasPicked",
539
+ computed: {
540
+ _tag: "relation-any",
541
+ path: "items",
542
+ filter: [{ t: "where", path: "items.-1.description", op: "contains", value: "picked" }]
543
+ }
544
+ }]
545
+ )
546
+ expect(result.sql).toContain("EXISTS(SELECT 1 FROM jsonb_array_elements(data->'items') AS _items")
547
+ expect(result.sql).toContain(`AS "hasPicked"`)
548
+ })
549
+
550
+ it("computed relation-every (pg)", () => {
551
+ const result = buildWhereSQLQuery(
552
+ pgDialect,
553
+ "id",
554
+ [],
555
+ "users",
556
+ {},
557
+ [{
558
+ key: "allPicked",
559
+ computed: {
560
+ _tag: "relation-every",
561
+ path: "items",
562
+ filter: [{ t: "where", path: "items.-1.state._tag", op: "eq", value: "picked" }]
563
+ }
564
+ }]
565
+ )
566
+ expect(result.sql).toContain(`NOT EXISTS(SELECT 1 FROM jsonb_array_elements(data->'items') AS _items WHERE NOT (`)
567
+ expect(result.sql).toContain(`AS "allPicked"`)
568
+ })
569
+
570
+ it("computed relation-distinct-count (pg)", () => {
571
+ const result = buildWhereSQLQuery(
572
+ pgDialect,
573
+ "id",
574
+ [],
575
+ "users",
576
+ {},
577
+ [{
578
+ key: "positions",
579
+ computed: { _tag: "relation-distinct-count", path: "items", field: "rowId", filter: [] }
580
+ }]
581
+ )
582
+ expect(result.sql).toContain(`COUNT(DISTINCT _items->>'rowId')`)
583
+ expect(result.sql).toContain(`AS "positions"`)
584
+ })
585
+
586
+ it("computed relation-sum (pg casts via ::numeric)", () => {
587
+ const result = buildWhereSQLQuery(
588
+ pgDialect,
589
+ "id",
590
+ [],
591
+ "users",
592
+ {},
593
+ [{
594
+ key: "totalWeight",
595
+ computed: { _tag: "relation-sum", path: "items", field: "weight", filter: [] }
596
+ }]
597
+ )
598
+ expect(result.sql).toContain(`COALESCE(SUM((_items->>'weight')::numeric), 0)`)
599
+ expect(result.sql).toContain(`AS "totalWeight"`)
600
+ })
601
+
602
+ it("computed relation-sum-expr (pg)", () => {
603
+ const result = buildWhereSQLQuery(
604
+ pgDialect,
605
+ "id",
606
+ [],
607
+ "users",
608
+ {},
609
+ [{
610
+ key: "totalWeighted",
611
+ computed: {
612
+ _tag: "relation-sum-expr",
613
+ path: "items",
614
+ expression: {
615
+ _tag: "mul",
616
+ left: { _tag: "field", field: "weight" },
617
+ right: { _tag: "field", field: "tradeUnit.amount" }
618
+ },
619
+ filter: []
620
+ }
621
+ }]
622
+ )
623
+ expect(result.sql).toContain(
624
+ `COALESCE(SUM(((_items->>'weight')::numeric * (_items->'tradeUnit'->>'amount')::numeric)), 0)`
625
+ )
626
+ expect(result.sql).toContain(`AS "totalWeighted"`)
627
+ })
628
+
629
+ it("computed relation-sum-expr-by (pg)", () => {
630
+ const result = buildWhereSQLQuery(
631
+ pgDialect,
632
+ "id",
633
+ [],
634
+ "users",
635
+ {},
636
+ [{
637
+ key: "totalsByUnit",
638
+ computed: {
639
+ _tag: "relation-sum-expr-by",
640
+ path: "items",
641
+ expression: {
642
+ _tag: "mul",
643
+ left: { _tag: "field", field: "weight" },
644
+ right: { _tag: "field", field: "tradeUnit.amount" }
645
+ },
646
+ unit: "tradeUnit.unit",
647
+ filter: []
648
+ }
649
+ }]
650
+ )
651
+ expect(result.sql).toContain(`jsonb_agg(jsonb_build_object('unit', __unit, 'total', __total))`)
652
+ expect(result.sql).toContain(`GROUP BY _items->'tradeUnit'->>'unit'`)
653
+ expect(result.sql).toContain(`AS "totalsByUnit"`)
654
+ })
655
+
656
+ it("computed relation-sum-expr-normalized (pg)", () => {
657
+ const result = buildWhereSQLQuery(
658
+ pgDialect,
659
+ "id",
660
+ [],
661
+ "users",
662
+ {},
663
+ [{
664
+ key: "totalKg",
665
+ computed: {
666
+ _tag: "relation-sum-expr-normalized",
667
+ path: "items",
668
+ expression: {
669
+ _tag: "mul",
670
+ left: { _tag: "field", field: "weight" },
671
+ right: { _tag: "field", field: "tradeUnit.amount" }
672
+ },
673
+ unit: "tradeUnit.unit",
674
+ toBase: "kg",
675
+ factors: { g: 0.001 },
676
+ filter: []
677
+ }
678
+ }]
679
+ )
680
+ expect(result.sql).toContain(`CASE _items->'tradeUnit'->>'unit' WHEN 'kg' THEN 1 WHEN 'g' THEN 0.001 ELSE NULL END`)
681
+ expect(result.sql).toContain(`AS "totalKg"`)
682
+ })
683
+
684
+ it("computed relation-collect (pg jsonb_agg)", () => {
685
+ const result = buildWhereSQLQuery(
686
+ pgDialect,
687
+ "id",
688
+ [],
689
+ "users",
690
+ {},
691
+ [{
692
+ key: "ids",
693
+ computed: { _tag: "relation-collect", path: "items", field: "articleId", distinct: false, filter: [] }
694
+ }]
695
+ )
696
+ expect(result.sql).toContain(`COALESCE(jsonb_agg(_items->>'articleId'), '[]'::jsonb)`)
697
+ expect(result.sql).toContain(`AS "ids"`)
698
+ })
699
+
700
+ it("computed relation-collect distinct (pg jsonb_agg DISTINCT)", () => {
701
+ const result = buildWhereSQLQuery(
702
+ pgDialect,
703
+ "id",
704
+ [],
705
+ "users",
706
+ {},
707
+ [{
708
+ key: "ids",
709
+ computed: { _tag: "relation-collect", path: "items", field: "articleId", distinct: true, filter: [] }
710
+ }]
711
+ )
712
+ expect(result.sql).toContain(`COALESCE(jsonb_agg(DISTINCT _items->>'articleId'), '[]'::jsonb)`)
713
+ })
714
+ })
715
+
716
+ // --- Integration tests with in-memory SQLite (direct, no Effect SQL client) ---
717
+
718
+ describe("SQL Store (SQLite integration)", () => {
719
+ const withDb = (fn: (db: Sqlite.Database) => void) => {
720
+ const db = new BetterSqlite(":memory:")
721
+ db.pragma("journal_mode = WAL")
722
+ try {
723
+ fn(db)
724
+ } finally {
725
+ db.close()
726
+ }
727
+ }
728
+
729
+ it("creates table and seeds data", () =>
730
+ withDb((db) => {
731
+ db.exec(
732
+ `CREATE TABLE IF NOT EXISTS "test_items" (id TEXT PRIMARY KEY, _etag TEXT, data JSON NOT NULL)`
733
+ )
734
+ db
735
+ .prepare(`INSERT INTO "test_items" (id, _etag, data) VALUES (?, ?, ?)`)
736
+ .run("1", "etag1", JSON.stringify({ name: "Alice", age: 30 }))
737
+
738
+ const rows = db.prepare(`SELECT * FROM "test_items"`).all()
739
+ expect(rows.length).toBe(1)
740
+ expect((rows[0] as any).id).toBe("1")
741
+ }))
742
+
743
+ it("data column should not contain _etag or id", () =>
744
+ withDb((db) => {
745
+ db.exec(
746
+ `CREATE TABLE IF NOT EXISTS "test_clean" (id TEXT PRIMARY KEY, _etag TEXT, data JSON NOT NULL)`
747
+ )
748
+ // Simulate what toRow now produces: data without id or _etag
749
+ const data = { name: "Alice", age: 30, tags: ["admin"] }
750
+ db
751
+ .prepare(`INSERT INTO "test_clean" (id, _etag, data) VALUES (?, ?, ?)`)
752
+ .run("1", "etag1", JSON.stringify(data))
753
+
754
+ const row = db.prepare(`SELECT * FROM "test_clean" WHERE id = ?`).get("1") as any
755
+ const parsed = JSON.parse(row.data) as any
756
+ expect(parsed).not.toHaveProperty("id")
757
+ expect(parsed).not.toHaveProperty("_etag")
758
+ expect(parsed.name).toBe("Alice")
759
+ expect(parsed.age).toBe(30)
760
+ expect(parsed.tags).toEqual(["admin"])
761
+ // id and _etag come from their own columns
762
+ expect(row.id).toBe("1")
763
+ expect(row._etag).toBe("etag1")
764
+ }))
765
+
766
+ it("backward compat: rows with id/_etag in data still work with queries", () =>
767
+ withDb((db) => {
768
+ db.exec(
769
+ `CREATE TABLE IF NOT EXISTS "test_compat" (id TEXT PRIMARY KEY, _etag TEXT, data JSON NOT NULL)`
770
+ )
771
+ // Old format: id and _etag inside data
772
+ db
773
+ .prepare(`INSERT INTO "test_compat" (id, _etag, data) VALUES (?, ?, ?)`)
774
+ .run("1", "etag1", JSON.stringify({ id: "1", _etag: "old_etag", name: "Alice", age: 30 }))
775
+ // New format: id and _etag stripped from data
776
+ db
777
+ .prepare(`INSERT INTO "test_compat" (id, _etag, data) VALUES (?, ?, ?)`)
778
+ .run("2", "etag2", JSON.stringify({ name: "Bob", age: 25 }))
779
+
780
+ // Both should be queryable by name
781
+ const q1 = buildWhereSQLQuery(
782
+ sqliteDialect,
783
+ "id",
784
+ [{ t: "where", path: "name", op: "eq", value: "Alice" }],
785
+ "test_compat",
786
+ {}
787
+ )
788
+ const r1 = query(db, q1.sql, q1.params)
789
+ expect(r1.length).toBe(1)
790
+ expect((r1[0] as any).id).toBe("1")
791
+
792
+ const q2 = buildWhereSQLQuery(
793
+ sqliteDialect,
794
+ "id",
795
+ [{ t: "where", path: "name", op: "eq", value: "Bob" }],
796
+ "test_compat",
797
+ {}
798
+ )
799
+ const r2 = query(db, q2.sql, q2.params)
800
+ expect(r2.length).toBe(1)
801
+ expect((r2[0] as any).id).toBe("2")
802
+
803
+ // Both queryable by id column
804
+ const q3 = buildWhereSQLQuery(
805
+ sqliteDialect,
806
+ "id",
807
+ [{ t: "where", path: "id", op: "in", value: ["1", "2"] as any }],
808
+ "test_compat",
809
+ {}
810
+ )
811
+ expect(query(db, q3.sql, q3.params).length).toBe(2)
812
+ }))
813
+
814
+ it("queries work when data does not contain id", () =>
815
+ withDb((db) => {
816
+ db.exec(
817
+ `CREATE TABLE IF NOT EXISTS "test_noid" (id TEXT PRIMARY KEY, _etag TEXT, data JSON NOT NULL)`
818
+ )
819
+ const people = [
820
+ { name: "Alice", age: 30 },
821
+ { name: "Bob", age: 25 },
822
+ { name: "Charlie", age: 35 }
823
+ ]
824
+ const insert = db.prepare(
825
+ `INSERT INTO "test_noid" (id, _etag, data) VALUES (?, ?, ?)`
826
+ )
827
+ people.forEach((p, i) => insert.run(String(i + 1), `etag_${i + 1}`, JSON.stringify(p)))
828
+
829
+ // Filter by field in data
830
+ const q1 = buildWhereSQLQuery(
831
+ sqliteDialect,
832
+ "id",
833
+ [{ t: "where", path: "age", op: "gt", value: 28 as any }],
834
+ "test_noid",
835
+ {}
836
+ )
837
+ expect(query(db, q1.sql, q1.params).length).toBe(2) // Alice(30), Charlie(35)
838
+
839
+ // Filter by id column
840
+ const q2 = buildWhereSQLQuery(
841
+ sqliteDialect,
842
+ "id",
843
+ [{ t: "where", path: "id", op: "eq", value: "2" }],
844
+ "test_noid",
845
+ {}
846
+ )
847
+ const r2 = query(db, q2.sql, q2.params)
848
+ expect(r2.length).toBe(1)
849
+ expect((r2[0] as any).id).toBe("2")
850
+ expect((JSON.parse((r2[0] as any).data) as any).name).toBe("Bob")
851
+
852
+ // Order + limit still works
853
+ const q3 = buildWhereSQLQuery(
854
+ sqliteDialect,
855
+ "id",
856
+ [],
857
+ "test_noid",
858
+ {},
859
+ undefined,
860
+ [{ key: "age", direction: "ASC" }] as any,
861
+ undefined,
862
+ 2
863
+ )
864
+ const r3 = query(db, q3.sql, q3.params)
865
+ expect(r3.length).toBe(2)
866
+ expect((JSON.parse((r3[0] as any).data) as any).name).toBe("Bob") // youngest first
867
+ }))
868
+
869
+ it("query builder generates valid SQL for SQLite", () =>
870
+ withDb((db) => {
871
+ db.exec(
872
+ `CREATE TABLE IF NOT EXISTS "test_people" (id TEXT PRIMARY KEY, _etag TEXT, data JSON NOT NULL)`
873
+ )
874
+
875
+ const people = [
876
+ { id: "1", name: "Alice", age: 30, tags: ["admin", "user"] },
877
+ { id: "2", name: "Bob", age: 25, tags: ["user"] },
878
+ { id: "3", name: "Charlie", age: 35, tags: ["admin"] },
879
+ { id: "4", name: "Diana", age: 28, tags: ["user", "editor"] }
880
+ ]
881
+
882
+ const insert = db.prepare(
883
+ `INSERT INTO "test_people" (id, _etag, data) VALUES (?, ?, ?)`
884
+ )
885
+ for (const p of people) {
886
+ const { id, ...data } = p
887
+ insert.run(id, `etag_${id}`, JSON.stringify(data))
888
+ }
889
+
890
+ // Test eq
891
+ const q1 = buildWhereSQLQuery(
892
+ sqliteDialect,
893
+ "id",
894
+ [{ t: "where", path: "name", op: "eq", value: "Alice" }],
895
+ "test_people",
896
+ {}
897
+ )
898
+ expect(query(db, q1.sql, q1.params).length).toBe(1)
899
+ expect((JSON.parse((query(db, q1.sql, q1.params)[0] as any).data) as any).name).toBe("Alice")
900
+
901
+ // Test gt
902
+ const q2 = buildWhereSQLQuery(
903
+ sqliteDialect,
904
+ "id",
905
+ [{ t: "where", path: "age", op: "gt", value: 28 as any }],
906
+ "test_people",
907
+ {}
908
+ )
909
+ expect(query(db, q2.sql, q2.params).length).toBe(2)
910
+
911
+ // Test OR
912
+ const q3 = buildWhereSQLQuery(
913
+ sqliteDialect,
914
+ "id",
915
+ [
916
+ { t: "where", path: "name", op: "eq", value: "Alice" },
917
+ { t: "or", path: "name", op: "eq", value: "Bob" }
918
+ ],
919
+ "test_people",
920
+ {}
921
+ )
922
+ expect(query(db, q3.sql, q3.params).length).toBe(2)
923
+
924
+ // Test AND
925
+ const q4 = buildWhereSQLQuery(
926
+ sqliteDialect,
927
+ "id",
928
+ [
929
+ { t: "where", path: "name", op: "eq", value: "Alice" },
930
+ { t: "and", path: "age", op: "gt", value: 25 as any }
931
+ ],
932
+ "test_people",
933
+ {}
934
+ )
935
+ const r4 = query(db, q4.sql, q4.params)
936
+ expect(r4.length).toBe(1)
937
+ expect((JSON.parse((r4[0] as any).data) as any).name).toBe("Alice")
938
+
939
+ // Test IN
940
+ const q5 = buildWhereSQLQuery(
941
+ sqliteDialect,
942
+ "id",
943
+ [{ t: "where", path: "id", op: "in", value: ["1", "3"] as any }],
944
+ "test_people",
945
+ {}
946
+ )
947
+ expect(query(db, q5.sql, q5.params).length).toBe(2)
948
+
949
+ // Test contains (string)
950
+ const q6 = buildWhereSQLQuery(
951
+ sqliteDialect,
952
+ "id",
953
+ [{ t: "where", path: "name", op: "contains", value: "li" }],
954
+ "test_people",
955
+ {}
956
+ )
957
+ expect(query(db, q6.sql, q6.params).length).toBe(2) // Alice, Charlie
958
+
959
+ // Test startsWith
960
+ const q7 = buildWhereSQLQuery(
961
+ sqliteDialect,
962
+ "id",
963
+ [{ t: "where", path: "name", op: "startsWith", value: "Al" }],
964
+ "test_people",
965
+ {}
966
+ )
967
+ const r7 = query(db, q7.sql, q7.params)
968
+ expect(r7.length).toBe(1)
969
+ expect((JSON.parse((r7[0] as any).data) as any).name).toBe("Alice")
970
+
971
+ // Test includes (array)
972
+ const q8 = buildWhereSQLQuery(
973
+ sqliteDialect,
974
+ "id",
975
+ [{ t: "where", path: "tags", op: "includes", value: "admin" }],
976
+ "test_people",
977
+ {}
978
+ )
979
+ expect(query(db, q8.sql, q8.params).length).toBe(2) // Alice, Charlie
980
+
981
+ // Test nested scope: where name = Alice OR (age > 30 AND name contains 'ar')
982
+ const q9 = buildWhereSQLQuery(
983
+ sqliteDialect,
984
+ "id",
985
+ [
986
+ { t: "where", path: "name", op: "eq", value: "Alice" },
987
+ {
988
+ t: "or-scope",
989
+ result: [
990
+ { t: "where", path: "age", op: "gt", value: 30 as any },
991
+ { t: "and", path: "name", op: "contains", value: "ar" }
992
+ ],
993
+ relation: "some"
994
+ }
995
+ ],
996
+ "test_people",
997
+ {}
998
+ )
999
+ expect(query(db, q9.sql, q9.params).length).toBe(2) // Alice + Charlie
1000
+
1001
+ // Test order + limit
1002
+ const q10 = buildWhereSQLQuery(
1003
+ sqliteDialect,
1004
+ "id",
1005
+ [],
1006
+ "test_people",
1007
+ {},
1008
+ undefined,
1009
+ [{ key: "age", direction: "DESC" }] as any,
1010
+ undefined,
1011
+ 2
1012
+ )
1013
+ const r10 = query(db, q10.sql, q10.params)
1014
+ expect(r10.length).toBe(2)
1015
+ expect((JSON.parse((r10[0] as any).data) as any).name).toBe("Charlie") // oldest first
1016
+ }))
1017
+
1018
+ it("computed relation-every / distinct-count / sum / collect run on SQLite", () =>
1019
+ withDb((db) => {
1020
+ db.exec(`CREATE TABLE "test_orders" (id TEXT PRIMARY KEY, _etag TEXT, data JSON NOT NULL)`)
1021
+ const orders = [
1022
+ {
1023
+ id: "o1",
1024
+ items: [
1025
+ { rowId: "r1", articleId: "A", weight: 1.5, state: { _tag: "picked" } },
1026
+ { rowId: "r2", articleId: "A", weight: 2.5, state: { _tag: "picked" } },
1027
+ { rowId: "r2", articleId: "B", weight: 0.25, state: { _tag: "picking" } }
1028
+ ]
1029
+ },
1030
+ {
1031
+ id: "o2",
1032
+ items: [
1033
+ { rowId: "r9", articleId: "Z", weight: 10, state: { _tag: "packed" } }
1034
+ ]
1035
+ }
1036
+ ]
1037
+ const insert = db.prepare(`INSERT INTO "test_orders" (id, _etag, data) VALUES (?, ?, ?)`)
1038
+ for (const o of orders) {
1039
+ const { id, ...data } = o
1040
+ insert.run(id, `etag_${id}`, JSON.stringify(data))
1041
+ }
1042
+
1043
+ const q = buildWhereSQLQuery(
1044
+ sqliteDialect,
1045
+ "id",
1046
+ [],
1047
+ "test_orders",
1048
+ {},
1049
+ [
1050
+ {
1051
+ key: "allPicked",
1052
+ computed: {
1053
+ _tag: "relation-every",
1054
+ path: "items",
1055
+ filter: [{ t: "where", path: "items.-1.state._tag", op: "eq", value: "picked" }]
1056
+ }
1057
+ },
1058
+ {
1059
+ key: "positionCount",
1060
+ computed: { _tag: "relation-distinct-count", path: "items", field: "rowId", filter: [] }
1061
+ },
1062
+ {
1063
+ key: "totalWeight",
1064
+ computed: { _tag: "relation-sum", path: "items", field: "weight", filter: [] }
1065
+ },
1066
+ {
1067
+ key: "articleIds",
1068
+ computed: {
1069
+ _tag: "relation-collect",
1070
+ path: "items",
1071
+ field: "articleId",
1072
+ distinct: true,
1073
+ filter: []
1074
+ }
1075
+ }
1076
+ ] as any,
1077
+ [{ key: "id", direction: "ASC" }] as any
1078
+ )
1079
+ const rows = query(db, q.sql, q.params) as Array<Record<string, unknown>>
1080
+ expect(rows.length).toBe(2)
1081
+ // o1: not all picked (one is "picking")
1082
+ expect(JSON.parse(rows[0]!["allPicked"] as string)).toBe(false)
1083
+ expect(rows[0]!["positionCount"]).toBe(2)
1084
+ expect(rows[0]!["totalWeight"]).toBeCloseTo(4.25)
1085
+ expect((JSON.parse(rows[0]!["articleIds"] as string) as string[]).sort()).toEqual(["A", "B"])
1086
+ // o2: all packed (so not "picked"), allPicked = !exists(NOT picked) = false
1087
+ expect(JSON.parse(rows[1]!["allPicked"] as string)).toBe(false)
1088
+ expect(rows[1]!["positionCount"]).toBe(1)
1089
+ expect(rows[1]!["totalWeight"]).toBeCloseTo(10)
1090
+ expect(JSON.parse(rows[1]!["articleIds"] as string)).toEqual(["Z"])
1091
+ }))
1092
+
1093
+ it("computed relation-every is true when all items match filter", () =>
1094
+ withDb((db) => {
1095
+ db.exec(`CREATE TABLE "test_every" (id TEXT PRIMARY KEY, _etag TEXT, data JSON NOT NULL)`)
1096
+ db.prepare(`INSERT INTO "test_every" (id, _etag, data) VALUES (?, ?, ?)`).run(
1097
+ "1",
1098
+ "e",
1099
+ JSON.stringify({
1100
+ items: [
1101
+ { state: { _tag: "picked" } },
1102
+ { state: { _tag: "picked" } }
1103
+ ]
1104
+ })
1105
+ )
1106
+
1107
+ const q = buildWhereSQLQuery(
1108
+ sqliteDialect,
1109
+ "id",
1110
+ [],
1111
+ "test_every",
1112
+ {},
1113
+ [{
1114
+ key: "allPicked",
1115
+ computed: {
1116
+ _tag: "relation-every",
1117
+ path: "items",
1118
+ filter: [{ t: "where", path: "items.-1.state._tag", op: "eq", value: "picked" }]
1119
+ }
1120
+ }]
1121
+ )
1122
+ const rows = query(db, q.sql, q.params) as Array<Record<string, unknown>>
1123
+ expect(JSON.parse(rows[0]!["allPicked"] as string)).toBe(true)
1124
+ }))
1125
+
1126
+ it("namespace param is in correct position for SQLite positional placeholders", () =>
1127
+ withDb((db) => {
1128
+ db.exec(
1129
+ `CREATE TABLE IF NOT EXISTS "test_ns" (id TEXT NOT NULL, _namespace TEXT NOT NULL DEFAULT 'primary', _etag TEXT, data JSON NOT NULL, PRIMARY KEY (id, _namespace))`
1130
+ )
1131
+ const insert = db.prepare(
1132
+ `INSERT INTO "test_ns" (id, _namespace, _etag, data) VALUES (?, ?, ?, ?)`
1133
+ )
1134
+ insert.run("1", "primary", "e1", JSON.stringify({ name: "Alice", role: "admin" }))
1135
+ insert.run("2", "primary", "e2", JSON.stringify({ name: "Bob", role: "user" }))
1136
+ insert.run("3", "other", "e3", JSON.stringify({ name: "Charlie", role: "admin" }))
1137
+
1138
+ // Build a filter query: role != 'deleted'
1139
+ const q = buildWhereSQLQuery(
1140
+ sqliteDialect,
1141
+ "id",
1142
+ [{ t: "where", path: "role", op: "neq", value: "deleted" }],
1143
+ "test_ns",
1144
+ {}
1145
+ )
1146
+
1147
+ // Simulate what SQL.ts does: prepend _namespace = ? and put ns FIRST in params
1148
+ const hasWhere = q.sql.includes("WHERE")
1149
+ const nsSql = hasWhere
1150
+ ? q.sql.replace("WHERE", `WHERE _namespace = ? AND`)
1151
+ : q.sql.replace(`FROM "test_ns"`, `FROM "test_ns" WHERE _namespace = ?`)
1152
+ const params = ["primary", ...q.params]
1153
+
1154
+ const results = query(db, nsSql, params)
1155
+ // Should only get Alice and Bob (primary namespace), not Charlie (other namespace)
1156
+ expect(results.length).toBe(2)
1157
+ const names = results.map((r) => (JSON.parse((r as any).data) as any).name).sort()
1158
+ expect(names).toEqual(["Alice", "Bob"])
1159
+ }))
1160
+ })
1161
+
1162
+ // --- boolean WHERE clause tests (regression: where("x", true) must work per dialect) ---
1163
+
1164
+ describe("boolean WHERE clauses — query builder", () => {
1165
+ it("sqlite: eq true serializes bool → 1 integer", () => {
1166
+ const result = buildWhereSQLQuery(
1167
+ sqliteDialect,
1168
+ "id",
1169
+ [{ t: "where", path: "flag", op: "eq", value: true as any }],
1170
+ "t",
1171
+ {}
1172
+ )
1173
+ expect(result.sql).toContain("json_extract(data, '$.flag') = ?")
1174
+ expect(result.params).toEqual([1])
1175
+ })
1176
+
1177
+ it("sqlite: eq false serializes bool → 0 integer", () => {
1178
+ const result = buildWhereSQLQuery(
1179
+ sqliteDialect,
1180
+ "id",
1181
+ [{ t: "where", path: "flag", op: "eq", value: false as any }],
1182
+ "t",
1183
+ {}
1184
+ )
1185
+ expect(result.params).toEqual([0])
1186
+ })
1187
+
1188
+ it("sqlite: neq/in also serialize booleans", () => {
1189
+ const r1 = buildWhereSQLQuery(
1190
+ sqliteDialect,
1191
+ "id",
1192
+ [{ t: "where", path: "flag", op: "neq", value: true as any }],
1193
+ "t",
1194
+ {}
1195
+ )
1196
+ expect(r1.params).toEqual([1])
1197
+
1198
+ const r2 = buildWhereSQLQuery(
1199
+ sqliteDialect,
1200
+ "id",
1201
+ [{ t: "where", path: "flag", op: "in", value: [true, false] as any }],
1202
+ "t",
1203
+ {}
1204
+ )
1205
+ expect(r2.params).toEqual([1, 0])
1206
+ })
1207
+
1208
+ it("pg: eq true serializes bool → 'true' text (matches ->> output)", () => {
1209
+ const result = buildWhereSQLQuery(
1210
+ pgDialect,
1211
+ "id",
1212
+ [{ t: "where", path: "flag", op: "eq", value: true as any }],
1213
+ "t",
1214
+ {}
1215
+ )
1216
+ expect(result.sql).toContain("data->>'flag' = $1")
1217
+ expect(result.params).toEqual(["true"])
1218
+ })
1219
+
1220
+ it("pg: eq false serializes bool → 'false' text", () => {
1221
+ const result = buildWhereSQLQuery(
1222
+ pgDialect,
1223
+ "id",
1224
+ [{ t: "where", path: "flag", op: "eq", value: false as any }],
1225
+ "t",
1226
+ {}
1227
+ )
1228
+ expect(result.params).toEqual(["false"])
1229
+ })
1230
+
1231
+ it("pg: in with booleans serializes as text list", () => {
1232
+ const result = buildWhereSQLQuery(
1233
+ pgDialect,
1234
+ "id",
1235
+ [{ t: "where", path: "flag", op: "in", value: [true, false] as any }],
1236
+ "t",
1237
+ {}
1238
+ )
1239
+ expect(result.params).toEqual(["true", "false"])
1240
+ })
1241
+
1242
+ it("non-boolean scalars pass through unchanged (sqlite)", () => {
1243
+ const result = buildWhereSQLQuery(
1244
+ sqliteDialect,
1245
+ "id",
1246
+ [{ t: "where", path: "name", op: "eq", value: "Alice" }],
1247
+ "t",
1248
+ {}
1249
+ )
1250
+ expect(result.params).toEqual(["Alice"])
1251
+ })
1252
+
1253
+ it("non-boolean scalars pass through unchanged (pg)", () => {
1254
+ const result = buildWhereSQLQuery(
1255
+ pgDialect,
1256
+ "id",
1257
+ [{ t: "where", path: "age", op: "gt", value: 18 as any }],
1258
+ "t",
1259
+ {}
1260
+ )
1261
+ expect(result.params).toEqual([18])
1262
+ })
1263
+ })
1264
+
1265
+ describe("boolean WHERE clauses — SQLite integration (end-to-end)", () => {
1266
+ const withDb = (fn: (db: Sqlite.Database) => void) => {
1267
+ const db = new BetterSqlite(":memory:")
1268
+ try {
1269
+ fn(db)
1270
+ } finally {
1271
+ db.close()
1272
+ }
1273
+ }
1274
+
1275
+ it("where flag = true matches only true rows", () =>
1276
+ withDb((db) => {
1277
+ db.exec(`CREATE TABLE "t" (id TEXT PRIMARY KEY, _etag TEXT, data JSON NOT NULL)`)
1278
+ db
1279
+ .prepare(`INSERT INTO "t" (id, _etag, data) VALUES (?, ?, ?)`)
1280
+ .run("1", "e", JSON.stringify({ flag: true, name: "Alice" }))
1281
+ db
1282
+ .prepare(`INSERT INTO "t" (id, _etag, data) VALUES (?, ?, ?)`)
1283
+ .run("2", "e", JSON.stringify({ flag: false, name: "Bob" }))
1284
+
1285
+ const q = buildWhereSQLQuery(
1286
+ sqliteDialect,
1287
+ "id",
1288
+ [{ t: "where", path: "flag", op: "eq", value: true as any }],
1289
+ "t",
1290
+ {}
1291
+ )
1292
+ const rows = query(db, q.sql, q.params)
1293
+ expect(rows.length).toBe(1)
1294
+ expect((JSON.parse((rows[0] as any).data) as any).name).toBe("Alice")
1295
+ }))
1296
+
1297
+ it("where flag = false matches only false rows", () =>
1298
+ withDb((db) => {
1299
+ db.exec(`CREATE TABLE "t" (id TEXT PRIMARY KEY, _etag TEXT, data JSON NOT NULL)`)
1300
+ db
1301
+ .prepare(`INSERT INTO "t" (id, _etag, data) VALUES (?, ?, ?)`)
1302
+ .run("1", "e", JSON.stringify({ flag: true, name: "Alice" }))
1303
+ db
1304
+ .prepare(`INSERT INTO "t" (id, _etag, data) VALUES (?, ?, ?)`)
1305
+ .run("2", "e", JSON.stringify({ flag: false, name: "Bob" }))
1306
+
1307
+ const q = buildWhereSQLQuery(
1308
+ sqliteDialect,
1309
+ "id",
1310
+ [{ t: "where", path: "flag", op: "eq", value: false as any }],
1311
+ "t",
1312
+ {}
1313
+ )
1314
+ const rows = query(db, q.sql, q.params)
1315
+ expect(rows.length).toBe(1)
1316
+ expect((JSON.parse((rows[0] as any).data) as any).name).toBe("Bob")
1317
+ }))
1318
+
1319
+ it("where nested boolean path works", () =>
1320
+ withDb((db) => {
1321
+ db.exec(`CREATE TABLE "t" (id TEXT PRIMARY KEY, _etag TEXT, data JSON NOT NULL)`)
1322
+ db
1323
+ .prepare(`INSERT INTO "t" (id, _etag, data) VALUES (?, ?, ?)`)
1324
+ .run("1", "e", JSON.stringify({ meta: { active: true }, name: "Alice" }))
1325
+ db
1326
+ .prepare(`INSERT INTO "t" (id, _etag, data) VALUES (?, ?, ?)`)
1327
+ .run("2", "e", JSON.stringify({ meta: { active: false }, name: "Bob" }))
1328
+
1329
+ const q = buildWhereSQLQuery(
1330
+ sqliteDialect,
1331
+ "id",
1332
+ [{ t: "where", path: "meta.active", op: "eq", value: true as any }],
1333
+ "t",
1334
+ {}
1335
+ )
1336
+ const rows = query(db, q.sql, q.params)
1337
+ expect(rows.length).toBe(1)
1338
+ expect((JSON.parse((rows[0] as any).data) as any).name).toBe("Alice")
1339
+ }))
1340
+
1341
+ it("where neq boolean works", () =>
1342
+ withDb((db) => {
1343
+ db.exec(`CREATE TABLE "t" (id TEXT PRIMARY KEY, _etag TEXT, data JSON NOT NULL)`)
1344
+ db
1345
+ .prepare(`INSERT INTO "t" (id, _etag, data) VALUES (?, ?, ?)`)
1346
+ .run("1", "e", JSON.stringify({ flag: true, name: "Alice" }))
1347
+ db
1348
+ .prepare(`INSERT INTO "t" (id, _etag, data) VALUES (?, ?, ?)`)
1349
+ .run("2", "e", JSON.stringify({ flag: false, name: "Bob" }))
1350
+
1351
+ const q = buildWhereSQLQuery(
1352
+ sqliteDialect,
1353
+ "id",
1354
+ [{ t: "where", path: "flag", op: "neq", value: true as any }],
1355
+ "t",
1356
+ {}
1357
+ )
1358
+ const rows = query(db, q.sql, q.params)
1359
+ expect(rows.length).toBe(1)
1360
+ expect((JSON.parse((rows[0] as any).data) as any).name).toBe("Bob")
1361
+ }))
1362
+ })
1363
+
1364
+ // --- jsonExtractJson dialect tests (regression: booleans must survive round-trip) ---
1365
+
1366
+ describe("sqliteDialect.jsonExtractJson boolean round-trip", () => {
1367
+ const withDb = (fn: (db: Sqlite.Database) => void) => {
1368
+ const db = new BetterSqlite(":memory:")
1369
+ try {
1370
+ fn(db)
1371
+ } finally {
1372
+ db.close()
1373
+ }
1374
+ }
1375
+
1376
+ it("SQL uses json_type CASE to preserve booleans", () => {
1377
+ const sql = sqliteDialect.jsonExtractJson("flag")
1378
+ expect(sql).toContain("json_type(data, '$.flag')")
1379
+ expect(sql).toContain("'true'")
1380
+ expect(sql).toContain("'false'")
1381
+ expect(sql).toContain("json_quote(json_extract(data, '$.flag'))")
1382
+ })
1383
+
1384
+ it("boolean false round-trips as false (not 0)", () =>
1385
+ withDb((db) => {
1386
+ db.exec(`CREATE TABLE "t" (data JSON NOT NULL)`)
1387
+ db.prepare(`INSERT INTO "t" (data) VALUES (?)`).run(JSON.stringify({ flag: false }))
1388
+
1389
+ const row = db.prepare(`SELECT ${sqliteDialect.jsonExtractJson("flag")} AS v FROM "t"`).get() as any
1390
+ const parsed = JSON.parse(row.v)
1391
+ expect(parsed).toBe(false)
1392
+ expect(typeof parsed).toBe("boolean")
1393
+ }))
1394
+
1395
+ it("boolean true round-trips as true (not 1)", () =>
1396
+ withDb((db) => {
1397
+ db.exec(`CREATE TABLE "t" (data JSON NOT NULL)`)
1398
+ db.prepare(`INSERT INTO "t" (data) VALUES (?)`).run(JSON.stringify({ flag: true }))
1399
+
1400
+ const row = db.prepare(`SELECT ${sqliteDialect.jsonExtractJson("flag")} AS v FROM "t"`).get() as any
1401
+ const parsed = JSON.parse(row.v)
1402
+ expect(parsed).toBe(true)
1403
+ expect(typeof parsed).toBe("boolean")
1404
+ }))
1405
+
1406
+ it("nested boolean round-trips correctly", () =>
1407
+ withDb((db) => {
1408
+ db.exec(`CREATE TABLE "t" (data JSON NOT NULL)`)
1409
+ db.prepare(`INSERT INTO "t" (data) VALUES (?)`).run(
1410
+ JSON.stringify({ nested: { flag: false } })
1411
+ )
1412
+
1413
+ const row = db
1414
+ .prepare(`SELECT ${sqliteDialect.jsonExtractJson("nested.flag")} AS v FROM "t"`)
1415
+ .get() as any
1416
+ const parsed = JSON.parse(row.v)
1417
+ expect(parsed).toBe(false)
1418
+ expect(typeof parsed).toBe("boolean")
1419
+ }))
1420
+
1421
+ it("numbers 0 and 1 still parse as numbers (not booleans)", () =>
1422
+ withDb((db) => {
1423
+ db.exec(`CREATE TABLE "t" (data JSON NOT NULL)`)
1424
+ db.prepare(`INSERT INTO "t" (data) VALUES (?)`).run(JSON.stringify({ zero: 0, one: 1 }))
1425
+
1426
+ const zRow = db.prepare(`SELECT ${sqliteDialect.jsonExtractJson("zero")} AS v FROM "t"`).get() as any
1427
+ const oRow = db.prepare(`SELECT ${sqliteDialect.jsonExtractJson("one")} AS v FROM "t"`).get() as any
1428
+ expect(JSON.parse(zRow.v)).toBe(0)
1429
+ expect(typeof JSON.parse(zRow.v)).toBe("number")
1430
+ expect(JSON.parse(oRow.v)).toBe(1)
1431
+ expect(typeof JSON.parse(oRow.v)).toBe("number")
1432
+ }))
1433
+
1434
+ it("strings, arrays, and null still round-trip correctly", () =>
1435
+ withDb((db) => {
1436
+ db.exec(`CREATE TABLE "t" (data JSON NOT NULL)`)
1437
+ db.prepare(`INSERT INTO "t" (data) VALUES (?)`).run(
1438
+ JSON.stringify({ s: "hello", arr: [1, 2, 3], nil: null })
1439
+ )
1440
+
1441
+ const sRow = db.prepare(`SELECT ${sqliteDialect.jsonExtractJson("s")} AS v FROM "t"`).get() as any
1442
+ const aRow = db.prepare(`SELECT ${sqliteDialect.jsonExtractJson("arr")} AS v FROM "t"`).get() as any
1443
+ const nRow = db.prepare(`SELECT ${sqliteDialect.jsonExtractJson("nil")} AS v FROM "t"`).get() as any
1444
+ expect(JSON.parse(sRow.v)).toBe("hello")
1445
+ expect(JSON.parse(aRow.v)).toEqual([1, 2, 3])
1446
+ expect(JSON.parse(nRow.v)).toBe(null)
1447
+ }))
1448
+ })
1449
+
1450
+ // --- toRow stripping and parseRow reconstruction tests ---
1451
+
1452
+ describe("toRow strips _etag and id from data", () => {
1453
+ // Replicate the toRow logic from SQL.ts to test in isolation
1454
+ const toRow = <IdKey extends PropertyKey>(e: any, idKey: IdKey) => {
1455
+ const newE = makeETag(e)
1456
+ const id = newE[idKey] as string
1457
+ const { _etag, [idKey]: _id, ...rest } = newE as any
1458
+ const data = JSON.stringify(rest)
1459
+ return { id, _etag: newE._etag!, data, item: newE }
1460
+ }
1461
+
1462
+ it("data JSON does not contain _etag", () => {
1463
+ const row = toRow({ id: "1", _etag: undefined, name: "Alice", age: 30 }, "id")
1464
+ const parsed = JSON.parse(row.data) as any
1465
+ expect(parsed).not.toHaveProperty("_etag")
1466
+ expect(parsed.name).toBe("Alice")
1467
+ expect(parsed.age).toBe(30)
1468
+ })
1469
+
1470
+ it("data JSON does not contain id field", () => {
1471
+ const row = toRow({ id: "1", _etag: undefined, name: "Alice" }, "id")
1472
+ const parsed = JSON.parse(row.data) as any
1473
+ expect(parsed).not.toHaveProperty("id")
1474
+ expect(parsed.name).toBe("Alice")
1475
+ })
1476
+
1477
+ it("data JSON does not contain custom idKey field", () => {
1478
+ const row = toRow({ myId: "abc", _etag: undefined, name: "Bob" }, "myId")
1479
+ const parsed = JSON.parse(row.data) as any
1480
+ expect(parsed).not.toHaveProperty("myId")
1481
+ expect(parsed.name).toBe("Bob")
1482
+ expect(row.id).toBe("abc")
1483
+ })
1484
+
1485
+ it("id and _etag are returned as separate fields", () => {
1486
+ const row = toRow({ id: "1", _etag: undefined, name: "Alice" }, "id")
1487
+ expect(row.id).toBe("1")
1488
+ expect(typeof row._etag).toBe("string")
1489
+ expect(row._etag.length).toBeGreaterThan(0)
1490
+ })
1491
+
1492
+ it("item still contains all fields including _etag and id", () => {
1493
+ const row = toRow({ id: "1", _etag: undefined, name: "Alice" }, "id")
1494
+ expect(row.item.id).toBe("1")
1495
+ expect(row.item._etag).toBe(row._etag)
1496
+ expect(row.item.name).toBe("Alice")
1497
+ })
1498
+
1499
+ it("preserves nested objects in data", () => {
1500
+ const row = toRow({ id: "1", _etag: undefined, address: { city: "NYC", zip: "10001" } }, "id")
1501
+ const parsed = JSON.parse(row.data) as any
1502
+ expect(parsed.address).toEqual({ city: "NYC", zip: "10001" })
1503
+ expect(parsed).not.toHaveProperty("id")
1504
+ expect(parsed).not.toHaveProperty("_etag")
1505
+ })
1506
+ })
1507
+
1508
+ describe("parseRow reconstructs full object from row", () => {
1509
+ it("re-injects id from row column using idKey", () => {
1510
+ const result: any = parseRow(
1511
+ { id: "42", _etag: "etag1", data: JSON.stringify({ name: "Alice", age: 30 }) },
1512
+ "id",
1513
+ {}
1514
+ )
1515
+ expect(result.id).toBe("42")
1516
+ expect(result.name).toBe("Alice")
1517
+ expect(result.age).toBe(30)
1518
+ expect(result._etag).toBe("etag1")
1519
+ })
1520
+
1521
+ it("re-injects custom idKey from row column", () => {
1522
+ const result: any = parseRow(
1523
+ { id: "abc", _etag: "etag2", data: JSON.stringify({ name: "Bob" }) },
1524
+ "myId",
1525
+ {}
1526
+ )
1527
+ expect(result.myId).toBe("abc")
1528
+ expect(result.name).toBe("Bob")
1529
+ expect(result._etag).toBe("etag2")
1530
+ })
1531
+
1532
+ it("uses _etag from row column, not from data", () => {
1533
+ const result: any = parseRow(
1534
+ { id: "1", _etag: "column_etag", data: JSON.stringify({ _etag: "stale_data_etag", name: "Alice" }) },
1535
+ "id",
1536
+ {}
1537
+ )
1538
+ expect(result._etag).toBe("column_etag")
1539
+ })
1540
+
1541
+ it("uses id from row column, not from data", () => {
1542
+ const result: any = parseRow(
1543
+ { id: "correct_id", _etag: "e1", data: JSON.stringify({ id: "wrong_id", name: "Alice" }) },
1544
+ "id",
1545
+ {}
1546
+ )
1547
+ expect(result.id).toBe("correct_id")
1548
+ })
1549
+
1550
+ it("applies defaultValues for missing fields", () => {
1551
+ const result: any = parseRow(
1552
+ { id: "1", _etag: "e1", data: JSON.stringify({ name: "Alice" }) },
1553
+ "id",
1554
+ { status: "active", role: "user" }
1555
+ )
1556
+ expect(result.name).toBe("Alice")
1557
+ expect(result.status).toBe("active")
1558
+ expect(result.role).toBe("user")
1559
+ })
1560
+
1561
+ it("data fields override defaultValues", () => {
1562
+ const result: any = parseRow(
1563
+ { id: "1", _etag: "e1", data: JSON.stringify({ name: "Alice", status: "inactive" }) },
1564
+ "id",
1565
+ { status: "active" }
1566
+ )
1567
+ expect(result.status).toBe("inactive")
1568
+ })
1569
+
1570
+ it("handles null _etag from row", () => {
1571
+ const result: any = parseRow(
1572
+ { id: "1", _etag: null, data: JSON.stringify({ name: "Alice" }) },
1573
+ "id",
1574
+ {}
1575
+ )
1576
+ expect(result._etag).toBeUndefined()
1577
+ })
1578
+
1579
+ it("round-trip: toRow then parseRow reconstructs the original", () => {
1580
+ const original = { id: "1", _etag: undefined as string | undefined, name: "Alice", age: 30, tags: ["admin"] }
1581
+ const newE = makeETag(original)
1582
+ const { _etag, id: _id, ...rest } = newE as any
1583
+ const row = { id: newE.id, _etag: newE._etag!, data: JSON.stringify(rest) }
1584
+
1585
+ const reconstructed: any = parseRow(row, "id", {})
1586
+ expect(reconstructed.id).toBe("1")
1587
+ expect(reconstructed.name).toBe("Alice")
1588
+ expect(reconstructed.age).toBe(30)
1589
+ expect(reconstructed.tags).toEqual(["admin"])
1590
+ expect(reconstructed._etag).toBe(newE._etag)
1591
+ })
1592
+ })