0
Fork 0
mirror of https://github.com/TryGhost/Ghost.git synced 2025-01-20 22:42:53 -05:00
ghost/core/server/utils/startup-check.js

216 lines
7.6 KiB
JavaScript
Raw Normal View History

var packages = require('../../../package.json'),
path = require('path'),
crypto = require('crypto'),
fs = require('fs'),
i18n = require('../i18n'),
mode = process.env.NODE_ENV === undefined ? 'development' : process.env.NODE_ENV,
appRoot = path.resolve(__dirname, '../../../'),
configFilePath = process.env.GHOST_CONFIG || path.join(appRoot, 'config.js'),
checks,
exitCodes = {
NODE_VERSION_UNSUPPORTED: 231,
NODE_ENV_CONFIG_MISSING: 232,
DEPENDENCIES_MISSING: 233,
CONTENT_PATH_NOT_ACCESSIBLE: 234,
CONTENT_PATH_NOT_WRITABLE: 235,
SQLITE_DB_NOT_WRITABLE: 236
};
checks = {
check: function check() {
this.nodeVersion();
this.nodeEnv();
this.packages();
this.contentPath();
this.sqlite();
},
// Make sure the node version is supported
nodeVersion: function checkNodeVersion() {
// Tell users if their node version is not supported, and exit
var semver = require('semver');
i18n.init();
if (process.env.GHOST_NODE_VERSION_CHECK !== 'false' &&
!semver.satisfies(process.versions.node, packages.engines.node) &&
!semver.satisfies(process.versions.node, packages.engines.iojs)) {
console.error(i18n.t('errors.utils.startupcheck.unsupportedNodeVersion.error'));
console.error(i18n.t('errors.utils.startupcheck.unsupportedNodeVersion.context',
{neededVersion: packages.engines.node, usedVersion: process.versions.node}));
console.error(i18n.t('errors.utils.startupcheck.unsupportedNodeVersion.help',
{url: 'http://support.ghost.org/supported-node-versions/'}));
process.exit(exitCodes.NODE_VERSION_UNSUPPORTED);
}
},
nodeEnv: function checkNodeEnvState() {
// Check if config path resolves, if not check for NODE_ENV in config.example.js prior to copy
var fd,
configFile,
config;
try {
fd = fs.openSync(configFilePath, 'r');
fs.closeSync(fd);
} catch (e) {
configFilePath = path.join(appRoot, 'config.example.js');
}
configFile = require(configFilePath);
config = configFile[mode];
if (!config) {
console.error(i18n.t('errors.utils.startupcheck.cannotFindConfigForCurrentNode.error',
{nodeEnv: process.env.NODE_ENV}));
console.error(i18n.t('errors.utils.startupcheck.cannotFindConfigForCurrentNode.help'));
process.exit(exitCodes.NODE_ENV_CONFIG_MISSING);
}
},
// Make sure package.json dependencies have been installed.
packages: function checkPackages() {
if (mode !== 'production' && mode !== 'development') {
return;
}
var errors = [];
Object.keys(packages.dependencies).forEach(function (p) {
try {
require.resolve(p);
} catch (e) {
errors.push(e.message);
}
});
if (!errors.length) {
return;
}
errors = errors.join('\n ');
console.error(i18n.t('errors.utils.startupcheck.ghostMissingDependencies.error', {error: errors}));
console.error(i18n.t('errors.utils.startupcheck.ghostMissingDependencies.explain'));
console.error(i18n.t('errors.utils.startupcheck.ghostMissingDependencies.help', {url: 'http://support.ghost.org'}));
process.exit(exitCodes.DEPENDENCIES_MISSING);
},
// Check content path permissions
contentPath: function checkContentPaths() {
if (mode !== 'production' && mode !== 'development') {
return;
}
var configFile,
config,
contentPath,
contentSubPaths = ['apps', 'data', 'images', 'themes'],
fd,
errorHeader = i18n.t('errors.utils.startupcheck.unableToAccessContentPath.error'),
errorHelp = i18n.t('errors.utils.startupcheck.unableToAccessContentPath.help', {url: 'http://support.ghost.org'});
// Get the content path to test. If it's defined in config.js use that, if not use the default
try {
configFile = require(configFilePath);
config = configFile[mode];
if (config && config.paths && config.paths.contentPath) {
contentPath = config.paths.contentPath;
} else {
contentPath = path.join(appRoot, 'content');
}
} catch (e) {
// If config.js doesn't exist yet, check the default content path location
contentPath = path.join(appRoot, 'content');
}
// Use all sync io calls so that we stay in this function until all checks are complete
// Check the root content path
try {
fd = fs.openSync(contentPath, 'r');
fs.closeSync(fd);
} catch (e) {
console.error(errorHeader);
console.error(' ' + e.message);
console.error('\n' + errorHelp);
process.exit(exitCodes.CONTENT_PATH_NOT_ACCESSIBLE);
}
// Check each of the content path subdirectories
try {
contentSubPaths.forEach(function (sub) {
var dir = path.join(contentPath, sub),
randomFile = path.join(dir, crypto.randomBytes(8).toString('hex'));
fd = fs.openSync(dir, 'r');
fs.closeSync(fd);
// Check write access to directory by attempting to create a random file
fd = fs.openSync(randomFile, 'wx+');
fs.closeSync(fd);
fs.unlinkSync(randomFile);
});
} catch (e) {
console.error(errorHeader);
console.error(' ' + e.message);
console.error('\n' + errorHelp);
process.exit(exitCodes.CONTENT_PATH_NOT_WRITABLE);
}
},
// Make sure sqlite3 database is available for read/write
sqlite: function checkSqlite() {
if (mode !== 'production' && mode !== 'development') {
return;
}
var configFile,
config,
appRoot = path.resolve(__dirname, '../../../'),
dbPath,
fd;
try {
configFile = require(configFilePath);
config = configFile[mode];
// Abort check if database type is not sqlite3
if (config && config.database && config.database.client !== 'sqlite3') {
return;
}
if (config && config.database && config.database.connection) {
dbPath = config.database.connection.filename;
}
} catch (e) {
// If config.js doesn't exist, use the default path
dbPath = path.join(appRoot, 'content', 'data', mode === 'production' ? 'ghost.db' : 'ghost-dev.db');
}
// Check for read/write access on sqlite db file
try {
fd = fs.openSync(dbPath, 'r+');
fs.closeSync(fd);
} catch (e) {
// Database file not existing is not an error as sqlite will create it.
if (e.code === 'ENOENT') {
return;
}
console.error(i18n.t('errors.utils.startupcheck.unableToOpenSqlite3Db.error'));
console.error(' ' + e.message);
console.error(i18n.t('errors.utils.startupcheck.unableToOpenSqlite3Db.help', {url: 'http://support.ghost.org'}));
process.exit(exitCodes.SQLITE_DB_NOT_WRITABLE);
}
}
};
module.exports = checks;