2014-09-01 14:46:37 -05:00
var packages = require ( '../../../package.json' ) ,
path = require ( 'path' ) ,
crypto = require ( 'crypto' ) ,
fs = require ( 'fs' ) ,
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 03:20:11 -05: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 ,
2016-02-13 14:06:52 -05:00
SQLITE _DB _NOT _WRITABLE : 236 ,
BUILT _FILES _DO _NOT _EXIST : 237
2015-11-26 03:20:11 -05:00
} ;
2014-09-01 14:46:37 -05:00
checks = {
check : function check ( ) {
2015-06-18 10:21:16 -05:00
this . nodeVersion ( ) ;
this . nodeEnv ( ) ;
2014-09-01 14:46:37 -05:00
this . packages ( ) ;
this . contentPath ( ) ;
2016-02-07 18:35:47 -05:00
this . mail ( ) ;
2014-09-01 14:46:37 -05:00
this . sqlite ( ) ;
2016-02-13 14:06:52 -05:00
this . builtFilesExist ( ) ;
2014-09-01 14:46:37 -05:00
} ,
2015-06-18 10:21:16 -05:00
// Make sure the node version is supported
nodeVersion : function checkNodeVersion ( ) {
// Tell users if their node version is not supported, and exit
2015-11-10 20:56:10 -05:00
var semver = require ( 'semver' ) ;
if ( process . env . GHOST _NODE _VERSION _CHECK !== 'false' &&
! semver . satisfies ( process . versions . node , packages . engines . node ) &&
! semver . satisfies ( process . versions . node , packages . engines . iojs ) ) {
2016-01-13 08:38:34 -05:00
console . error ( '\x1B[31mERROR: Unsupported version of Node' ) ;
console . error ( '\x1B[31mGhost needs Node version ' + packages . engines . node +
' you are using version ' + process . versions . node + '\033[0m\n' ) ;
console . error ( '\x1B[32mPlease see http://support.ghost.org/supported-node-versions/ for more information\033[0m' ) ;
2015-11-10 20:56:10 -05:00
2015-11-26 03:20:11 -05:00
process . exit ( exitCodes . NODE _VERSION _UNSUPPORTED ) ;
2015-06-18 10:21:16 -05: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 ) {
2016-01-13 08:38:34 -05:00
console . error ( '\x1B[31mERROR: Cannot find the configuration for the current NODE_ENV: ' +
process . env . NODE _ENV + '\033[0m\n' ) ;
console . error ( '\x1B[32mEnsure your config.js has a section for the current NODE_ENV value' +
' and is formatted properly.\033[0m' ) ;
2015-06-18 10:21:16 -05:00
2015-11-26 03:20:11 -05:00
process . exit ( exitCodes . NODE _ENV _CONFIG _MISSING ) ;
2015-06-18 10:21:16 -05:00
}
} ,
2014-09-01 14:46:37 -05: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 ' ) ;
2016-01-13 08:38:34 -05:00
console . error ( '\x1B[31mERROR: Ghost is unable to start due to missing dependencies:\033[0m\n ' + errors ) ;
console . error ( '\x1B[32m\nPlease run `npm install --production` and try starting Ghost again.' ) ;
console . error ( '\x1B[32mHelp and documentation can be found at http://support.ghost.org.\033[0m\n' ) ;
2014-09-01 14:46:37 -05:00
2015-11-26 03:20:11 -05:00
process . exit ( exitCodes . DEPENDENCIES _MISSING ) ;
2014-09-01 14:46:37 -05:00
} ,
// Check content path permissions
contentPath : function checkContentPaths ( ) {
if ( mode !== 'production' && mode !== 'development' ) {
return ;
}
var configFile ,
config ,
contentPath ,
contentSubPaths = [ 'apps' , 'data' , 'images' , 'themes' ] ,
fd ,
2016-01-13 08:38:34 -05:00
errorHeader = '\x1B[31mERROR: Unable to access Ghost\'s content path:\033[0m' ,
errorHelp = '\x1B[32mCheck that the content path exists and file system permissions are correct.' +
'\nHelp and documentation can be found at http://support.ghost.org.\033[0m' ;
2014-09-01 14:46:37 -05: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 03:20:11 -05:00
process . exit ( exitCodes . CONTENT _PATH _NOT _ACCESSIBLE ) ;
2014-09-01 14:46:37 -05: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 03:20:11 -05:00
process . exit ( exitCodes . CONTENT _PATH _NOT _WRITABLE ) ;
2014-09-01 14:46:37 -05: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 ;
}
2016-01-13 08:38:34 -05:00
console . error ( '\x1B[31mERROR: Unable to open sqlite3 database file for read/write\033[0m' ) ;
2014-09-01 14:46:37 -05:00
console . error ( ' ' + e . message ) ;
2016-01-13 08:38:34 -05:00
console . error ( '\n\x1B[32mCheck that the sqlite3 database file permissions allow read and write access.' ) ;
console . error ( 'Help and documentation can be found at http://support.ghost.org.\033[0m' ) ;
2014-09-01 14:46:37 -05:00
2015-11-26 03:20:11 -05:00
process . exit ( exitCodes . SQLITE _DB _NOT _WRITABLE ) ;
2014-09-01 14:46:37 -05:00
}
2016-02-07 18:35:47 -05:00
} ,
mail : function checkMail ( ) {
var configFile ,
config ;
try {
configFile = require ( configFilePath ) ;
config = configFile [ mode ] ;
} catch ( e ) {
configFilePath = path . join ( appRoot , 'config.example.js' ) ;
}
if ( ! config . mail || ! config . mail . transport ) {
console . error ( '\x1B[31mWARNING: Ghost is attempting to use a direct method to send email. \nIt is recommended that you explicitly configure an email service.\033[0m' ) ;
console . error ( '\x1B[32mHelp and documentation can be found at http://support.ghost.org/mail.\033[0m\n' ) ;
}
2016-02-13 14:06:52 -05:00
} ,
builtFilesExist : function builtFilesExist ( ) {
var configFile ,
config ,
location ,
fileNames = [ 'ghost.js' , 'vendor.js' , 'ghost.css' , 'vendor.css' ] ;
try {
configFile = require ( configFilePath ) ;
config = configFile [ mode ] ;
if ( config . paths && config . paths . clientAssets ) {
location = config . paths . clientAssets ;
} else {
location = path . join ( appRoot , '/core/built/assets/' ) ;
}
} catch ( e ) {
location = path . join ( appRoot , '/core/built/assets/' ) ;
}
if ( process . env . NODE _ENV === 'production' ) {
// Production uses `.min` files
fileNames = fileNames . map ( function ( file ) {
return file . replace ( '.' , '.min.' ) ;
} ) ;
}
function checkExist ( fileName ) {
try {
fs . statSync ( fileName ) ;
} catch ( e ) {
console . error ( '\x1B[31mERROR: Javascript files have not been built.\033[0m' ) ;
console . error ( '\n\x1B[32mPlease read the getting started instructions at:' ) ;
console . error ( 'https://github.com/TryGhost/Ghost#getting-started\033[0m' ) ;
process . exit ( exitCodes . BUILT _FILES _DO _NOT _EXIST ) ;
}
}
fileNames . forEach ( function ( fileName ) {
checkExist ( location + fileName ) ;
} ) ;
2014-09-01 14:46:37 -05:00
}
} ;
module . exports = checks ;