2021-06-15 19:01:22 +03:00
|
|
|
const debug = require('@tryghost/debug')('test:dbUtils');
|
2021-02-16 19:22:58 +03:00
|
|
|
|
|
|
|
// Utility Packages
|
2021-12-06 15:50:35 +03:00
|
|
|
const fs = require('fs-extra');
|
2021-02-16 19:22:58 +03:00
|
|
|
const Promise = require('bluebird');
|
|
|
|
const KnexMigrator = require('knex-migrator');
|
|
|
|
const knexMigrator = new KnexMigrator();
|
2022-03-17 17:36:33 +03:00
|
|
|
const DatabaseInfo = require('@tryghost/database-info');
|
2021-02-16 19:22:58 +03:00
|
|
|
|
|
|
|
// Ghost Internals
|
|
|
|
const config = require('../../core/shared/config');
|
|
|
|
const db = require('../../core/server/data/db');
|
|
|
|
const schema = require('../../core/server/data/schema').tables;
|
|
|
|
const schemaTables = Object.keys(schema);
|
|
|
|
|
2021-02-17 20:36:27 +03:00
|
|
|
// Other Test Utilities
|
|
|
|
const urlServiceUtils = require('./url-service-utils');
|
2021-02-16 19:22:58 +03:00
|
|
|
|
2021-12-06 15:50:35 +03:00
|
|
|
const dbHash = Date.now();
|
|
|
|
|
2022-03-17 19:53:08 +03:00
|
|
|
/**
|
|
|
|
* Checks if the current active connection is a MySQL database
|
|
|
|
* @returns {Boolean} isMySQL
|
|
|
|
*/
|
2022-03-17 17:36:33 +03:00
|
|
|
module.exports.isMySQL = () => {
|
|
|
|
return DatabaseInfo.isMySQL(db.knex);
|
|
|
|
};
|
|
|
|
|
2022-03-17 19:53:08 +03:00
|
|
|
/**
|
|
|
|
* Checks if the current active connection is a SQLite database
|
|
|
|
* @returns {Boolean} isSQLite
|
|
|
|
*/
|
2022-03-17 17:36:33 +03:00
|
|
|
module.exports.isSQLite = () => {
|
|
|
|
return DatabaseInfo.isSQLite(db.knex);
|
|
|
|
};
|
|
|
|
|
2022-03-17 19:53:08 +03:00
|
|
|
/**
|
|
|
|
* Reset
|
|
|
|
* - restores the DB to a fresh state with the default fixtures in place
|
|
|
|
* - has many behind the scenes tricks to try to do this as fast as possible
|
|
|
|
*
|
|
|
|
* @param {Object} options
|
|
|
|
* @param {Boolean} options.truncate whether to truncate rather thann fully reset
|
|
|
|
*/
|
2022-02-07 18:46:35 +03:00
|
|
|
module.exports.reset = async ({truncate} = {truncate: false}) => {
|
2021-12-06 15:50:35 +03:00
|
|
|
// Only run this copy in CI until it gets fleshed out
|
2022-03-17 17:36:33 +03:00
|
|
|
if (process.env.CI && module.exports.isSQLite()) {
|
2021-12-06 15:50:35 +03:00
|
|
|
const filename = config.get('database:connection:filename');
|
|
|
|
const filenameOrig = `${filename}.${dbHash}-orig`;
|
|
|
|
|
|
|
|
const dbExists = await fs.pathExists(filenameOrig);
|
|
|
|
|
|
|
|
if (dbExists) {
|
2021-12-07 11:23:51 +03:00
|
|
|
await db.knex.destroy();
|
2021-12-06 15:50:35 +03:00
|
|
|
await fs.copyFile(filenameOrig, filename);
|
|
|
|
} else {
|
2022-03-17 19:53:08 +03:00
|
|
|
// Do a full database reset & initialisation
|
|
|
|
await forceReinit();
|
2021-12-06 15:50:35 +03:00
|
|
|
|
|
|
|
await fs.copyFile(filename, filenameOrig);
|
|
|
|
}
|
|
|
|
} else {
|
2022-02-07 18:46:35 +03:00
|
|
|
if (truncate) {
|
2022-03-17 12:04:57 +03:00
|
|
|
// Perform a fast reset by tearing down all the tables and inserting the fixtures
|
|
|
|
try {
|
2022-03-17 19:36:43 +03:00
|
|
|
await truncateAll();
|
2022-03-17 12:04:57 +03:00
|
|
|
await knexMigrator.init({only: 2});
|
|
|
|
} catch (err) {
|
|
|
|
// If it fails, try a normal restore
|
2022-03-17 19:53:08 +03:00
|
|
|
await forceReinit();
|
2022-03-17 12:04:57 +03:00
|
|
|
}
|
2022-02-07 18:46:35 +03:00
|
|
|
} else {
|
|
|
|
// Do a full database reset + initialisation
|
2022-03-17 19:53:08 +03:00
|
|
|
await forceReinit();
|
2022-02-07 18:46:35 +03:00
|
|
|
}
|
2021-12-06 15:50:35 +03:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2022-03-17 19:53:08 +03:00
|
|
|
/**
|
|
|
|
* Teardown
|
|
|
|
* - restores the DB to empty tables only - no default fixtures, settings or permissions
|
|
|
|
* - has behind the scenes tricks to try to do this as fast as possible
|
|
|
|
*/
|
|
|
|
module.exports.teardown = async () => {
|
|
|
|
try {
|
|
|
|
await truncateAll();
|
|
|
|
} catch (err) {
|
|
|
|
await knexMigrator.reset({force: true});
|
|
|
|
}
|
2021-02-16 19:22:58 +03:00
|
|
|
};
|
|
|
|
|
2022-03-17 19:53:08 +03:00
|
|
|
/**
|
|
|
|
* Truncate
|
|
|
|
* - truncate a single table
|
|
|
|
* @param {string} tableName - the table to truncate
|
|
|
|
*/
|
2021-02-17 23:37:49 +03:00
|
|
|
module.exports.truncate = async (tableName) => {
|
2022-03-17 17:36:33 +03:00
|
|
|
if (module.exports.isSQLite()) {
|
2021-03-01 21:19:24 +03:00
|
|
|
const [foreignKeysEnabled] = await db.knex.raw('PRAGMA foreign_keys;');
|
|
|
|
if (foreignKeysEnabled.foreign_keys) {
|
|
|
|
await db.knex.raw('PRAGMA foreign_keys = OFF;');
|
|
|
|
}
|
2021-02-17 23:37:49 +03:00
|
|
|
await db.knex(tableName).truncate();
|
2021-03-01 21:19:24 +03:00
|
|
|
if (foreignKeysEnabled.foreign_keys) {
|
|
|
|
await db.knex.raw('PRAGMA foreign_keys = ON;');
|
|
|
|
}
|
2021-02-17 23:37:49 +03:00
|
|
|
return;
|
2021-02-16 19:22:58 +03:00
|
|
|
}
|
|
|
|
|
2021-02-17 23:37:49 +03:00
|
|
|
await db.knex.raw('SET FOREIGN_KEY_CHECKS=0;');
|
|
|
|
await db.knex(tableName).truncate();
|
|
|
|
await db.knex.raw('SET FOREIGN_KEY_CHECKS=1;');
|
2021-02-16 19:22:58 +03:00
|
|
|
};
|
|
|
|
|
2022-03-17 19:36:43 +03:00
|
|
|
/**
|
2022-03-17 19:53:08 +03:00
|
|
|
* Internal helper to do a safe-but-slow knex-based forced reinit of the DB.
|
2022-03-17 19:36:43 +03:00
|
|
|
*/
|
2022-03-17 19:53:08 +03:00
|
|
|
const forceReinit = async () => {
|
|
|
|
await knexMigrator.reset({force: true});
|
|
|
|
await knexMigrator.init();
|
2022-03-17 19:36:43 +03:00
|
|
|
};
|
|
|
|
|
2021-02-16 19:22:58 +03:00
|
|
|
/**
|
2022-03-17 19:53:08 +03:00
|
|
|
* Internal helper to attempt to truncate all tables as fast as possible
|
2021-02-16 19:22:58 +03:00
|
|
|
* Has to run in a transaction for MySQL, otherwise the foreign key check does not work.
|
|
|
|
* Sqlite3 has no truncate command.
|
|
|
|
*/
|
2022-03-17 19:36:43 +03:00
|
|
|
const truncateAll = () => {
|
2021-02-16 19:22:58 +03:00
|
|
|
debug('Database teardown');
|
2021-02-17 20:36:27 +03:00
|
|
|
urlServiceUtils.reset();
|
2021-02-16 19:22:58 +03:00
|
|
|
|
|
|
|
const tables = schemaTables.concat(['migrations']);
|
|
|
|
|
2022-03-17 17:36:33 +03:00
|
|
|
if (module.exports.isSQLite()) {
|
2021-02-16 19:22:58 +03:00
|
|
|
return Promise
|
|
|
|
.mapSeries(tables, function createTable(table) {
|
2021-03-01 21:19:24 +03:00
|
|
|
return (async function () {
|
|
|
|
const [foreignKeysEnabled] = await db.knex.raw('PRAGMA foreign_keys;');
|
|
|
|
if (foreignKeysEnabled.foreign_keys) {
|
|
|
|
await db.knex.raw('PRAGMA foreign_keys = OFF;');
|
|
|
|
}
|
|
|
|
await db.knex.raw('DELETE FROM ' + table + ';');
|
|
|
|
if (foreignKeysEnabled.foreign_keys) {
|
|
|
|
await db.knex.raw('PRAGMA foreign_keys = ON;');
|
|
|
|
}
|
|
|
|
})();
|
2021-02-16 19:22:58 +03:00
|
|
|
})
|
|
|
|
.catch(function (err) {
|
|
|
|
// CASE: table does not exist
|
|
|
|
if (err.errno === 1) {
|
|
|
|
return Promise.resolve();
|
|
|
|
}
|
|
|
|
|
|
|
|
throw err;
|
2021-11-18 18:28:46 +03:00
|
|
|
})
|
|
|
|
.finally(() => {
|
|
|
|
debug('Database teardown end');
|
2021-02-16 19:22:58 +03:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
return db.knex.transaction(function (trx) {
|
|
|
|
return db.knex.raw('SET FOREIGN_KEY_CHECKS=0;').transacting(trx)
|
|
|
|
.then(function () {
|
|
|
|
return Promise
|
|
|
|
.each(tables, function createTable(table) {
|
|
|
|
return db.knex.raw('TRUNCATE ' + table + ';').transacting(trx);
|
|
|
|
});
|
|
|
|
})
|
|
|
|
.then(function () {
|
|
|
|
return db.knex.raw('SET FOREIGN_KEY_CHECKS=1;').transacting(trx);
|
|
|
|
})
|
|
|
|
.catch(function (err) {
|
2022-03-17 12:04:57 +03:00
|
|
|
// CASE: table does not exist || DB does not exist
|
|
|
|
// If the table or DB are not present, we can safely ignore
|
|
|
|
if (err.errno === 1146 || err.errno === 1049) {
|
2021-02-16 19:22:58 +03:00
|
|
|
return Promise.resolve();
|
|
|
|
}
|
|
|
|
|
|
|
|
throw err;
|
2022-03-17 12:04:57 +03:00
|
|
|
})
|
|
|
|
.finally(() => {
|
|
|
|
debug('Database teardown end');
|
2021-02-16 19:22:58 +03:00
|
|
|
});
|
2022-03-17 12:04:57 +03:00
|
|
|
})
|
|
|
|
.catch(function (err) {
|
|
|
|
// CASE: table does not exist || DB does not exist
|
|
|
|
// If the table or DB are not present, we can safely ignore
|
|
|
|
if (err.errno === 1146 || err.errno === 1049) {
|
|
|
|
return Promise.resolve();
|
|
|
|
}
|
|
|
|
|
|
|
|
throw err;
|
|
|
|
});
|
2021-02-16 19:22:58 +03:00
|
|
|
};
|
2022-03-17 19:53:08 +03:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @deprecated Use teardown or reset instead
|
|
|
|
* Old method for clearing data from the database that also mixes in url service behaviour
|
|
|
|
*/
|
|
|
|
module.exports.clearData = async () => {
|
|
|
|
debug('Database reset');
|
|
|
|
await knexMigrator.reset({force: true});
|
|
|
|
urlServiceUtils.reset();
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @deprecated Use reset instead
|
|
|
|
* Old method for clearing data from the database that also mixes in url service behaviour
|
|
|
|
*/
|
|
|
|
module.exports.initData = async () => {
|
|
|
|
await knexMigrator.init();
|
|
|
|
await urlServiceUtils.reset();
|
|
|
|
await urlServiceUtils.init();
|
|
|
|
await urlServiceUtils.isFinished();
|
|
|
|
};
|
2022-05-06 14:15:12 +03:00
|
|
|
|
|
|
|
module.exports.knex = db.knex;
|