|
|
|
const spawn = require('child_process').spawn;
|
|
|
|
|
|
|
|
module.exports = (shepherd) => {
|
|
|
|
const getConf = (flock, coind) => {
|
|
|
|
let DaemonConfPath = '';
|
|
|
|
let nativeCoindDir;
|
|
|
|
|
|
|
|
if (flock === 'CHIPS') {
|
|
|
|
flock = 'chipsd';
|
|
|
|
}
|
|
|
|
|
|
|
|
shepherd.log(flock);
|
|
|
|
shepherd.log(`getconf coind ${coind}`);
|
|
|
|
shepherd.writeLog(`getconf flock: ${flock}`);
|
|
|
|
|
|
|
|
if (coind) {
|
|
|
|
switch (shepherd.os.platform()) {
|
|
|
|
case 'darwin':
|
|
|
|
nativeCoindDir = `${process.env.HOME}/Library/Application Support/${shepherd.nativeCoindList[coind.toLowerCase()].bin}`;
|
|
|
|
break;
|
|
|
|
case 'linux':
|
|
|
|
nativeCoindDir = coind ? `${process.env.HOME}/.${shepherd.nativeCoindList[coind.toLowerCase()].bin.toLowerCase()}` : null;
|
|
|
|
break;
|
|
|
|
case 'win32':
|
|
|
|
nativeCoindDir = coind ? `${process.env.APPDATA}/${shepherd.nativeCoindList[coind.toLowerCase()].bin}` : null;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (flock) {
|
|
|
|
case 'komodod':
|
|
|
|
DaemonConfPath = shepherd.komodoDir;
|
|
|
|
if (shepherd.os.platform() === 'win32') {
|
|
|
|
DaemonConfPath = shepherd.path.normalize(DaemonConfPath);
|
|
|
|
shepherd.log('===>>> SHEPHERD API OUTPUT ===>>>');
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'zcashd':
|
|
|
|
DaemonConfPath = shepherd.ZcashDir;
|
|
|
|
if (shepherd.os.platform() === 'win32') {
|
|
|
|
DaemonConfPath = shepherd.path.normalize(DaemonConfPath);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'chipsd':
|
|
|
|
DaemonConfPath = shepherd.chipsDir;
|
|
|
|
if (shepherd.os.platform() === 'win32') {
|
|
|
|
DaemonConfPath = shepherd.path.normalize(DaemonConfPath);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'coind':
|
|
|
|
DaemonConfPath = shepherd.os.platform() === 'win32' ? shepherd.path.normalize(`${shepherd.coindRootDir}/${coind.toLowerCase()}`) : `${shepherd.coindRootDir}/${coind.toLowerCase()}`;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
DaemonConfPath = `${shepherd.komodoDir}/${flock}`;
|
|
|
|
if (shepherd.os.platform() === 'win32') {
|
|
|
|
DaemonConfPath = shepherd.path.normalize(DaemonConfPath);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
shepherd.writeLog(`getconf path: ${DaemonConfPath}`);
|
|
|
|
shepherd.log(`daemon path: ${DaemonConfPath}`);
|
|
|
|
|
|
|
|
return DaemonConfPath;
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: json.stringify wrapper
|
|
|
|
|
|
|
|
const herder = (flock, data, coind) => {
|
|
|
|
if (data === undefined) {
|
|
|
|
data = 'none';
|
|
|
|
shepherd.log('it is undefined');
|
|
|
|
}
|
|
|
|
|
|
|
|
shepherd.log(`herder flock: ${flock} coind: ${coind}`);
|
|
|
|
shepherd.log(`selected data: ${JSON.stringify(data, null, '\t')}`);
|
|
|
|
|
|
|
|
// TODO: notify gui that reindex/rescan param is used to reflect on the screen
|
|
|
|
// asset chain debug.log unlink
|
|
|
|
if (flock === 'komodod') {
|
|
|
|
let kmdDebugLogLocation = (data.ac_name !== 'komodod' ? `${shepherd.komodoDir}/${data.ac_name}` : shepherd.komodoDir) + '/debug.log';
|
|
|
|
|
|
|
|
shepherd.log('komodod flock selected...');
|
|
|
|
shepherd.log(`selected data: ${JSON.stringify(data, null, '\t')}`);
|
|
|
|
shepherd.writeLog('komodod flock selected...');
|
|
|
|
shepherd.writeLog(`selected data: ${data}`);
|
|
|
|
|
|
|
|
// datadir case, check if komodo/chain folder exists
|
|
|
|
if (shepherd.appConfig.dataDir.length &&
|
|
|
|
data.ac_name !== 'komodod') {
|
|
|
|
const _dir = data.ac_name !== 'komodod' ? `${shepherd.komodoDir}/${data.ac_name}` : shepherd.komodoDir;
|
|
|
|
|
|
|
|
try {
|
|
|
|
shepherd._fs.accessSync(_dir, shepherd.fs.R_OK | shepherd.fs.W_OK);
|
|
|
|
|
|
|
|
shepherd.log(`komodod datadir ${_dir} exists`);
|
|
|
|
} catch (e) {
|
|
|
|
shepherd.log(`komodod datadir ${_dir} access err: ${e}`);
|
|
|
|
shepherd.log(`attempting to create komodod datadir ${_dir}`);
|
|
|
|
|
|
|
|
shepherd.fs.mkdirSync(_dir);
|
|
|
|
|
|
|
|
if (shepherd.fs.existsSync(_dir)) {
|
|
|
|
shepherd.log(`created komodod datadir folder at ${_dir}`);
|
|
|
|
} else {
|
|
|
|
shepherd.log(`unable to create komodod datadir folder at ${_dir}`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// truncate debug.log
|
|
|
|
if (!shepherd.kmdMainPassiveMode) {
|
|
|
|
try {
|
|
|
|
const _confFileAccess = shepherd._fs.accessSync(kmdDebugLogLocation, shepherd.fs.R_OK | shepherd.fs.W_OK);
|
|
|
|
|
|
|
|
if (_confFileAccess) {
|
|
|
|
shepherd.log(`error accessing ${kmdDebugLogLocation}`);
|
|
|
|
shepherd.writeLog(`error accessing ${kmdDebugLogLocation}`);
|
|
|
|
} else {
|
|
|
|
try {
|
|
|
|
shepherd.fs.unlinkSync(kmdDebugLogLocation);
|
|
|
|
shepherd.log(`truncate ${kmdDebugLogLocation}`);
|
|
|
|
shepherd.writeLog(`truncate ${kmdDebugLogLocation}`);
|
|
|
|
} catch (e) {
|
|
|
|
shepherd.log('cant unlink debug.log');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} catch (e) {
|
|
|
|
shepherd.log(`komodod debug.log access err: ${e}`);
|
|
|
|
shepherd.writeLog(`komodod debug.log access err: ${e}`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// get komodod instance port
|
|
|
|
const _port = shepherd.assetChainPorts[data.ac_name];
|
|
|
|
|
|
|
|
try {
|
|
|
|
// check if komodod instance is already running
|
|
|
|
shepherd.portscanner.checkPortStatus(_port, '127.0.0.1', (error, status) => {
|
|
|
|
// Status is 'open' if currently in use or 'closed' if available
|
|
|
|
if (status === 'closed' ||
|
|
|
|
!shepherd.appConfig.stopNativeDaemonsOnQuit) {
|
|
|
|
// start komodod via exec
|
|
|
|
const _customParamDict = {
|
|
|
|
silent: '&',
|
|
|
|
reindex: '-reindex',
|
|
|
|
change: '-pubkey=',
|
|
|
|
datadir: '-datadir=',
|
|
|
|
rescan: '-rescan',
|
|
|
|
};
|
|
|
|
let _customParam = '';
|
|
|
|
|
|
|
|
if (data.ac_custom_param === 'silent' ||
|
|
|
|
data.ac_custom_param === 'reindex' ||
|
|
|
|
data.ac_custom_param === 'rescan') {
|
|
|
|
_customParam = ` ${_customParamDict[data.ac_custom_param]}`;
|
|
|
|
} else if (data.ac_custom_param === 'change' && data.ac_custom_param_value) {
|
|
|
|
_customParam = ` ${_customParamDict[data.ac_custom_param]}${data.ac_custom_param_value}`;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (shepherd.appConfig.dataDir.length) {
|
|
|
|
_customParam = _customParam + ' -datadir=' + shepherd.appConfig.dataDir + (data.ac_name !== 'komodod' ? '/' + data.ac_name : '');
|
|
|
|
}
|
|
|
|
|
|
|
|
shepherd.log(`exec ${shepherd.komododBin} ${data.ac_options.join(' ')}${_customParam}`);
|
|
|
|
shepherd.writeLog(`exec ${shepherd.komododBin} ${data.ac_options.join(' ')}${_customParam}`);
|
|
|
|
|
|
|
|
const isChain = data.ac_name.match(/^[A-Z]*$/);
|
|
|
|
const coindACParam = isChain ? ` -ac_name=${data.ac_name} ` : '';
|
|
|
|
shepherd.log(`daemon param ${data.ac_custom_param}`);
|
|
|
|
|
|
|
|
shepherd.coindInstanceRegistry[data.ac_name] = true;
|
|
|
|
if (!shepherd.kmdMainPassiveMode) {
|
|
|
|
let _arg = `${coindACParam}${data.ac_options.join(' ')}${_customParam}`;
|
|
|
|
_arg = _arg.trim().split(' ');
|
|
|
|
|
|
|
|
if (!shepherd.appConfig.stopNativeDaemonsOnQuit) {
|
|
|
|
spawn(shepherd.komododBin, _arg, {
|
|
|
|
stdio: 'ignore', // piping all stdio to /dev/null
|
|
|
|
detached: true,
|
|
|
|
}).unref();
|
|
|
|
} else {
|
|
|
|
shepherd.execFile(`${shepherd.komododBin}`, _arg, {
|
|
|
|
maxBuffer: 1024 * 1000000 // 1000 mb
|
|
|
|
}, (error, stdout, stderr) => {
|
|
|
|
shepherd.writeLog(`stdout: ${stdout}`);
|
|
|
|
shepherd.writeLog(`stderr: ${stderr}`);
|
|
|
|
|
|
|
|
if (error !== null) {
|
|
|
|
shepherd.log(`exec error: ${error}`);
|
|
|
|
shepherd.writeLog(`exec error: ${error}`);
|
|
|
|
|
|
|
|
if (error.toString().indexOf('using -reindex') > -1) {
|
|
|
|
shepherd.io.emit('service', {
|
|
|
|
komodod: {
|
|
|
|
error: 'run -reindex',
|
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (shepherd.kmdMainPassiveMode) {
|
|
|
|
shepherd.coindInstanceRegistry[data.ac_name] = true;
|
|
|
|
}
|
|
|
|
shepherd.log(`port ${_port} (${data.ac_name}) is already in use`);
|
|
|
|
shepherd.writeLog(`port ${_port} (${data.ac_name}) is already in use`);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
} catch(e) {
|
|
|
|
shepherd.log(`failed to start komodod err: ${e}`);
|
|
|
|
shepherd.writeLog(`failed to start komodod err: ${e}`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: refactor
|
|
|
|
if (flock === 'chipsd') {
|
|
|
|
let kmdDebugLogLocation = `${shepherd.chipsDir}/debug.log`;
|
|
|
|
|
|
|
|
shepherd.log('chipsd flock selected...');
|
|
|
|
shepherd.log(`selected data: ${JSON.stringify(data, null, '\t')}`);
|
|
|
|
shepherd.writeLog('chipsd flock selected...');
|
|
|
|
shepherd.writeLog(`selected data: ${data}`);
|
|
|
|
|
|
|
|
// truncate debug.log
|
|
|
|
try {
|
|
|
|
const _confFileAccess = shepherd._fs.accessSync(kmdDebugLogLocation, shepherd.fs.R_OK | shepherd.fs.W_OK);
|
|
|
|
|
|
|
|
if (_confFileAccess) {
|
|
|
|
shepherd.log(`error accessing ${kmdDebugLogLocation}`);
|
|
|
|
shepherd.writeLog(`error accessing ${kmdDebugLogLocation}`);
|
|
|
|
} else {
|
|
|
|
try {
|
|
|
|
shepherd.fs.unlinkSync(kmdDebugLogLocation);
|
|
|
|
shepherd.log(`truncate ${kmdDebugLogLocation}`);
|
|
|
|
shepherd.writeLog(`truncate ${kmdDebugLogLocation}`);
|
|
|
|
} catch (e) {
|
|
|
|
shepherd.log('cant unlink debug.log');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} catch(e) {
|
|
|
|
shepherd.log(`chipsd debug.log access err: ${e}`);
|
|
|
|
shepherd.writeLog(`chipsd debug.log access err: ${e}`);
|
|
|
|
}
|
|
|
|
|
|
|
|
// get komodod instance port
|
|
|
|
const _port = shepherd.assetChainPorts.chipsd;
|
|
|
|
|
|
|
|
try {
|
|
|
|
// check if komodod instance is already running
|
|
|
|
shepherd.portscanner.checkPortStatus(_port, '127.0.0.1', (error, status) => {
|
|
|
|
// Status is 'open' if currently in use or 'closed' if available
|
|
|
|
if (status === 'closed') {
|
|
|
|
// start komodod via exec
|
|
|
|
const _customParamDict = {
|
|
|
|
silent: '&',
|
|
|
|
reindex: '-reindex',
|
|
|
|
change: '-pubkey=',
|
|
|
|
rescan: '-rescan',
|
|
|
|
};
|
|
|
|
let _customParam = '';
|
|
|
|
|
|
|
|
if (data.ac_custom_param === 'silent' ||
|
|
|
|
data.ac_custom_param === 'reindex' ||
|
|
|
|
data.ac_custom_param === 'rescan') {
|
|
|
|
_customParam = ` ${_customParamDict[data.ac_custom_param]}`;
|
|
|
|
} else if (data.ac_custom_param === 'change' && data.ac_custom_param_value) {
|
|
|
|
_customParam = ` ${_customParamDict[data.ac_custom_param]}${data.ac_custom_param_value}`;
|
|
|
|
}
|
|
|
|
|
|
|
|
shepherd.log(`exec ${shepherd.chipsBin} ${_customParam}`);
|
|
|
|
shepherd.writeLog(`exec ${shepherd.chipsBin} ${_customParam}`);
|
|
|
|
|
|
|
|
shepherd.log(`daemon param ${data.ac_custom_param}`);
|
|
|
|
|
|
|
|
shepherd.coindInstanceRegistry['CHIPS'] = true;
|
|
|
|
let _arg = `${_customParam}`;
|
|
|
|
_arg = _arg.trim().split(' ');
|
|
|
|
|
|
|
|
if (_arg &&
|
|
|
|
_arg.length > 1) {
|
|
|
|
shepherd.execFile(`${shepherd.chipsBin}`, _arg, {
|
|
|
|
maxBuffer: 1024 * 1000000 // 1000 mb
|
|
|
|
}, (error, stdout, stderr) => {
|
|
|
|
shepherd.writeLog(`stdout: ${stdout}`);
|
|
|
|
shepherd.writeLog(`stderr: ${stderr}`);
|
|
|
|
|
|
|
|
if (error !== null) {
|
|
|
|
shepherd.log(`exec error: ${error}`);
|
|
|
|
shepherd.writeLog(`exec error: ${error}`);
|
|
|
|
|
|
|
|
if (error.toString().indexOf('using -reindex') > -1) {
|
|
|
|
shepherd.io.emit('service', {
|
|
|
|
komodod: {
|
|
|
|
error: 'run -reindex',
|
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
shepherd.execFile(`${shepherd.chipsBin}`, {
|
|
|
|
maxBuffer: 1024 * 1000000 // 1000 mb
|
|
|
|
}, (error, stdout, stderr) => {
|
|
|
|
shepherd.writeLog(`stdout: ${stdout}`);
|
|
|
|
shepherd.writeLog(`stderr: ${stderr}`);
|
|
|
|
|
|
|
|
if (error !== null) {
|
|
|
|
shepherd.log(`exec error: ${error}`);
|
|
|
|
shepherd.writeLog(`exec error: ${error}`);
|
|
|
|
|
|
|
|
if (error.toString().indexOf('using -reindex') > -1) {
|
|
|
|
shepherd.io.emit('service', {
|
|
|
|
komodod: {
|
|
|
|
error: 'run -reindex',
|
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
} catch(e) {
|
|
|
|
shepherd.log(`failed to start chipsd err: ${e}`);
|
|
|
|
shepherd.writeLog(`failed to start chipsd err: ${e}`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (flock === 'zcashd') { // TODO: fix(?)
|
|
|
|
let kmdDebugLogLocation = `${shepherd.zcashDir}/debug.log`;
|
|
|
|
|
|
|
|
shepherd.log('zcashd flock selected...');
|
|
|
|
shepherd.log(`selected data: ${data}`);
|
|
|
|
shepherd.writeLog('zcashd flock selected...');
|
|
|
|
shepherd.writeLog(`selected data: ${data}`);
|
|
|
|
|
|
|
|
/*pm2.connect(true, function(err) { // start up pm2 god
|
|
|
|
if (err) {
|
|
|
|
shepherd.error(err);
|
|
|
|
process.exit(2);
|
|
|
|
}
|
|
|
|
|
|
|
|
pm2.start({
|
|
|
|
script: shepherd.zcashdBin, // path to binary
|
|
|
|
name: data.ac_name, // REVS, USD, EUR etc.
|
|
|
|
exec_mode: 'fork',
|
|
|
|
cwd: shepherd.zcashDir,
|
|
|
|
args: data.ac_options
|
|
|
|
}, function(err, apps) {
|
|
|
|
shepherd.writeLog(`zcashd fork started ${data.ac_name} ${JSON.stringify(data.ac_options)}`);
|
|
|
|
|
|
|
|
pm2.disconnect(); // Disconnect from PM2
|
|
|
|
if (err) {
|
|
|
|
shepherd.writeLog(`pm2.disconnect err: ${err}`);
|
|
|
|
shepherd.log(`pm2.disconnect err: ${err}`);
|
|
|
|
}
|
|
|
|
// throw err;
|
|
|
|
});
|
|
|
|
});*/
|
|
|
|
}
|
|
|
|
|
|
|
|
if (flock === 'coind') {
|
|
|
|
const _osHome = shepherd.os.platform === 'win32' ? process.env.APPDATA : process.env.HOME;
|
|
|
|
let coindDebugLogLocation = `${_osHome}/.${shepherd.nativeCoindList[coind.toLowerCase()].bin.toLowerCase()}/debug.log`;
|
|
|
|
|
|
|
|
shepherd.log(`coind ${coind} flock selected...`);
|
|
|
|
shepherd.log(`selected data: ${JSON.stringify(data, null, '\t')}`);
|
|
|
|
shepherd.writeLog(`coind ${coind} flock selected...`);
|
|
|
|
shepherd.writeLog(`selected data: ${data}`);
|
|
|
|
|
|
|
|
// truncate debug.log
|
|
|
|
try {
|
|
|
|
shepherd._fs.access(coindDebugLogLocation, shepherd.fs.constants.R_OK, (err) => {
|
|
|
|
if (err) {
|
|
|
|
shepherd.log(`error accessing ${coindDebugLogLocation}`);
|
|
|
|
shepherd.writeLog(`error accessing ${coindDebugLogLocation}`);
|
|
|
|
} else {
|
|
|
|
shepherd.log(`truncate ${coindDebugLogLocation}`);
|
|
|
|
shepherd.writeLog(`truncate ${coindDebugLogLocation}`);
|
|
|
|
shepherd.fs.unlink(coindDebugLogLocation);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
} catch(e) {
|
|
|
|
shepherd.log(`coind ${coind} debug.log access err: ${e}`);
|
|
|
|
shepherd.writeLog(`coind ${coind} debug.log access err: ${e}`);
|
|
|
|
}
|
|
|
|
|
|
|
|
// get komodod instance port
|
|
|
|
const _port = shepherd.nativeCoindList[coind.toLowerCase()].port;
|
|
|
|
const coindBin = `${shepherd.coindRootDir}/${coind.toLowerCase()}/${shepherd.nativeCoindList[coind.toLowerCase()].bin.toLowerCase()}d`;
|
|
|
|
|
|
|
|
try {
|
|
|
|
// check if coind instance is already running
|
|
|
|
shepherd.portscanner.checkPortStatus(_port, '127.0.0.1', (error, status) => {
|
|
|
|
// Status is 'open' if currently in use or 'closed' if available
|
|
|
|
if (status === 'closed') {
|
|
|
|
shepherd.log(`exec ${coindBin} ${data.ac_options.join(' ')}`);
|
|
|
|
shepherd.writeLog(`exec ${coindBin} ${data.ac_options.join(' ')}`);
|
|
|
|
|
|
|
|
shepherd.coindInstanceRegistry[coind] = true;
|
|
|
|
let _arg = `${data.ac_options.join(' ')}`;
|
|
|
|
_arg = _arg.trim().split(' ');
|
|
|
|
shepherd.execFile(`${coindBin}`, _arg, {
|
|
|
|
maxBuffer: 1024 * 1000000 // 1000 mb
|
|
|
|
}, (error, stdout, stderr) => {
|
|
|
|
shepherd.writeLog(`stdout: ${stdout}`);
|
|
|
|
shepherd.writeLog(`stderr: ${stderr}`);
|
|
|
|
|
|
|
|
if (error !== null) {
|
|
|
|
shepherd.log(`exec error: ${error}`);
|
|
|
|
shepherd.writeLog(`exec error: ${error}`);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
shepherd.log(`port ${_port} (${coind}) is already in use`);
|
|
|
|
shepherd.writeLog(`port ${_port} (${coind}) is already in use`);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
} catch(e) {
|
|
|
|
shepherd.log(`failed to start ${coind} err: ${e}`);
|
|
|
|
shepherd.writeLog(`failed to start ${coind} err: ${e}`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const setConf = (flock, coind) => {
|
|
|
|
let nativeCoindDir;
|
|
|
|
let DaemonConfPath;
|
|
|
|
|
|
|
|
shepherd.log(flock);
|
|
|
|
shepherd.writeLog(`setconf ${flock}`);
|
|
|
|
|
|
|
|
if (shepherd.os.platform() === 'darwin') {
|
|
|
|
nativeCoindDir = coind ? `${process.env.HOME}/Library/Application Support/${shepherd.nativeCoindList[coind.toLowerCase()].bin}` : null;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (shepherd.os.platform() === 'linux') {
|
|
|
|
nativeCoindDir = coind ? `${process.env.HOME}/.${shepherd.nativeCoindList[coind.toLowerCase()].bin.toLowerCase()}` : null;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (shepherd.os.platform() === 'win32') {
|
|
|
|
nativeCoindDir = coind ? `${process.env.APPDATA}/${shepherd.nativeCoindList[coind.toLowerCase()].bin}` : null;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (flock) {
|
|
|
|
case 'komodod':
|
|
|
|
DaemonConfPath = `${shepherd.komodoDir}/komodo.conf`;
|
|
|
|
|
|
|
|
if (shepherd.os.platform() === 'win32') {
|
|
|
|
DaemonConfPath = shepherd.path.normalize(DaemonConfPath);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'zcashd':
|
|
|
|
DaemonConfPath = `${shepherd.ZcashDir}/zcash.conf`;
|
|
|
|
|
|
|
|
if (shepherd.os.platform() === 'win32') {
|
|
|
|
DaemonConfPath = shepherd.path.normalize(DaemonConfPath);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'chipsd':
|
|
|
|
DaemonConfPath = `${shepherd.chipsDir}/chips.conf`;
|
|
|
|
|
|
|
|
if (shepherd.os.platform() === 'win32') {
|
|
|
|
DaemonConfPath = shepherd.path.normalize(DaemonConfPath);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'coind':
|
|
|
|
DaemonConfPath = `${nativeCoindDir}/${shepherd.nativeCoindList[coind.toLowerCase()].bin.toLowerCase()}.conf`;
|
|
|
|
|
|
|
|
if (shepherd.os.platform() === 'win32') {
|
|
|
|
DaemonConfPath = shepherd.path.normalize(DaemonConfPath);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
DaemonConfPath = `${shepherd.komodoDir}/${flock}/${flock}.conf`;
|
|
|
|
|
|
|
|
if (shepherd.os.platform() === 'win32') {
|
|
|
|
DaemonConfPath = shepherd.path.normalize(DaemonConfPath);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
shepherd.log(DaemonConfPath);
|
|
|
|
shepherd.writeLog(`setconf ${DaemonConfPath}`);
|
|
|
|
|
|
|
|
const CheckFileExists = () => {
|
|
|
|
return new shepherd.Promise((resolve, reject) => {
|
|
|
|
const result = 'Check Conf file exists is done';
|
|
|
|
|
|
|
|
const confFileExist = shepherd.fs.ensureFileSync(DaemonConfPath);
|
|
|
|
if (confFileExist) {
|
|
|
|
shepherd.log(result);
|
|
|
|
shepherd.writeLog(`setconf ${result}`);
|
|
|
|
|
|
|
|
resolve(result);
|
|
|
|
} else {
|
|
|
|
shepherd.log('conf file doesnt exist');
|
|
|
|
resolve('conf file doesnt exist');
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
const FixFilePermissions = () => {
|
|
|
|
return new shepherd.Promise((resolve, reject) => {
|
|
|
|
const result = 'Conf file permissions updated to Read/Write';
|
|
|
|
|
|
|
|
shepherd.fsnode.chmodSync(DaemonConfPath, '0666');
|
|
|
|
shepherd.log(result);
|
|
|
|
shepherd.writeLog(`setconf ${result}`);
|
|
|
|
|
|
|
|
resolve(result);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
const RemoveLines = () => {
|
|
|
|
return new shepherd.Promise((resolve, reject) => {
|
|
|
|
const result = 'RemoveLines is done';
|
|
|
|
|
|
|
|
shepherd.fs.readFile(DaemonConfPath, 'utf8', (err, data) => {
|
|
|
|
if (err) {
|
|
|
|
shepherd.writeLog(`setconf error ${err}`);
|
|
|
|
return shepherd.log(err);
|
|
|
|
}
|
|
|
|
|
|
|
|
const rmlines = data.replace(/(?:(?:\r\n|\r|\n)\s*){2}/gm, '\n');
|
|
|
|
|
|
|
|
shepherd.fs.writeFile(DaemonConfPath, rmlines, 'utf8', (err) => {
|
|
|
|
if (err)
|
|
|
|
return shepherd.log(err);
|
|
|
|
|
|
|
|
shepherd.fsnode.chmodSync(DaemonConfPath, '0666');
|
|
|
|
shepherd.writeLog(`setconf ${result}`);
|
|
|
|
shepherd.log(result);
|
|
|
|
resolve(result);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
const CheckConf = () => {
|
|
|
|
return new shepherd.Promise((resolve, reject) => {
|
|
|
|
const result = 'CheckConf is done';
|
|
|
|
|
|
|
|
shepherd.setconf.status(DaemonConfPath, (err, status) => {
|
|
|
|
const rpcuser = () => {
|
|
|
|
return new shepherd.Promise((resolve, reject) => {
|
|
|
|
const result = 'checking rpcuser...';
|
|
|
|
|
|
|
|
if (status[0].hasOwnProperty('rpcuser')) {
|
|
|
|
shepherd.log('rpcuser: OK');
|
|
|
|
shepherd.writeLog('rpcuser: OK');
|
|
|
|
} else {
|
|
|
|
const randomstring = shepherd.md5((Math.random() * Math.random() * 999).toString());
|
|
|
|
|
|
|
|
shepherd.log('rpcuser: NOT FOUND');
|
|
|
|
shepherd.writeLog('rpcuser: NOT FOUND');
|
|
|
|
|
|
|
|
shepherd.fs.appendFile(DaemonConfPath, `\nrpcuser=user${randomstring.substring(0, 16)}`, (err) => {
|
|
|
|
if (err) {
|
|
|
|
shepherd.writeLog(`append daemon conf err: ${err}`);
|
|
|
|
shepherd.log(`append daemon conf err: ${err}`);
|
|
|
|
}
|
|
|
|
// throw err;
|
|
|
|
shepherd.log('rpcuser: ADDED');
|
|
|
|
shepherd.writeLog('rpcuser: ADDED');
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
resolve(result);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
const rpcpass = () => {
|
|
|
|
return new shepherd.Promise((resolve, reject) => {
|
|
|
|
const result = 'checking rpcpassword...';
|
|
|
|
|
|
|
|
if (status[0].hasOwnProperty('rpcpassword')) {
|
|
|
|
shepherd.log('rpcpassword: OK');
|
|
|
|
shepherd.writeLog('rpcpassword: OK');
|
|
|
|
} else {
|
|
|
|
const randomstring = shepherd.md5((Math.random() * Math.random() * 999).toString());
|
|
|
|
|
|
|
|
shepherd.log('rpcpassword: NOT FOUND');
|
|
|
|
shepherd.writeLog('rpcpassword: NOT FOUND');
|
|
|
|
|
|
|
|
shepherd.fs.appendFile(DaemonConfPath, `\nrpcpassword=${randomstring}`, (err) => {
|
|
|
|
if (err) {
|
|
|
|
shepherd.writeLog(`append daemon conf err: ${err}`);
|
|
|
|
shepherd.log(`append daemon conf err: ${err}`);
|
|
|
|
}
|
|
|
|
// throw err;
|
|
|
|
shepherd.log('rpcpassword: ADDED');
|
|
|
|
shepherd.writeLog('rpcpassword: ADDED');
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
resolve(result);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
const rpcbind = () => {
|
|
|
|
return new shepherd.Promise((resolve, reject) => {
|
|
|
|
const result = 'checking rpcbind...';
|
|
|
|
|
|
|
|
if (status[0].hasOwnProperty('rpcbind')) {
|
|
|
|
shepherd.log('rpcbind: OK');
|
|
|
|
shepherd.writeLog('rpcbind: OK');
|
|
|
|
} else {
|
|
|
|
shepherd.log('rpcbind: NOT FOUND');
|
|
|
|
shepherd.writeLog('rpcbind: NOT FOUND');
|
|
|
|
|
|
|
|
shepherd.fs.appendFile(DaemonConfPath, '\nrpcbind=127.0.0.1', (err) => {
|
|
|
|
if (err) {
|
|
|
|
shepherd.writeLog(`append daemon conf err: ${err}`);
|
|
|
|
shepherd.log(`append daemon conf err: ${err}`);
|
|
|
|
}
|
|
|
|
// throw err;
|
|
|
|
shepherd.log('rpcbind: ADDED');
|
|
|
|
shepherd.writeLog('rpcbind: ADDED');
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
resolve(result);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
const server = () => {
|
|
|
|
return new shepherd.Promise((resolve, reject) => {
|
|
|
|
const result = 'checking server...';
|
|
|
|
|
|
|
|
if (status[0].hasOwnProperty('server')) {
|
|
|
|
shepherd.log('server: OK');
|
|
|
|
shepherd.writeLog('server: OK');
|
|
|
|
} else {
|
|
|
|
shepherd.log('server: NOT FOUND');
|
|
|
|
shepherd.writeLog('server: NOT FOUND');
|
|
|
|
|
|
|
|
shepherd.fs.appendFile(DaemonConfPath, '\nserver=1', (err) => {
|
|
|
|
if (err) {
|
|
|
|
shepherd.writeLog(`append daemon conf err: ${err}`);
|
|
|
|
shepherd.log(`append daemon conf err: ${err}`);
|
|
|
|
}
|
|
|
|
// throw err;
|
|
|
|
shepherd.log('server: ADDED');
|
|
|
|
shepherd.writeLog('server: ADDED');
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
resolve(result);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
const addnode = () => {
|
|
|
|
return new shepherd.Promise((resolve, reject) => {
|
|
|
|
const result = 'checking addnode...';
|
|
|
|
|
|
|
|
if (flock === 'chipsd' ||
|
|
|
|
flock === 'komodod') {
|
|
|
|
if (status[0].hasOwnProperty('addnode')) {
|
|
|
|
shepherd.log('addnode: OK');
|
|
|
|
shepherd.writeLog('addnode: OK');
|
|
|
|
} else {
|
|
|
|
let nodesList;
|
|
|
|
|
|
|
|
if (flock === 'chipsd') {
|
|
|
|
nodesList = '\naddnode=95.110.191.193' +
|
|
|
|
'\naddnode=144.76.167.66' +
|
|
|
|
'\naddnode=158.69.248.93' +
|
|
|
|
'\naddnode=149.202.49.218' +
|
|
|
|
'\naddnode=95.213.205.222' +
|
|
|
|
'\naddnode=5.9.253.198' +
|
|
|
|
'\naddnode=164.132.224.253' +
|
|
|
|
'\naddnode=163.172.4.66' +
|
|
|
|
'\naddnode=217.182.194.216' +
|
|
|
|
'\naddnode=94.130.96.114' +
|
|
|
|
'\naddnode=5.9.253.195';
|
|
|
|
} else if (flock === 'komodod') {
|
|
|
|
nodesList = '\naddnode=78.47.196.146' +
|
|
|
|
'\naddnode=5.9.102.210' +
|
|
|
|
'\naddnode=178.63.69.164' +
|
|
|
|
'\naddnode=88.198.65.74' +
|
|
|
|
'\naddnode=5.9.122.241' +
|
|
|
|
'\naddnode=144.76.94.3';
|
|
|
|
}
|
|
|
|
|
|
|
|
shepherd.log('addnode: NOT FOUND');
|
|
|
|
shepherd.fs.appendFile(DaemonConfPath, nodesList, (err) => {
|
|
|
|
if (err) {
|
|
|
|
shepherd.writeLog(`append daemon conf err: ${err}`);
|
|
|
|
shepherd.log(`append daemon conf err: ${err}`);
|
|
|
|
}
|
|
|
|
// throw err;
|
|
|
|
shepherd.log('addnode: ADDED');
|
|
|
|
shepherd.writeLog('addnode: ADDED');
|
|
|
|
});
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
result = 'skip addnode';
|
|
|
|
}
|
|
|
|
|
|
|
|
resolve(result);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
rpcuser()
|
|
|
|
.then((result) => {
|
|
|
|
return rpcpass();
|
|
|
|
})
|
|
|
|
.then(server)
|
|
|
|
.then(rpcbind)
|
|
|
|
.then(addnode);
|
|
|
|
});
|
|
|
|
|
|
|
|
shepherd.log(result);
|
|
|
|
shepherd.writeLog(`checkconf addnode ${result}`);
|
|
|
|
|
|
|
|
resolve(result);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
CheckFileExists()
|
|
|
|
.then((result) => {
|
|
|
|
return FixFilePermissions();
|
|
|
|
})
|
|
|
|
.then(RemoveLines)
|
|
|
|
.then(CheckConf);
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* type: POST
|
|
|
|
* params: herd
|
|
|
|
*/
|
|
|
|
shepherd.post('/herd', (req, res) => {
|
|
|
|
shepherd.log('======= req.body =======');
|
|
|
|
shepherd.log(req.body);
|
|
|
|
|
|
|
|
if (req.body.options &&
|
|
|
|
!shepherd.kmdMainPassiveMode) {
|
|
|
|
const testCoindPort = (skipError) => {
|
|
|
|
if (!shepherd.lockDownAddCoin) {
|
|
|
|
const _port = shepherd.assetChainPorts[req.body.options.ac_name];
|
|
|
|
|
|
|
|
shepherd.portscanner.checkPortStatus(_port, '127.0.0.1', (error, status) => {
|
|
|
|
// Status is 'open' if currently in use or 'closed' if available
|
|
|
|
if (status === 'open' &&
|
|
|
|
shepherd.appConfig.stopNativeDaemonsOnQuit) {
|
|
|
|
if (!skipError) {
|
|
|
|
shepherd.log(`komodod service start error at port ${_port}, reason: port is closed`);
|
|
|
|
shepherd.writeLog(`komodod service start error at port ${_port}, reason: port is closed`);
|
|
|
|
shepherd.io.emit('service', {
|
|
|
|
komodod: {
|
|
|
|
error: `error starting ${req.body.herd} ${req.body.options.ac_name} daemon. Port ${_port} is already taken!`,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
const obj = {
|
|
|
|
msg: 'error',
|
|
|
|
result: `error starting ${req.body.herd} ${req.body.options.ac_name} daemon. Port ${_port} is already taken!`,
|
|
|
|
};
|
|
|
|
|
|
|
|
res.status(500);
|
|
|
|
res.end(JSON.stringify(obj));
|
|
|
|
} else {
|
|
|
|
shepherd.log(`komodod service start success at port ${_port}`);
|
|
|
|
shepherd.writeLog(`komodod service start success at port ${_port}`);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (!skipError) {
|
|
|
|
herder(req.body.herd, req.body.options);
|
|
|
|
|
|
|
|
const obj = {
|
|
|
|
msg: 'success',
|
|
|
|
result: 'result',
|
|
|
|
};
|
|
|
|
|
|
|
|
res.end(JSON.stringify(obj));
|
|
|
|
} else {
|
|
|
|
shepherd.log(`komodod service start error at port ${_port}, reason: unknown`);
|
|
|
|
shepherd.writeLog(`komodod service start error at port ${_port}, reason: unknown`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (req.body.herd === 'komodod') {
|
|
|
|
// check if komodod instance is already running
|
|
|
|
testCoindPort();
|
|
|
|
setTimeout(() => {
|
|
|
|
testCoindPort(true);
|
|
|
|
}, 10000);
|
|
|
|
} else {
|
|
|
|
herder(req.body.herd, req.body.options, req.body.coind);
|
|
|
|
|
|
|
|
const obj = {
|
|
|
|
msg: 'success',
|
|
|
|
result: 'result',
|
|
|
|
};
|
|
|
|
|
|
|
|
res.end(JSON.stringify(obj));
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// (?)
|
|
|
|
herder(req.body.herd, req.body.options);
|
|
|
|
|
|
|
|
const obj = {
|
|
|
|
msg: 'success',
|
|
|
|
result: 'result',
|
|
|
|
};
|
|
|
|
|
|
|
|
res.end(JSON.stringify(obj));
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
/*
|
|
|
|
* type: POST
|
|
|
|
*/
|
|
|
|
shepherd.post('/setconf', (req, res) => {
|
|
|
|
shepherd.log('======= req.body =======');
|
|
|
|
shepherd.log(req.body);
|
|
|
|
|
|
|
|
if (shepherd.os.platform() === 'win32' &&
|
|
|
|
req.body.chain == 'komodod') {
|
|
|
|
setkomodoconf = spawn(shepherd.path.join(__dirname, '../assets/bin/win64/genkmdconf.bat'));
|
|
|
|
} else {
|
|
|
|
shepherd.setConf(req.body.chain);
|
|
|
|
}
|
|
|
|
|
|
|
|
const obj = {
|
|
|
|
msg: 'success',
|
|
|
|
result: 'result',
|
|
|
|
};
|
|
|
|
|
|
|
|
res.end(JSON.stringify(obj));
|
|
|
|
});
|
|
|
|
|
|
|
|
/*
|
|
|
|
* type: POST
|
|
|
|
*/
|
|
|
|
shepherd.post('/getconf', (req, res) => {
|
|
|
|
shepherd.log('======= req.body =======');
|
|
|
|
shepherd.log(req.body);
|
|
|
|
|
|
|
|
const confpath = getConf(req.body.chain, req.body.coind);
|
|
|
|
|
|
|
|
shepherd.log('got conf path is:');
|
|
|
|
shepherd.log(confpath);
|
|
|
|
shepherd.writeLog('got conf path is:');
|
|
|
|
shepherd.writeLog(confpath);
|
|
|
|
|
|
|
|
const obj = {
|
|
|
|
msg: 'success',
|
|
|
|
result: confpath,
|
|
|
|
};
|
|
|
|
|
|
|
|
res.end(JSON.stringify(obj));
|
|
|
|
});
|
|
|
|
|
|
|
|
shepherd.setConfKMD = (isChips) => {
|
|
|
|
// check if kmd conf exists
|
|
|
|
shepherd._fs.access(isChips ? `${shepherd.chipsDir}/chips.conf` : `${shepherd.komodoDir}/komodo.conf`, shepherd.fs.constants.R_OK, (err) => {
|
|
|
|
if (err) {
|
|
|
|
shepherd.log(isChips ? 'creating chips conf' : 'creating komodo conf');
|
|
|
|
shepherd.writeLog(isChips ? `creating chips conf in ${shepherd.chipsDir}/chips.conf` : `creating komodo conf in ${shepherd.komodoDir}/komodo.conf`);
|
|
|
|
setConf(isChips ? 'chipsd' : 'komodod');
|
|
|
|
} else {
|
|
|
|
const _confSize = shepherd.fs.lstatSync(isChips ? `${shepherd.chipsDir}/chips.conf` : `${shepherd.komodoDir}/komodo.conf`);
|
|
|
|
|
|
|
|
if (_confSize.size === 0) {
|
|
|
|
shepherd.log(isChips ? 'err: chips conf file is empty, creating chips conf' : 'err: komodo conf file is empty, creating komodo conf');
|
|
|
|
shepherd.writeLog(isChips ? `creating chips conf in ${shepherd.chipsDir}/chips.conf` : `creating komodo conf in ${shepherd.komodoDir}/komodo.conf`);
|
|
|
|
setConf(isChips ? 'chipsd' : 'komodod');
|
|
|
|
} else {
|
|
|
|
shepherd.writeLog(isChips ? 'chips conf exists' : 'komodo conf exists');
|
|
|
|
shepherd.log(isChips ? 'chips conf exists' : 'komodo conf exists');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
return shepherd;
|
|
|
|
};
|