2014-09-01 19:46:37 +00:00
|
|
|
var packages = require('../../../package.json'),
|
|
|
|
path = require('path'),
|
|
|
|
crypto = require('crypto'),
|
|
|
|
fs = require('fs'),
|
2015-11-12 13:29:45 +01:00
|
|
|
i18n = require('../i18n'),
|
2014-09-01 19:46:37 +00:00
|
|
|
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'),
|
2015-11-26 16:20:11 +08:00
|
|
|
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
|
|
|
|
};
|
2014-09-01 19:46:37 +00:00
|
|
|
|
|
|
|
checks = {
|
|
|
|
check: function check() {
|
2015-06-18 16:21:16 +01:00
|
|
|
this.nodeVersion();
|
|
|
|
this.nodeEnv();
|
2014-09-01 19:46:37 +00:00
|
|
|
this.packages();
|
|
|
|
this.contentPath();
|
|
|
|
this.sqlite();
|
|
|
|
},
|
|
|
|
|
2015-06-18 16:21:16 +01:00
|
|
|
// Make sure the node version is supported
|
|
|
|
nodeVersion: function checkNodeVersion() {
|
|
|
|
// Tell users if their node version is not supported, and exit
|
2015-11-11 02:56:10 +01:00
|
|
|
var semver = require('semver');
|
2015-11-12 13:29:45 +01:00
|
|
|
i18n.init();
|
2015-11-11 02:56:10 +01:00
|
|
|
|
|
|
|
if (process.env.GHOST_NODE_VERSION_CHECK !== 'false' &&
|
|
|
|
!semver.satisfies(process.versions.node, packages.engines.node) &&
|
|
|
|
!semver.satisfies(process.versions.node, packages.engines.iojs)) {
|
2015-11-12 13:29:45 +01:00
|
|
|
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/'}));
|
2015-11-11 02:56:10 +01:00
|
|
|
|
2015-11-26 16:20:11 +08:00
|
|
|
process.exit(exitCodes.NODE_VERSION_UNSUPPORTED);
|
2015-06-18 16:21:16 +01:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
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) {
|
2015-11-12 13:29:45 +01:00
|
|
|
console.error(i18n.t('errors.utils.startupcheck.cannotFindConfigForCurrentNode.error',
|
|
|
|
{nodeEnv: process.env.NODE_ENV}));
|
|
|
|
console.error(i18n.t('errors.utils.startupcheck.cannotFindConfigForCurrentNode.help'));
|
2015-06-18 16:21:16 +01:00
|
|
|
|
2015-11-26 16:20:11 +08:00
|
|
|
process.exit(exitCodes.NODE_ENV_CONFIG_MISSING);
|
2015-06-18 16:21:16 +01:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2014-09-01 19:46:37 +00:00
|
|
|
// 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 ');
|
|
|
|
|
2015-11-12 13:29:45 +01:00
|
|
|
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'}));
|
2014-09-01 19:46:37 +00:00
|
|
|
|
2015-11-26 16:20:11 +08:00
|
|
|
process.exit(exitCodes.DEPENDENCIES_MISSING);
|
2014-09-01 19:46:37 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
// Check content path permissions
|
|
|
|
contentPath: function checkContentPaths() {
|
|
|
|
if (mode !== 'production' && mode !== 'development') {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
var configFile,
|
|
|
|
config,
|
|
|
|
contentPath,
|
|
|
|
contentSubPaths = ['apps', 'data', 'images', 'themes'],
|
|
|
|
fd,
|
2015-11-12 13:29:45 +01:00
|
|
|
errorHeader = i18n.t('errors.utils.startupcheck.unableToAccessContentPath.error'),
|
|
|
|
errorHelp = i18n.t('errors.utils.startupcheck.unableToAccessContentPath.help', {url: 'http://support.ghost.org'});
|
2014-09-01 19:46:37 +00:00
|
|
|
|
|
|
|
// 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);
|
|
|
|
|
2015-11-26 16:20:11 +08:00
|
|
|
process.exit(exitCodes.CONTENT_PATH_NOT_ACCESSIBLE);
|
2014-09-01 19:46:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// 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);
|
|
|
|
|
2015-11-26 16:20:11 +08:00
|
|
|
process.exit(exitCodes.CONTENT_PATH_NOT_WRITABLE);
|
2014-09-01 19:46:37 +00:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
// 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;
|
|
|
|
}
|
|
|
|
|
2015-11-12 13:29:45 +01:00
|
|
|
console.error(i18n.t('errors.utils.startupcheck.unableToOpenSqlite3Db.error'));
|
2014-09-01 19:46:37 +00:00
|
|
|
console.error(' ' + e.message);
|
2015-11-12 13:29:45 +01:00
|
|
|
console.error(i18n.t('errors.utils.startupcheck.unableToOpenSqlite3Db.help', {url: 'http://support.ghost.org'}));
|
2014-09-01 19:46:37 +00:00
|
|
|
|
2015-11-26 16:20:11 +08:00
|
|
|
process.exit(exitCodes.SQLITE_DB_NOT_WRITABLE);
|
2014-09-01 19:46:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
module.exports = checks;
|