2016-09-14 17:50:17 +03:00
|
|
|
/**
|
|
|
|
* make sure overrides get's called first!
|
2018-07-17 19:10:05 +03:00
|
|
|
* - keeping the overrides import here works for installing Ghost as npm!
|
2016-09-14 17:50:17 +03:00
|
|
|
*
|
|
|
|
* the call order is the following:
|
|
|
|
* - root index requires core module
|
|
|
|
* - core index requires server
|
|
|
|
* - overrides is the first package to load
|
|
|
|
*/
|
|
|
|
require('./overrides');
|
|
|
|
|
2018-07-17 19:10:05 +03:00
|
|
|
const debug = require('ghost-ignition').debug('boot:init');
|
|
|
|
const Promise = require('bluebird');
|
2020-05-27 20:47:53 +03:00
|
|
|
const config = require('../shared/config');
|
2020-05-28 21:30:23 +03:00
|
|
|
const {events, i18n} = require('./lib/common');
|
|
|
|
const logging = require('../shared/logging');
|
2018-07-17 19:10:05 +03:00
|
|
|
const migrator = require('./data/db/migrator');
|
2020-05-28 13:57:02 +03:00
|
|
|
const urlUtils = require('./../shared/url-utils');
|
2015-11-12 15:29:45 +03:00
|
|
|
|
2019-11-05 12:03:27 +03:00
|
|
|
// Frontend Components
|
|
|
|
const themeService = require('../frontend/services/themes');
|
2020-03-23 22:43:01 +03:00
|
|
|
const appService = require('../frontend/services/apps');
|
2020-09-09 15:28:12 +03:00
|
|
|
const frontendSettings = require('../frontend/services/settings');
|
2019-11-05 12:03:27 +03:00
|
|
|
|
2021-01-27 01:54:19 +03:00
|
|
|
async function initialiseServices() {
|
2018-08-06 18:18:59 +03:00
|
|
|
// CASE: When Ghost is ready with bootstrapping (db migrations etc.), we can trigger the router creation.
|
|
|
|
// Reason is that the routers access the routes.yaml, which shouldn't and doesn't have to be validated to
|
|
|
|
// start Ghost in maintenance mode.
|
2019-11-05 12:03:27 +03:00
|
|
|
// Routing is a bridge between the frontend and API
|
2019-06-19 12:30:28 +03:00
|
|
|
const routing = require('../frontend/services/routing');
|
2019-11-05 12:03:27 +03:00
|
|
|
// We pass the themeService API version here, so that the frontend services are less tightly-coupled
|
|
|
|
routing.bootstrap.start(themeService.getApiVersion());
|
2018-08-06 18:18:59 +03:00
|
|
|
|
2020-09-09 15:28:12 +03:00
|
|
|
const settings = require('./services/settings');
|
2020-04-29 18:44:27 +03:00
|
|
|
const permissions = require('./services/permissions');
|
|
|
|
const xmlrpc = require('./services/xmlrpc');
|
|
|
|
const slack = require('./services/slack');
|
|
|
|
const {mega} = require('./services/mega');
|
|
|
|
const webhooks = require('./services/webhooks');
|
|
|
|
const scheduling = require('./adapters/scheduling');
|
2018-04-16 00:53:18 +03:00
|
|
|
|
2018-07-17 19:10:05 +03:00
|
|
|
debug('`initialiseServices` Start...');
|
2020-09-09 15:28:12 +03:00
|
|
|
const getRoutesHash = () => frontendSettings.getCurrentHash('routes');
|
2018-04-16 00:53:18 +03:00
|
|
|
|
2021-01-27 01:54:19 +03:00
|
|
|
await Promise.all([
|
2014-05-07 04:49:25 +04:00
|
|
|
// Initialize the permissions actions and objects
|
2018-07-17 19:10:05 +03:00
|
|
|
permissions.init(),
|
|
|
|
xmlrpc.listen(),
|
|
|
|
slack.listen(),
|
2019-11-04 10:38:40 +03:00
|
|
|
mega.listen(),
|
2018-07-17 19:10:05 +03:00
|
|
|
webhooks.listen(),
|
2020-09-09 15:28:12 +03:00
|
|
|
settings.syncRoutesHash(getRoutesHash),
|
2020-03-23 22:43:01 +03:00
|
|
|
appService.init(),
|
2018-07-17 19:10:05 +03:00
|
|
|
scheduling.init({
|
2019-05-06 13:26:08 +03:00
|
|
|
// NOTE: When changing API version need to consider how to migrate custom scheduling adapters
|
|
|
|
// that rely on URL to lookup persisted scheduled records (jobs, etc.). Ref: https://github.com/TryGhost/Ghost/pull/10726#issuecomment-489557162
|
2020-04-05 19:52:48 +03:00
|
|
|
apiUrl: urlUtils.urlFor('api', {version: 'v3', versionType: 'admin'}, true)
|
2018-07-17 19:10:05 +03:00
|
|
|
})
|
2021-01-27 01:54:19 +03:00
|
|
|
]);
|
2016-10-06 15:27:35 +03:00
|
|
|
|
2021-01-27 01:54:19 +03:00
|
|
|
debug('XMLRPC, Slack, MEGA, Webhooks, Scheduling, Permissions done');
|
|
|
|
|
|
|
|
// Initialise analytics events
|
|
|
|
if (config.get('segment:key')) {
|
|
|
|
require('./analytics-events').init();
|
|
|
|
}
|
|
|
|
|
|
|
|
debug('...`initialiseServices` End');
|
2013-11-23 21:54:47 +04:00
|
|
|
}
|
|
|
|
|
2020-12-02 11:17:44 +03:00
|
|
|
async function initializeRecurringJobs() {
|
2020-11-26 16:09:38 +03:00
|
|
|
// we don't want to kick off scheduled/recurring jobs that will interfere with tests
|
|
|
|
if (process.env.NODE_ENV.match(/^testing/)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-12-01 12:10:43 +03:00
|
|
|
if (config.get('backgroundJobs:emailAnalytics')) {
|
2020-12-02 11:17:44 +03:00
|
|
|
const emailAnalyticsJobs = require('./services/email-analytics/jobs');
|
|
|
|
await emailAnalyticsJobs.scheduleRecurringJobs();
|
2020-12-01 12:10:43 +03:00
|
|
|
}
|
2020-11-26 16:09:38 +03:00
|
|
|
}
|
|
|
|
|
2021-01-27 03:33:55 +03:00
|
|
|
const initExpressApps = async () => {
|
|
|
|
await frontendSettings.init();
|
|
|
|
debug('Frontend settings done');
|
|
|
|
|
|
|
|
await themeService.init();
|
|
|
|
debug('Themes done');
|
|
|
|
|
|
|
|
const parentApp = require('./web/parent/app')();
|
|
|
|
debug('Express Apps done');
|
|
|
|
|
|
|
|
return parentApp;
|
|
|
|
};
|
|
|
|
|
2018-07-17 19:10:05 +03:00
|
|
|
/**
|
|
|
|
* - initialise models
|
|
|
|
* - initialise i18n
|
2021-01-27 03:33:55 +03:00
|
|
|
* - start the ghost server
|
2018-07-17 19:10:05 +03:00
|
|
|
* - load all settings into settings cache (almost every component makes use of this cache)
|
2021-01-27 03:33:55 +03:00
|
|
|
* - enable maintenance mode if migrations are missing
|
2018-07-17 19:10:05 +03:00
|
|
|
* - load active theme
|
|
|
|
* - create our express apps (site, admin, api)
|
|
|
|
*/
|
2021-01-27 01:54:19 +03:00
|
|
|
const minimalRequiredSetupToStartGhost = async (dbState) => {
|
2018-07-17 19:10:05 +03:00
|
|
|
const settings = require('./services/settings');
|
2020-08-11 23:26:34 +03:00
|
|
|
const jobService = require('./services/jobs');
|
2018-07-17 19:10:05 +03:00
|
|
|
const models = require('./models');
|
|
|
|
const GhostServer = require('./ghost-server');
|
|
|
|
|
|
|
|
// Initialize Ghost core internationalization
|
2020-04-30 22:26:12 +03:00
|
|
|
i18n.init();
|
2018-07-17 19:10:05 +03:00
|
|
|
debug('Default i18n done for core');
|
|
|
|
|
|
|
|
models.init();
|
|
|
|
debug('Models done');
|
|
|
|
|
2021-01-27 03:33:55 +03:00
|
|
|
const ghostServer = new GhostServer();
|
2021-01-27 01:54:19 +03:00
|
|
|
|
|
|
|
ghostServer.registerCleanupTask(async () => {
|
|
|
|
await jobService.shutdown();
|
|
|
|
});
|
|
|
|
|
|
|
|
// CASE: all good or db was just initialised
|
|
|
|
if (dbState === 1 || dbState === 2) {
|
2021-01-27 03:33:55 +03:00
|
|
|
await settings.init();
|
|
|
|
debug('Settings done');
|
|
|
|
const parentApp = await initExpressApps();
|
|
|
|
ghostServer.rootApp = parentApp;
|
|
|
|
|
2021-01-27 01:54:19 +03:00
|
|
|
events.emit('db.ready');
|
|
|
|
|
|
|
|
await initialiseServices();
|
|
|
|
}
|
|
|
|
|
|
|
|
// CASE: migrations required, put blog into maintenance mode
|
|
|
|
if (dbState === 4) {
|
|
|
|
config.set('maintenance:enabled', true);
|
2021-01-27 02:16:00 +03:00
|
|
|
logging.info('Blog is in maintenance mode.');
|
2021-01-27 01:54:19 +03:00
|
|
|
|
2021-01-28 06:07:13 +03:00
|
|
|
ghostServer.rootApp = require('./web/maintenance');
|
2021-01-27 01:54:19 +03:00
|
|
|
|
2021-01-27 06:40:31 +03:00
|
|
|
try {
|
|
|
|
migrator.migrate()
|
|
|
|
.then(async () => {
|
|
|
|
await settings.init();
|
|
|
|
debug('Settings done');
|
2021-01-27 03:33:55 +03:00
|
|
|
|
2021-01-27 06:40:31 +03:00
|
|
|
const parentApp = await initExpressApps();
|
|
|
|
ghostServer.swapHttpApp(parentApp);
|
2021-01-27 03:33:55 +03:00
|
|
|
|
2021-01-27 06:40:31 +03:00
|
|
|
events.emit('db.ready');
|
2021-01-27 01:54:19 +03:00
|
|
|
|
2021-01-27 06:40:31 +03:00
|
|
|
await initialiseServices();
|
2021-01-27 01:54:19 +03:00
|
|
|
|
2021-01-27 06:40:31 +03:00
|
|
|
config.set('maintenance:enabled', false);
|
|
|
|
logging.info('Blog is out of maintenance mode.');
|
2021-01-27 01:54:19 +03:00
|
|
|
|
2021-01-27 06:40:31 +03:00
|
|
|
await GhostServer.announceServerReadiness();
|
|
|
|
});
|
2021-01-27 01:54:19 +03:00
|
|
|
} catch (err) {
|
|
|
|
try {
|
|
|
|
await GhostServer.announceServerReadiness(err);
|
|
|
|
} finally {
|
|
|
|
logging.error(err);
|
|
|
|
setTimeout(() => {
|
2021-02-05 23:13:50 +03:00
|
|
|
process.exit(1);
|
2021-01-27 01:54:19 +03:00
|
|
|
}, 100);
|
2018-07-17 19:10:05 +03:00
|
|
|
}
|
2021-01-27 01:54:19 +03:00
|
|
|
}
|
|
|
|
}
|
2021-01-27 02:16:00 +03:00
|
|
|
|
|
|
|
initializeRecurringJobs();
|
|
|
|
|
|
|
|
return ghostServer;
|
2018-07-17 19:10:05 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Connect to database.
|
|
|
|
* Check db state.
|
|
|
|
*/
|
2021-01-27 01:54:19 +03:00
|
|
|
const isDatabaseInitialisationRequired = async () => {
|
2018-07-17 19:10:05 +03:00
|
|
|
const db = require('./data/db/connection');
|
|
|
|
|
2021-01-27 01:54:19 +03:00
|
|
|
let dbState = await migrator.getState();
|
2018-07-17 19:10:05 +03:00
|
|
|
|
2021-01-27 01:54:19 +03:00
|
|
|
// CASE: db initialisation required, wait till finished
|
|
|
|
if (dbState === 2) {
|
2021-01-27 01:59:59 +03:00
|
|
|
await migrator.dbInit();
|
2021-01-27 01:54:19 +03:00
|
|
|
}
|
2018-07-17 19:10:05 +03:00
|
|
|
|
2021-01-27 01:54:19 +03:00
|
|
|
// CASE: is db incompatible? e.g. you can't connect a 0.11 database with Ghost 1.0 or 2.0
|
|
|
|
if (dbState === 3) {
|
|
|
|
await migrator.isDbCompatible(db);
|
|
|
|
|
|
|
|
dbState = 2;
|
|
|
|
await migrator.dbInit();
|
|
|
|
}
|
|
|
|
|
|
|
|
return minimalRequiredSetupToStartGhost(dbState);
|
2018-07-17 19:10:05 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
module.exports = isDatabaseInitialisationRequired;
|