@strapi/database 4.5.1 → 4.5.3

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.
@@ -29,6 +29,10 @@ class Dialect {
29
29
  return false;
30
30
  }
31
31
 
32
+ supportsWindowFunctions() {
33
+ return true;
34
+ }
35
+
32
36
  async startSchemaUpdate() {
33
37
  // noop
34
38
  }
@@ -0,0 +1,6 @@
1
+ 'use strict';
2
+
3
+ module.exports = {
4
+ MYSQL: 'MYSQL',
5
+ MARIADB: 'MARIADB',
6
+ };
@@ -0,0 +1,37 @@
1
+ 'use strict';
2
+
3
+ const { MARIADB, MYSQL } = require('./constants');
4
+
5
+ const SQL_QUERIES = {
6
+ VERSION: `SELECT version() as version`,
7
+ };
8
+
9
+ class MysqlDatabaseInspector {
10
+ constructor(db) {
11
+ this.db = db;
12
+ }
13
+
14
+ async getInformation() {
15
+ let database;
16
+ let versionNumber;
17
+ try {
18
+ const [results] = await this.db.connection.raw(SQL_QUERIES.VERSION);
19
+ const versionSplit = results[0].version.split('-');
20
+ const databaseName = versionSplit[1];
21
+ versionNumber = versionSplit[0];
22
+ database = databaseName && databaseName.toLowerCase() === 'mariadb' ? MARIADB : MYSQL;
23
+ } catch (e) {
24
+ return {
25
+ database: null,
26
+ version: null,
27
+ };
28
+ }
29
+
30
+ return {
31
+ database,
32
+ version: versionNumber,
33
+ };
34
+ }
35
+ }
36
+
37
+ module.exports = MysqlDatabaseInspector;
@@ -1,13 +1,19 @@
1
1
  'use strict';
2
2
 
3
+ const semver = require('semver');
4
+
3
5
  const { Dialect } = require('../dialect');
4
6
  const MysqlSchemaInspector = require('./schema-inspector');
7
+ const MysqlDatabaseInspector = require('./database-inspector');
8
+ const { MYSQL } = require('./constants');
5
9
 
