// # Bootup // This file needs serious love & refactoring /** * make sure overrides get's called first! * - keeping the overrides require here works for installing Ghost as npm! * * the call order is the following: * - root index requires core module * - core index requires server * - overrides is the first package to load */ require('./overrides'); // Module dependencies var debug = require('debug')('ghost:boot:init'), uuid = require('uuid'), Promise = require('bluebird'), KnexMigrator = require('knex-migrator'), config = require('./config'), logging = require('./logging'), errors = require('./errors'), i18n = require('./i18n'), api = require('./api'), models = require('./models'), permissions = require('./permissions'), apps = require('./apps'), auth = require('./auth'), xmlrpc = require('./data/xml/xmlrpc'), slack = require('./data/slack'), GhostServer = require('./ghost-server'), scheduling = require('./scheduling'), readDirectory = require('./utils/read-directory'), utils = require('./utils'), knexMigrator = new KnexMigrator({ knexMigratorFilePath: config.get('paths:appRoot') }), dbHash; function initDbHashAndFirstRun() { return api.settings.read({key: 'dbHash', context: {internal: true}}).then(function (response) { var hash = response.settings[0].value, initHash; dbHash = hash; if (dbHash === null) { initHash = uuid.v4(); return api.settings.edit({settings: [{key: 'dbHash', value: initHash}]}, {context: {internal: true}}) .then(function (response) { dbHash = response.settings[0].value; return dbHash; // Use `then` here to do 'first run' actions }); } return dbHash; }); } // ## Initialise Ghost // Sets up the express server instances, runs init on a bunch of stuff, configures views, helpers, routes and more // Finally it returns an instance of GhostServer function init(options) { debug('Init Start...'); options = options || {}; var ghostServer, parentApp; // ### Initialisation // The server and its dependencies require a populated config // It returns a promise that is resolved when the application // has finished starting up. // Initialize Internationalization i18n.init(); debug('I18n done'); return readDirectory(config.getContentPath('apps')).then(function loadThemes(result) { config.set('paths:availableApps', result); return api.themes.loadThemes(); }).then(function () { debug('Themes & apps done'); models.init(); }).then(function () { return knexMigrator.isDatabaseOK() .catch(function (outerErr) { if (outerErr.code === 'DB_NOT_INITIALISED') { throw outerErr; } // CASE: migration table does not exist, figure out if database is compatible return models.Settings.findOne({key: 'databaseVersion', context: {internal: true}}) .then(function (response) { // CASE: no db version key, database is compatible if (!response) { throw outerErr; } throw new errors.DatabaseVersionError({ message: 'Your database version is not compatible with Ghost 1.0.0 Alpha (master branch)', context: 'Want to keep your DB? Use Ghost < 1.0.0 or the "stable" branch. Otherwise please delete your DB and restart Ghost.', help: 'More information on the Ghost 1.0.0 Alpha at https://support.ghost.org/v1-0-alpha' }); }) .catch(function (err) { // CASE: settings table does not exist if (err.errno === 1 || err.errno === 1146) { throw outerErr; } throw err; }); }); }).then(function () { // Populate any missing default settings return models.Settings.populateDefaults(); }).then(function () { debug('Models & database done'); return api.settings.updateSettingsCache(); }).then(function () { debug('Update settings cache done'); // Initialize the permissions actions and objects // NOTE: Must be done before initDbHashAndFirstRun calls return permissions.init(); }).then(function () { debug('Permissions done'); return Promise.join( // Check for or initialise a dbHash. initDbHashAndFirstRun(), // Initialize apps apps.init(), // Initialize xmrpc ping xmlrpc.listen(), // Initialize slack ping slack.listen() ); }).then(function () { debug('Apps, XMLRPC, Slack done'); // Setup our collection of express apps parentApp = require('./app')(); debug('Express Apps done'); // runs asynchronous auth.init({ authType: config.get('auth:type'), ghostAuthUrl: config.get('auth:url'), redirectUri: utils.url.urlFor('admin', true), clientUri: utils.url.urlFor('home', true), clientName: api.settings.cache.get('title'), clientDescription: api.settings.cache.get('description') }).then(function (response) { parentApp.use(response.auth); }).catch(function onAuthError(err) { logging.error(err); }); }).then(function () { debug('Auth done'); return new GhostServer(parentApp); }).then(function (_ghostServer) { ghostServer = _ghostServer; // scheduling can trigger api requests, that's why we initialize the module after the ghost server creation // scheduling module can create x schedulers with different adapters debug('Server done'); return scheduling.init({ schedulerUrl: config.get('scheduling').schedulerUrl, active: config.get('scheduling').active, apiUrl: utils.url.apiUrl(), internalPath: config.get('paths').internalSchedulingPath, contentPath: config.getContentPath('scheduling') }); }).then(function () { debug('Scheduling done'); debug('...Init End'); return ghostServer; }); } module.exports = init;