mirror of
				https://github.com/NodeBB/NodeBB.git
				synced 2025-10-31 11:05:54 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			300 lines
		
	
	
		
			7.3 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			300 lines
		
	
	
		
			7.3 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| 'use strict';
 | |
| 
 | |
| var winston = require('winston');
 | |
| var express = require('express');
 | |
| var bodyParser = require('body-parser');
 | |
| var fs = require('fs');
 | |
| var path = require('path');
 | |
| var childProcess = require('child_process');
 | |
| var less = require('less');
 | |
| var async = require('async');
 | |
| var uglify = require('uglify-es');
 | |
| var nconf = require('nconf');
 | |
| var Benchpress = require('benchpressjs');
 | |
| 
 | |
| var app = express();
 | |
| var server;
 | |
| 
 | |
| var formats = [
 | |
| 	winston.format.colorize(),
 | |
| ];
 | |
| 
 | |
| const timestampFormat = winston.format((info) => {
 | |
| 	var dateString = new Date().toISOString() + ' [' + global.process.pid + ']';
 | |
| 	info.level = dateString + ' - ' + info.level;
 | |
| 	return info;
 | |
| });
 | |
| formats.push(timestampFormat());
 | |
| formats.push(winston.format.splat());
 | |
| formats.push(winston.format.simple());
 | |
| 
 | |
| winston.configure({
 | |
| 	level: 'verbose',
 | |
| 	format: winston.format.combine.apply(null, formats),
 | |
| 	transports: [
 | |
| 		new winston.transports.Console({
 | |
| 			handleExceptions: true,
 | |
| 		}),
 | |
| 		new winston.transports.File({
 | |
| 			filename: 'logs/webinstall.log',
 | |
| 			handleExceptions: true,
 | |
| 		}),
 | |
| 	],
 | |
| });
 | |
| 
 | |
| var web = module.exports;
 | |
| 
 | |
| var scripts = [
 | |
| 	'node_modules/jquery/dist/jquery.js',
 | |
| 	'public/vendor/xregexp/xregexp.js',
 | |
| 	'public/vendor/xregexp/unicode/unicode-base.js',
 | |
| 	'public/src/utils.js',
 | |
| 	'public/src/installer/install.js',
 | |
| 	'node_modules/zxcvbn/dist/zxcvbn.js',
 | |
| ];
 | |
| 
 | |
| var installing = false;
 | |
| var success = false;
 | |
| var error = false;
 | |
| var launchUrl;
 | |
| 
 | |
| web.install = function (port) {
 | |
| 	port = port || 4567;
 | |
| 	winston.info('Launching web installer on port ' + port);
 | |
| 
 | |
| 	app.use(express.static('public', {}));
 | |
| 	app.engine('tpl', function (filepath, options, callback) {
 | |
| 		async.waterfall([
 | |
| 			function (next) {
 | |
| 				fs.readFile(filepath, 'utf-8', next);
 | |
| 			},
 | |
| 			function (buffer, next) {
 | |
| 				Benchpress.compileParse(buffer.toString(), options, next);
 | |
| 			},
 | |
| 		], callback);
 | |
| 	});
 | |
| 	app.set('view engine', 'tpl');
 | |
| 	app.set('views', path.join(__dirname, '../src/views'));
 | |
| 	app.use(bodyParser.urlencoded({
 | |
| 		extended: true,
 | |
| 	}));
 | |
| 
 | |
| 	async.parallel([compileLess, compileJS, copyCSS, loadDefaults], function (err) {
 | |
| 		if (err) {
 | |
| 			winston.error(err.stack);
 | |
| 		}
 | |
| 		setupRoutes();
 | |
| 		launchExpress(port);
 | |
| 	});
 | |
| };
 | |
| 
 | |
| 
 | |
| function launchExpress(port) {
 | |
| 	server = app.listen(port, function () {
 | |
| 		winston.info('Web installer listening on http://%s:%s', '0.0.0.0', port);
 | |
| 	});
 | |
| }
 | |
| 
 | |
| function setupRoutes() {
 | |
| 	app.get('/', welcome);
 | |
| 	app.post('/', install);
 | |
| 	app.post('/launch', launch);
 | |
| 	app.get('/ping', ping);
 | |
| 	app.get('/sping', ping);
 | |
| }
 | |