6
10
  class MysqlDialect extends Dialect {
7
11
  constructor(db) {
8
12
  super(db);
9
13
 
10
14
  this.schemaInspector = new MysqlSchemaInspector(db);
15
+ this.databaseInspector = new MysqlDatabaseInspector(db);
16
+ this.info = null;
11
17
  }
12
18
 
13
19
  configure() {
@@ -38,6 +44,8 @@ class MysqlDialect extends Dialect {
38
44
  } catch (err) {
39
45
  // Ignore error due to lack of session permissions
40
46
  }
47
+
48
+ this.info = await this.databaseInspector.getInformation();
41
49
  }
42
50
 
43
51
  async startSchemaUpdate() {
@@ -57,6 +65,17 @@ class MysqlDialect extends Dialect {
57
65
  return true;
58
66
  }
59
67
 
68
+ supportsWindowFunctions() {
69
+ const isMysqlDB = !this.info.database || this.info.database === MYSQL;
70
+ const isBeforeV8 = !semver.valid(this.info.version) || semver.lt(this.info.version, '8.0.0');
71
+
72
+ if (isMysqlDB && isBeforeV8) {
73
+ return false;
74
+ }
75
+
76
+ return true;
77
+ }
78
+
60
79
  usesForeignKeys() {
61
80
  return true;
62
81
  }
@@ -15,7 +15,7 @@ class PostgresDialect extends Dialect {
15
15
  return true;
16
16
  }
17
17
 
18
- initialize() {
18
+ async initialize() {
19
19
  this.db.connection.client.driver.types.setTypeParser(1082, 'text', (v) => v); // Don't cast DATE string to Date()
20
20
  this.db.connection.client.driver.types.setTypeParser(1700, 'text', parseFloat);
21
21
  }
@@ -5,13 +5,13 @@ const fse = require('fs-extra');
5
5
 
6
6
  const errors = require('../../errors');
7
7
  const { Dialect } = require('../dialect');
8
- const SqliteSchmeaInspector = require('./schema-inspector');
8
+ const SqliteSchemaInspector = require('./schema-inspector');
9
9
 
10
10
  class SqliteDialect extends Dialect {
11
11
  constructor(db) {
12
12
  super(db);
13
13
 
14
- this.schemaInspector = new SqliteSchmeaInspector(db);
14
+ this.schemaInspector = new SqliteSchemaInspector(db);
15
15
  }
16
16
 
17
17
  configure() {
@@ -1,6 +1,7 @@
1
1
  'use strict';
2
2
 
3
3
  const { map, isEmpty } = require('lodash/fp');
4
+
4
5
  const {
5
6
  isBidirectional,
6
7
  isOneToAny,
@@ -10,6 +11,7 @@ const {
10
11
  hasInverseOrderColumn,
11
12
  } = require('../metadata/relations');
12
13
  const { createQueryBuilder } = require('../query');
14
+ const { addSchema } = require('../utils/knex');
13
15
 
14
16
  /**
15
17
  * If some relations currently exist for this oneToX relation, on the one side, this function removes them and update the inverse order if needed.
@@ -195,6 +197,12 @@ const cleanOrderColumns = async ({ id, attribute, db, inverseRelIds, transaction
195
197
  return;
196
198
  }
197
199
 
200
+ // Handle databases that don't support window function ROW_NUMBER
201
+ if (!strapi.db.dialect.supportsWindowFunctions()) {
202
+ await cleanOrderColumnsForOldDatabases({ id, attribute, db, inverseRelIds, transaction: trx });
203
+ return;
204
+ }
205
+
198
206
  const { joinTable } = attribute;
199
207
  const { joinColumn, inverseJoinColumn, orderColumnName, inverseOrderColumnName } = joinTable;
200
208
  const update = [];
@@ -241,7 +249,8 @@ const cleanOrderColumns = async ({ id, attribute, db, inverseRelIds, transaction
241
249
  )
242
250
  .transacting(trx);
243
251
  break;
244
- default:
252
+ default: {
253
+ const joinTableName = addSchema(joinTable.name);
245
254
  await db.connection
246
255
  .raw(
247
256
  `UPDATE ?? as a
@@ -252,9 +261,10 @@ const cleanOrderColumns = async ({ id, attribute, db, inverseRelIds, transaction
252
261
  WHERE ${where.join(' OR ')}
253
262
  ) AS b
254
263
  WHERE b.id = a.id`,
255
- [joinTable.name, ...updateBinding, ...selectBinding, joinTable.name, ...whereBinding]
264
+ [joinTableName, ...updateBinding, ...selectBinding, joinTableName, ...whereBinding]
256
265
  )
257
266
  .transacting(trx);
267
+ }
258
268
  /*
259
269
  `UPDATE :joinTable: as a
260
270
  SET :orderColumn: = b.src_order, :inverseOrderColumn: = b.inv_order
@@ -271,6 +281,103 @@ const cleanOrderColumns = async ({ id, attribute, db, inverseRelIds, transaction
271
281
  }
272
282
  };
273
283
 
284
+ const cleanOrderColumnsForOldDatabases = async ({
285
+ id,
286
+ attribute,
287
+ db,
288
+ inverseRelIds,
289
+ transaction: trx,
290
+ }) => {
291
+ const { joinTable } = attribute;
292
+ const { joinColumn, inverseJoinColumn, orderColumnName, inverseOrderColumnName } = joinTable;
293
+
294
+ const now = new Date().valueOf();
295
+
296
+ if (hasOrderColumn(attribute) && id) {
297
+ const tempOrderTableName = `tempOrderTableName_${now}`;
298
+ try {
299
+ await db.connection
300
+ .raw(
301
+ `
302
+ CREATE TEMPORARY TABLE :tempOrderTableName:
303
+ SELECT
304
+ id,
305
+ (
306
+ SELECT count(*)
307
+ FROM :joinTableName: b
308
+ WHERE a.:orderColumnName: >= b.:orderColumnName: AND a.:joinColumnName: = b.:joinColumnName: AND a.:joinColumnName: = :id
309
+ ) AS src_order
310
+ FROM :joinTableName: a`,
311
+ {
312
+ tempOrderTableName,
313
+ joinTableName: joinTable.name,
314
+ orderColumnName,
315
+ joinColumnName: joinColumn.name,
316
+ id,
317
+ }
318
+ )
319
+ .transacting(trx);
320
+ await db.connection
321
+ .raw(
322
+ `UPDATE ?? as a, (SELECT * FROM ??) AS b
323
+ SET ?? = b.src_order
324
+ WHERE a.id = b.id`,
325
+ [joinTable.name, tempOrderTableName, orderColumnName]
326
+ )
327
+ .transacting(trx);
328
+ } finally {
329
+ await db.connection
330
+ .raw(`DROP TEMPORARY TABLE IF EXISTS ??`, [tempOrderTableName])
331
+ .transacting(trx);
332
+ }
333
+ }
334
+
335
+ if (hasInverseOrderColumn(attribute) && !isEmpty(inverseRelIds)) {
336
+ const tempInvOrderTableName = `tempInvOrderTableName_${now}`;
337
+ try {
338
+ await db.connection
339
+ .raw(
340
+ `
341
+ CREATE TEMPORARY TABLE ??
342
+ SELECT
343
+ id,
344
+ (
345
+ SELECT count(*)
346
+ FROM ?? b
347
+ WHERE a.?? >= b.?? AND a.?? = b.?? AND a.?? IN (${inverseRelIds
348
+ .map(() => '?')
349
+ .join(', ')})
350
+ ) AS inv_order
351
+ FROM ?? a`,
352
+ [
353
+ tempInvOrderTableName,
354
+ joinTable.name,
355
+ inverseOrderColumnName,
356
+ inverseOrderColumnName,
357
+ inverseJoinColumn.name,
358
+ inverseJoinColumn.name,
359
+ inverseJoinColumn.name,
360
+ ...inverseRelIds,
361
+ joinTable.name,
362
+ ]
363
+ )
364
+ .transacting(trx);
365
+ await db.connection
366
+ .raw(
367
+ `UPDATE ?? as a, (SELECT * FROM ??) AS b
368
+ SET ?? = b.inv_order
369
+ WHERE a.id = b.id`,
370
+ [joinTable.name, tempInvOrderTableName, inverseOrderColumnName]
371
+ )
372
+ .transacting(trx);
373
+ } finally {
374
+ await db.connection
375
+ .raw(`DROP TEMPORARY TABLE IF EXISTS ??`, [tempInvOrderTableName])
376
+ .transacting(trx);
377
+ }
378
+ }
379
+ };
380
+
274
381
  module.exports = {
275
382
  deletePreviousOneToAnyRelations,
276
383
  deletePreviousAnyToOneRelations,
@@ -446,6 +446,11 @@ const morphToMany = async (input, ctx) => {
446
446
  .where({
447
447
  [joinColumn.name]: referencedValues,
448
448
  ...(joinTable.on || {}),
449
+ // If the populateValue contains an "on" property,
450
+ // only populate the types defined in it
451
+ ...('on' in populateValue
452
+ ? { [morphColumn.typeColumn.name]: Object.keys(populateValue.on) }
453
+ : {}),
449
454
  })
450
455
  .orderBy([joinColumn.name, 'order'])
451
456
  .execute({ mapResults: false });
@@ -470,6 +475,8 @@ const morphToMany = async (input, ctx) => {
470
475
  }, {});
471
476
 
472
477
  const map = {};
478
+ const { on, ...typePopulate } = populateValue;
479
+
473
480
  for (const type of Object.keys(idsByType)) {
474
481
  const ids = idsByType[type];
475
482
 
@@ -483,7 +490,7 @@ const morphToMany = async (input, ctx) => {
483
490
  const qb = db.entityManager.createQueryBuilder(type);
484
491
 
485
492
  const rows = await qb
486
- .init(populateValue)
493
+ .init(on?.[type] ?? typePopulate)
487
494
  .addSelect(`${qb.alias}.${idColumn.referencedColumn}`)
488
495
  .where({ [idColumn.referencedColumn]: ids })
489
496
  .execute({ mapResults: false });
@@ -540,6 +547,8 @@ const morphToOne = async (input, ctx) => {
540
547
  }, {});
541
548
 
542
549
  const map = {};
550
+ const { on, ...typePopulate } = populateValue;
551
+
543
552
  for (const type of Object.keys(idsByType)) {
544
553
  const ids = idsByType[type];
545
554
 
@@ -552,7 +561,7 @@ const morphToOne = async (input, ctx) => {
552
561
  const qb = db.entityManager.createQueryBuilder(type);
553
562
 
554
563
  const rows = await qb
555
- .init(populateValue)
564
+ .init(on?.[type] ?? typePopulate)
556
565
  .addSelect(`${qb.alias}.${idColumn.referencedColumn}`)
557
566
  .where({ [idColumn.referencedColumn]: ids })
558
567
  .execute({ mapResults: false });
@@ -579,7 +588,16 @@ const morphToOne = async (input, ctx) => {
579
588
 
580
589
  // TODO: Omit limit & offset to avoid needing a query per result to avoid making too many queries
581
590
  const pickPopulateParams = (populate) => {
582
- const fieldsToPick = ['select', 'count', 'where', 'populate', 'orderBy', 'filters', 'ordering'];
591
+ const fieldsToPick = [
592
+ 'select',
593
+ 'count',
594
+ 'where',
595
+ 'populate',
596
+ 'orderBy',
597
+ 'filters',
598
+ 'ordering',
599
+ 'on',
600
+ ];
583
601
 
584
602
  if (populate.count !== true) {
585
603
  fieldsToPick.push('limit', 'offset');
package/lib/utils/knex.js CHANGED
@@ -7,6 +7,16 @@ const isKnexQuery = (value) => {
7
7
  return value instanceof KnexBuilder || value instanceof KnexRaw;
8
8
  };
9
9
 
10
+ /**
11
+ * Adds the name of the schema to the table name if the schema was defined by the user.
12
+ * Users can set the db schema only for Postgres in strapi database config.
13
+ */
14
+ const addSchema = (tableName) => {
15
+ const schemaName = strapi.db.connection.getSchemaName();
16
+ return schemaName ? `${schemaName}.${tableName}` : tableName;
17
+ };
18
+
10
19
  module.exports = {
11
20
  isKnexQuery,
21
+ addSchema,
12
22
  };
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@strapi/database",
3
- "version": "4.5.1",
3
+ "version": "4.5.3",
4
4
  "description": "Strapi's database layer",
5
5
  "homepage": "https://strapi.io",
6
6
  "bugs": {
@@ -36,11 +36,12 @@
36
36
  "fs-extra": "10.0.0",
37
37
  "knex": "1.0.7",
38
38
  "lodash": "4.17.21",
39
+ "semver": "7.3.8",
39
40
  "umzug": "3.1.1"
40
41
  },
41
42
  "engines": {
42
43
  "node": ">=14.19.1 <=18.x.x",
43
44
  "npm": ">=6.0.0"
44
45
  },
45
- "gitHead": "8c20ea2b5c5a115b78454086ea270dcd59b06004"
46
+ "gitHead": "5453885f7aa329f98d047a6030a8e98ba3cbdb41"
46
47
  }