s-nomp/init.js

283 lines
8.6 KiB
JavaScript
Raw Normal View History

2014-02-20 15:13:50 -08:00
var fs = require('fs');
var path = require('path');
var os = require('os');
var cluster = require('cluster');
var async = require('async');
var PoolLogger = require('./libs/logUtil.js');
var BlocknotifyListener = require('./libs/blocknotifyListener.js');
2014-03-14 12:02:55 -07:00
var RedisBlocknotifyListener = require('./libs/redisblocknotifyListener.js');
var WorkerListener = require('./libs/workerListener.js');
var PoolWorker = require('./libs/poolWorker.js');
var PaymentProcessor = require('./libs/paymentProcessor.js');
var Website = require('./libs/website.js');
var algos = require('stratum-pool/lib/algoProperties.js');
JSON.minify = JSON.minify || require("node-json-minify");
if (!fs.existsSync('config.json')){
console.log('config.json file does not exist. Read the installation/setup instructions.');
return;
}
2014-03-13 14:03:28 -07:00
var portalConfig = JSON.parse(JSON.minify(fs.readFileSync("config.json", {encoding: 'utf8'})));
2014-03-20 15:25:59 -07:00
var logger = new PoolLogger({
2014-03-13 14:03:28 -07:00
logLevel: portalConfig.logLevel
});
2014-01-13 17:32:54 -08:00
2014-01-13 17:32:54 -08:00
2014-03-20 15:25:59 -07:00
try {
require('newrelic');
if (cluster.isMaster)
logger.debug('NewRelic', 'Monitor', 'New Relic initiated');
2014-03-20 15:25:59 -07:00
} catch(e) {}
2014-03-09 19:31:58 -07:00
//Try to give process ability to handle 100k concurrent connections
try{
var posix = require('posix');
try {
posix.setrlimit('nofile', { soft: 100000, hard: 100000 });
}
catch(e){
if (cluster.isMaster)
logger.warning('POSIX', 'Connection Limit', '(Safe to ignore) Must be ran as root to increase resource limits');
}
2014-03-09 19:31:58 -07:00
}
catch(e){
if (cluster.isMaster)
logger.debug('POSIX', 'Connection Limit', '(Safe to ignore) POSIX module not installed and resource (connection) limit was not raised');
2014-03-09 19:31:58 -07:00
}
if (cluster.isWorker){
2014-03-14 12:02:55 -07:00
2014-03-09 19:31:58 -07:00
switch(process.env.workerType){
case 'pool':
new PoolWorker(logger);
2014-03-09 19:31:58 -07:00
break;
case 'paymentProcessor':
new PaymentProcessor(logger);
2014-03-09 19:31:58 -07:00
break;
2014-03-12 23:37:27 -07:00
case 'website':
new Website(logger);
2014-03-12 23:37:27 -07:00
break;
2014-03-09 19:31:58 -07:00
}
2014-01-13 17:32:54 -08:00
2014-03-09 19:31:58 -07:00
return;
2014-03-14 14:58:21 -07:00
} /* else {
2014-03-14 12:02:55 -07:00
var coinNames = ['alphacoin','frankocoin','emerald','kittehcoin'];
var curIndex = 0;
setInterval(function () {
var newCoinName = coinNames[++curIndex % coinNames.length];
console.log("SWITCHING to "+newCoinName);
var ipcMessage = {type:'switch', coin: newCoinName};
Object.keys(cluster.workers).forEach(function(id) {
cluster.workers[id].send(ipcMessage);
});
2014-03-14 14:56:54 -07:00
}, 20000);
2014-03-14 14:58:21 -07:00
} */
2014-01-13 17:32:54 -08:00
2014-02-27 15:59:49 -08:00
2014-03-09 19:31:58 -07:00
//Read all pool configs from pool_configs and join them with their coin profile
var buildPoolConfigs = function(){
var configs = {};
var configDir = 'pool_configs/';
fs.readdirSync(configDir).forEach(function(file){
if (!fs.existsSync(configDir + file) || path.extname(configDir + file) !== '.json') return;
var poolOptions = JSON.parse(JSON.minify(fs.readFileSync(configDir + file, {encoding: 'utf8'})));
if (!poolOptions.enabled) return;
2014-03-09 19:31:58 -07:00
var coinFilePath = 'coins/' + poolOptions.coin;
if (!fs.existsSync(coinFilePath)){
logger.error('Master', poolOptions.coin, 'could not find file: ' + coinFilePath);
2014-03-09 19:31:58 -07:00
return;
}
2014-03-09 19:31:58 -07:00
var coinProfile = JSON.parse(JSON.minify(fs.readFileSync(coinFilePath, {encoding: 'utf8'})));
poolOptions.coin = coinProfile;
configs[poolOptions.coin.name] = poolOptions;
if (!(coinProfile.algorithm in algos)){
logger.error('Master', coinProfile.name, 'Cannot run a pool for unsupported algorithm "' + coinProfile.algorithm + '"');
delete configs[poolOptions.coin.name];
}
2014-03-09 19:31:58 -07:00
});
return configs;
};
2014-02-27 15:59:49 -08:00
2014-01-13 17:32:54 -08:00
2014-03-09 19:31:58 -07:00
var spawnPoolWorkers = function(portalConfig, poolConfigs){
Object.keys(poolConfigs).forEach(function(coin){
var p = poolConfigs[coin];
var internalEnabled = p.shareProcessing && p.shareProcessing.internal && p.shareProcessing.internal.enabled;
2014-03-28 17:05:15 -07:00
var mposEnabled = p.shareProcessing && p.shareProcessing.mpos && p.shareProcessing.mpos.enabled;
if (!internalEnabled && !mposEnabled){
logger.error('Master', coin, 'Share processing is not configured so a pool cannot be started for this coin.');
delete poolConfigs[coin];
}
});
if (Object.keys(poolConfigs).length === 0){
logger.warning('Master', 'PoolSpawner', 'No pool configs exists or are enabled in pool_configs folder. No pools spawned.');
return;
}
var serializedConfigs = JSON.stringify(poolConfigs);
2014-02-27 15:59:49 -08:00
var numForks = (function(){
2014-03-09 19:31:58 -07:00
if (!portalConfig.clustering || !portalConfig.clustering.enabled)
return 1;
2014-03-09 19:31:58 -07:00
if (portalConfig.clustering.forks === 'auto')
return os.cpus().length;
2014-03-09 19:31:58 -07:00
if (!portalConfig.clustering.forks || isNaN(portalConfig.clustering.forks))
return 1;
2014-03-09 19:31:58 -07:00
return portalConfig.clustering.forks;
})();
2014-01-13 17:32:54 -08:00
2014-03-09 19:31:58 -07:00
var createPoolWorker = function(forkId){
var worker = cluster.fork({
workerType: 'pool',
forkId: forkId,
pools: serializedConfigs,
portalConfig: JSON.stringify(portalConfig)
});
2014-03-09 19:31:58 -07:00
worker.on('exit', function(code, signal){
2014-03-22 23:46:59 -07:00
logger.error('Master', 'PoolSpanwer', 'Fork ' + forkId + ' died, spawning replacement worker...');
setTimeout(function(){
createPoolWorker(forkId);
}, 2000);
2014-03-09 19:31:58 -07:00
});
};
2014-03-22 23:46:59 -07:00
var i = 0;
var spawnInterval = setInterval(function(){
2014-03-09 19:31:58 -07:00
createPoolWorker(i);
2014-03-22 23:46:59 -07:00
i++;
if (i === numForks){
clearInterval(spawnInterval);
logger.debug('Master', 'PoolSpawner', 'Spawned ' + Object.keys(poolConfigs).length + ' pool(s) on ' + numForks + ' thread(s)');
2014-03-22 23:46:59 -07:00
}
}, 250);
2014-01-13 17:32:54 -08:00
2014-03-09 19:31:58 -07:00
};
2014-02-27 15:59:49 -08:00
2014-03-09 19:31:58 -07:00
var startWorkerListener = function(poolConfigs){
var workerListener = new WorkerListener(logger, poolConfigs);
workerListener.init();
2014-03-09 19:31:58 -07:00
};
2014-03-09 19:31:58 -07:00
var startBlockListener = function(portalConfig){
//block notify options
//setup block notify here and use IPC to tell appropriate pools
2014-03-09 19:31:58 -07:00
var listener = new BlocknotifyListener(portalConfig.blockNotifyListener);
listener.on('log', function(text){
logger.debug('Master', 'Blocknotify', text);
});
listener.on('hash', function(message){
2014-03-04 12:24:02 -08:00
var ipcMessage = {type:'blocknotify', coin: message.coin, hash: message.hash};
Object.keys(cluster.workers).forEach(function(id) {
2014-03-04 12:24:02 -08:00
cluster.workers[id].send(ipcMessage);
});
});
listener.start();
2014-03-09 19:31:58 -07:00
};
2014-03-14 12:02:55 -07:00
var startRedisBlockListener = function(portalConfig){
//block notify options
//setup block notify here and use IPC to tell appropriate pools
if (!portalConfig.redisBlockNotifyListener.enabled) return;
2014-03-14 12:02:55 -07:00
var listener = new RedisBlocknotifyListener(portalConfig.redisBlockNotifyListener);
listener.on('log', function(text){
logger.debug('Master', 'blocknotify', text);
2014-03-14 12:02:55 -07:00
}).on('hash', function (message) {
var ipcMessage = {type:'blocknotify', coin: message.coin, hash: message.hash};
Object.keys(cluster.workers).forEach(function(id) {
cluster.workers[id].send(ipcMessage);
});
});
listener.start();
};
2014-03-07 15:29:16 -08:00
2014-03-09 19:31:58 -07:00
var startPaymentProcessor = function(poolConfigs){
var enabledForAny = false;
for (var pool in poolConfigs){
var p = poolConfigs[pool];
var enabled = p.enabled && p.shareProcessing && p.shareProcessing.internal && p.shareProcessing.internal.enabled;
if (enabled){
enabledForAny = true;
break;
}
}
if (!enabledForAny)
return;
2014-03-09 19:31:58 -07:00
var worker = cluster.fork({
workerType: 'paymentProcessor',
pools: JSON.stringify(poolConfigs)
});
worker.on('exit', function(code, signal){
logger.error('Master', 'Payment Processor', 'Payment processor died, spawning replacement...');
setTimeout(function(){
2014-03-19 13:24:29 -07:00
startPaymentProcessor(poolConfigs);
2014-03-12 23:37:27 -07:00
}, 2000);
});
};
var startWebsite = function(portalConfig, poolConfigs){
2014-03-14 14:56:54 -07:00
2014-03-12 23:37:27 -07:00
if (!portalConfig.website.enabled) return;
var worker = cluster.fork({
workerType: 'website',
pools: JSON.stringify(poolConfigs),
portalConfig: JSON.stringify(portalConfig)
});
worker.on('exit', function(code, signal){
logger.error('Master', 'Website', 'Website process died, spawning replacement...');
2014-03-12 23:37:27 -07:00
setTimeout(function(){
2014-03-19 13:24:29 -07:00
startWebsite(portalConfig, poolConfigs);
}, 2000);
2014-03-09 19:31:58 -07:00
});
};
2014-03-09 19:31:58 -07:00
(function init(){
var poolConfigs = buildPoolConfigs();
spawnPoolWorkers(portalConfig, poolConfigs);
startPaymentProcessor(poolConfigs);
startBlockListener(portalConfig);
2014-03-14 12:02:55 -07:00
startRedisBlockListener(portalConfig);
2014-03-09 19:31:58 -07:00
startWorkerListener(poolConfigs);
2014-03-12 23:37:27 -07:00
startWebsite(portalConfig, poolConfigs);
2014-03-14 12:02:55 -07:00
})();