| 
 | |
| function ping(req, res) {
 | |
| 	res.status(200).send(req.path === '/sping' ? 'healthy' : '200');
 | |
| }
 | |
| 
 | |
| function welcome(req, res) {
 | |
| 	var dbs = ['redis', 'mongo', 'postgres'];
 | |
| 	var databases = dbs.map(function (databaseName) {
 | |
| 		var questions = require('../src/database/' + databaseName).questions.filter(function (question) {
 | |
| 			return question && !question.hideOnWebInstall;
 | |
| 		});
 | |
| 
 | |
| 		return {
 | |
| 			name: databaseName,
 | |
| 			questions: questions,
 | |
| 		};
 | |
| 	});
 | |
| 
 | |
| 	var defaults = require('./data/defaults');
 | |
| 
 | |
| 	res.render('install/index', {
 | |
| 		url: nconf.get('url') || (req.protocol + '://' + req.get('host')),
 | |
| 		launchUrl: launchUrl,
 | |
| 		skipGeneralSetup: !!nconf.get('url'),
 | |
| 		databases: databases,
 | |
| 		skipDatabaseSetup: !!nconf.get('database'),
 | |
| 		error: error,
 | |
| 		success: success,
 | |
| 		values: req.body,
 | |
| 		minimumPasswordLength: defaults.minimumPasswordLength,
 | |
| 		minimumPasswordStrength: defaults.minimumPasswordStrength,
 | |
| 		installing: installing,
 | |
| 	});
 | |
| }
 | |
| 
 | |
