@strapi/database 5.12.1 → 5.12.2

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 (321) hide show
  1. package/dist/connection.js +43 -0
  2. package/dist/connection.js.map +1 -0
  3. package/dist/connection.mjs +41 -0
  4. package/dist/connection.mjs.map +1 -0
  5. package/dist/dialects/dialect.js +54 -0
  6. package/dist/dialects/dialect.js.map +1 -0
  7. package/dist/dialects/dialect.mjs +52 -0
  8. package/dist/dialects/dialect.mjs.map +1 -0
  9. package/dist/dialects/index.js +44 -0
  10. package/dist/dialects/index.js.map +1 -0
  11. package/dist/dialects/index.mjs +42 -0
  12. package/dist/dialects/index.mjs.map +1 -0
  13. package/dist/dialects/mysql/constants.js +8 -0
  14. package/dist/dialects/mysql/constants.js.map +1 -0
  15. package/dist/dialects/mysql/constants.mjs +5 -0
  16. package/dist/dialects/mysql/constants.mjs.map +1 -0
  17. package/dist/dialects/mysql/database-inspector.js +35 -0
  18. package/dist/dialects/mysql/database-inspector.js.map +1 -0
  19. package/dist/dialects/mysql/database-inspector.mjs +33 -0
  20. package/dist/dialects/mysql/database-inspector.mjs.map +1 -0
  21. package/dist/dialects/mysql/index.js +75 -0
  22. package/dist/dialects/mysql/index.js.map +1 -0
  23. package/dist/dialects/mysql/index.mjs +73 -0
  24. package/dist/dialects/mysql/index.mjs.map +1 -0
  25. package/dist/dialects/mysql/schema-inspector.js +297 -0
  26. package/dist/dialects/mysql/schema-inspector.js.map +1 -0
  27. package/dist/dialects/mysql/schema-inspector.mjs +295 -0
  28. package/dist/dialects/mysql/schema-inspector.mjs.map +1 -0
  29. package/dist/dialects/postgresql/index.js +62 -0
  30. package/dist/dialects/postgresql/index.js.map +1 -0
  31. package/dist/dialects/postgresql/index.mjs +60 -0
  32. package/dist/dialects/postgresql/index.mjs.map +1 -0
  33. package/dist/dialects/postgresql/schema-inspector.js +316 -0
  34. package/dist/dialects/postgresql/schema-inspector.js.map +1 -0
  35. package/dist/dialects/postgresql/schema-inspector.mjs +314 -0
  36. package/dist/dialects/postgresql/schema-inspector.mjs.map +1 -0
  37. package/dist/dialects/sqlite/index.js +82 -0
  38. package/dist/dialects/sqlite/index.js.map +1 -0
  39. package/dist/dialects/sqlite/index.mjs +80 -0
  40. package/dist/dialects/sqlite/index.mjs.map +1 -0
  41. package/dist/dialects/sqlite/schema-inspector.js +211 -0
  42. package/dist/dialects/sqlite/schema-inspector.js.map +1 -0
  43. package/dist/dialects/sqlite/schema-inspector.mjs +209 -0
  44. package/dist/dialects/sqlite/schema-inspector.mjs.map +1 -0
  45. package/dist/entity-manager/entity-repository.js +139 -0
  46. package/dist/entity-manager/entity-repository.js.map +1 -0
  47. package/dist/entity-manager/entity-repository.mjs +137 -0
  48. package/dist/entity-manager/entity-repository.mjs.map +1 -0
  49. package/dist/entity-manager/index.js +1186 -0
  50. package/dist/entity-manager/index.js.map +1 -0
  51. package/dist/entity-manager/index.mjs +1184 -0
  52. package/dist/entity-manager/index.mjs.map +1 -0
  53. package/dist/entity-manager/morph-relations.js +73 -0
  54. package/dist/entity-manager/morph-relations.js.map +1 -0
  55. package/dist/entity-manager/morph-relations.mjs +69 -0
  56. package/dist/entity-manager/morph-relations.mjs.map +1 -0
  57. package/dist/entity-manager/regular-relations.js +247 -0
  58. package/dist/entity-manager/regular-relations.js.map +1 -0
  59. package/dist/entity-manager/regular-relations.mjs +242 -0
  60. package/dist/entity-manager/regular-relations.mjs.map +1 -0
  61. package/dist/entity-manager/relations-orderer.js +221 -0
  62. package/dist/entity-manager/relations-orderer.js.map +1 -0
  63. package/dist/entity-manager/relations-orderer.mjs +218 -0
  64. package/dist/entity-manager/relations-orderer.mjs.map +1 -0
  65. package/dist/errors/database.js +13 -0
  66. package/dist/errors/database.js.map +1 -0
  67. package/dist/errors/database.mjs +11 -0
  68. package/dist/errors/database.mjs.map +1 -0
  69. package/dist/errors/index.js +18 -0
  70. package/dist/errors/index.js.map +1 -0
  71. package/dist/errors/index.mjs +7 -0
  72. package/dist/errors/index.mjs.map +1 -0
  73. package/dist/errors/invalid-date.js +13 -0
  74. package/dist/errors/invalid-date.js.map +1 -0
  75. package/dist/errors/invalid-date.mjs +11 -0
  76. package/dist/errors/invalid-date.mjs.map +1 -0
  77. package/dist/errors/invalid-datetime.js +13 -0
  78. package/dist/errors/invalid-datetime.js.map +1 -0
  79. package/dist/errors/invalid-datetime.mjs +11 -0
  80. package/dist/errors/invalid-datetime.mjs.map +1 -0
  81. package/dist/errors/invalid-relation.js +13 -0
  82. package/dist/errors/invalid-relation.js.map +1 -0
  83. package/dist/errors/invalid-relation.mjs +11 -0
  84. package/dist/errors/invalid-relation.mjs.map +1 -0
  85. package/dist/errors/invalid-time.js +13 -0
  86. package/dist/errors/invalid-time.js.map +1 -0
  87. package/dist/errors/invalid-time.mjs +11 -0
  88. package/dist/errors/invalid-time.mjs.map +1 -0
  89. package/dist/errors/not-null.js +17 -0
  90. package/dist/errors/not-null.js.map +1 -0
  91. package/dist/errors/not-null.mjs +15 -0
  92. package/dist/errors/not-null.mjs.map +1 -0
  93. package/dist/fields/biginteger.js +9 -0
  94. package/dist/fields/biginteger.js.map +1 -0
  95. package/dist/fields/biginteger.mjs +7 -0
  96. package/dist/fields/biginteger.mjs.map +1 -0
  97. package/dist/fields/boolean.js +48 -0
  98. package/dist/fields/boolean.js.map +1 -0
  99. package/dist/fields/boolean.mjs +46 -0
  100. package/dist/fields/boolean.mjs.map +1 -0
  101. package/dist/fields/date.js +16 -0
  102. package/dist/fields/date.js.map +1 -0
  103. package/dist/fields/date.mjs +14 -0
  104. package/dist/fields/date.mjs.map +1 -0
  105. package/dist/fields/datetime.js +37 -0
  106. package/dist/fields/datetime.js.map +1 -0
  107. package/dist/fields/datetime.mjs +16 -0
  108. package/dist/fields/datetime.mjs.map +1 -0
  109. package/dist/fields/field.js +16 -0
  110. package/dist/fields/field.js.map +1 -0
  111. package/dist/fields/field.mjs +14 -0
  112. package/dist/fields/field.mjs.map +1 -0
  113. package/dist/fields/index.js +45 -0
  114. package/dist/fields/index.js.map +1 -0
  115. package/dist/fields/index.mjs +43 -0
  116. package/dist/fields/index.mjs.map +1 -0
  117. package/dist/fields/json.js +36 -0
  118. package/dist/fields/json.js.map +1 -0
  119. package/dist/fields/json.mjs +34 -0
  120. package/dist/fields/json.mjs.map +1 -0
  121. package/dist/fields/number.js +20 -0
  122. package/dist/fields/number.js.map +1 -0
  123. package/dist/fields/number.mjs +18 -0
  124. package/dist/fields/number.mjs.map +1 -0
  125. package/dist/fields/shared/parsers.js +91 -0
  126. package/dist/fields/shared/parsers.js.map +1 -0
  127. package/dist/fields/shared/parsers.mjs +68 -0
  128. package/dist/fields/shared/parsers.mjs.map +1 -0
  129. package/dist/fields/string.js +16 -0
  130. package/dist/fields/string.js.map +1 -0
  131. package/dist/fields/string.mjs +14 -0
  132. package/dist/fields/string.mjs.map +1 -0
  133. package/dist/fields/time.js +17 -0
  134. package/dist/fields/time.js.map +1 -0
  135. package/dist/fields/time.mjs +15 -0
  136. package/dist/fields/time.mjs.map +1 -0
  137. package/dist/fields/timestamp.js +37 -0
  138. package/dist/fields/timestamp.js.map +1 -0
  139. package/dist/fields/timestamp.mjs +16 -0
  140. package/dist/fields/timestamp.mjs.map +1 -0
  141. package/dist/index.js +33 -8569
  142. package/dist/index.js.map +1 -1
  143. package/dist/index.mjs +16 -8532
  144. package/dist/index.mjs.map +1 -1
  145. package/dist/lifecycles/index.js +73 -0
  146. package/dist/lifecycles/index.js.map +1 -0
  147. package/dist/lifecycles/index.mjs +71 -0
  148. package/dist/lifecycles/index.mjs.map +1 -0
  149. package/dist/lifecycles/subscribers/index.js +10 -0
  150. package/dist/lifecycles/subscribers/index.js.map +1 -0
  151. package/dist/lifecycles/subscribers/index.mjs +8 -0
  152. package/dist/lifecycles/subscribers/index.mjs.map +1 -0
  153. package/dist/lifecycles/subscribers/models-lifecycles.js +13 -0
  154. package/dist/lifecycles/subscribers/models-lifecycles.js.map +1 -0
  155. package/dist/lifecycles/subscribers/models-lifecycles.mjs +11 -0
  156. package/dist/lifecycles/subscribers/models-lifecycles.mjs.map +1 -0
  157. package/dist/lifecycles/subscribers/timestamps.js +55 -0
  158. package/dist/lifecycles/subscribers/timestamps.js.map +1 -0
  159. package/dist/lifecycles/subscribers/timestamps.mjs +53 -0
  160. package/dist/lifecycles/subscribers/timestamps.mjs.map +1 -0
  161. package/dist/metadata/index.js +24 -0
  162. package/dist/metadata/index.js.map +1 -0
  163. package/dist/metadata/index.mjs +16 -0
  164. package/dist/metadata/index.mjs.map +1 -0
  165. package/dist/metadata/metadata.js +100 -0
  166. package/dist/metadata/metadata.js.map +1 -0
  167. package/dist/metadata/metadata.mjs +98 -0
  168. package/dist/metadata/metadata.mjs.map +1 -0
  169. package/dist/metadata/relations.js +545 -0
  170. package/dist/metadata/relations.js.map +1 -0
  171. package/dist/metadata/relations.mjs +536 -0
  172. package/dist/metadata/relations.mjs.map +1 -0
  173. package/dist/migrations/common.js +8 -0
  174. package/dist/migrations/common.js.map +1 -0
  175. package/dist/migrations/common.mjs +6 -0
  176. package/dist/migrations/common.mjs.map +1 -0
  177. package/dist/migrations/index.js +39 -0
  178. package/dist/migrations/index.js.map +1 -0
  179. package/dist/migrations/index.mjs +37 -0
  180. package/dist/migrations/index.mjs.map +1 -0
  181. package/dist/migrations/internal-migrations/5.0.0-01-convert-identifiers-long-than-max-length.js +179 -0
  182. package/dist/migrations/internal-migrations/5.0.0-01-convert-identifiers-long-than-max-length.js.map +1 -0
  183. package/dist/migrations/internal-migrations/5.0.0-01-convert-identifiers-long-than-max-length.mjs +177 -0
  184. package/dist/migrations/internal-migrations/5.0.0-01-convert-identifiers-long-than-max-length.mjs.map +1 -0
  185. package/dist/migrations/internal-migrations/5.0.0-02-document-id.js +125 -0
  186. package/dist/migrations/internal-migrations/5.0.0-02-document-id.js.map +1 -0
  187. package/dist/migrations/internal-migrations/5.0.0-02-document-id.mjs +123 -0
  188. package/dist/migrations/internal-migrations/5.0.0-02-document-id.mjs.map +1 -0
  189. package/dist/migrations/internal-migrations/5.0.0-03-locale.js +41 -0
  190. package/dist/migrations/internal-migrations/5.0.0-03-locale.js.map +1 -0
  191. package/dist/migrations/internal-migrations/5.0.0-03-locale.mjs +39 -0
  192. package/dist/migrations/internal-migrations/5.0.0-03-locale.mjs.map +1 -0
  193. package/dist/migrations/internal-migrations/5.0.0-04-published-at.js +45 -0
  194. package/dist/migrations/internal-migrations/5.0.0-04-published-at.js.map +1 -0
  195. package/dist/migrations/internal-migrations/5.0.0-04-published-at.mjs +43 -0
  196. package/dist/migrations/internal-migrations/5.0.0-04-published-at.mjs.map +1 -0
  197. package/dist/migrations/internal-migrations/5.0.0-05-drop-slug-unique-index.js +43 -0
  198. package/dist/migrations/internal-migrations/5.0.0-05-drop-slug-unique-index.js.map +1 -0
  199. package/dist/migrations/internal-migrations/5.0.0-05-drop-slug-unique-index.mjs +41 -0
  200. package/dist/migrations/internal-migrations/5.0.0-05-drop-slug-unique-index.mjs.map +1 -0
  201. package/dist/migrations/internal-migrations/index.js +26 -0
  202. package/dist/migrations/internal-migrations/index.js.map +1 -0
  203. package/dist/migrations/internal-migrations/index.mjs +24 -0
  204. package/dist/migrations/internal-migrations/index.mjs.map +1 -0
  205. package/dist/migrations/internal.js +63 -0
  206. package/dist/migrations/internal.js.map +1 -0
  207. package/dist/migrations/internal.mjs +61 -0
  208. package/dist/migrations/internal.mjs.map +1 -0
  209. package/dist/migrations/logger.js +24 -0
  210. package/dist/migrations/logger.js.map +1 -0
  211. package/dist/migrations/logger.mjs +22 -0
  212. package/dist/migrations/logger.mjs.map +1 -0
  213. package/dist/migrations/storage.js +39 -0
  214. package/dist/migrations/storage.js.map +1 -0
  215. package/dist/migrations/storage.mjs +37 -0
  216. package/dist/migrations/storage.mjs.map +1 -0
  217. package/dist/migrations/users.js +87 -0
  218. package/dist/migrations/users.js.map +1 -0
  219. package/dist/migrations/users.mjs +85 -0
  220. package/dist/migrations/users.mjs.map +1 -0
  221. package/dist/query/helpers/join.js +127 -0
  222. package/dist/query/helpers/join.js.map +1 -0
  223. package/dist/query/helpers/join.mjs +122 -0
  224. package/dist/query/helpers/join.mjs.map +1 -0
  225. package/dist/query/helpers/order-by.js +167 -0
  226. package/dist/query/helpers/order-by.js.map +1 -0
  227. package/dist/query/helpers/order-by.mjs +163 -0
  228. package/dist/query/helpers/order-by.mjs.map +1 -0
  229. package/dist/query/helpers/populate/apply.js +592 -0
  230. package/dist/query/helpers/populate/apply.js.map +1 -0
  231. package/dist/query/helpers/populate/apply.mjs +590 -0
  232. package/dist/query/helpers/populate/apply.mjs.map +1 -0
  233. package/dist/query/helpers/populate/process.js +92 -0
  234. package/dist/query/helpers/populate/process.js.map +1 -0
  235. package/dist/query/helpers/populate/process.mjs +90 -0
  236. package/dist/query/helpers/populate/process.mjs.map +1 -0
  237. package/dist/query/helpers/search.js +67 -0
  238. package/dist/query/helpers/search.js.map +1 -0
  239. package/dist/query/helpers/search.mjs +65 -0
  240. package/dist/query/helpers/search.mjs.map +1 -0
  241. package/dist/query/helpers/streams/readable.js +131 -0
  242. package/dist/query/helpers/streams/readable.js.map +1 -0
  243. package/dist/query/helpers/streams/readable.mjs +129 -0
  244. package/dist/query/helpers/streams/readable.mjs.map +1 -0
  245. package/dist/query/helpers/transform.js +77 -0
  246. package/dist/query/helpers/transform.js.map +1 -0
  247. package/dist/query/helpers/transform.mjs +73 -0
  248. package/dist/query/helpers/transform.mjs.map +1 -0
  249. package/dist/query/helpers/where.js +372 -0
  250. package/dist/query/helpers/where.js.map +1 -0
  251. package/dist/query/helpers/where.mjs +369 -0
  252. package/dist/query/helpers/where.mjs.map +1 -0
  253. package/dist/query/query-builder.js +507 -0
  254. package/dist/query/query-builder.js.map +1 -0
  255. package/dist/query/query-builder.mjs +505 -0
  256. package/dist/query/query-builder.mjs.map +1 -0
  257. package/dist/repairs/index.js +13 -0
  258. package/dist/repairs/index.js.map +1 -0
  259. package/dist/repairs/index.mjs +11 -0
  260. package/dist/repairs/index.mjs.map +1 -0
  261. package/dist/repairs/operations/remove-orphan-morph-types.js +54 -0
  262. package/dist/repairs/operations/remove-orphan-morph-types.js.map +1 -0
  263. package/dist/repairs/operations/remove-orphan-morph-types.mjs +52 -0
  264. package/dist/repairs/operations/remove-orphan-morph-types.mjs.map +1 -0
  265. package/dist/schema/builder.js +354 -0
  266. package/dist/schema/builder.js.map +1 -0
  267. package/dist/schema/builder.mjs +352 -0
  268. package/dist/schema/builder.mjs.map +1 -0
  269. package/dist/schema/diff.js +379 -0
  270. package/dist/schema/diff.js.map +1 -0
  271. package/dist/schema/diff.mjs +377 -0
  272. package/dist/schema/diff.mjs.map +1 -0
  273. package/dist/schema/index.js +93 -0
  274. package/dist/schema/index.js.map +1 -0
  275. package/dist/schema/index.mjs +91 -0
  276. package/dist/schema/index.mjs.map +1 -0
  277. package/dist/schema/schema.js +266 -0
  278. package/dist/schema/schema.js.map +1 -0
  279. package/dist/schema/schema.mjs +264 -0
  280. package/dist/schema/schema.mjs.map +1 -0
  281. package/dist/schema/storage.js +58 -0
  282. package/dist/schema/storage.js.map +1 -0
  283. package/dist/schema/storage.mjs +56 -0
  284. package/dist/schema/storage.mjs.map +1 -0
  285. package/dist/transaction-context.js +65 -0
  286. package/dist/transaction-context.js.map +1 -0
  287. package/dist/transaction-context.mjs +63 -0
  288. package/dist/transaction-context.mjs.map +1 -0
  289. package/dist/utils/async-curry.js +19 -0
  290. package/dist/utils/async-curry.js.map +1 -0
  291. package/dist/utils/async-curry.mjs +17 -0
  292. package/dist/utils/async-curry.mjs.map +1 -0
  293. package/dist/utils/identifiers/hash.js +30 -0
  294. package/dist/utils/identifiers/hash.js.map +1 -0
  295. package/dist/utils/identifiers/hash.mjs +28 -0
  296. package/dist/utils/identifiers/hash.mjs.map +1 -0
  297. package/dist/utils/identifiers/index.js +414 -0
  298. package/dist/utils/identifiers/index.js.map +1 -0
  299. package/dist/utils/identifiers/index.mjs +411 -0
  300. package/dist/utils/identifiers/index.mjs.map +1 -0
  301. package/dist/utils/knex.js +21 -0
  302. package/dist/utils/knex.js.map +1 -0
  303. package/dist/utils/knex.mjs +18 -0
  304. package/dist/utils/knex.mjs.map +1 -0
  305. package/dist/utils/types.js +51 -0
  306. package/dist/utils/types.js.map +1 -0
  307. package/dist/utils/types.mjs +44 -0
  308. package/dist/utils/types.mjs.map +1 -0
  309. package/dist/validations/index.js +12 -0
  310. package/dist/validations/index.js.map +1 -0
  311. package/dist/validations/index.mjs +10 -0
  312. package/dist/validations/index.mjs.map +1 -0
  313. package/dist/validations/relations/bidirectional.js +64 -0
  314. package/dist/validations/relations/bidirectional.js.map +1 -0
  315. package/dist/validations/relations/bidirectional.mjs +62 -0
  316. package/dist/validations/relations/bidirectional.mjs.map +1 -0
  317. package/dist/validations/relations/index.js +13 -0
  318. package/dist/validations/relations/index.js.map +1 -0
  319. package/dist/validations/relations/index.mjs +11 -0
  320. package/dist/validations/relations/index.mjs.map +1 -0
  321. package/package.json +4 -4
