'use strict'; var nconf = require('nconf'), fs = require('fs'), url = require('url'), path = require('path'), fork = require('child_process').fork, async = require('async'), logrotate = require('logrotate-stream'), pkg = require('./package.json'), pidFilePath = __dirname + '/pidfile', output = logrotate({ file: __dirname + '/logs/output.log', size: '1m', keep: 3, compress: true }), silent = process.env.NODE_ENV !== 'development', numProcs, workers = [], Loader = { timesStarted: 0, js: { cache: undefined, map: undefined }, css: { cache: undefined, acpCache: undefined } }; Loader.init = function(callback) { if (silent) { console.log = function(value) { output.write(value + '\n'); }; } process.on('SIGHUP', Loader.restart); process.on('SIGUSR2', Loader.reload); process.on('SIGTERM', Loader.stop); callback(); }; Loader.displayStartupMessages = function(callback) { console.log(''); console.log('NodeBB v' + pkg.version + ' Copyright (C) 2013-2014 NodeBB Inc.'); console.log('This program comes with ABSOLUTELY NO WARRANTY.'); console.log('This is free software, and you are welcome to redistribute it under certain conditions.'); console.log('For the full license, please visit: http://www.gnu.org/copyleft/gpl.html'); console.log(''); callback(); }; Loader.addWorkerEvents = function(worker) { worker.on('exit', function(code, signal) { if (code !== 0) { if (Loader.timesStarted < numProcs*3) { Loader.timesStarted++; if (Loader.crashTimer) { clearTimeout(Loader.crashTimer); } Loader.crashTimer = setTimeout(function() { Loader.timesStarted = 0; }, 10000); } else { console.log(numProcs*3 + ' restarts in 10 seconds, most likely an error on startup. Halting.'); process.exit(); } } console.log('[cluster] Child Process (' + worker.pid + ') has exited (code: ' + code + ', signal: ' + signal +')'); if (!(worker.suicide || code === 0)) { console.log('[cluster] Spinning up another process...'); forkWorker(worker.index, worker.isPrimary); } }); worker.on('message', function(message) { if (message && typeof message === 'object' && message.action) { switch (message.action) { case 'ready': if (Loader.js.cache && !worker.isPrimary) { worker.send({ action: 'js-propagate', cache: Loader.js.cache, map: Loader.js.map, hash: Loader.js.hash }); } if (Loader.css.cache && !worker.isPrimary) { worker.send({ action: 'css-propagate', cache: Loader.css.cache, acpCache: Loader.css.acpCache, hash: Loader.css.hash }); } break; case 'restart': console.log('[cluster] Restarting...'); Loader.restart(); break; case 'reload': console.log('[cluster] Reloading...'); Loader.reload(); break; case 'js-propagate': Loader.js.cache = message.cache; Loader.js.map = message.map; Loader.js.hash = message.hash; Loader.notifyWorkers({ action: 'js-propagate', cache: message.cache, map: message.map, hash: message.hash }, worker.pid); break; case 'css-propagate': Loader.css.cache = message.cache; Loader.css.acpCache = message.acpCache; Loader.css.hash = message.hash; Loader.notifyWorkers({ action: 'css-propagate', cache: message.cache, acpCache: message.acpCache, hash: message.hash }, worker.pid); break; case 'templates:compiled': Loader.notifyWorkers({ action: 'templates:compiled', }, worker.pid); break; } } }); }; Loader.start = function(callback) { numProcs = getPorts().length; console.log('Clustering enabled: Spinning up ' + numProcs + ' process(es).\n'); for (var x=0; x