@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.
- package/lib/dialects/dialect.js +4 -0
- package/lib/dialects/mysql/constants.js +6 -0
- package/lib/dialects/mysql/database-inspector.js +37 -0
- package/lib/dialects/mysql/index.js +19 -0
- package/lib/dialects/postgresql/index.js +1 -1
- package/lib/dialects/sqlite/index.js +2 -2
- package/lib/entity-manager/regular-relations.js +109 -2
- package/lib/query/helpers/populate/apply.js +21 -3
- package/lib/utils/knex.js +10 -0
- package/package.json +3 -2
package/lib/dialects/dialect.js
CHANGED
|
@@ -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
|
|
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
|
|
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
|
-
[
|
|
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(
|
|
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(
|
|
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 = [
|
|
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.
|
|
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": "
|
|
46
|
+
"gitHead": "5453885f7aa329f98d047a6030a8e98ba3cbdb41"
|
|
46
47
|
}
|