@@ -0,0 +1,122 @@
1
+ const createPivotJoin = (ctx, { alias, refAlias, joinTable, targetMeta })=>{
2
+ const { qb } = ctx;
3
+ const joinAlias = qb.getAlias();
4
+ qb.join({
5
+ alias: joinAlias,
6
+ referencedTable: joinTable.name,
7
+ referencedColumn: joinTable.joinColumn.name,
8
+ rootColumn: joinTable.joinColumn.referencedColumn,
9
+ rootTable: alias,
10
+ on: joinTable.on
11
+ });
12
+ const subAlias = refAlias || qb.getAlias();
13
+ qb.join({
14
+ alias: subAlias,
15
+ referencedTable: targetMeta.tableName,
16
+ referencedColumn: joinTable.inverseJoinColumn.referencedColumn,
17
+ rootColumn: joinTable.inverseJoinColumn.name,
18
+ rootTable: joinAlias
19
+ });
20
+ return subAlias;
21
+ };
22
+ const createJoin = (ctx, { alias, refAlias, attributeName, attribute })=>{
23
+ const { db, qb, uid } = ctx;
24
+ if (attribute.type !== 'relation') {
25
+ throw new Error(`Cannot join on non relational field ${attributeName}`);
26
+ }
27
+ const targetMeta = db.metadata.get(attribute.target);
28
+ if ([
29
+ 'morphOne',
30
+ 'morphMany'
31
+ ].includes(attribute.relation)) {
32
+ const targetAttribute = targetMeta.attributes[attribute.morphBy];
33
+ // @ts-expect-error - morphBy is not defined on the attribute
34
+ const { joinTable, morphColumn } = targetAttribute;
35
+ if (morphColumn) {
36
+ const subAlias = refAlias || qb.getAlias();
37
+ qb.join({
38
+ alias: subAlias,
39
+ referencedTable: targetMeta.tableName,
40
+ referencedColumn: morphColumn.idColumn.name,
41
+ rootColumn: morphColumn.idColumn.referencedColumn,
42
+ rootTable: alias,
43
+ on: {
44
+ [morphColumn.typeColumn.name]: uid,
45
+ ...morphColumn.on
46
+ }
47
+ });
48
+ return subAlias;
49
+ }
50
+ if (joinTable) {
51
+ const joinAlias = qb.getAlias();
52
+ qb.join({
53
+ alias: joinAlias,
54
+ referencedTable: joinTable.name,
55
+ referencedColumn: joinTable.morphColumn.idColumn.name,
56
+ rootColumn: joinTable.morphColumn.idColumn.referencedColumn,
57
+ rootTable: alias,
58
+ on: {
59
+ [joinTable.morphColumn.typeColumn.name]: uid,
60
+ field: attributeName
61
+ }
62
+ });
63
+ const subAlias = refAlias || qb.getAlias();
64
+ qb.join({
65
+ alias: subAlias,
66
+ referencedTable: targetMeta.tableName,
67
+ referencedColumn: joinTable.joinColumn.referencedColumn,
68
+ rootColumn: joinTable.joinColumn.name,
69
+ rootTable: joinAlias
70
+ });
71
+ return subAlias;
72
+ }
73
+ return alias;
74
+ }
75
+ const { joinColumn } = attribute;
76
+ if (joinColumn) {
77
+ const subAlias = refAlias || qb.getAlias();
78
+ qb.join({
79
+ alias: subAlias,
80
+ referencedTable: targetMeta.tableName,
81
+ referencedColumn: joinColumn.referencedColumn,
82
+ rootColumn: joinColumn.name,
83
+ rootTable: alias
84
+ });
85
+ return subAlias;
86
+ }
87
+ const { joinTable } = attribute;
88
+ if (joinTable) {
89
+ return createPivotJoin(ctx, {
90
+ alias,
91
+ refAlias,
92
+ joinTable,
93
+ targetMeta
94
+ });
95
+ }
96
+ return alias;
97
+ };
98
+ // TODO: toColumnName for orderBy & on
99
+ const applyJoin = (qb, join)=>{
100
+ const { method = 'leftJoin', alias, referencedTable, referencedColumn, rootColumn, // FIXME: qb.alias can't exist here
101
+ rootTable, on, orderBy } = join;
102
+ qb[method](`${referencedTable} as ${alias}`, (inner)=>{
103
+ inner.on(`${rootTable}.${rootColumn}`, `${alias}.${referencedColumn}`);
104
+ if (on) {
105
+ for (const key of Object.keys(on)){
106
+ inner.onVal(`${alias}.${key}`, on[key]);
107
+ }
108
+ }
109
+ });
110
+ if (orderBy) {
111
+ Object.keys(orderBy).forEach((column)=>{
112
+ const direction = orderBy[column];
113
+ qb.orderBy(`${alias}.${column}`, direction);
114
+ });
115
+ }
116
+ };
117
+ const applyJoins = (qb, joins)=>{
118
+ return joins.forEach((join)=>applyJoin(qb, join));
119
+ };
120
+
121
+ export { applyJoin, applyJoins, createJoin, createPivotJoin };
122
+ //# sourceMappingURL=join.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"join.mjs","sources":["../../../src/query/helpers/join.ts"],"sourcesContent":["import type { Knex } from 'knex';\nimport type { Ctx } from '../types';\n\nexport interface Join {\n method?: 'leftJoin' | 'innerJoin';\n alias: string;\n referencedTable: string;\n referencedColumn: string;\n rootColumn: string;\n rootTable?: string;\n on?: Record<string, any>;\n orderBy?: Record<string, 'asc' | 'desc'>;\n}\n\ninterface JoinOptions {\n alias: string;\n refAlias?: string;\n attributeName: string;\n attribute: any;\n}\n\ninterface PivotJoinOptions {\n alias: string;\n refAlias?: string;\n joinTable: any;\n targetMeta: any;\n}\n\nconst createPivotJoin = (\n ctx: Ctx,\n { alias, refAlias, joinTable, targetMeta }: PivotJoinOptions\n) => {\n const { qb } = ctx;\n const joinAlias = qb.getAlias();\n qb.join({\n alias: joinAlias,\n referencedTable: joinTable.name,\n referencedColumn: joinTable.joinColumn.name,\n rootColumn: joinTable.joinColumn.referencedColumn,\n rootTable: alias,\n on: joinTable.on,\n });\n\n const subAlias = refAlias || qb.getAlias();\n qb.join({\n alias: subAlias,\n referencedTable: targetMeta.tableName,\n referencedColumn: joinTable.inverseJoinColumn.referencedColumn,\n rootColumn: joinTable.inverseJoinColumn.name,\n rootTable: joinAlias,\n });\n\n return subAlias;\n};\n\nconst createJoin = (ctx: Ctx, { alias, refAlias, attributeName, attribute }: JoinOptions) => {\n const { db, qb, uid } = ctx;\n\n if (attribute.type !== 'relation') {\n throw new Error(`Cannot join on non relational field ${attributeName}`);\n }\n\n const targetMeta = db.metadata.get(attribute.target);\n\n if (['morphOne', 'morphMany'].includes(attribute.relation)) {\n const targetAttribute = targetMeta.attributes[attribute.morphBy];\n\n // @ts-expect-error - morphBy is not defined on the attribute\n const { joinTable, morphColumn } = targetAttribute;\n\n if (morphColumn) {\n const subAlias = refAlias || qb.getAlias();\n\n qb.join({\n alias: subAlias,\n referencedTable: targetMeta.tableName,\n referencedColumn: morphColumn.idColumn.name,\n rootColumn: morphColumn.idColumn.referencedColumn,\n rootTable: alias,\n on: {\n [morphColumn.typeColumn.name]: uid,\n ...morphColumn.on,\n },\n });\n\n return subAlias;\n }\n\n if (joinTable) {\n const joinAlias = qb.getAlias();\n\n qb.join({\n alias: joinAlias,\n referencedTable: joinTable.name,\n referencedColumn: joinTable.morphColumn.idColumn.name,\n rootColumn: joinTable.morphColumn.idColumn.referencedColumn,\n rootTable: alias,\n on: {\n [joinTable.morphColumn.typeColumn.name]: uid,\n field: attributeName,\n },\n });\n\n const subAlias = refAlias || qb.getAlias();\n\n qb.join({\n alias: subAlias,\n referencedTable: targetMeta.tableName,\n referencedColumn: joinTable.joinColumn.referencedColumn,\n rootColumn: joinTable.joinColumn.name,\n rootTable: joinAlias,\n });\n\n return subAlias;\n }\n\n return alias;\n }\n\n const { joinColumn } = attribute;\n\n if (joinColumn) {\n const subAlias = refAlias || qb.getAlias();\n qb.join({\n alias: subAlias,\n referencedTable: targetMeta.tableName,\n referencedColumn: joinColumn.referencedColumn,\n rootColumn: joinColumn.name,\n rootTable: alias,\n });\n return subAlias;\n }\n\n const { joinTable } = attribute;\n if (joinTable) {\n return createPivotJoin(ctx, { alias, refAlias, joinTable, targetMeta });\n }\n\n return alias;\n};\n\n// TODO: toColumnName for orderBy & on\nconst applyJoin = (qb: Knex.QueryBuilder, join: Join) => {\n const {\n method = 'leftJoin',\n alias,\n referencedTable,\n referencedColumn,\n rootColumn,\n // FIXME: qb.alias can't exist here\n rootTable, // = qb.alias\n on,\n orderBy,\n } = join;\n\n qb[method](`${referencedTable} as ${alias}`, (inner) => {\n inner.on(`${rootTable}.${rootColumn}`, `${alias}.${referencedColumn}`);\n\n if (on) {\n for (const key of Object.keys(on)) {\n inner.onVal(`${alias}.${key}`, on[key]);\n }\n }\n });\n\n if (orderBy) {\n Object.keys(orderBy).forEach((column) => {\n const direction = orderBy[column];\n qb.orderBy(`${alias}.${column}`, direction);\n });\n }\n};\n\nconst applyJoins = (qb: Knex.QueryBuilder, joins: Join[]) => {\n return joins.forEach((join) => applyJoin(qb, join));\n};\n\nexport { createJoin, createPivotJoin, applyJoins, applyJoin };\n"],"names":["createPivotJoin","ctx","alias","refAlias","joinTable","targetMeta","qb","joinAlias","getAlias","join","referencedTable","name","referencedColumn","joinColumn","rootColumn","rootTable","on","subAlias","tableName","inverseJoinColumn","createJoin","attributeName","attribute","db","uid","type","Error","metadata","get","target","includes","relation","targetAttribute","attributes","morphBy","morphColumn","idColumn","typeColumn","field","applyJoin","method","orderBy","inner","key","Object","keys","onVal","forEach","column","direction","applyJoins","joins"],"mappings":"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"}
@@ -0,0 +1,167 @@
1
+ 'use strict';
2
+
3
+ var _ = require('lodash/fp');
4
+ var types = require('../../utils/types.js');
5
+ var join = require('./join.js');
6
+ var transform = require('./transform.js');
7
+
8
+ const COL_STRAPI_ROW_NUMBER = '__strapi_row_number';
9
+ const COL_STRAPI_ORDER_BY_PREFIX = '__strapi_order_by';
10
+ const processOrderBy = (orderBy, ctx)=>{
11
+ const { db, uid, qb, alias } = ctx;
12
+ const meta = db.metadata.get(uid);
13
+ const { attributes } = meta;
14
+ if (typeof orderBy === 'string') {
15
+ const attribute = attributes[orderBy];
16
+ if (!attribute) {
17
+ throw new Error(`Attribute ${orderBy} not found on model ${uid}`);
18
+ }
19
+ const columnName = transform.toColumnName(meta, orderBy);
20
+ return [
21
+ {
22
+ column: qb.aliasColumn(columnName, alias)
23
+ }
24
+ ];
25
+ }
26
+ if (Array.isArray(orderBy)) {
27
+ return orderBy.flatMap((value)=>processOrderBy(value, ctx));
28
+ }
29
+ if (_.isPlainObject(orderBy)) {
30
+ return Object.entries(orderBy).flatMap(([key, direction])=>{
31
+ const value = orderBy[key];
32
+ const attribute = attributes[key];
33
+ if (!attribute) {
34
+ throw new Error(`Attribute ${key} not found on model ${uid}`);
35
+ }
36
+ if (types.isScalar(attribute.type)) {
37
+ const columnName = transform.toColumnName(meta, key);
38
+ return {
39
+ column: qb.aliasColumn(columnName, alias),
40
+ order: direction
41
+ };
42
+ }
43
+ if (attribute.type === 'relation' && 'target' in attribute) {
44
+ const subAlias = join.createJoin(ctx, {
45
+ alias: alias || qb.alias,
46
+ attributeName: key,
47
+ attribute
48
+ });
49
+ return processOrderBy(value, {
50
+ db,
51
+ qb,
52
+ alias: subAlias,
53
+ uid: attribute.target
54
+ });
55
+ }
56
+ throw new Error(`You cannot order on ${attribute.type} types`);
57
+ });
58
+ }
59
+ throw new Error('Invalid orderBy syntax');
60
+ };
61
+ const getStrapiOrderColumnAlias = (column)=>{
62
+ const trimmedColumnName = column.replaceAll('.', '_');
63
+ return `${COL_STRAPI_ORDER_BY_PREFIX}__${trimmedColumnName}`;
64
+ };
65
+ /**
66
+ * Wraps the original Knex query with deep sorting functionality.
67
+ *
68
+ * The function takes an original query and an OrderByCtx object as parameters and returns a new Knex query with deep sorting applied.
69
+ */ const wrapWithDeepSort = (originalQuery, ctx)=>{
70
+ /**
71
+ * Notes:
72
+ * - The generated query has the following flow: baseQuery (filtered unsorted data) -> T (partitioned/sorted data) --> resultQuery (distinct, paginated, sorted data)
73
+ * - Pagination and selection are transferred from the original query to the outer one to avoid pruning rows too early
74
+ * - Filtering (where) has to be done in the deepest sub query possible to avoid processing invalid rows and corrupting the final results
75
+ * - We assume that all necessary joins are done in the original query (`originalQuery`), and every needed column is available with the right name and alias.
76
+ */ const { db, qb, uid } = ctx;
77
+ const { tableName } = db.metadata.get(uid);
78
+ // The orderBy is cloned to avoid unwanted mutations of the original object
79
+ const orderBy = _.cloneDeep(qb.state.orderBy);
80
+ // 0. Init a new Knex query instance (referenced as resultQuery) using the DB connection
81
+ // The connection reuse the original table name (aliased if needed)
82
+ const resultQueryAlias = qb.getAlias();
83
+ const aliasedTableName = qb.mustUseAlias() ? alias(resultQueryAlias, tableName) : tableName;
84
+ const resultQuery = db.getConnection(aliasedTableName);
85
+ // 1. Clone the original query to create the sub-query (referenced as baseQuery) and avoid any mutation on the initial object
86
+ const baseQuery = originalQuery.clone();
87
+ const baseQueryAlias = qb.getAlias();
88
+ // Clear unwanted statements from the sub-query 'baseQuery'
89
+ // Note: `first()` is cleared through the combination of `baseQuery.clear('limit')` and calling `baseQuery.select(...)` again
90
+ // Note: Those statements will be re-applied when duplicates are removed from the final selection
91
+ baseQuery// Columns selection
92
+ .clear('select')// Pagination and sorting
93
+ .clear('order').clear('limit').clear('offset');
94
+ // Override the initial select and return only the columns needed for the partitioning.
95
+ baseQuery.select(// Always select the row id for future manipulation
96
+ prefix(qb.alias, 'id'), // Select every column used in an order by clause, but alias it for future reference
97
+ // i.e. if t2.name is present in an order by clause:
98
+ // Then, "t2.name" will become "t2.name as __strapi_order_by__t2_name"
99
+ ...orderBy.map((orderByClause)=>alias(getStrapiOrderColumnAlias(orderByClause.column), orderByClause.column)));
100
+ // 2. Create a sub-query callback to extract and sort the partitions using row number
101
+ const partitionedQueryAlias = qb.getAlias();
102
+ const selectRowsAsNumberedPartitions = (partitionedQuery)=>{
103
+ // Transform order by clause to their alias to reference them from baseQuery
104
+ const prefixedOrderBy = orderBy.map((orderByClause)=>({
105
+ column: prefix(baseQueryAlias, getStrapiOrderColumnAlias(orderByClause.column)),
106
+ order: orderByClause.order
107
+ }));
108
+ // partitionedQuery select must contain every column used for sorting
109
+ const orderByColumns = prefixedOrderBy.map(_.prop('column'));
110
+ partitionedQuery.select(// Always select baseQuery.id
111
+ prefix(baseQueryAlias, 'id'), // Sort columns
112
+ ...orderByColumns)// The row number is used to assign an index to every row in every partition
113
+ .rowNumber(COL_STRAPI_ROW_NUMBER, (subQuery)=>{
114
+ for (const orderByClause of prefixedOrderBy){
115
+ subQuery.orderBy(orderByClause.column, orderByClause.order, 'last');
116
+ }
117
+ // And each partition/group is created based on baseQuery.id
118
+ subQuery.partitionBy(`${baseQueryAlias}.id`);
119
+ }).from(baseQuery.as(baseQueryAlias)).as(partitionedQueryAlias);
120
+ };
121
+ // 3. Create the final resultQuery query, that select and sort the wanted data using T
122
+ const originalSelect = _.difference(qb.state.select, // Remove order by columns from the initial select
123
+ qb.state.orderBy.map(_.prop('column')))// Alias everything in resultQuery
124
+ .map(prefix(resultQueryAlias));
125
+ resultQuery.select(originalSelect)// Join T to resultQuery to access sorted data
126
+ // Notes:
127
+ // - Only select the first row for each partition
128
+ // - Since we're applying the "where" statement directly on baseQuery (and not on resultQuery), we're using an inner join to avoid unwanted rows
129
+ .innerJoin(selectRowsAsNumberedPartitions, function() {
130
+ this// Only select rows that are returned by T
131
+ .on(`${partitionedQueryAlias}.id`, `${resultQueryAlias}.id`)// By only selecting the rows number equal to 1, we make sure we don't have duplicate, and that
132
+ // we're selecting rows in the correct order amongst the groups created by the "partition by"
133
+ .andOnVal(`${partitionedQueryAlias}.${COL_STRAPI_ROW_NUMBER}`, '=', 1);
134
+ });
135
+ // Re-apply pagination params
136
+ if (qb.state.limit) {
137
+ resultQuery.limit(qb.state.limit);
138
+ }
139
+ if (qb.state.offset) {
140
+ resultQuery.offset(qb.state.offset);
141
+ }
142
+ if (qb.state.first) {
143
+ resultQuery.first();
144
+ }
145
+ // Re-apply the sort using T values
146
+ resultQuery.orderBy([
147
+ // Transform "order by" clause to their T alias and prefix them with T alias
148
+ ...orderBy.map((orderByClause)=>({
149
+ column: prefix(partitionedQueryAlias, getStrapiOrderColumnAlias(orderByClause.column)),
150
+ order: orderByClause.order
151
+ })),
152
+ // Add T.id to the order by clause to get consistent results in case several rows have the exact same order
153
+ {
154
+ column: `${partitionedQueryAlias}.id`,
155
+ order: 'asc'
156
+ }
157
+ ]);
158
+ return resultQuery;
159
+ };
160
+ // Utils
161
+ const alias = _.curry((alias, value)=>`${value} as ${alias}`);
162
+ const prefix = _.curry((prefix, value)=>`${prefix}.${value}`);
163
+
164
+ exports.getStrapiOrderColumnAlias = getStrapiOrderColumnAlias;
165
+ exports.processOrderBy = processOrderBy;
166
+ exports.wrapWithDeepSort = wrapWithDeepSort;
167
+ //# sourceMappingURL=order-by.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"order-by.js","sources":["../../../src/query/helpers/order-by.ts"],"sourcesContent":["import _ from 'lodash/fp';\nimport knex from 'knex';\n\nimport * as types from '../../utils/types';\nimport { createJoin } from './join';\nimport { toColumnName } from './transform';\n\nimport type { Ctx } from '../types';\n\ntype OrderByCtx = Ctx & { alias?: string };\ntype OrderBy = string | { [key: string]: 'asc' | 'desc' } | OrderBy[];\ntype OrderByValue = { column: string; order?: 'asc' | 'desc' };\n\nconst COL_STRAPI_ROW_NUMBER = '__strapi_row_number';\nconst COL_STRAPI_ORDER_BY_PREFIX = '__strapi_order_by';\n\nexport const processOrderBy = (orderBy: OrderBy, ctx: OrderByCtx): OrderByValue[] => {\n const { db, uid, qb, alias } = ctx;\n const meta = db.metadata.get(uid);\n const { attributes } = meta;\n\n if (typeof orderBy === 'string') {\n const attribute = attributes[orderBy];\n\n if (!attribute) {\n throw new Error(`Attribute ${orderBy} not found on model ${uid}`);\n }\n\n const columnName = toColumnName(meta, orderBy);\n\n return [{ column: qb.aliasColumn(columnName, alias) }];\n }\n\n if (Array.isArray(orderBy)) {\n return orderBy.flatMap((value) => processOrderBy(value, ctx));\n }\n\n if (_.isPlainObject(orderBy)) {\n return Object.entries(orderBy).flatMap(([key, direction]) => {\n const value = orderBy[key];\n const attribute = attributes[key];\n\n if (!attribute) {\n throw new Error(`Attribute ${key} not found on model ${uid}`);\n }\n\n if (types.isScalar(attribute.type)) {\n const columnName = toColumnName(meta, key);\n\n return { column: qb.aliasColumn(columnName, alias), order: direction };\n }\n\n if (attribute.type === 'relation' && 'target' in attribute) {\n const subAlias = createJoin(ctx, {\n alias: alias || qb.alias,\n attributeName: key,\n attribute,\n });\n\n return processOrderBy(value, {\n db,\n qb,\n alias: subAlias,\n uid: attribute.target,\n });\n }\n\n throw new Error(`You cannot order on ${attribute.type} types`);\n });\n }\n\n throw new Error('Invalid orderBy syntax');\n};\n\nexport const getStrapiOrderColumnAlias = (column: string) => {\n const trimmedColumnName = column.replaceAll('.', '_');\n\n return `${COL_STRAPI_ORDER_BY_PREFIX}__${trimmedColumnName}`;\n};\n\n/**\n * Wraps the original Knex query with deep sorting functionality.\n *\n * The function takes an original query and an OrderByCtx object as parameters and returns a new Knex query with deep sorting applied.\n */\nexport const wrapWithDeepSort = (originalQuery: knex.Knex.QueryBuilder, ctx: OrderByCtx) => {\n /**\n * Notes:\n * - The generated query has the following flow: baseQuery (filtered unsorted data) -> T (partitioned/sorted data) --> resultQuery (distinct, paginated, sorted data)\n * - Pagination and selection are transferred from the original query to the outer one to avoid pruning rows too early\n * - Filtering (where) has to be done in the deepest sub query possible to avoid processing invalid rows and corrupting the final results\n * - We assume that all necessary joins are done in the original query (`originalQuery`), and every needed column is available with the right name and alias.\n */\n\n const { db, qb, uid } = ctx;\n\n const { tableName } = db.metadata.get(uid);\n\n // The orderBy is cloned to avoid unwanted mutations of the original object\n const orderBy = _.cloneDeep<OrderByValue[]>(qb.state.orderBy);\n\n // 0. Init a new Knex query instance (referenced as resultQuery) using the DB connection\n // The connection reuse the original table name (aliased if needed)\n const resultQueryAlias = qb.getAlias();\n const aliasedTableName = qb.mustUseAlias() ? alias(resultQueryAlias, tableName) : tableName;\n\n const resultQuery = db.getConnection(aliasedTableName);\n\n // 1. Clone the original query to create the sub-query (referenced as baseQuery) and avoid any mutation on the initial object\n const baseQuery = originalQuery.clone();\n const baseQueryAlias = qb.getAlias();\n\n // Clear unwanted statements from the sub-query 'baseQuery'\n // Note: `first()` is cleared through the combination of `baseQuery.clear('limit')` and calling `baseQuery.select(...)` again\n // Note: Those statements will be re-applied when duplicates are removed from the final selection\n baseQuery\n // Columns selection\n .clear('select')\n // Pagination and sorting\n .clear('order')\n .clear('limit')\n .clear('offset');\n\n // Override the initial select and return only the columns needed for the partitioning.\n baseQuery.select(\n // Always select the row id for future manipulation\n prefix(qb.alias, 'id'),\n // Select every column used in an order by clause, but alias it for future reference\n // i.e. if t2.name is present in an order by clause:\n // Then, \"t2.name\" will become \"t2.name as __strapi_order_by__t2_name\"\n ...orderBy.map((orderByClause) =>\n alias(getStrapiOrderColumnAlias(orderByClause.column), orderByClause.column)\n )\n );\n\n // 2. Create a sub-query callback to extract and sort the partitions using row number\n const partitionedQueryAlias = qb.getAlias();\n\n const selectRowsAsNumberedPartitions = (partitionedQuery: knex.Knex.QueryBuilder) => {\n // Transform order by clause to their alias to reference them from baseQuery\n const prefixedOrderBy = orderBy.map((orderByClause) => ({\n column: prefix(baseQueryAlias, getStrapiOrderColumnAlias(orderByClause.column)),\n order: orderByClause.order,\n }));\n\n // partitionedQuery select must contain every column used for sorting\n const orderByColumns = prefixedOrderBy.map<string>(_.prop('column'));\n\n partitionedQuery\n .select(\n // Always select baseQuery.id\n prefix(baseQueryAlias, 'id'),\n // Sort columns\n ...orderByColumns\n )\n // The row number is used to assign an index to every row in every partition\n .rowNumber(COL_STRAPI_ROW_NUMBER, (subQuery) => {\n for (const orderByClause of prefixedOrderBy) {\n subQuery.orderBy(orderByClause.column, orderByClause.order, 'last');\n }\n\n // And each partition/group is created based on baseQuery.id\n subQuery.partitionBy(`${baseQueryAlias}.id`);\n })\n .from(baseQuery.as(baseQueryAlias))\n .as(partitionedQueryAlias);\n };\n\n // 3. Create the final resultQuery query, that select and sort the wanted data using T\n\n const originalSelect = _.difference(\n qb.state.select,\n // Remove order by columns from the initial select\n qb.state.orderBy.map(_.prop('column'))\n )\n // Alias everything in resultQuery\n .map(prefix(resultQueryAlias));\n\n resultQuery\n .select(originalSelect)\n // Join T to resultQuery to access sorted data\n // Notes:\n // - Only select the first row for each partition\n // - Since we're applying the \"where\" statement directly on baseQuery (and not on resultQuery), we're using an inner join to avoid unwanted rows\n .innerJoin(selectRowsAsNumberedPartitions, function () {\n this\n // Only select rows that are returned by T\n .on(`${partitionedQueryAlias}.id`, `${resultQueryAlias}.id`)\n // By only selecting the rows number equal to 1, we make sure we don't have duplicate, and that\n // we're selecting rows in the correct order amongst the groups created by the \"partition by\"\n .andOnVal(`${partitionedQueryAlias}.${COL_STRAPI_ROW_NUMBER}`, '=', 1);\n });\n\n // Re-apply pagination params\n\n if (qb.state.limit) {\n resultQuery.limit(qb.state.limit);\n }\n\n if (qb.state.offset) {\n resultQuery.offset(qb.state.offset);\n }\n\n if (qb.state.first) {\n resultQuery.first();\n }\n\n // Re-apply the sort using T values\n resultQuery.orderBy([\n // Transform \"order by\" clause to their T alias and prefix them with T alias\n ...orderBy.map((orderByClause) => ({\n column: prefix(partitionedQueryAlias, getStrapiOrderColumnAlias(orderByClause.column)),\n order: orderByClause.order,\n })),\n // Add T.id to the order by clause to get consistent results in case several rows have the exact same order\n { column: `${partitionedQueryAlias}.id`, order: 'asc' },\n ]);\n\n return resultQuery;\n};\n\n// Utils\nconst alias = _.curry((alias: string, value: string) => `${value} as ${alias}`);\nconst prefix = _.curry((prefix: string, value: string) => `${prefix}.${value}`);\n"],"names":["COL_STRAPI_ROW_NUMBER","COL_STRAPI_ORDER_BY_PREFIX","processOrderBy","orderBy","ctx","db","uid","qb","alias","meta","metadata","get","attributes","attribute","Error","columnName","toColumnName","column","aliasColumn","Array","isArray","flatMap","value","_","isPlainObject","Object","entries","key","direction","types","type","order","subAlias","createJoin","attributeName","target","getStrapiOrderColumnAlias","trimmedColumnName","replaceAll","wrapWithDeepSort","originalQuery","tableName","cloneDeep","state","resultQueryAlias","getAlias","aliasedTableName","mustUseAlias","resultQuery","getConnection","baseQuery","clone","baseQueryAlias","clear","select","prefix","map","orderByClause","partitionedQueryAlias","selectRowsAsNumberedPartitions","partitionedQuery","prefixedOrderBy","orderByColumns","prop","rowNumber","subQuery","partitionBy","from","as","originalSelect","difference","innerJoin","on","andOnVal","limit","offset","first","curry"],"mappings":"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"}
@@ -0,0 +1,163 @@
1
+ import _ from 'lodash/fp';
2
+ import { isScalar } from '../../utils/types.mjs';
3
+ import { createJoin } from './join.mjs';
4
+ import { toColumnName } from './transform.mjs';
5
+
6
+ const COL_STRAPI_ROW_NUMBER = '__strapi_row_number';
7
+ const COL_STRAPI_ORDER_BY_PREFIX = '__strapi_order_by';
8
+ const processOrderBy = (orderBy, ctx)=>{
9
+ const { db, uid, qb, alias } = ctx;
10
+ const meta = db.metadata.get(uid);
11
+ const { attributes } = meta;
12
+ if (typeof orderBy === 'string') {
13
+ const attribute = attributes[orderBy];
14
+ if (!attribute) {
15
+ throw new Error(`Attribute ${orderBy} not found on model ${uid}`);
16
+ }
17
+ const columnName = toColumnName(meta, orderBy);
18
+ return [
19
+ {
20
+ column: qb.aliasColumn(columnName, alias)
21
+ }
22
+ ];
23
+ }
24
+ if (Array.isArray(orderBy)) {
25
+ return orderBy.flatMap((value)=>processOrderBy(value, ctx));
26
+ }
27
+ if (_.isPlainObject(orderBy)) {
28
+ return Object.entries(orderBy).flatMap(([key, direction])=>{
29
+ const value = orderBy[key];
30
+ const attribute = attributes[key];
31
+ if (!attribute) {
32
+ throw new Error(`Attribute ${key} not found on model ${uid}`);
33
+ }
34
+ if (isScalar(attribute.type)) {
35
+ const columnName = toColumnName(meta, key);
36
+ return {
37
+ column: qb.aliasColumn(columnName, alias),
38
+ order: direction
39
+ };
40
+ }
41
+ if (attribute.type === 'relation' && 'target' in attribute) {
42
+ const subAlias = createJoin(ctx, {
43
+ alias: alias || qb.alias,
44
+ attributeName: key,
45
+ attribute
46
+ });
47
+ return processOrderBy(value, {
48
+ db,
49
+ qb,
50
+ alias: subAlias,
51
+ uid: attribute.target
52
+ });
53
+ }
54
+ throw new Error(`You cannot order on ${attribute.type} types`);
55
+ });
56
+ }
57
+ throw new Error('Invalid orderBy syntax');
58
+ };
59
+ const getStrapiOrderColumnAlias = (column)=>{
60
+ const trimmedColumnName = column.replaceAll('.', '_');
61
+ return `${COL_STRAPI_ORDER_BY_PREFIX}__${trimmedColumnName}`;
62
+ };
63
+ /**
64
+ * Wraps the original Knex query with deep sorting functionality.
65
+ *
66
+ * The function takes an original query and an OrderByCtx object as parameters and returns a new Knex query with deep sorting applied.
67
+ */ const wrapWithDeepSort = (originalQuery, ctx)=>{
68
+ /**
69
+ * Notes:
70
+ * - The generated query has the following flow: baseQuery (filtered unsorted data) -> T (partitioned/sorted data) --> resultQuery (distinct, paginated, sorted data)
71
+ * - Pagination and selection are transferred from the original query to the outer one to avoid pruning rows too early
72
+ * - Filtering (where) has to be done in the deepest sub query possible to avoid processing invalid rows and corrupting the final results
73
+ * - We assume that all necessary joins are done in the original query (`originalQuery`), and every needed column is available with the right name and alias.
74
+ */ const { db, qb, uid } = ctx;
75
+ const { tableName } = db.metadata.get(uid);
76
+ // The orderBy is cloned to avoid unwanted mutations of the original object
77
+ const orderBy = _.cloneDeep(qb.state.orderBy);
78
+ // 0. Init a new Knex query instance (referenced as resultQuery) using the DB connection
79
+ // The connection reuse the original table name (aliased if needed)
80
+ const resultQueryAlias = qb.getAlias();
81
+ const aliasedTableName = qb.mustUseAlias() ? alias(resultQueryAlias, tableName) : tableName;
82
+ const resultQuery = db.getConnection(aliasedTableName);
83
+ // 1. Clone the original query to create the sub-query (referenced as baseQuery) and avoid any mutation on the initial object
84
+ const baseQuery = originalQuery.clone();
85
+ const baseQueryAlias = qb.getAlias();
86
+ // Clear unwanted statements from the sub-query 'baseQuery'
87
+ // Note: `first()` is cleared through the combination of `baseQuery.clear('limit')` and calling `baseQuery.select(...)` again
88
+ // Note: Those statements will be re-applied when duplicates are removed from the final selection
89
+ baseQuery// Columns selection
90
+ .clear('select')// Pagination and sorting
91
+ .clear('order').clear('limit').clear('offset');
92
+ // Override the initial select and return only the columns needed for the partitioning.
93
+ baseQuery.select(// Always select the row id for future manipulation
94
+ prefix(qb.alias, 'id'), // Select every column used in an order by clause, but alias it for future reference
95
+ // i.e. if t2.name is present in an order by clause:
96
+ // Then, "t2.name" will become "t2.name as __strapi_order_by__t2_name"
97
+ ...orderBy.map((orderByClause)=>alias(getStrapiOrderColumnAlias(orderByClause.column), orderByClause.column)));
98
+ // 2. Create a sub-query callback to extract and sort the partitions using row number
99
+ const partitionedQueryAlias = qb.getAlias();
100
+ const selectRowsAsNumberedPartitions = (partitionedQuery)=>{
101
+ // Transform order by clause to their alias to reference them from baseQuery
102
+ const prefixedOrderBy = orderBy.map((orderByClause)=>({
103
+ column: prefix(baseQueryAlias, getStrapiOrderColumnAlias(orderByClause.column)),
104
+ order: orderByClause.order
105
+ }));
106
+ // partitionedQuery select must contain every column used for sorting
107
+ const orderByColumns = prefixedOrderBy.map(_.prop('column'));
108
+ partitionedQuery.select(// Always select baseQuery.id
109
+ prefix(baseQueryAlias, 'id'), // Sort columns
110
+ ...orderByColumns)// The row number is used to assign an index to every row in every partition
111
+ .rowNumber(COL_STRAPI_ROW_NUMBER, (subQuery)=>{
112
+ for (const orderByClause of prefixedOrderBy){
113
+ subQuery.orderBy(orderByClause.column, orderByClause.order, 'last');
114
+ }
115
+ // And each partition/group is created based on baseQuery.id
116
+ subQuery.partitionBy(`${baseQueryAlias}.id`);
117
+ }).from(baseQuery.as(baseQueryAlias)).as(partitionedQueryAlias);
118
+ };
119
+ // 3. Create the final resultQuery query, that select and sort the wanted data using T
120
+ const originalSelect = _.difference(qb.state.select, // Remove order by columns from the initial select
121
+ qb.state.orderBy.map(_.prop('column')))// Alias everything in resultQuery
122
+ .map(prefix(resultQueryAlias));
123
+ resultQuery.select(originalSelect)// Join T to resultQuery to access sorted data
124
+ // Notes:
125
+ // - Only select the first row for each partition
126
+ // - Since we're applying the "where" statement directly on baseQuery (and not on resultQuery), we're using an inner join to avoid unwanted rows
127
+ .innerJoin(selectRowsAsNumberedPartitions, function() {
128
+ this// Only select rows that are returned by T
129
+ .on(`${partitionedQueryAlias}.id`, `${resultQueryAlias}.id`)// By only selecting the rows number equal to 1, we make sure we don't have duplicate, and that
130
+ // we're selecting rows in the correct order amongst the groups created by the "partition by"
131
+ .andOnVal(`${partitionedQueryAlias}.${COL_STRAPI_ROW_NUMBER}`, '=', 1);
132
+ });
133
+ // Re-apply pagination params
134
+ if (qb.state.limit) {
135
+ resultQuery.limit(qb.state.limit);
136
+ }
137
+ if (qb.state.offset) {
138
+ resultQuery.offset(qb.state.offset);
139
+ }
140
+ if (qb.state.first) {
141
+ resultQuery.first();
142
+ }
143
+ // Re-apply the sort using T values
144
+ resultQuery.orderBy([
145
+ // Transform "order by" clause to their T alias and prefix them with T alias
146
+ ...orderBy.map((orderByClause)=>({
147
+ column: prefix(partitionedQueryAlias, getStrapiOrderColumnAlias(orderByClause.column)),
148
+ order: orderByClause.order
149
+ })),
150
+ // Add T.id to the order by clause to get consistent results in case several rows have the exact same order
151
+ {
152
+ column: `${partitionedQueryAlias}.id`,
153
+ order: 'asc'
154
+ }
155
+ ]);
156
+ return resultQuery;
157
+ };
158
+ // Utils
159
+ const alias = _.curry((alias, value)=>`${value} as ${alias}`);
160
+ const prefix = _.curry((prefix, value)=>`${prefix}.${value}`);
161
+
162
+ export { getStrapiOrderColumnAlias, processOrderBy, wrapWithDeepSort };
163
+ //# sourceMappingURL=order-by.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"order-by.mjs","sources":["../../../src/query/helpers/order-by.ts"],"sourcesContent":["import _ from 'lodash/fp';\nimport knex from 'knex';\n\nimport * as types from '../../utils/types';\nimport { createJoin } from './join';\nimport { toColumnName } from './transform';\n\nimport type { Ctx } from '../types';\n\ntype OrderByCtx = Ctx & { alias?: string };\ntype OrderBy = string | { [key: string]: 'asc' | 'desc' } | OrderBy[];\ntype OrderByValue = { column: string; order?: 'asc' | 'desc' };\n\nconst COL_STRAPI_ROW_NUMBER = '__strapi_row_number';\nconst COL_STRAPI_ORDER_BY_PREFIX = '__strapi_order_by';\n\nexport const processOrderBy = (orderBy: OrderBy, ctx: OrderByCtx): OrderByValue[] => {\n const { db, uid, qb, alias } = ctx;\n const meta = db.metadata.get(uid);\n const { attributes } = meta;\n\n if (typeof orderBy === 'string') {\n const attribute = attributes[orderBy];\n\n if (!attribute) {\n throw new Error(`Attribute ${orderBy} not found on model ${uid}`);\n }\n\n const columnName = toColumnName(meta, orderBy);\n\n return [{ column: qb.aliasColumn(columnName, alias) }];\n }\n\n if (Array.isArray(orderBy)) {\n return orderBy.flatMap((value) => processOrderBy(value, ctx));\n }\n\n if (_.isPlainObject(orderBy)) {\n return Object.entries(orderBy).flatMap(([key, direction]) => {\n const value = orderBy[key];\n const attribute = attributes[key];\n\n if (!attribute) {\n throw new Error(`Attribute ${key} not found on model ${uid}`);\n }\n\n if (types.isScalar(attribute.type)) {\n const columnName = toColumnName(meta, key);\n\n return { column: qb.aliasColumn(columnName, alias), order: direction };\n }\n\n if (attribute.type === 'relation' && 'target' in attribute) {\n const subAlias = createJoin(ctx, {\n alias: alias || qb.alias,\n attributeName: key,\n attribute,\n });\n\n return processOrderBy(value, {\n db,\n qb,\n alias: subAlias,\n uid: attribute.target,\n });\n }\n\n throw new Error(`You cannot order on ${attribute.type} types`);\n });\n }\n\n throw new Error('Invalid orderBy syntax');\n};\n\nexport const getStrapiOrderColumnAlias = (column: string) => {\n const trimmedColumnName = column.replaceAll('.', '_');\n\n return `${COL_STRAPI_ORDER_BY_PREFIX}__${trimmedColumnName}`;\n};\n\n/**\n * Wraps the original Knex query with deep sorting functionality.\n *\n * The function takes an original query and an OrderByCtx object as parameters and returns a new Knex query with deep sorting applied.\n */\nexport const wrapWithDeepSort = (originalQuery: knex.Knex.QueryBuilder, ctx: OrderByCtx) => {\n /**\n * Notes:\n * - The generated query has the following flow: baseQuery (filtered unsorted data) -> T (partitioned/sorted data) --> resultQuery (distinct, paginated, sorted data)\n * - Pagination and selection are transferred from the original query to the outer one to avoid pruning rows too early\n * - Filtering (where) has to be done in the deepest sub query possible to avoid processing invalid rows and corrupting the final results\n * - We assume that all necessary joins are done in the original query (`originalQuery`), and every needed column is available with the right name and alias.\n */\n\n const { db, qb, uid } = ctx;\n\n const { tableName } = db.metadata.get(uid);\n\n // The orderBy is cloned to avoid unwanted mutations of the original object\n const orderBy = _.cloneDeep<OrderByValue[]>(qb.state.orderBy);\n\n // 0. Init a new Knex query instance (referenced as resultQuery) using the DB connection\n // The connection reuse the original table name (aliased if needed)\n const resultQueryAlias = qb.getAlias();\n const aliasedTableName = qb.mustUseAlias() ? alias(resultQueryAlias, tableName) : tableName;\n\n const resultQuery = db.getConnection(aliasedTableName);\n\n // 1. Clone the original query to create the sub-query (referenced as baseQuery) and avoid any mutation on the initial object\n const baseQuery = originalQuery.clone();\n const baseQueryAlias = qb.getAlias();\n\n // Clear unwanted statements from the sub-query 'baseQuery'\n // Note: `first()` is cleared through the combination of `baseQuery.clear('limit')` and calling `baseQuery.select(...)` again\n // Note: Those statements will be re-applied when duplicates are removed from the final selection\n baseQuery\n // Columns selection\n .clear('select')\n // Pagination and sorting\n .clear('order')\n .clear('limit')\n .clear('offset');\n\n // Override the initial select and return only the columns needed for the partitioning.\n baseQuery.select(\n // Always select the row id for future manipulation\n prefix(qb.alias, 'id'),\n // Select every column used in an order by clause, but alias it for future reference\n // i.e. if t2.name is present in an order by clause:\n // Then, \"t2.name\" will become \"t2.name as __strapi_order_by__t2_name\"\n ...orderBy.map((orderByClause) =>\n alias(getStrapiOrderColumnAlias(orderByClause.column), orderByClause.column)\n )\n );\n\n // 2. Create a sub-query callback to extract and sort the partitions using row number\n const partitionedQueryAlias = qb.getAlias();\n\n const selectRowsAsNumberedPartitions = (partitionedQuery: knex.Knex.QueryBuilder) => {\n // Transform order by clause to their alias to reference them from baseQuery\n const prefixedOrderBy = orderBy.map((orderByClause) => ({\n column: prefix(baseQueryAlias, getStrapiOrderColumnAlias(orderByClause.column)),\n order: orderByClause.order,\n }));\n\n // partitionedQuery select must contain every column used for sorting\n const orderByColumns = prefixedOrderBy.map<string>(_.prop('column'));\n\n partitionedQuery\n .select(\n // Always select baseQuery.id\n prefix(baseQueryAlias, 'id'),\n // Sort columns\n ...orderByColumns\n )\n // The row number is used to assign an index to every row in every partition\n .rowNumber(COL_STRAPI_ROW_NUMBER, (subQuery) => {\n for (const orderByClause of prefixedOrderBy) {\n subQuery.orderBy(orderByClause.column, orderByClause.order, 'last');\n }\n\n // And each partition/group is created based on baseQuery.id\n subQuery.partitionBy(`${baseQueryAlias}.id`);\n })\n .from(baseQuery.as(baseQueryAlias))\n .as(partitionedQueryAlias);\n };\n\n // 3. Create the final resultQuery query, that select and sort the wanted data using T\n\n const originalSelect = _.difference(\n qb.state.select,\n // Remove order by columns from the initial select\n qb.state.orderBy.map(_.prop('column'))\n )\n // Alias everything in resultQuery\n .map(prefix(resultQueryAlias));\n\n resultQuery\n .select(originalSelect)\n // Join T to resultQuery to access sorted data\n // Notes:\n // - Only select the first row for each partition\n // - Since we're applying the \"where\" statement directly on baseQuery (and not on resultQuery), we're using an inner join to avoid unwanted rows\n .innerJoin(selectRowsAsNumberedPartitions, function () {\n this\n // Only select rows that are returned by T\n .on(`${partitionedQueryAlias}.id`, `${resultQueryAlias}.id`)\n // By only selecting the rows number equal to 1, we make sure we don't have duplicate, and that\n // we're selecting rows in the correct order amongst the groups created by the \"partition by\"\n .andOnVal(`${partitionedQueryAlias}.${COL_STRAPI_ROW_NUMBER}`, '=', 1);\n });\n\n // Re-apply pagination params\n\n if (qb.state.limit) {\n resultQuery.limit(qb.state.limit);\n }\n\n if (qb.state.offset) {\n resultQuery.offset(qb.state.offset);\n }\n\n if (qb.state.first) {\n resultQuery.first();\n }\n\n // Re-apply the sort using T values\n resultQuery.orderBy([\n // Transform \"order by\" clause to their T alias and prefix them with T alias\n ...orderBy.map((orderByClause) => ({\n column: prefix(partitionedQueryAlias, getStrapiOrderColumnAlias(orderByClause.column)),\n order: orderByClause.order,\n })),\n // Add T.id to the order by clause to get consistent results in case several rows have the exact same order\n { column: `${partitionedQueryAlias}.id`, order: 'asc' },\n ]);\n\n return resultQuery;\n};\n\n// Utils\nconst alias = _.curry((alias: string, value: string) => `${value} as ${alias}`);\nconst prefix = _.curry((prefix: string, value: string) => `${prefix}.${value}`);\n"],"names":["COL_STRAPI_ROW_NUMBER","COL_STRAPI_ORDER_BY_PREFIX","processOrderBy","orderBy","ctx","db","uid","qb","alias","meta","metadata","get","attributes","attribute","Error","columnName","toColumnName","column","aliasColumn","Array","isArray","flatMap","value","_","isPlainObject","Object","entries","key","direction","types","type","order","subAlias","createJoin","attributeName","target","getStrapiOrderColumnAlias","trimmedColumnName","replaceAll","wrapWithDeepSort","originalQuery","tableName","cloneDeep","state","resultQueryAlias","getAlias","aliasedTableName","mustUseAlias","resultQuery","getConnection","baseQuery","clone","baseQueryAlias","clear","select","prefix","map","orderByClause","partitionedQueryAlias","selectRowsAsNumberedPartitions","partitionedQuery","prefixedOrderBy","orderByColumns","prop","rowNumber","subQuery","partitionBy","from","as","originalSelect","difference","innerJoin","on","andOnVal","limit","offset","first","curry"],"mappings":"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"}