| function install(req, res) {
 | |
| 	if (installing) {
 | |
| 		return welcome(req, res);
 | |
| 	}
 | |
| 	req.setTimeout(0);
 | |
| 	installing = true;
 | |
| 	var setupEnvVars = nconf.get();
 | |
| 	for (var i in req.body) {
 | |
| 		if (req.body.hasOwnProperty(i) && !process.env.hasOwnProperty(i)) {
 | |
| 			setupEnvVars[i.replace(':', '__')] = req.body[i];
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	// Flatten any objects in setupEnvVars
 | |
| 	const pushToRoot = function (parentKey, key) {
 | |
| 		setupEnvVars[parentKey + '__' + key] = setupEnvVars[parentKey][key];
 | |
| 	};
 | |
| 	for (var j in setupEnvVars) {
 | |
| 		if (setupEnvVars.hasOwnProperty(j) && typeof setupEnvVars[j] === 'object' && setupEnvVars[j] !== null && !Array.isArray(setupEnvVars[j])) {
 | |
| 			Object.keys(setupEnvVars[j]).forEach(pushToRoot.bind(null, j));
 | |
| 			delete setupEnvVars[j];
 | |
| 		} else if (Array.isArray(setupEnvVars[j])) {
 | |
| 			setupEnvVars[j] = JSON.stringify(setupEnvVars[j]);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	winston.info('Starting setup process');
 | |
| 	winston.info(setupEnvVars);
 | |
| 	launchUrl = setupEnvVars.url;
 | |
| 
 | |
| 	var child = require('child_process').fork('app', ['--setup'], {
 | |
| 		env: setupEnvVars,
 | |
| 	});
 | |
| 
 | |
| 	child.on('close', function (data) {
 | |
| 		installing = false;
 | |
| 		success = data === 0;
 | |
| 		error = data !== 0;
 | |
| 
 | |
| 		welcome(req, res);
 | |
| 	});
 | |
| }
 | |
| 
 | |
| function launch(req, res) {
 | |
| 	res.json({});
 | |
| 	server.close();
 | |
| 	req.setTimeout(0);
 | |
| 	var child;
 | |
| 
 | |
| 	if (!nconf.get('launchCmd')) {
 | |
| 		child = childProcess.spawn('node', ['loader.js'], {
 | |
| 			detached: true,
 | |
| 			stdio: ['ignore', 'ignore', 'ignore'],
 | |
| 		});
 | |
| 
 | |
| 		console.log('\nStarting NodeBB');
 | |
| 		console.log('    "./nodebb stop" to stop the NodeBB server');
 | |
| 		console.log('    "./nodebb log" to view server output');
 | |
| 		console.log('    "./nodebb restart" to restart NodeBB');
 | |
| 	} else {
 | |
| 		// Use launchCmd instead, if specified
 | |
| 		child = childProcess.exec(nconf.get('launchCmd'), {
 | |
| 			detached: true,
 | |
| 			stdio: ['ignore', 'ignore', 'ignore'],
 | |
| 		});
 | |
| 	}
 | |
| 
 | |
| 	var filesToDelete = [
 | |
| 		'installer.css',
 | |
| 		'installer.min.js',
 | |
| 		'bootstrap.min.css',
 | |
| 	];
 | |
| 
 | |
| 	async.each(filesToDelete, function (filename, next) {
 | |
| 		fs.unlink(path.join(__dirname, '../public', filename), next);
 | |
| 	}, function (err) {
 | |
| 		if (err) {
 | |
| 			winston.warn('Unable to remove installer files');
 | |
| 		}
 | |
| 
 | |
| 		child.unref();
 | |
| 		process.exit(0);
 | |
| 	});
 | |
| }
 | |
| 
 | |
| function compileLess(callback) {
 | |
| 	var installSrc = path.join(__dirname, '../public/less/install.less');
 | |
| 	fs.readFile(installSrc, function (err, style) {
 | |
| 		if (err) {
 | |
| 			return winston.error('Unable to read LESS install file: ', err.stack);
 | |
| 		}
 | |
| 
 | |
| 		less.render(style.toString(), {
 | |
| 			filename: path.resolve(installSrc),
 | |
| 		}, function (err, css) {
 | |
| 			if (err) {
 | |
| 				return winston.error('Unable to compile LESS: ', err.stack);
 | |
| 			}
 | |
| 
 | |
| 			fs.writeFile(path.join(__dirname, '../public/installer.css'), css.css, callback);
 | |
| 		});
 | |
| 	});
 | |
| }
 | |
| 
 | |
| function compileJS(callback) {
 | |
| 	var code = '';
 | |
| 	async.eachSeries(scripts, function (srcPath, next) {
 | |
| 		fs.readFile(path.join(__dirname, '..', srcPath), function (err, buffer) {
 | |
| 			if (err) {
 | |
| 				return next(err);
 | |
| 			}
 | |
| 
 | |
| 			code += buffer.toString();
 | |
| 			next();
 | |
| 		});
 | |
| 	}, function (err) {
 | |
| 		if (err) {
 | |
| 			return callback(err);
 | |
| 		}
 | |
| 		try {
 | |
| 			var minified = uglify.minify(code, {
 | |
| 				compress: false,
 | |
| 			});
 | |
| 			if (!minified.code) {
 | |
| 				return callback(new Error('[[error:failed-to-minify]]'));
 | |
| 			}
 | |
| 			fs.writeFile(path.join(__dirname, '../public/installer.min.js'), minified.code, callback);
 | |
| 		} catch (e) {
 | |
| 			callback(e);
 | |
| 		}
 | |
| 	});
 | |
| }
 | |
| 
 | |
| function copyCSS(next) {
 | |
| 	async.waterfall([
 | |
| 		function (next) {
 | |
| 			fs.readFile(path.join(__dirname, '../node_modules/bootstrap/dist/css/bootstrap.min.css'), 'utf8', next);
 | |
| 		},
 | |
| 		function (src, next) {
 | |
| 			fs.writeFile(path.join(__dirname, '../public/bootstrap.min.css'), src, next);
 | |
| 		},
 | |
| 	], next);
 | |
| }
 | |
| 
 | |
| function loadDefaults(next) {
 | |
| 	var setupDefaultsPath = path.join(__dirname, '../setup.json');
 | |
| 	fs.access(setupDefaultsPath, fs.constants.F_OK | fs.constants.R_OK, function (err) {
 | |
| 		if (err) {
 | |
| 			// setup.json not found or inaccessible, proceed with no defaults
 | |
| 			return setImmediate(next);
 | |
| 		}
 | |
| 
 | |
| 		winston.info('[installer] Found setup.json, populating default values');
 | |
| 		nconf.file({
 | |
| 			file: setupDefaultsPath,
 | |
| 		});
 | |
| 
 | |
| 		next();
 | |
| 	});
 | |
| }
 |