@casekit/orm2 0.0.0-20250322230249

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 (275) hide show
  1. package/build/builders/buildCount.d.ts +23 -0
  2. package/build/builders/buildCount.js +63 -0
  3. package/build/builders/buildCount.test.d.ts +1 -0
  4. package/build/builders/buildCount.test.js +144 -0
  5. package/build/builders/buildCreate.d.ts +5 -0
  6. package/build/builders/buildCreate.js +28 -0
  7. package/build/builders/buildCreate.test.d.ts +1 -0
  8. package/build/builders/buildCreate.test.js +47 -0
  9. package/build/builders/buildDelete.d.ts +5 -0
  10. package/build/builders/buildDelete.js +28 -0
  11. package/build/builders/buildDelete.test.d.ts +1 -0
  12. package/build/builders/buildDelete.test.js +108 -0
  13. package/build/builders/buildFind.d.ts +8 -0
  14. package/build/builders/buildFind.js +185 -0
  15. package/build/builders/buildFind.test.d.ts +1 -0
  16. package/build/builders/buildFind.test.js +263 -0
  17. package/build/builders/buildUpdate.d.ts +5 -0
  18. package/build/builders/buildUpdate.js +34 -0
  19. package/build/builders/buildUpdate.test.d.ts +1 -0
  20. package/build/builders/buildUpdate.test.js +130 -0
  21. package/build/builders/buildWhere.d.ts +6 -0
  22. package/build/builders/buildWhere.js +63 -0
  23. package/build/builders/buildWhere.test.d.ts +1 -0
  24. package/build/builders/buildWhere.test.js +154 -0
  25. package/build/builders/types.d.ts +87 -0
  26. package/build/builders/types.js +1 -0
  27. package/build/connection.d.ts +31 -0
  28. package/build/connection.js +206 -0
  29. package/build/index.d.ts +10 -0
  30. package/build/index.js +5 -0
  31. package/build/operators.d.ts +59 -0
  32. package/build/operators.js +44 -0
  33. package/build/orm.count.d.ts +14 -0
  34. package/build/orm.count.js +22 -0
  35. package/build/orm.createMany.d.ts +5 -0
  36. package/build/orm.createMany.js +26 -0
  37. package/build/orm.createOne.d.ts +5 -0
  38. package/build/orm.createOne.js +34 -0
  39. package/build/orm.d.ts +81 -0
  40. package/build/orm.deleteMany.d.ts +5 -0
  41. package/build/orm.deleteMany.js +26 -0
  42. package/build/orm.deleteOne.d.ts +5 -0
  43. package/build/orm.deleteOne.js +32 -0
  44. package/build/orm.findMany.d.ts +8 -0
  45. package/build/orm.findMany.js +64 -0
  46. package/build/orm.findOne.d.ts +5 -0
  47. package/build/orm.findOne.js +20 -0
  48. package/build/orm.js +243 -0
  49. package/build/orm.restrict.d.ts +6 -0
  50. package/build/orm.restrict.js +52 -0
  51. package/build/orm.updateMany.d.ts +5 -0
  52. package/build/orm.updateMany.js +29 -0
  53. package/build/orm.updateOne.d.ts +5 -0
  54. package/build/orm.updateOne.js +35 -0
  55. package/build/sql/countToSql.d.ts +3 -0
  56. package/build/sql/countToSql.js +11 -0
  57. package/build/sql/countToSql.test.d.ts +1 -0
  58. package/build/sql/countToSql.test.js +218 -0
  59. package/build/sql/createToSql.d.ts +3 -0
  60. package/build/sql/createToSql.js +27 -0
  61. package/build/sql/createToSql.test.d.ts +1 -0
  62. package/build/sql/createToSql.test.js +186 -0
  63. package/build/sql/deleteToSql.d.ts +3 -0
  64. package/build/sql/deleteToSql.js +15 -0
  65. package/build/sql/deleteToSql.test.d.ts +1 -0
  66. package/build/sql/deleteToSql.test.js +93 -0
  67. package/build/sql/findToSql.d.ts +3 -0
  68. package/build/sql/findToSql.js +33 -0
  69. package/build/sql/findToSql.test.d.ts +1 -0
  70. package/build/sql/findToSql.test.js +409 -0
  71. package/build/sql/updateToSql.d.ts +3 -0
  72. package/build/sql/updateToSql.js +16 -0
  73. package/build/sql/updateToSql.test.d.ts +1 -0
  74. package/build/sql/updateToSql.test.js +165 -0
  75. package/build/sql/util.d.ts +11 -0
  76. package/build/sql/util.js +36 -0
  77. package/build/sql/util.test.d.ts +1 -0
  78. package/build/sql/util.test.js +163 -0
  79. package/build/tests/connection.test.d.ts +1 -0
  80. package/build/tests/connection.test.js +304 -0
  81. package/build/tests/datatypes.test.d.ts +1 -0
  82. package/build/tests/datatypes.test.js +239 -0
  83. package/build/tests/operators.test.d.ts +1 -0
  84. package/build/tests/operators.test.js +125 -0
  85. package/build/tests/orm.count.middleware.test.d.ts +1 -0
  86. package/build/tests/orm.count.middleware.test.js +132 -0
  87. package/build/tests/orm.count.test-d.d.ts +1 -0
  88. package/build/tests/orm.count.test-d.js +60 -0
  89. package/build/tests/orm.count.test.d.ts +1 -0
  90. package/build/tests/orm.count.test.js +151 -0
  91. package/build/tests/orm.createMany.middleware.test.d.ts +1 -0
  92. package/build/tests/orm.createMany.middleware.test.js +63 -0
  93. package/build/tests/orm.createMany.test-d.d.ts +1 -0
  94. package/build/tests/orm.createMany.test-d.js +131 -0
  95. package/build/tests/orm.createMany.test.d.ts +1 -0
  96. package/build/tests/orm.createMany.test.js +392 -0
  97. package/build/tests/orm.createOne.middleware.test.d.ts +1 -0
  98. package/build/tests/orm.createOne.middleware.test.js +54 -0
  99. package/build/tests/orm.createOne.test-d.d.ts +1 -0
  100. package/build/tests/orm.createOne.test-d.js +113 -0
  101. package/build/tests/orm.createOne.test.d.ts +1 -0
  102. package/build/tests/orm.createOne.test.js +268 -0
  103. package/build/tests/orm.deleteMany.middleware.test.d.ts +1 -0
  104. package/build/tests/orm.deleteMany.middleware.test.js +77 -0
  105. package/build/tests/orm.deleteMany.test-d.d.ts +1 -0
  106. package/build/tests/orm.deleteMany.test-d.js +179 -0
  107. package/build/tests/orm.deleteMany.test.d.ts +1 -0
  108. package/build/tests/orm.deleteMany.test.js +394 -0
  109. package/build/tests/orm.deleteOne.middleware.test.d.ts +1 -0
  110. package/build/tests/orm.deleteOne.middleware.test.js +61 -0
  111. package/build/tests/orm.deleteOne.test-d.d.ts +1 -0
  112. package/build/tests/orm.deleteOne.test-d.js +179 -0
  113. package/build/tests/orm.deleteOne.test.d.ts +1 -0
  114. package/build/tests/orm.deleteOne.test.js +360 -0
  115. package/build/tests/orm.findMany.includeManyToMany.test.d.ts +1 -0
  116. package/build/tests/orm.findMany.includeManyToMany.test.js +335 -0
  117. package/build/tests/orm.findMany.includeManyToOne.test.d.ts +1 -0
  118. package/build/tests/orm.findMany.includeManyToOne.test.js +286 -0
  119. package/build/tests/orm.findMany.includeOneToMany.test.d.ts +1 -0
  120. package/build/tests/orm.findMany.includeOneToMany.test.js +530 -0
  121. package/build/tests/orm.findMany.middleware.test.d.ts +1 -0
  122. package/build/tests/orm.findMany.middleware.test.js +66 -0
  123. package/build/tests/orm.findMany.offsetLimit.test.d.ts +1 -0
  124. package/build/tests/orm.findMany.offsetLimit.test.js +108 -0
  125. package/build/tests/orm.findMany.orderBy.test.d.ts +1 -0
  126. package/build/tests/orm.findMany.orderBy.test.js +304 -0
  127. package/build/tests/orm.findMany.select.test.d.ts +1 -0
  128. package/build/tests/orm.findMany.select.test.js +278 -0
  129. package/build/tests/orm.findMany.test-d.d.ts +1 -0
  130. package/build/tests/orm.findMany.test-d.js +374 -0
  131. package/build/tests/orm.findMany.where.test.d.ts +1 -0
  132. package/build/tests/orm.findMany.where.test.js +383 -0
  133. package/build/tests/orm.findOne.middleware.test.d.ts +1 -0
  134. package/build/tests/orm.findOne.middleware.test.js +57 -0
  135. package/build/tests/orm.findOne.test-d.d.ts +1 -0
  136. package/build/tests/orm.findOne.test-d.js +377 -0
  137. package/build/tests/orm.findOne.test.d.ts +1 -0
  138. package/build/tests/orm.findOne.test.js +247 -0
  139. package/build/tests/orm.restrict.test-d.d.ts +1 -0
  140. package/build/tests/orm.restrict.test-d.js +105 -0
  141. package/build/tests/orm.restrict.test.d.ts +1 -0
  142. package/build/tests/orm.restrict.test.js +259 -0
  143. package/build/tests/orm.transact.test.d.ts +1 -0
  144. package/build/tests/orm.transact.test.js +48 -0
  145. package/build/tests/orm.updateMany.middleware.test.d.ts +1 -0
  146. package/build/tests/orm.updateMany.middleware.test.js +72 -0
  147. package/build/tests/orm.updateMany.test.d.ts +1 -0
  148. package/build/tests/orm.updateMany.test.js +210 -0
  149. package/build/tests/orm.updateOne.middleware.test.d.ts +1 -0
  150. package/build/tests/orm.updateOne.middleware.test.js +62 -0
  151. package/build/tests/orm.updateOne.test.d.ts +1 -0
  152. package/build/tests/orm.updateOne.test.js +209 -0
  153. package/build/tests/util/db.d.ts +1571 -0
  154. package/build/tests/util/db.js +10 -0
  155. package/build/tests/util/logger.d.ts +19 -0
  156. package/build/tests/util/logger.js +40 -0
  157. package/build/types/BaseFindParams.d.ts +1 -0
  158. package/build/types/BaseFindParams.js +1 -0
  159. package/build/types/CountParams.d.ts +16 -0
  160. package/build/types/CountParams.js +1 -0
  161. package/build/types/CountParams.test-d.d.ts +1 -0
  162. package/build/types/CountParams.test-d.js +89 -0
  163. package/build/types/CreateManyParams.d.ts +11 -0
  164. package/build/types/CreateManyParams.js +1 -0
  165. package/build/types/CreateManyParams.test-d.d.ts +1 -0
  166. package/build/types/CreateManyParams.test-d.js +83 -0
  167. package/build/types/CreateManyResult.d.ts +5 -0
  168. package/build/types/CreateManyResult.js +1 -0
  169. package/build/types/CreateManyResult.test-d.d.ts +1 -0
  170. package/build/types/CreateManyResult.test-d.js +22 -0
  171. package/build/types/CreateOneParams.d.ts +11 -0
  172. package/build/types/CreateOneParams.js +1 -0
  173. package/build/types/CreateOneParams.test-d.d.ts +1 -0
  174. package/build/types/CreateOneParams.test-d.js +56 -0
  175. package/build/types/CreateOneResult.d.ts +5 -0
  176. package/build/types/CreateOneResult.js +1 -0
  177. package/build/types/CreateOneResult.test-d.d.ts +1 -0
  178. package/build/types/CreateOneResult.test-d.js +23 -0
  179. package/build/types/CreateValues.d.ts +6 -0
  180. package/build/types/CreateValues.js +1 -0
  181. package/build/types/CreateValues.test-d.d.ts +1 -0
  182. package/build/types/CreateValues.test-d.js +60 -0
  183. package/build/types/DeleteManyResult.d.ts +5 -0
  184. package/build/types/DeleteManyResult.js +1 -0
  185. package/build/types/DeleteManyResult.test-d.d.ts +1 -0
  186. package/build/types/DeleteManyResult.test-d.js +23 -0
  187. package/build/types/DeleteOneResult.d.ts +5 -0
  188. package/build/types/DeleteOneResult.js +1 -0
  189. package/build/types/DeleteOneResult.test-d.d.ts +1 -0
  190. package/build/types/DeleteOneResult.test-d.js +23 -0
  191. package/build/types/DeleteParams.d.ts +7 -0
  192. package/build/types/DeleteParams.js +1 -0
  193. package/build/types/DeleteParams.test-d.d.ts +1 -0
  194. package/build/types/DeleteParams.test-d.js +46 -0
  195. package/build/types/FindParams.d.ts +22 -0
  196. package/build/types/FindParams.js +1 -0
  197. package/build/types/FindParams.test-d.d.ts +1 -0
  198. package/build/types/FindParams.test-d.js +107 -0
  199. package/build/types/FindResult.d.ts +13 -0
  200. package/build/types/FindResult.js +1 -0
  201. package/build/types/FindResult.test-d.d.ts +1 -0
  202. package/build/types/FindResult.test-d.js +30 -0
  203. package/build/types/IncludeClause.d.ts +5 -0
  204. package/build/types/IncludeClause.js +1 -0
  205. package/build/types/IncludeClause.test-d.d.ts +1 -0
  206. package/build/types/IncludeClause.test-d.js +69 -0
  207. package/build/types/Middleware.d.ts +51 -0
  208. package/build/types/Middleware.js +37 -0
  209. package/build/types/OptionalValues.d.ts +4 -0
  210. package/build/types/OptionalValues.js +1 -0
  211. package/build/types/OptionalValues.test-d.d.ts +1 -0
  212. package/build/types/OptionalValues.test-d.js +12 -0
  213. package/build/types/OrderByClause.d.ts +9 -0
  214. package/build/types/OrderByClause.js +1 -0
  215. package/build/types/OrderByClause.test-d.d.ts +1 -0
  216. package/build/types/OrderByClause.test-d.js +44 -0
  217. package/build/types/RequiredValues.d.ts +4 -0
  218. package/build/types/RequiredValues.js +1 -0
  219. package/build/types/RequiredValues.test-d.d.ts +1 -0
  220. package/build/types/RequiredValues.test-d.js +22 -0
  221. package/build/types/RestrictModels.d.ts +23 -0
  222. package/build/types/RestrictModels.js +1 -0
  223. package/build/types/RestrictModels.test-d.d.ts +1 -0
  224. package/build/types/RestrictModels.test-d.js +44 -0
  225. package/build/types/ReturningClause.d.ts +3 -0
  226. package/build/types/ReturningClause.js +1 -0
  227. package/build/types/ReturningClause.test-d.d.ts +1 -0
  228. package/build/types/ReturningClause.test-d.js +15 -0
  229. package/build/types/SelectClause.d.ts +3 -0
  230. package/build/types/SelectClause.js +1 -0
  231. package/build/types/SelectClause.test-d.d.ts +1 -0
  232. package/build/types/SelectClause.test-d.js +15 -0
  233. package/build/types/UpdateManyResult.d.ts +5 -0
  234. package/build/types/UpdateManyResult.js +1 -0
  235. package/build/types/UpdateManyResult.test-d.d.ts +1 -0
  236. package/build/types/UpdateManyResult.test-d.js +15 -0
  237. package/build/types/UpdateOneResult.d.ts +5 -0
  238. package/build/types/UpdateOneResult.js +1 -0
  239. package/build/types/UpdateOneResult.test-d.d.ts +1 -0
  240. package/build/types/UpdateOneResult.test-d.js +15 -0
  241. package/build/types/UpdateParams.d.ts +9 -0
  242. package/build/types/UpdateParams.js +1 -0
  243. package/build/types/UpdateParams.test-d.d.ts +1 -0
  244. package/build/types/UpdateParams.test-d.js +29 -0
  245. package/build/types/UpdateValues.d.ts +6 -0
  246. package/build/types/UpdateValues.js +1 -0
  247. package/build/types/UpdateValues.test-d.d.ts +1 -0
  248. package/build/types/UpdateValues.test-d.js +33 -0
  249. package/build/types/WhereClause.d.ts +29 -0
  250. package/build/types/WhereClause.js +1 -0
  251. package/build/types/WhereClause.test-d.d.ts +1 -0
  252. package/build/types/WhereClause.test-d.js +137 -0
  253. package/build/util/getIncludedToManySubqueries.d.ts +12 -0
  254. package/build/util/getIncludedToManySubqueries.js +63 -0
  255. package/build/util/getIncludedToManySubqueries.test.d.ts +1 -0
  256. package/build/util/getIncludedToManySubqueries.test.js +121 -0
  257. package/build/util/getLateralJoinValues.d.ts +4 -0
  258. package/build/util/getLateralJoinValues.js +30 -0
  259. package/build/util/getLateralJoinValues.test.d.ts +1 -0
  260. package/build/util/getLateralJoinValues.test.js +99 -0
  261. package/build/util/hasClauses.d.ts +1 -0
  262. package/build/util/hasClauses.js +9 -0
  263. package/build/util/hasClauses.test.d.ts +1 -0
  264. package/build/util/hasClauses.test.js +15 -0
  265. package/build/util/makeTableAlias.d.ts +1 -0
  266. package/build/util/makeTableAlias.js +10 -0
  267. package/build/util/makeTableAlias.test.d.ts +1 -0
  268. package/build/util/makeTableAlias.test.js +18 -0
  269. package/build/util/resultSchema.d.ts +10 -0
  270. package/build/util/resultSchema.js +42 -0
  271. package/build/util/rowToObject.d.ts +8 -0
  272. package/build/util/rowToObject.js +10 -0
  273. package/build/util/rowToObject.test.d.ts +1 -0
  274. package/build/util/rowToObject.test.js +68 -0
  275. package/package.json +67 -0
