mirror of
https://github.com/TryGhost/Ghost.git
synced 2025-01-04 17:04:59 +03:00
3d989eba23
refs https://github.com/TryGhost/Toolbox/issues/354 - this commit turns the Ghost repo into a monorepo so we can bring our internal packages back in, which makes life easier when working on Ghost
248 lines
8.3 KiB
JavaScript
248 lines
8.3 KiB
JavaScript
// Utility Packages
|
|
const debug = require('@tryghost/debug')('test');
|
|
const Promise = require('bluebird');
|
|
const _ = require('lodash');
|
|
const fs = require('fs-extra');
|
|
const path = require('path');
|
|
const os = require('os');
|
|
const uuid = require('uuid');
|
|
const KnexMigrator = require('knex-migrator');
|
|
const knexMigrator = new KnexMigrator();
|
|
|
|
// Ghost Internals
|
|
const config = require('../../core/shared/config');
|
|
const boot = require('../../core/boot');
|
|
const db = require('../../core/server/data/db');
|
|
const models = require('../../core/server/models');
|
|
const urlService = require('../../core/server/services/url');
|
|
const settingsService = require('../../core/server/services/settings/settings-service');
|
|
const routeSettingsService = require('../../core/server/services/route-settings');
|
|
const themeService = require('../../core/server/services/themes');
|
|
const limits = require('../../core/server/services/limits');
|
|
const customRedirectsService = require('../../core/server/services/redirects');
|
|
|
|
// Other Test Utilities
|
|
const configUtils = require('./configUtils');
|
|
const dbUtils = require('./db-utils');
|
|
const urlServiceUtils = require('./url-service-utils');
|
|
const redirects = require('./redirects');
|
|
const context = require('./fixtures/context');
|
|
|
|
let ghostServer;
|
|
let existingData = {};
|
|
let totalStartTime = 0;
|
|
let totalBoots = 0;
|
|
|
|
/**
|
|
* Because we use ObjectID we don't know the ID of fixtures ahead of time
|
|
* This function fetches all of our fixtures and exposes them so that tests can use them
|
|
* @TODO: Optimize this by making it optional / selective
|
|
*/
|
|
const exposeFixtures = async () => {
|
|
const fixturePromises = {
|
|
roles: models.Role.findAll({columns: ['id']}),
|
|
users: models.User.findAll({columns: ['id', 'email', 'slug']}),
|
|
tags: models.Tag.findAll({columns: ['id']}),
|
|
apiKeys: models.ApiKey.findAll({withRelated: 'integration'})
|
|
};
|
|
const keys = Object.keys(fixturePromises);
|
|
existingData = {};
|
|
|
|
return Promise
|
|
.all(Object.values(fixturePromises))
|
|
.then((results) => {
|
|
for (let i = 0; i < keys.length; i += 1) {
|
|
existingData[keys[i]] = results[i].toJSON(context.internal);
|
|
}
|
|
})
|
|
.catch((err) => {
|
|
console.error('Unable to expose fixtures', err); // eslint-disable-line no-console
|
|
process.exit(1);
|
|
});
|
|
};
|
|
|
|
const prepareContentFolder = (options) => {
|
|
const contentFolderForTests = options.contentFolder;
|
|
|
|
/**
|
|
* We never use the root content folder for testing!
|
|
* We use a tmp folder.
|
|
*/
|
|
configUtils.set('paths:contentPath', contentFolderForTests);
|
|
|
|
fs.ensureDirSync(contentFolderForTests);
|
|
fs.ensureDirSync(path.join(contentFolderForTests, 'data'));
|
|
fs.ensureDirSync(path.join(contentFolderForTests, 'themes'));
|
|
fs.ensureDirSync(path.join(contentFolderForTests, 'images'));
|
|
fs.ensureDirSync(path.join(contentFolderForTests, 'logs'));
|
|
fs.ensureDirSync(path.join(contentFolderForTests, 'adapters'));
|
|
fs.ensureDirSync(path.join(contentFolderForTests, 'settings'));
|
|
|
|
if (options.copyThemes) {
|
|
// Copy all themes into the new test content folder. Default active theme is always casper. If you want to use a different theme, you have to set the active theme (e.g. stub)
|
|
fs.copySync(path.join(__dirname, 'fixtures', 'themes'), path.join(contentFolderForTests, 'themes'));
|
|
}
|
|
|
|
if (options.redirectsFile) {
|
|
redirects.setupFile(contentFolderForTests, options.redirectsFileExt);
|
|
}
|
|
|
|
if (options.routesFilePath) {
|
|
fs.copySync(options.routesFilePath, path.join(contentFolderForTests, 'settings', 'routes.yaml'));
|
|
} else if (options.copySettings) {
|
|
fs.copySync(path.join(__dirname, 'fixtures', 'settings', 'routes.yaml'), path.join(contentFolderForTests, 'settings', 'routes.yaml'));
|
|
}
|
|
};
|
|
|
|
// CASE: Ghost Server is Running
|
|
// In this case we need to reset things so it's as though Ghost just booted:
|
|
// - truncate database
|
|
// - re-run default fixtures
|
|
// - reload affected services
|
|
const restartModeGhostStart = async ({frontend}) => {
|
|
debug('Reload Mode');
|
|
|
|
// TODO: figure out why we need this if we reset again later?
|
|
urlServiceUtils.reset();
|
|
|
|
await dbUtils.reset({truncate: true});
|
|
|
|
debug('init done');
|
|
|
|
// Reset the settings cache
|
|
await settingsService.init();
|
|
debug('settings done');
|
|
|
|
if (frontend) {
|
|
// Load the frontend-related components
|
|
await routeSettingsService.init();
|
|
await themeService.init();
|
|
debug('frontend done');
|
|
}
|
|
|
|
// Reload the URL service & wait for it to be ready again
|
|
// @TODO: why/how is this different to urlService.resetGenerators?
|
|
urlServiceUtils.reset();
|
|
urlServiceUtils.init({urlCache: !frontend});
|
|
|
|
if (frontend) {
|
|
await urlServiceUtils.isFinished();
|
|
}
|
|
|
|
debug('routes done');
|
|
|
|
await customRedirectsService.init();
|
|
|
|
// Reload limits service
|
|
limits.init();
|
|
};
|
|
|
|
// CASE: Ghost Server needs Starting
|
|
// In this case we need to ensure that Ghost is started cleanly:
|
|
// - ensure the DB is reset
|
|
// - CASE: If we are in force start mode the server is already running so we
|
|
// - stop the server (if we are in force start mode it will be running)
|
|
// - reload affected services - just settings and not the frontend!?
|
|
// - Start Ghost: Uses OLD Boot process
|
|
const freshModeGhostStart = async (options) => {
|
|
if (options.forceStart) {
|
|
debug('Forced Restart Mode');
|
|
} else {
|
|
debug('Fresh Start Mode');
|
|
}
|
|
|
|
// Stop the server (forceStart Mode)
|
|
await stopGhost();
|
|
|
|
// Reset the settings cache and disable listeners so they don't get triggered further
|
|
settingsService.reset();
|
|
|
|
await dbUtils.reset();
|
|
|
|
await settingsService.init();
|
|
|
|
// Actually boot Ghost
|
|
ghostServer = await boot({
|
|
backend: options.backend,
|
|
frontend: options.frontend,
|
|
server: options.server
|
|
});
|
|
|
|
// Wait for the URL service to be ready, which happens after boot
|
|
if (options.frontend) {
|
|
await urlServiceUtils.isFinished();
|
|
}
|
|
};
|
|
|
|
/**
|
|
*
|
|
* @param {Object} [options]
|
|
* @param {boolean} [options.backend]
|
|
* @param {boolean} [options.frontend]
|
|
* @param {boolean} [options.redirectsFile]
|
|
* @param {String} [options.redirectsFileExt]
|
|
* @param {boolean} [options.forceStart]
|
|
* @param {boolean} [options.copyThemes]
|
|
* @param {boolean} [options.copySettings]
|
|
* @param {String} [options.routesFilePath] - path to a routes configuration file to start the instance with
|
|
* @param {String} [options.contentFolder]
|
|
* @param {boolean} [options.subdir]
|
|
* @returns {Promise<GhostServer>}
|
|
*/
|
|
const startGhost = async (options) => {
|
|
const startTime = Date.now();
|
|
debug('Start Ghost');
|
|
options = _.merge({
|
|
backend: true,
|
|
frontend: true,
|
|
redirectsFile: true,
|
|
redirectsFileExt: '.json',
|
|
forceStart: false,
|
|
copyThemes: true,
|
|
copySettings: true,
|
|
contentFolder: path.join(os.tmpdir(), uuid.v4(), 'ghost-test'),
|
|
subdir: false
|
|
}, options);
|
|
|
|
// Ensure we have tmp content folders populated ready for testing
|
|
// @TODO: tidy up the tmp folders after tests
|
|
prepareContentFolder(options);
|
|
|
|
if (ghostServer && ghostServer.httpServer && !options.forceStart) {
|
|
await restartModeGhostStart(options);
|
|
} else {
|
|
await freshModeGhostStart(options);
|
|
}
|
|
|
|
// Expose fixture data, wrap-up and return
|
|
await exposeFixtures();
|
|
|
|
// Reporting
|
|
const totalTime = Date.now() - startTime;
|
|
totalStartTime += totalTime;
|
|
totalBoots += 1;
|
|
const averageBootTime = Math.round(totalStartTime / totalBoots);
|
|
debug(`Started Ghost in ${totalTime / 1000}s`);
|
|
debug(`Accumulated start time across ${totalBoots} boots is ${totalStartTime / 1000}s (average = ${averageBootTime}ms)`);
|
|
return ghostServer;
|
|
};
|
|
|
|
const stopGhost = async () => {
|
|
if (ghostServer && ghostServer.httpServer) {
|
|
await ghostServer.stop();
|
|
delete require.cache[require.resolve('../../core/app')];
|
|
// NOTE: similarly to urlService.reset() there doesn't seem to be a need for this call
|
|
// probable best location for this type of cleanup if it's needed is registering
|
|
// a hood during the "server cleanup" phase of the server stop
|
|
urlService.resetGenerators();
|
|
}
|
|
};
|
|
|
|
module.exports = {
|
|
startGhost,
|
|
stopGhost,
|
|
getExistingData: () => {
|
|
return existingData;
|
|
}
|
|
};
|