2013-12-06 09:13:15 -05:00
// This file manages the root level config.js.
// It will create config.js from config.exampe.js
// if it doesn't exist and then always attempt to load
// config.js into memory, error and quitting if config.js
// has an improper format.
2013-09-23 22:37:36 -05:00
var fs = require ( 'fs' ) ,
url = require ( 'url' ) ,
when = require ( 'when' ) ,
2014-05-27 21:57:53 +00:00
validator = require ( 'validator' ) ,
2014-05-09 12:11:29 +02:00
errors = require ( './server/errors' ) ,
2014-01-05 01:40:53 -05:00
config = require ( './server/config' ) ,
2013-10-10 12:44:31 +02:00
2014-01-05 01:40:53 -05:00
appRoot = config ( ) . paths . appRoot ,
configExample = config ( ) . paths . configExample ,
2014-02-08 10:41:15 -05:00
configFile ;
2013-12-28 17:53:05 +00:00
function readConfigFile ( envVal ) {
return require ( configFile ) [ envVal ] ;
}
2013-09-06 16:54:50 +01:00
function writeConfigFile ( ) {
var written = when . defer ( ) ;
/ * C h e c k f o r c o n f i g f i l e a n d c o p y f r o m c o n f i g . e x a m p l e . j s
if one doesn ' t exist . After that , start the server . * /
2013-12-28 17:53:05 +00:00
fs . exists ( configExample , function checkTemplate ( templateExists ) {
2013-09-06 16:54:50 +01:00
var read ,
2014-06-18 00:25:19 +00:00
write ,
error ;
2013-09-06 16:54:50 +01:00
if ( ! templateExists ) {
2014-06-18 00:25:19 +00:00
error = new Error ( 'Could not locate a configuration file.' ) ;
error . context = appRoot ;
error . help = 'Please check your deployment for config.js or config.example.js.' ;
return written . reject ( error ) ;
2013-09-06 16:54:50 +01:00
}
// Copy config.example.js => config.js
2013-12-28 17:53:05 +00:00
read = fs . createReadStream ( configExample ) ;
2013-09-06 16:54:50 +01:00
read . on ( 'error' , function ( err ) {
2014-06-18 00:25:19 +00:00
errors . logError ( new Error ( 'Could not open config.example.js for read.' ) , appRoot , 'Please check your deployment for config.js or config.example.js.' ) ;
return written . reject ( err ) ;
2013-09-06 16:54:50 +01:00
} ) ;
2013-10-16 02:38:30 -04:00
write = fs . createWriteStream ( configFile ) ;
2013-09-06 16:54:50 +01:00
write . on ( 'error' , function ( err ) {
2014-06-18 00:25:19 +00:00
errors . logError ( new Error ( 'Could not open config.js for write.' ) , appRoot , 'Please check your deployment for config.js or config.example.js.' ) ;
return written . reject ( err ) ;
2013-09-06 16:54:50 +01:00
} ) ;
2014-06-18 00:25:19 +00:00
write . on ( 'finish' , written . resolve ) ;
2013-09-06 16:54:50 +01:00
read . pipe ( write ) ;
} ) ;
return written . promise ;
}
2013-09-23 22:37:36 -05:00
function validateConfigEnvironment ( ) {
2013-12-28 16:01:08 +00:00
var envVal = process . env . NODE _ENV || undefined ,
2013-10-10 16:11:35 +01:00
hasHostAndPort ,
hasSocket ,
2013-09-23 22:37:36 -05:00
config ,
parsedUrl ;
try {
2013-12-28 17:53:05 +00:00
config = readConfigFile ( envVal ) ;
2014-06-18 00:25:19 +00:00
}
catch ( e ) {
return when . reject ( e ) ;
2013-09-23 22:37:36 -05:00
}
// Check if we don't even have a config
if ( ! config ) {
2014-06-18 00:25:19 +00:00
errors . logError ( new Error ( 'Cannot find the configuration for the current NODE_ENV' ) , 'NODE_ENV=' + envVal ,
2013-10-21 23:38:41 -07:00
'Ensure your config.js has a section for the current NODE_ENV value and is formatted properly.' ) ;
2014-06-18 00:25:19 +00:00
return when . reject ( new Error ( 'Unable to load config for NODE_ENV=' + envVal ) ) ;
2013-09-23 22:37:36 -05:00
}
// Check that our url is valid
2014-05-27 21:57:53 +00:00
if ( ! validator . isURL ( config . url , { protocols : [ 'http' , 'https' ] , require _protocol : true } ) ) {
2013-09-23 22:37:36 -05:00
errors . logError ( new Error ( 'Your site url in config.js is invalid.' ) , config . url , 'Please make sure this is a valid url before restarting' ) ;
2014-06-18 00:25:19 +00:00
return when . reject ( new Error ( 'invalid site url' ) ) ;
2013-09-23 22:37:36 -05:00
}
2014-05-27 21:57:53 +00:00
parsedUrl = url . parse ( config . url || 'invalid' , false , true ) ;
2013-12-28 17:53:05 +00:00
if ( /\/ghost(\/|$)/ . test ( parsedUrl . pathname ) ) {
errors . logError ( new Error ( 'Your site url in config.js cannot contain a subdirectory called ghost.' ) , config . url , 'Please rename the subdirectory before restarting' ) ;
2014-06-18 00:25:19 +00:00
return when . reject ( new Error ( 'ghost subdirectory not allowed' ) ) ;
2013-12-28 17:53:05 +00:00
}
2013-09-23 22:37:36 -05:00
// Check that we have database values
2013-12-28 17:53:05 +00:00
if ( ! config . database || ! config . database . client ) {
2013-09-23 22:37:36 -05:00
errors . logError ( new Error ( 'Your database configuration in config.js is invalid.' ) , JSON . stringify ( config . database ) , 'Please make sure this is a valid Bookshelf database configuration' ) ;
2014-06-18 00:25:19 +00:00
return when . reject ( new Error ( 'invalid database configuration' ) ) ;
2013-09-23 22:37:36 -05:00
}
2013-10-10 16:11:35 +01:00
hasHostAndPort = config . server && ! ! config . server . host && ! ! config . server . port ;
hasSocket = config . server && ! ! config . server . socket ;
2013-09-23 22:37:36 -05:00
// Check for valid server host and port values
2013-10-10 16:11:35 +01:00
if ( ! config . server || ! ( hasHostAndPort || hasSocket ) ) {
errors . logError ( new Error ( 'Your server values (socket, or host and port) in config.js are invalid.' ) , JSON . stringify ( config . server ) , 'Please provide them before restarting.' ) ;
2014-06-18 00:25:19 +00:00
return when . reject ( new Error ( 'invalid server configuration' ) ) ;
2013-09-23 22:37:36 -05:00
}
2013-11-25 22:22:59 -05:00
return when . resolve ( config ) ;
2013-09-23 22:37:36 -05:00
}
2014-02-08 10:41:15 -05:00
function loadConfig ( configFilePath ) {
2013-11-25 22:22:59 -05:00
var loaded = when . defer ( ) ,
pendingConfig ;
2014-02-08 10:41:15 -05:00
// Allow config file path to be taken from, in order of importance:
// environment process, passed in value, default location
configFile = process . env . GHOST _CONFIG || configFilePath || config ( ) . paths . config ;
2013-09-06 16:54:50 +01:00
/ * C h e c k f o r c o n f i g f i l e a n d c o p y f r o m c o n f i g . e x a m p l e . j s
if one doesn ' t exist . After that , start the server . * /
2013-10-16 02:38:30 -04:00
fs . exists ( configFile , function checkConfig ( configExists ) {
2013-11-25 22:22:59 -05:00
if ( ! configExists ) {
pendingConfig = writeConfigFile ( ) ;
2013-09-06 16:54:50 +01:00
}
2014-06-18 00:25:19 +00:00
2014-01-05 01:40:53 -05:00
when ( pendingConfig ) . then ( validateConfigEnvironment ) . then ( function ( rawConfig ) {
return config . init ( rawConfig ) . then ( loaded . resolve ) ;
2014-06-18 00:25:19 +00:00
} ) . catch ( loaded . reject ) ;
2013-09-06 16:54:50 +01:00
} ) ;
2014-01-05 01:40:53 -05:00
2013-09-06 16:54:50 +01:00
return loaded . promise ;
2013-11-20 08:58:52 -05:00
}
module . exports = loadConfig ;