@@ -0,0 +1,377 @@
1
+ import { describe, expectTypeOf, test } from "vitest";
2
+ import { $and, $gt, $is, $like, $or } from "../operators.js";
3
+ import { createTestDB } from "./util/db.js";
4
+ describe("orm.findOne", () => {
5
+ const { db } = createTestDB();
6
+ test("handles basic select fields", () => {
7
+ expectTypeOf(db.findOne("post", {
8
+ select: ["id", "title", "content"],
9
+ })).resolves.toEqualTypeOf();
10
+ });
11
+ test("handles array fields in select and where clauses", async () => {
12
+ expectTypeOf(db.findOne("post", {
13
+ select: ["tags"],
14
+ where: {
15
+ tags: ["news", "tech"],
16
+ },
17
+ })).resolves.toEqualTypeOf();
18
+ await db.findOne("post", {
19
+ select: ["id"],
20
+ where: {
21
+ // @ts-expect-error tags must be string array
22
+ tags: "not-an-array",
23
+ },
24
+ });
25
+ });
26
+ test("handles enum fields in select and where clauses", async () => {
27
+ expectTypeOf(db.findOne("user", {
28
+ select: ["role"],
29
+ where: {
30
+ role: "admin",
31
+ },
32
+ })).resolves.toEqualTypeOf();
33
+ await db.findOne("user", {
34
+ select: ["id"],
35
+ where: {
36
+ // @ts-expect-error invalid enum value
37
+ role: "superadmin",
38
+ },
39
+ });
40
+ });
41
+ test("handles JSON fields in select and where clauses", async () => {
42
+ expectTypeOf(db.findOne("post", {
43
+ select: ["metadata"],
44
+ where: {
45
+ metadata: {
46
+ foo: "a",
47
+ bar: [
48
+ {
49
+ baz: "good",
50
+ quux: true,
51
+ },
52
+ ],
53
+ },
54
+ },
55
+ })).resolves.toEqualTypeOf();
56
+ await db.findOne("post", {
57
+ select: ["id"],
58
+ where: {
59
+ metadata: {
60
+ // @ts-expect-error invalid JSON structure
61
+ foo: "invalid",
62
+ bar: [
63
+ {
64
+ // @ts-expect-error invalid JSON structure
65
+ baz: "invalid",
66
+ // @ts-expect-error invalid JSON structure
67
+ quux: "not-a-boolean",
68
+ },
69
+ ],
70
+ },
71
+ },
72
+ });
73
+ });
74
+ test("handles deeply nested where clauses with logical operators", () => {
75
+ expectTypeOf(db.findOne("post", {
76
+ select: ["id"],
77
+ where: {
78
+ [$and]: [
79
+ {
80
+ [$or]: [
81
+ {
82
+ [$and]: [
83
+ {
84
+ [$or]: [
85
+ {
86
+ [$and]: [
87
+ {
88
+ [$or]: [
89
+ {
90
+ [$and]: [
91
+ {
92
+ [$or]: [
93
+ {
94
+ [$and]: [
95
+ {
96
+ [$or]: [
97
+ {
98
+ [$and]: [
99
+ {
100
+ [$or]: [
101
+ {
102
+ [$and]: [
103
+ {
104
+ [$or]: [
105
+ {
106
+ [$and]: [
107
+ {
108
+ [$or]: [
109
+ {
110
+ title: {
111
+ [$like]: "test%",
112
+ },
113
+ id: {
114
+ [$gt]: 100,
115
+ },
116
+ },
117
+ ],
118
+ },
119
+ ],
120
+ },
121
+ ],
122
+ },
123
+ ],
124
+ },
125
+ ],
126
+ },
127
+ ],
128
+ },
129
+ ],
130
+ },
131
+ ],
132
+ },
133
+ ],
134
+ },
135
+ ],
136
+ },
137
+ ],
138
+ },
139
+ ],
140
+ },
141
+ ],
142
+ },
143
+ ],
144
+ },
145
+ ],
146
+ },
147
+ ],
148
+ },
149
+ })).resolves.toEqualTypeOf();
150
+ });
151
+ test("handles deeply nested includes", async () => {
152
+ const result = await db.findOne("user", {
153
+ select: ["id"],
154
+ include: {
155
+ // Level 1: user -> posts
156
+ posts: {
157
+ select: ["id"],
158
+ include: {
159
+ // Level 2: posts -> author
160
+ author: {
161
+ select: ["id"],
162
+ include: {
163
+ // Level 3: author -> posts
164
+ posts: {
165
+ select: ["id"],
166
+ include: {
167
+ // Level 4: posts -> author
168
+ author: {
169
+ select: ["id"],
170
+ include: {
171
+ // Level 5: author -> posts
172
+ posts: {
173
+ select: ["id"],
174
+ include: {
175
+ // Level 6: posts -> author
176
+ author: {
177
+ select: ["id"],
178
+ include: {
179
+ // Level 7: author -> posts
180
+ posts: {
181
+ select: [
182
+ "id",
183
+ ],
184
+ include: {
185
+ // Level 8: posts -> author
186
+ author: {
187
+ select: [
188
+ "id",
189
+ ],
190
+ include: {
191
+ // Level 9: author -> posts
192
+ posts: {
193
+ select: [
194
+ "id",
195
+ ],
196
+ include: {
197
+ // Level 10: posts -> author
198
+ author: {
199
+ select: [
200
+ "id",
201
+ ],
202
+ include: {
203
+ // Level 11: author -> posts
204
+ posts: {
205
+ select: [
206
+ "id",
207
+ ],
208
+ include: {
209
+ // Level 12: posts -> author
210
+ author: {
211
+ select: [
212
+ "id",
213
+ ],
214
+ include: {
215
+ // Level 13: author -> friends and author -> posts
216
+ friends: {
217
+ select: [
218
+ "id",
219
+ "name",
220
+ ],
221
+ include: {
222
+ posts: {
223
+ select: [
224
+ "id",
225
+ "title",
226
+ "tags",
227
+ ],
228
+ include: {
229
+ // Level 14: posts -> backgroundColor
230
+ backgroundColor: {
231
+ select: [
232
+ "hex",
233
+ "name",
234
+ ],
235
+ },
236
+ },
237
+ },
238
+ },
239
+ },
240
+ posts: {
241
+ select: [
242
+ "id",
243
+ "title",
244
+ "tags",
245
+ ],
246
+ include: {
247
+ // Level 14: posts -> backgroundColor
248
+ backgroundColor: {
249
+ select: [
250
+ "hex",
251
+ "name",
252
+ ],
253
+ },
254
+ },
255
+ },
256
+ },
257
+ },
258
+ },
259
+ },
260
+ },
261
+ },
262
+ },
263
+ },
264
+ },
265
+ },
266
+ },
267
+ },
268
+ },
269
+ },
270
+ },
271
+ },
272
+ },
273
+ },
274
+ },
275
+ },
276
+ },
277
+ },
278
+ },
279
+ },
280
+ },
281
+ });
282
+ const z = result.posts[0].author.posts[0].author.posts[0].author.posts[0]
283
+ .author.posts[0].author.posts[0].author.friends[0].posts[0]
284
+ .backgroundColor?.hex;
285
+ expectTypeOf(result.posts[0].author.posts[0].author.posts[0].author.posts[0]
286
+ .author.posts[0].author.posts[0].author.friends[0].posts[0]
287
+ .backgroundColor).toEqualTypeOf();
288
+ expectTypeOf(result.posts[0].author.posts[0].author.posts[0].author.posts[0]
289
+ .author.posts[0].author.posts[0].author.friends[0]).toEqualTypeOf();
290
+ });
291
+ test("handles complex combinations of fields, operators and includes", () => {
292
+ expectTypeOf(db.findOne("post", {
293
+ select: ["id", "title", "content", "tags", "metadata"],
294
+ where: {
295
+ [$or]: [
296
+ {
297
+ title: { [$like]: "test%" },
298
+ tags: ["news"],
299
+ metadata: {
300
+ foo: "a",
301
+ bar: [{ baz: "good", quux: true }],
302
+ },
303
+ },
304
+ {
305
+ publishedAt: { [$is]: null },
306
+ authorId: { [$gt]: 100 },
307
+ },
308
+ ],
309
+ },
310
+ include: {
311
+ author: {
312
+ select: ["id", "role"],
313
+ include: {
314
+ posts: {
315
+ select: ["id"],
316
+ },
317
+ },
318
+ },
319
+ backgroundColor: {
320
+ select: ["hex", "name"],
321
+ },
322
+ },
323
+ })).resolves.toEqualTypeOf();
324
+ });
325
+ test("enforces select clause is required", async () => {
326
+ // @ts-expect-error select is required
327
+ await db.findOne("post", {
328
+ where: {
329
+ id: 1,
330
+ },
331
+ });
332
+ });
333
+ test("prevents invalid field names in select clause", async () => {
334
+ await db.findOne("post", {
335
+ // @ts-expect-error invalid field name
336
+ select: ["invalid"],
337
+ where: {
338
+ id: 1,
339
+ },
340
+ });
341
+ });
342
+ test("prevents invalid field names in where clause", async () => {
343
+ await db.findOne("post", {
344
+ select: ["id"],
345
+ where: {
346
+ // @ts-expect-error invalid field name
347
+ invalid: 1,
348
+ },
349
+ });
350
+ });
351
+ test("prevents invalid model names", async () => {
352
+ // @ts-expect-error invalid model name
353
+ await db.findOne("invalid", {
354
+ select: ["id"],
355
+ });
356
+ });
357
+ test("prevents invalid relation names in include", async () => {
358
+ await db.findOne("post", {
359
+ select: ["id"],
360
+ include: {
361
+ // @ts-expect-error invalid relation name
362
+ invalid: {
363
+ select: ["id"],
364
+ },
365
+ },
366
+ });
367
+ });
368
+ test("prevents invalid operator usage", async () => {
369
+ await db.findOne("post", {
370
+ select: ["id"],
371
+ where: {
372
+ // @ts-expect-error $like cannot be used with number
373
+ id: { [$like]: 100 },
374
+ },
375
+ });
376
+ });
377
+ });
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,247 @@
1
+ import { afterAll, beforeAll, beforeEach, describe, expect, test, } from "vitest";
2
+ import { config } from "@casekit/orm2-fixtures";
3
+ import { $like } from "../operators.js";
4
+ import { orm } from "../orm.js";
5
+ import { mockLogger } from "./util/logger.js";
6
+ describe("findOne", () => {
7
+ const logger = mockLogger();
8
+ let db;
9
+ beforeEach(() => {
10
+ logger.clear();
11
+ });
12
+ beforeAll(async () => {
13
+ db = orm({ ...config, logger });
14
+ await db.connect();
15
+ });
16
+ afterAll(async () => {
17
+ await db.close();
18
+ });
19
+ test("finds a single record with basic select", async () => {
20
+ await db.transact(async (db) => {
21
+ const user = await db.createOne("user", {
22
+ values: {
23
+ id: 1,
24
+ name: "Test User",
25
+ email: "test@example.com",
26
+ role: "user",
27
+ },
28
+ returning: ["id"],
29
+ });
30
+ const foundUser = await db.findOne("user", {
31
+ select: ["id", "name", "email"],
32
+ where: { id: user.id },
33
+ });
34
+ expect(foundUser).toEqual({
35
+ id: 1,
36
+ name: "Test User",
37
+ email: "test@example.com",
38
+ });
39
+ }, { rollback: true });
40
+ });
41
+ test("finds a record with N:1 relation", async () => {
42
+ await db.transact(async (db) => {
43
+ const user = await db.createOne("user", {
44
+ values: {
45
+ id: 1,
46
+ name: "Test User",
47
+ email: "test@example.com",
48
+ role: "user",
49
+ },
50
+ returning: ["id"],
51
+ });
52
+ await db.createOne("post", {
53
+ values: {
54
+ title: "Test Post",
55
+ content: "Test Content",
56
+ authorId: user.id,
57
+ },
58
+ returning: ["id"],
59
+ });
60
+ const post = await db.findOne("post", {
61
+ select: ["id", "title"],
62
+ include: {
63
+ author: {
64
+ select: ["id", "name"],
65
+ },
66
+ },
67
+ where: { authorId: user.id },
68
+ });
69
+ expect(post).toEqual({
70
+ id: expect.any(Number),
71
+ title: "Test Post",
72
+ author: {
73
+ id: 1,
74
+ name: "Test User",
75
+ },
76
+ });
77
+ }, { rollback: true });
78
+ });
79
+ test("finds a record with optional N:1 relation", async () => {
80
+ await db.transact(async (db) => {
81
+ const user = await db.createOne("user", {
82
+ values: {
83
+ id: 1,
84
+ name: "Stewart Home",
85
+ email: "stewart@example.com",
86
+ role: "user",
87
+ },
88
+ returning: ["id"],
89
+ });
90
+ const color = await db.createOne("color", {
91
+ values: {
92
+ name: "Red",
93
+ hex: "#FF0000",
94
+ },
95
+ returning: ["hex"],
96
+ });
97
+ const posts = await db.createMany("post", {
98
+ values: [
99
+ {
100
+ title: "Test Post",
101
+ content: "Test Content",
102
+ authorId: user.id,
103
+ },
104
+ {
105
+ title: "Test Post 2",
106
+ content: "Test Content 2",
107
+ authorId: user.id,
108
+ backgroundColorValue: color.hex,
109
+ },
110
+ ],
111
+ returning: ["id"],
112
+ });
113
+ const postWithoutBgColor = await db.findOne("post", {
114
+ select: ["id", "title"],
115
+ include: {
116
+ backgroundColor: {
117
+ select: ["hex", "name"],
118
+ },
119
+ },
120
+ where: { id: posts[0].id },
121
+ });
122
+ expect(postWithoutBgColor).toEqual({
123
+ id: expect.any(Number),
124
+ title: "Test Post",
125
+ backgroundColor: null,
126
+ });
127
+ const postWithBgColor = await db.findOne("post", {
128
+ select: ["id", "title"],
129
+ include: {
130
+ backgroundColor: {
131
+ select: ["hex", "name"],
132
+ },
133
+ },
134
+ where: { id: posts[1].id },
135
+ });
136
+ expect(postWithBgColor).toEqual({
137
+ id: expect.any(Number),
138
+ title: "Test Post 2",
139
+ backgroundColor: { hex: "#FF0000", name: "Red" },
140
+ });
141
+ }, { rollback: true });
142
+ });
143
+ test("throws error when no records match criteria", async () => {
144
+ await db.transact(async (db) => {
145
+ await expect(db.findOne("user", {
146
+ select: ["id"],
147
+ where: { id: 999999 },
148
+ })).rejects.toThrow("Expected one row, but found none");
149
+ }, { rollback: true });
150
+ });
151
+ test("throws error when multiple records match criteria", async () => {
152
+ await db.transact(async (db) => {
153
+ // Create multiple users with similar names
154
+ await db.createMany("user", {
155
+ values: [
156
+ {
157
+ id: 1,
158
+ name: "John Doe",
159
+ email: "john1@example.com",
160
+ role: "user",
161
+ },
162
+ {
163
+ id: 2,
164
+ name: "John Doe",
165
+ email: "john2@example.com",
166
+ role: "user",
167
+ },
168
+ ],
169
+ });
170
+ await expect(db.findOne("user", {
171
+ select: ["id", "name"],
172
+ where: {
173
+ name: { [$like]: "John%" },
174
+ },
175
+ })).rejects.toThrow("Expected one row, but found more");
176
+ }, { rollback: true });
177
+ });
178
+ test("handles complex where clauses with array and JSON fields", async () => {
179
+ await db.transact(async (db) => {
180
+ const user = await db.createOne("user", {
181
+ values: {
182
+ id: 1,
183
+ name: "Test User",
184
+ email: "test@example.com",
185
+ role: "user",
186
+ },
187
+ returning: ["id"],
188
+ });
189
+ await db.createOne("post", {
190
+ values: {
191
+ title: "Test Post",
192
+ content: "Test Content",
193
+ authorId: user.id,
194
+ tags: ["test", "example"],
195
+ metadata: {
196
+ foo: "a",
197
+ bar: [{ baz: "good", quux: true }],
198
+ },
199
+ },
200
+ });
201
+ const post = await db.findOne("post", {
202
+ select: ["id", "title", "tags", "metadata"],
203
+ where: {
204
+ tags: ["test", "example"],
205
+ metadata: {
206
+ foo: "a",
207
+ bar: [{ baz: "good", quux: true }],
208
+ },
209
+ },
210
+ });
211
+ expect(post).toEqual({
212
+ id: expect.any(Number),
213
+ title: "Test Post",
214
+ tags: ["test", "example"],
215
+ metadata: {
216
+ foo: "a",
217
+ bar: [{ baz: "good", quux: true }],
218
+ },
219
+ });
220
+ }, { rollback: true });
221
+ });
222
+ test("respects enum field types", async () => {
223
+ await db.transact(async (db) => {
224
+ await db.createOne("user", {
225
+ values: {
226
+ id: 1,
227
+ name: "Test Admin",
228
+ email: "admin@example.com",
229
+ role: "admin",
230
+ },
231
+ });
232
+ const admin = await db.findOne("user", {
233
+ select: ["id", "name", "role"],
234
+ where: { role: "admin" },
235
+ });
236
+ expect(admin.role).toBe("admin");
237
+ // TypeScript should prevent this, but we test at runtime too
238
+ await expect(db.findOne("user", {
239
+ select: ["id"],
240
+ where: {
241
+ // @ts-expect-error - invalid role
242
+ role: "superadmin",
243
+ },
244
+ })).rejects.toThrow();
245
+ }, { rollback: true });
246
+ });
247
+ });
@@ -0,0 +1 @@
1
+ export {};