|
|
|
const portscanner = require('portscanner');
|
|
|
|
const execFile = require('child_process').execFile;
|
|
|
|
|
|
|
|
module.exports = (shepherd) => {
|
|
|
|
shepherd.quitKomodod = (timeout = 100) => {
|
|
|
|
// if komodod is under heavy load it may not respond to cli stop the first time
|
|
|
|
// exit komodod gracefully
|
|
|
|
let coindExitInterval = {};
|
|
|
|
shepherd.lockDownAddCoin = true;
|
|
|
|
|
|
|
|
for (let key in shepherd.coindInstanceRegistry) {
|
|
|
|
if (shepherd.appConfig.stopNativeDaemonsOnQuit) {
|
|
|
|
const chain = key !== 'komodod' ? key : null;
|
|
|
|
let _coindQuitCmd = shepherd.komodocliBin;
|
|
|
|
|
|
|
|
// any coind
|
|
|
|
if (shepherd.nativeCoindList[key.toLowerCase()]) {
|
|
|
|
_coindQuitCmd = `${shepherd.coindRootDir}/${key.toLowerCase()}/${shepherd.nativeCoindList[key.toLowerCase()].bin.toLowerCase()}-cli`;
|
|
|
|
}
|
|
|
|
if (key === 'CHIPS') {
|
|
|
|
_coindQuitCmd = shepherd.chipscliBin;
|
|
|
|
}
|
|
|
|
|
|
|
|
const execCliStop = () => {
|
|
|
|
let _arg = [];
|
|
|
|
if (chain &&
|
|
|
|
!shepherd.nativeCoindList[key.toLowerCase()] && key !== 'CHIPS') {
|
|
|
|
_arg.push(`-ac_name=${chain}`);
|
|
|
|
|
|
|
|
if (shepherd.appConfig.dataDir.length) {
|
|
|
|
_arg.push(`-datadir=${shepherd.appConfig.dataDir + (key !== 'komodod' ? '/' + key : '')}`);
|
|
|
|
}
|
|
|
|
} else if (key === 'komodod' && shepherd.appConfig.dataDir.length) {
|
|
|
|
_arg.push(`-datadir=${shepherd.appConfig.dataDir}`);
|
|
|
|
}
|
|
|
|
|
|
|
|
_arg.push('stop');
|
|
|
|
execFile(`${_coindQuitCmd}`, _arg, (error, stdout, stderr) => {
|
|
|
|
shepherd.log(`stdout: ${stdout}`);
|
|
|
|
shepherd.log(`stderr: ${stderr}`);
|
|
|
|
shepherd.log(`send stop sig to ${key}`);
|
|
|
|
|
|
|
|
if (stdout.indexOf('EOF reached') > -1 ||
|
|
|
|
stderr.indexOf('EOF reached') > -1 ||
|
|
|
|
(error && error.toString().indexOf('Command failed') > -1 && !stderr) || // win "special snowflake" case
|
|
|
|
stdout.indexOf('connect to server: unknown (code -1)') > -1 ||
|
|
|
|
stderr.indexOf('connect to server: unknown (code -1)') > -1) {
|
|
|
|
delete shepherd.coindInstanceRegistry[key];
|
|
|
|
clearInterval(coindExitInterval[key]);
|
|
|
|
}
|
|
|
|
|
|
|
|
// workaround for AGT-65
|
|
|
|
const _port = shepherd.assetChainPorts[key];
|
|
|
|
setTimeout(() => {
|
|
|
|
portscanner.checkPortStatus(_port, '127.0.0.1', (error, status) => {
|
|
|
|
// Status is 'open' if currently in use or 'closed' if available
|
|
|
|
if (status === 'closed') {
|
|
|
|
delete shepherd.coindInstanceRegistry[key];
|
|
|
|
clearInterval(coindExitInterval[key]);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}, 100);
|
|
|
|
|
|
|
|
if (error !== null) {
|
|
|
|
shepherd.log(`exec error: ${error}`);
|
|
|
|
}
|
|
|
|
|
|
|
|
setTimeout(() => {
|
|
|
|
shepherd.killRogueProcess(key === 'CHIPS' ? 'chips-cli' : 'komodo-cli');
|
|
|
|
}, 100);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
execCliStop();
|
|
|
|
coindExitInterval[key] = setInterval(() => {
|
|
|
|
execCliStop();
|
|
|
|
}, timeout);
|
|
|
|
} else {
|
|
|
|
delete shepherd.coindInstanceRegistry[key];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
shepherd.post('/coind/stop', (req, res) => {
|
|
|
|
if (shepherd.checkToken(req.body.token)) {
|
|
|
|
const _chain = req.body.chain;
|
|
|
|
let _coindQuitCmd = shepherd.komodocliBin;
|
|
|
|
let _arg = [];
|
|
|
|
|
|
|
|
if (_chain) {
|
|
|
|
_arg.push(`-ac_name=${_chain}`);
|
|
|
|
|
|
|
|
if (shepherd.appConfig.dataDir.length) {
|
|
|
|
_arg.push(`-datadir=${shepherd.appConfig.dataDir + (_chain ? '/' + _chain : '')}`);
|
|
|
|
}
|
|
|
|
} else if (!_chain && shepherd.appConfig.dataDir.length) {
|
|
|
|
_arg.push(`-datadir=${shepherd.appConfig.dataDir}`);
|
|
|
|
}
|
|
|
|
|
|
|
|
_arg.push('stop');
|
|
|
|
execFile(`${_coindQuitCmd}`, _arg, (error, stdout, stderr) => {
|
|
|
|
shepherd.log(`stdout: ${stdout}`);
|
|
|
|
shepherd.log(`stderr: ${stderr}`);
|
|
|
|
shepherd.log(`send stop sig to ${_chain ? _chain : 'komodo'}`);
|
|
|
|
|
|
|
|
if (stdout.indexOf('EOF reached') > -1 ||
|
|
|
|
stderr.indexOf('EOF reached') > -1 ||
|
|
|
|
(error && error.toString().indexOf('Command failed') > -1 && !stderr) || // win "special snowflake" case
|
|
|
|
stdout.indexOf('connect to server: unknown (code -1)') > -1 ||
|
|
|
|
stderr.indexOf('connect to server: unknown (code -1)') > -1) {
|
|
|
|
delete shepherd.coindInstanceRegistry[_chain ? _chain : 'komodod'];
|
|
|
|
|
|
|
|
const obj = {
|
|
|
|
msg: 'success',
|
|
|
|
result: 'result',
|
|
|
|
};
|
|
|
|
|
|
|
|
res.end(JSON.stringify(obj));
|
|
|
|
} else {
|
|
|
|
if (stdout.indexOf('Komodo server stopping') > -1) {
|
|
|
|
delete shepherd.coindInstanceRegistry[_chain ? _chain : 'komodod'];
|
|
|
|
|
|
|
|
const obj = {
|
|
|
|
msg: 'success',
|
|
|
|
result: 'result',
|
|
|
|
};
|
|
|
|
|
|
|
|
res.end(JSON.stringify(obj));
|
|
|
|
} else {
|
|
|
|
const obj = {
|
|
|
|
msg: 'error',
|
|
|
|
result: 'result',
|
|
|
|
};
|
|
|
|
|
|
|
|
res.end(JSON.stringify(obj));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
const errorObj = {
|
|
|
|
msg: 'error',
|
|
|
|
result: 'unauthorized access',
|
|
|
|
};
|
|
|
|
|
|
|
|
res.end(JSON.stringify(errorObj));
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
shepherd.post('/coins/remove', (req, res) => {
|
|
|
|
if (shepherd.checkToken(req.body.token)) {
|
|
|
|
const _chain = req.body.chain;
|
|
|
|
|
|
|
|
if (req.body.mode === 'native') {
|
|
|
|
delete shepherd.coindInstanceRegistry[_chain ? _chain : 'komodod'];
|
|
|
|
|
|
|
|
const obj = {
|
|
|
|
msg: 'success',
|
|
|
|
result: 'result',
|
|
|
|
};
|
|
|
|
|
|
|
|
res.end(JSON.stringify(obj));
|
|
|
|
} else {
|
|
|
|
delete shepherd.electrumCoins[_chain === 'komodo' ? 'KMD' : _chain];
|
|
|
|
|
|
|
|
if (Object.keys(shepherd.electrumCoins).length - 1 === 0) {
|
|
|
|
shepherd.electrumCoins.auth = false;
|
|
|
|
shepherd.electrumKeys = {};
|
|
|
|
}
|
|
|
|
|
|
|
|
const obj = {
|
|
|
|
msg: 'success',
|
|
|
|
result: 'result',
|
|
|
|
};
|
|
|
|
|
|
|
|
res.end(JSON.stringify(obj));
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
const errorObj = {
|
|
|
|
msg: 'error',
|
|
|
|
result: 'unauthorized access',
|
|
|
|
};
|
|
|
|
|
|
|
|
res.end(JSON.stringify(errorObj));
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
return shepherd;
|
|
|
|
};
|