/ *
NodeBB - A better forum platform for the modern web
https : //github.com/NodeBB/NodeBB/
Copyright ( C ) 2013 - 2017 NodeBB Inc .
This program is free software : you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation , either version 3 of the License , or
( at your option ) any later version .
This program is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
You should have received a copy of the GNU General Public License
along with this program . If not , see < http : //www.gnu.org/licenses/>.
* /
'use strict' ;
if ( require . main !== module ) {
require . main . require = function ( path ) {
return require ( path ) ;
} ;
}
var nconf = require ( 'nconf' ) ;
nconf . argv ( ) . env ( {
separator : '__' ,
lowerCase : true ,
} ) ;
var url = require ( 'url' ) ;
var async = require ( 'async' ) ;
var winston = require ( 'winston' ) ;
var path = require ( 'path' ) ;
var pkg = require ( './package.json' ) ;
var file = require ( './src/file' ) ;
var debug = require ( './src/meta/debugFork' ) . debugging ;
global . env = process . env . NODE _ENV || 'production' ;
winston . remove ( winston . transports . Console ) ;
winston . add ( winston . transports . Console , {
colorize : true ,
timestamp : function ( ) {
var date = new Date ( ) ;
return nconf . get ( 'json-logging' ) ? date . toJSON ( ) : date . getDate ( ) + '/' + ( date . getMonth ( ) + 1 ) + ' ' + date . toTimeString ( ) . substr ( 0 , 8 ) + ' [' + global . process . pid + ']' ;
} ,
level : nconf . get ( 'log-level' ) || ( global . env === 'production' ? 'info' : 'verbose' ) ,
json : ( ! ! nconf . get ( 'json-logging' ) ) ,
stringify : ( ! ! nconf . get ( 'json-logging' ) ) ,
} ) ;
if ( debug ) {
var winstonCommon = require ( 'winston/lib/winston/common' ) ;
// Override to use real console.log etc for VSCode debugger
winston . transports . Console . prototype . log = function ( level , message , meta , callback ) {
const output = winstonCommon . log ( Object . assign ( { } , this , {
level ,
message ,
meta ,
} ) ) ;
console [ level in console ? level : 'log' ] ( output ) ;
setImmediate ( callback , null , true ) ;
} ;
}
// Alternate configuration file support
var configFile = path . join ( _ _dirname , '/config.json' ) ;
if ( nconf . get ( 'config' ) ) {
configFile = path . resolve ( _ _dirname , nconf . get ( 'config' ) ) ;
}
var configExists = file . existsSync ( configFile ) || ( nconf . get ( 'url' ) && nconf . get ( 'secret' ) && nconf . get ( 'database' ) ) ;
loadConfig ( ) ;
versionCheck ( ) ;
if ( ! process . send ) {
// If run using `node app`, log GNU copyright info along with server info
winston . info ( 'NodeBB v' + nconf . get ( 'version' ) + ' Copyright (C) 2013-' + ( new Date ( ) ) . getFullYear ( ) + ' NodeBB Inc.' ) ;
winston . info ( 'This program comes with ABSOLUTELY NO WARRANTY.' ) ;
winston . info ( 'This is free software, and you are welcome to redistribute it under certain conditions.' ) ;
winston . info ( '' ) ;
}
if ( nconf . get ( 'setup' ) || nconf . get ( 'install' ) ) {
setup ( ) ;
} else if ( ! configExists ) {
require ( './install/web' ) . install ( nconf . get ( 'port' ) ) ;
} else if ( nconf . get ( 'upgrade' ) ) {
upgrade ( ) ;
} else if ( nconf . get ( 'reset' ) ) {
async . waterfall ( [
async . apply ( require ( './src/reset' ) . reset ) ,
async . apply ( require ( './src/meta/build' ) . buildAll ) ,
] , function ( err ) {
process . exit ( err ? 1 : 0 ) ;
} ) ;
} else if ( nconf . get ( 'activate' ) ) {
activate ( ) ;
} else if ( nconf . get ( 'plugins' ) ) {
listPlugins ( ) ;
} else if ( nconf . get ( 'build' ) ) {
require ( './src/meta/build' ) . build ( nconf . get ( 'build' ) ) ;
} else if ( nconf . get ( 'events' ) ) {
async . series ( [
async . apply ( require ( './src/database' ) . init ) ,
async . apply ( require ( './src/events' ) . output ) ,
] ) ;
} else {
require ( './src/start' ) . start ( ) ;
}
function loadConfig ( callback ) {
winston . verbose ( '* using configuration stored in: %s' , configFile ) ;
nconf . file ( {
file : configFile ,
} ) ;
nconf . defaults ( {
base _dir : _ _dirname ,
themes _path : path . join ( _ _dirname , 'node_modules' ) ,
upload _path : 'public/uploads' ,
views _dir : path . join ( _ _dirname , 'build/public/templates' ) ,
version : pkg . version ,
} ) ;
if ( ! nconf . get ( 'isCluster' ) ) {
nconf . set ( 'isPrimary' , 'true' ) ;
nconf . set ( 'isCluster' , 'false' ) ;
}
// Ensure themes_path is a full filepath
nconf . set ( 'themes_path' , path . resolve ( _ _dirname , nconf . get ( 'themes_path' ) ) ) ;
nconf . set ( 'core_templates_path' , path . join ( _ _dirname , 'src/views' ) ) ;
nconf . set ( 'base_templates_path' , path . join ( nconf . get ( 'themes_path' ) , 'nodebb-theme-persona/templates' ) ) ;
nconf . set ( 'upload_path' , path . resolve ( nconf . get ( 'base_dir' ) , nconf . get ( 'upload_path' ) ) ) ;
if ( nconf . get ( 'url' ) ) {
nconf . set ( 'url_parsed' , url . parse ( nconf . get ( 'url' ) ) ) ;
}
// Explicitly cast 'jobsDisabled' as Bool
var castAsBool = [ 'jobsDisabled' ] ;
nconf . stores . env . readOnly = false ;
castAsBool . forEach ( function ( prop ) {
var value = nconf . get ( prop ) ;
if ( value ) {
nconf . set ( prop , typeof value === 'boolean' ? value : String ( value ) . toLowerCase ( ) === 'true' ) ;
}
} ) ;
nconf . stores . env . readOnly = true ;
if ( typeof callback === 'function' ) {
callback ( ) ;
}
}
function setup ( ) {
winston . info ( 'NodeBB Setup Triggered via Command Line' ) ;
var install = require ( './src/install' ) ;
var build = require ( './src/meta/build' ) ;
process . stdout . write ( '\nWelcome to NodeBB!\n' ) ;
process . stdout . write ( '\nThis looks like a new installation, so you\'ll have to answer a few questions about your environment before we can proceed.\n' ) ;
process . stdout . write ( 'Press enter to accept the default setting (shown in brackets).\n' ) ;
async . series ( [
async . apply ( install . setup ) ,
async . apply ( loadConfig ) ,
async . apply ( build . buildAll ) ,
] , function ( err , data ) {
// Disregard build step data
data = data [ 0 ] ;
var separator = ' ' ;
if ( process . stdout . columns > 10 ) {
for ( var x = 0 , cols = process . stdout . columns - 10 ; x < cols ; x += 1 ) {
separator += '=' ;
}
}
process . stdout . write ( '\n' + separator + '\n\n' ) ;
if ( err ) {
winston . error ( 'There was a problem completing NodeBB setup: ' , err . message ) ;
} else {
if ( data . hasOwnProperty ( 'password' ) ) {
process . stdout . write ( 'An administrative user was automatically created for you:\n' ) ;
process . stdout . write ( ' Username: ' + data . username + '\n' ) ;
process . stdout . write ( ' Password: ' + data . password + '\n' ) ;
process . stdout . write ( '\n' ) ;
}
process . stdout . write ( 'NodeBB Setup Completed. Run \'./nodebb start\' to manually start your NodeBB server.\n' ) ;
// If I am a child process, notify the parent of the returned data before exiting (useful for notifying
// hosts of auto-generated username/password during headless setups)
if ( process . send ) {
process . send ( data ) ;
}
}
process . exit ( ) ;
} ) ;
}
function upgrade ( ) {
var db = require ( './src/database' ) ;
var meta = require ( './src/meta' ) ;
var upgrade = require ( './src/upgrade' ) ;
var build = require ( './src/meta/build' ) ;
var tasks = [ db . init , meta . configs . init ] ;
if ( nconf . get ( 'upgrade' ) !== true ) {
// Likely an upgrade script name passed in
tasks . push ( async . apply ( upgrade . runParticular , nconf . get ( 'upgrade' ) . split ( ',' ) ) ) ;
} else {
tasks . push ( upgrade . run , build . buildAll ) ;
}
// disable mongo timeouts during upgrade
nconf . set ( 'mongo:options:socketTimeoutMS' , 0 ) ;
async . series ( tasks , function ( err ) {
if ( err ) {
winston . error ( err . stack ) ;
process . exit ( 1 ) ;
} else {
process . exit ( 0 ) ;
}
} ) ;
}
function activate ( ) {
var db = require ( './src/database' ) ;
var plugins = require ( './src/plugins' ) ;
var events = require ( './src/events' ) ;
var plugin = nconf . get ( 'activate' ) ;
async . waterfall ( [
function ( next ) {
db . init ( next ) ;
} ,
function ( next ) {
if ( plugin . indexOf ( 'nodebb-' ) !== 0 ) {
// Allow omission of `nodebb-plugin-`
plugin = 'nodebb-plugin-' + plugin ;
}
plugins . isInstalled ( plugin , next ) ;
} ,
function ( isInstalled , next ) {
if ( ! isInstalled ) {
return next ( new Error ( 'plugin not installed' ) ) ;
}
winston . info ( 'Activating plugin `%s`' , plugin ) ;
db . sortedSetAdd ( 'plugins:active' , 0 , plugin , next ) ;
} ,
function ( next ) {
events . log ( {
type : 'plugin-activate' ,
text : plugin ,
} , next ) ;
} ,
] , function ( err ) {
if ( err ) {
winston . error ( err . message ) ;
}
process . exit ( err ? 1 : 0 ) ;
} ) ;
}
function listPlugins ( ) {
require ( './src/database' ) . init ( function ( err ) {
if ( err ) {
winston . error ( err . stack ) ;
process . exit ( 1 ) ;
}
var db = require ( './src/database' ) ;
db . getSortedSetRange ( 'plugins:active' , 0 , - 1 , function ( err , plugins ) {
if ( err ) {
winston . error ( err . stack ) ;
process . exit ( 1 ) ;
}
winston . info ( 'Active plugins: \n\t - ' + plugins . join ( '\n\t - ' ) ) ;
process . exit ( ) ;
} ) ;
} ) ;
}
function versionCheck ( ) {
var version = process . version . slice ( 1 ) ;
var range = pkg . engines . node ;
var semver = require ( 'semver' ) ;
var compatible = semver . satisfies ( version , range ) ;
if ( ! compatible ) {
winston . warn ( 'Your version of Node.js is too outdated for NodeBB. Please update your version of Node.js.' ) ;
winston . warn ( 'Recommended ' + range . green + ', ' . reset + version . yellow + ' provided\n' . reset ) ;
}
}