const electron = require('electron'), app = electron.app, BrowserWindow = electron.BrowserWindow, path = require('path'), url = require('url'), os = require('os'), fsnode = require('fs'), fs = require('fs-extra'), _fs = require('graceful-fs'), mkdirp = require('mkdirp'), express = require('express'), exec = require('child_process').exec, spawn = require('child_process').spawn, md5 = require('md5'), pm2 = require('pm2'), request = require('request'), async = require('async'), rimraf = require('rimraf'), portscanner = require('portscanner'), aes256 = require('nodejs-aes256'), Promise = require('bluebird'); const fixPath = require('fix-path'); var ps = require('ps-node'), setconf = require('../private/setconf.js'), assetChainPorts = require('./ports.js'), shepherd = express.Router(), iguanaInstanceRegistry = {}, syncOnlyIguanaInstanceInfo = {}, syncOnlyInstanceInterval = -1, guiLog = {}, rpcConf = {}; // IGUANA FILES AND CONFIG SETTINGS var iguanaConfsDirSrc = path.join(__dirname, '../assets/deps/confs'), CorsProxyBin = path.join(__dirname, '../node_modules/corsproxy/bin/corsproxy'); // SETTING OS DIR TO RUN IGUANA FROM // SETTING APP ICON FOR LINUX AND WINDOWS if (os.platform() === 'darwin') { fixPath(); var iguanaBin = path.join(__dirname, '../assets/bin/osx/iguana'), iguanaDir = `${process.env.HOME}/Library/Application Support/iguana`, iguanaConfsDir = `${iguanaDir}/confs`, komododBin = path.join(__dirname, '../assets/bin/osx/komodod'), komodocliBin = path.join(__dirname, '../assets/bin/osx/komodo-cli'), komodoDir = `${process.env.HOME}/Library/Application Support/Komodo`, zcashdBin = '/Applications/ZCashSwingWalletUI.app/Contents/MacOS/zcashd', zcashcliBin = '/Applications/ZCashSwingWalletUI.app/Contents/MacOS/zcash-cli', zcashDir = `${process.env.HOME}/Library/Application Support/Zcash`; } if (os.platform() === 'linux') { var iguanaBin = path.join(__dirname, '../assets/bin/linux64/iguana'), iguanaDir = `${process.env.HOME}/.iguana`, iguanaConfsDir = `${iguanaDir}/confs`, iguanaIcon = path.join(__dirname, '/assets/icons/agama_icons/128x128.png'), komododBin = path.join(__dirname, '../assets/bin/linux64/komodod'), komodocliBin = path.join(__dirname, '../assets/bin/linux64/komodo-cli'), komodoDir = `${process.env.HOME}/.komodo`; } if (os.platform() === 'win32') { var iguanaBin = path.join(__dirname, '../assets/bin/win64/iguana.exe'); iguanaBin = path.normalize(iguanaBin); iguanaDir = `${process.env.APPDATA}/iguana`; iguanaDir = path.normalize(iguanaDir); iguanaConfsDir = `${process.env.APPDATA}/iguana/confs`; iguanaConfsDir = path.normalize(iguanaConfsDir); iguanaIcon = path.join(__dirname, '/assets/icons/agama_icons/agama_app_icon.ico'), iguanaConfsDirSrc = path.normalize(iguanaConfsDirSrc), komododBin = path.join(__dirname, '../assets/bin/win64/komodod.exe'), komododBin = path.normalize(komododBin), komodocliBin = path.join(__dirname, '../assets/bin/win64/komodo-cli.exe'), komodocliBin = path.normalize(komodocliBin), komodoDir = `${process.env.APPDATA}/Komodo`, komodoDir = path.normalize(komodoDir); } shepherd.appConfig = { "edexGuiOnly": true, "iguanaGuiOnly": false, "manualIguanaStart": false, "skipBasiliskNetworkCheck": true, "minNotaries": 8, "host": "127.0.0.1", "agamaPort": 17777, "iguanaCorePort": 7778, "maxDescriptors": { "darwin": 90000, "linux": 1000000 }, "killIguanaOnStart": true, "dev": false, "v2": true, "useBasiliskInstance": true, "debug": true, "cli": { "passthru": false, "default": false } }; shepherd.writeLog = function(data) { const logLocation = `${iguanaDir}/shepherd`; const timeFormatted = new Date(Date.now()).toLocaleString('en-US', { hour12: false }); if (fs.existsSync(`${logLocation}/agamalog.txt`)) { fs.appendFile(`${logLocation}/agamalog.txt`, `${timeFormatted} ${data}\r\n`, function (err) { if (err) { console.log('error writing log file'); } }); } else { fs.writeFile(`${logLocation}/agamalog.txt`, `${timeFormatted} ${data}\r\n`, function (err) { if (err) { console.log('error writing log file'); } }); } } shepherd.createIguanaDirs = function() { if (!fs.existsSync(iguanaDir)) { fs.mkdirSync(iguanaDir); if (fs.existsSync(iguanaDir)) { console.log(`created iguana folder at ${iguanaDir}`); shepherd.writeLog(`created iguana folder at ${iguanaDir}`); } } else { console.log('iguana folder already exists'); } if (!fs.existsSync(`${iguanaDir}/shepherd`)) { fs.mkdirSync(`${iguanaDir}/shepherd`); if (fs.existsSync(`${iguanaDir}/shepherd`)) { console.log(`created shepherd folder at ${iguanaDir}/shepherd`); shepherd.writeLog(`create shepherd folder at ${iguanaDir}/shepherd`); } } else { console.log('iguana/shepherd folder already exists'); } if (!fs.existsSync(`${iguanaDir}/shepherd/pin`)) { fs.mkdirSync(`${iguanaDir}/shepherd/pin`); if (fs.existsSync(`${iguanaDir}/shepherd/pin`)) { console.log(`created pin folder at ${iguanaDir}/shepherd/pin`); shepherd.writeLog(`create pin folder at ${iguanaDir}/shepherd/pin`); } } else { console.log('shepherd/pin folder already exists'); } } shepherd.post('/encryptkey', function(req, res, next) { if (req.query.key && req.query.string && req.query.pubkey) { const encryptedString = aes256.encrypt(req.query.key, req.query.string); // test pin security // - at least 1 char in upper case // - at least 1 digit // - at least one special character // - min length 8 const _pin = req.query.key; const _pinTest = _pin.match('^(?=.*[A-Z])(?=.*[^<>{}\"/|;:.,~!?@#$%^=&*\\]\\\\()\\[_+]*$)(?=.*[0-9])(?=.*[a-z]).{8}$'); console.log(_pinTest); fs.writeFile(`${iguanaDir}/shepherd/pin/${req.query.pubkey}.pin`, encryptedString, function (err) { if (err) { console.log('error writing pin file'); } const returnObj = { 'msg': 'success', 'result': encryptedString }; res.end(JSON.stringify(returnObj)); }); } else { let errorObj = { 'msg': 'error', 'result': '' }; const _paramsList = [ 'key', 'string', 'pubkey' ]; let _errorParamsList = []; for (let i = 0; i < _paramsList.length; i++) { if (!req.query[_paramsList[i]]) { _errorParamsList.push(_paramsList[i]); } } errorObj.result = `missing param ${_errorParamsList.join(', ')}`; res.end(JSON.stringify(errorObj)); } }); shepherd.post('/decryptkey', function(req, res, next) { if (req.query.key && req.query.pubkey) { if (fs.existsSync(`${iguanaDir}/shepherd/pin/${req.query.pubkey}.pin`)) { fs.readFile(`${iguanaDir}/shepherd/pin/${req.query.pubkey}.pin`, 'utf8', function (err, data) { if (err) { const errorObj = { 'msg': 'error', 'result': err }; res.end(JSON.stringify(errorObj)); } else { const encryptedKey = aes256.decrypt(req.query.key, data); // test if stored encrypted passphrase is decrypted correctly // if not then the key is wrong const _regexTest = encryptedKey.match(/^[0-9a-zA-Z ]+$/g); let returnObj; if (!_regexTest) { returnObj = { 'msg': 'error', 'result': 'wrong key' }; } else { returnObj = { 'msg': 'success', 'result': encryptedKey }; } res.end(JSON.stringify(returnObj)); } }); } else { const errorObj = { 'msg': 'error', 'result': `file ${req.query.pubkey}.pin doesnt exist` }; res.end(JSON.stringify(errorObj)); } } else { const errorObj = { 'msg': 'error', 'result': 'missing key or pubkey param' }; res.end(JSON.stringify(errorObj)); } }); /* * type: GET * params: none */ shepherd.get('/coinslist', function(req, res, next) { if (fs.existsSync(`${iguanaDir}/shepherd/coinslist.json`)) { fs.readFile(`${iguanaDir}/shepherd/coinslist.json`, 'utf8', function (err, data) { if (err) { const errorObj = { 'msg': 'error', 'result': err }; res.end(JSON.stringify(errorObj)); } else { const successObj = { 'msg': 'success', 'result': data ? JSON.parse(data) : '' }; res.end(JSON.stringify(successObj)); } }); } else { const errorObj = { 'msg': 'error', 'result': 'coin list doesn\'t exist' }; res.end(JSON.stringify(errorObj)); } }); /* * type: POST * params: payload */ shepherd.post('/guilog', function(req, res, next) { const logLocation = `${iguanaDir}/shepherd`; if (!guiLog[shepherd.appSessionHash]) { guiLog[shepherd.appSessionHash] = {}; } if (guiLog[shepherd.appSessionHash][req.body.timestamp]) { guiLog[shepherd.appSessionHash][req.body.timestamp].status = req.body.status; guiLog[shepherd.appSessionHash][req.body.timestamp].response = req.body.response; } else { guiLog[shepherd.appSessionHash][req.body.timestamp] = { 'function': req.body.function, 'type': req.body.type, 'url': req.body.url, 'payload': req.body.payload, 'status': req.body.status, }; } fs.writeFile(`${logLocation}/agamalog.json`, JSON.stringify(guiLog), function (err) { if (err) { shepherd.writeLog('error writing gui log file'); } const returnObj = { 'msg': 'success', 'result': 'gui log entry is added' }; res.end(JSON.stringify(returnObj)); }); }); /* * type: GET * params: type */ shepherd.get('/getlog', function(req, res, next) { const logExt = req.query.type === 'txt' ? 'txt' : 'json'; if (fs.existsSync(`${iguanaDir}/shepherd/agamalog.${logExt}`)) { fs.readFile(`${iguanaDir}/shepherd/agamalog.${logExt}`, 'utf8', function (err, data) { if (err) { const errorObj = { 'msg': 'error', 'result': err }; res.end(JSON.stringify(errorObj)); } else { const successObj = { 'msg': 'success', 'result': data ? JSON.parse(data) : '' }; res.end(JSON.stringify(successObj)); } }); } else { const errorObj = { 'msg': 'error', 'result': `agama.${logExt} doesnt exist` }; res.end(JSON.stringify(errorObj)); } }); /* * type: POST * params: payload */ shepherd.post('/coinslist', function(req, res, next) { const _payload = req.body.payload; if (!_payload) { const errorObj = { 'msg': 'error', 'result': 'no payload provided' }; res.end(JSON.stringify(errorObj)); } else { fs.writeFile(`${cache.iguanaDir}/shepherd/coinslist.json`, JSON.stringify(_payload), function (err) { if (err) { const errorObj = { 'msg': 'error', 'result': err }; res.end(JSON.stringify(errorObj)); } else { const successObj = { 'msg': 'success', 'result': 'done' }; res.end(JSON.stringify(successObj)); } }); } }); // TODO: check if komodod is running shepherd.quitKomodod = function(chain) { // exit komodod gracefully console.log('exec ' + komodocliBin + (chain ? ' -ac_name=' + chain : '') + ' stop'); exec(komodocliBin + (chain ? ' -ac_name=' + chain : '') + ' stop', function(error, stdout, stderr) { console.log(`stdout: ${stdout}`); console.log(`stderr: ${stderr}`); if (error !== null) { console.log(`exec error: ${error}`); } }); } shepherd.getConf = function(chain) { const _confLocation = chain === 'komodod' ? `${komodoDir}/komodo.conf` : `${komodoDir}/${chain}/${chain}.conf`; if (fs.existsSync(_confLocation)) { const _port = assetChainPorts[chain]; const _rpcConf = fs.readFileSync(_confLocation, 'utf8'); if (_rpcConf.length) { let _match; let parsedRpcConfig = { user: '', pass: '', port: _port }; if (_match = _rpcConf.match(/rpcuser=\s*(.*)/)) { parsedRpcConfig.user = _match[1]; } if ((_match = _rpcConf.match(/rpcpass=\s*(.*)/)) || (_match = _rpcConf.match(/rpcpassword=\s*(.*)/))) { parsedRpcConfig.pass = _match[1]; } rpcConf[chain === 'komodod' ? 'KMD' : chain] = parsedRpcConfig; } else { console.log(`${_confLocation} is empty`); } } else { console.log(`${_confLocation} doesnt exist`); } } /* * type: POST * params: payload */ shepherd.post('/cli', function(req, res, next) { if (!req.body.payload) { const errorObj = { 'msg': 'error', 'result': 'no payload provided' }; res.end(JSON.stringify(errorObj)); } else if (!req.body.payload.cmd.match(/^[0-9a-zA-Z _\[\]"'/\\]+$/g)) { const errorObj = { 'msg': 'error', 'result': 'wrong cli string format' }; res.end(JSON.stringify(errorObj)); } else { const _mode = req.body.payload.mode === 'passthru' ? 'passthru' : 'default'; const _chain = req.body.payload.chain === 'KMD' ? null : req.body.payload.chain; const _cmd = req.body.payload.cmd; const _params = req.body.payload.params ? ` ${req.body.payload.params}` : ''; if (!rpcConf[_chain]) { shepherd.getConf(req.body.payload.chain === 'KMD' ? 'komodod' : req.body.payload.chain); } if (_mode === 'default') { let _body = { 'agent': 'bitcoinrpc', 'method': _cmd }; if (req.body.payload.params) { _body = { 'agent': 'bitcoinrpc', 'method': _cmd, 'params': req.body.payload.params === ' ' ? [''] : req.body.payload.params }; } const options = { url: `http://localhost:${rpcConf[req.body.payload.chain].port}`, method: 'POST', auth: { 'user': rpcConf[req.body.payload.chain].user, 'pass': rpcConf[req.body.payload.chain].pass }, body: JSON.stringify(_body) }; // send back body on both success and error // this bit replicates iguana core's behaviour request(options, function (error, response, body) { if (response && response.statusCode && response.statusCode === 200) { res.end(body); } else { res.end(body); } }); } else { exec(komodocliBin + (_chain ? ' -ac_name=' + _chain : '') + ' ' + _cmd + _params, function(error, stdout, stderr) { console.log(`stdout: ${stdout}`); console.log(`stderr: ${stderr}`); if (error !== null) { console.log(`exec error: ${error}`); } let responseObj; if (stderr) { responseObj = { 'msg': 'error', 'result': stderr }; } else { responseObj = { 'msg': 'success', 'result': stdout }; } res.end(JSON.stringify(responseObj)); }); } } }); /* * type: POST * params: payload */ shepherd.post('/appconf', function(req, res, next) { if (!req.body.payload) { const errorObj = { 'msg': 'error', 'result': 'no payload provided' }; res.end(JSON.stringify(errorObj)); } else { shepherd.saveLocalAppConf(req.body.payload); const errorObj = { 'msg': 'success', 'result': 'config saved' }; res.end(JSON.stringify(errorObj)); } }); shepherd.saveLocalAppConf = function(appSettings) { let appConfFileName = `${iguanaDir}/config.json`; _fs.access(iguanaDir, fs.constants.R_OK, function(err) { if (!err) { const FixFilePermissions = function() { return new Promise(function(resolve, reject) { const result = 'config.json file permissions updated to Read/Write'; fsnode.chmodSync(appConfFileName, '0666'); setTimeout(function() { console.log(result); shepherd.writeLog(result); resolve(result); }, 1000); }); } const FsWrite = function() { return new Promise(function(resolve, reject) { const result = 'config.json write file is done'; fs.writeFile(appConfFileName, JSON.stringify(appSettings) .replace(/,/g, ',\n') // format json in human readable form .replace(/:/g, ': ') .replace(/{/g, '{\n') .replace(/}/g, '\n}'), 'utf8', function(err) { if (err) return console.log(err); }); fsnode.chmodSync(appConfFileName, '0666'); setTimeout(function() { console.log(result); console.log(`app conf.json file is created successfully at: ${iguanaConfsDir}`); shepherd.writeLog(`app conf.json file is created successfully at: ${iguanaConfsDir}`); resolve(result); }, 2000); }); } FsWrite() .then(FixFilePermissions()); } }); } shepherd.loadLocalConfig = function() { if (fs.existsSync(`${iguanaDir}/config.json`)) { let localAppConfig = fs.readFileSync(`${iguanaDir}/config.json`, 'utf8'); console.log('app config set from local file'); shepherd.writeLog('app config set from local file'); // find diff between local and hardcoded configs // append diff to local config const compareJSON = function(obj1, obj2) { let result = {}; for (let i in obj1) { if (!obj2.hasOwnProperty(i)) { result[i] = obj1[i]; } } return result; }; if (localAppConfig) { const compareConfigs = compareJSON(shepherd.appConfig, JSON.parse(localAppConfig)); if (Object.keys(compareConfigs).length) { const newConfig = Object.assign(JSON.parse(localAppConfig), compareConfigs); console.log('config diff is found, updating local config'); console.log('config diff:'); console.log(compareConfigs); shepherd.writeLog('aconfig diff is found, updating local config'); shepherd.writeLog('config diff:'); shepherd.writeLog(compareConfigs); shepherd.saveLocalAppConf(newConfig); return newConfig; } else { return JSON.parse(localAppConfig); } } else { return shepherd.appConfig; } } else { console.log('local config file is not found!'); shepherd.writeLog('local config file is not found!'); shepherd.saveLocalAppConf(shepherd.appConfig); return shepherd.appConfig; } }; shepherd.appConfig = shepherd.loadLocalConfig(); console.log(`iguana dir: ${iguanaDir}`); console.log(`iguana bin: ${iguanaBin}`); console.log('--------------------------') console.log(`iguana dir: ${komododBin}`); console.log(`iguana bin: ${komodoDir}`); shepherd.writeLog(`iguana dir: ${iguanaDir}`); shepherd.writeLog(`iguana bin: ${iguanaBin}`); shepherd.writeLog(`iguana dir: ${komododBin}`); shepherd.writeLog(`iguana bin: ${komodoDir}`); // END IGUANA FILES AND CONFIG SETTINGS // default route shepherd.get('/', function(req, res, next) { res.send('Iguana app server'); }); /* * type: GET * */ shepherd.get('/appconf', function(req, res, next) { const obj = shepherd.loadLocalConfig(); res.send(obj); }); /* * type: GET * */ shepherd.get('/sysinfo', function(req, res, next) { const obj = shepherd.SystemInfo(); res.send(obj); }); /* * type: GET * */ shepherd.get('/appinfo', function(req, res, next) { const obj = shepherd.appInfo(); res.send(obj); }); shepherd.dumpCacheBeforeExit = function() { cache.dumpCacheBeforeExit(); } var cache = require('./cache'); var mock = require('./mock'); // expose sockets obj shepherd.setIO = function(io) { shepherd.io = io; cache.setVar('io', io); }; shepherd.setVar = function(_name, _body) { shepherd[_name] = _body; }; cache.setVar('iguanaDir', iguanaDir); cache.setVar('appConfig', shepherd.appConfig); // fetch sync only forks info shepherd.getSyncOnlyForksInfo = function() { async.forEachOf(iguanaInstanceRegistry, function(data, port) { if (iguanaInstanceRegistry[port].mode.indexOf('/sync') > -1) { syncOnlyIguanaInstanceInfo[port] = {}; request({ url: `http://localhost:${port}/api/bitcoinrpc/getinfo?userpass=tmpIgRPCUser@${shepherd.appSessionHash}`, method: 'GET' }, function (error, response, body) { if (response && response.statusCode && response.statusCode === 200) { // console.log(body); try { syncOnlyIguanaInstanceInfo[port].getinfo = JSON.parse(body); } catch(e) {} } else { // TODO: error } }); request({ url: `http://localhost:${port}/api/SuperNET/activehandle?userpass=${shepherd.appSessionHash}`, method: 'GET' }, function (error, response, body) { if (response && response.statusCode && response.statusCode === 200) { // console.log(body); try { syncOnlyIguanaInstanceInfo[port].activehandle = JSON.parse(body); } catch(e) {} } else { // TODO: error } }); syncOnlyIguanaInstanceInfo[port].registry = iguanaInstanceRegistry[port]; } }); } /* * type: GET * */ shepherd.get('/forks/info/start', function(req, res, next) { const successObj = { 'msg': 'success', 'result': 'started' }; res.end(JSON.stringify(successObj)); shepherd.getSyncOnlyForksInfo(); }); /* * type: GET * */ shepherd.get('/forks/info/show', function(req, res, next) { const successObj = { 'msg': 'success', 'result': JSON.stringify(syncOnlyIguanaInstanceInfo) }; res.end(JSON.stringify(successObj)); }); /* * type: GET * */ shepherd.get('/forks/restart', function(req, res, next) { const _pmid = req.query.pmid; pm2.connect(function(err) { if (err) { console.error(err); } pm2.restart(_pmid, function(err, ret) { if (err) { console.error(err); } pm2.disconnect(); const successObj = { 'msg': 'success', 'result': 'restarted' }; shepherd.writeLog(`iguana fork pmid ${_pmid} restarted`); res.end(JSON.stringify(successObj)); }); }); }); /* * type: GET * */ shepherd.get('/forks/stop', function(req, res, next) { const _pmid = req.query.pmid; pm2.connect(function(err) { if (err) { console.error(err); } pm2.stop(_pmid, function(err, ret) { if (err) { console.error(err); } pm2.disconnect(); const successObj = { 'msg': 'success', 'result': 'stopped' }; shepherd.writeLog(`iguana fork pmid ${_pmid} stopped`); res.end(JSON.stringify(successObj)); }); }); }); /* * type: GET * */ shepherd.get('/forks', function(req, res, next) { const successObj = { 'msg': 'success', 'result': iguanaInstanceRegistry }; res.end(JSON.stringify(successObj)); }); /* * type: POST * params: name */ shepherd.post('/forks', function(req, res, next) { const mode = req.body.mode, coin = req.body.coin, port = shepherd.appConfig.iguanaCorePort; portscanner.findAPortNotInUse(port, port + 100, '127.0.0.1', function(error, _port) { pm2.connect(true, function(err) { //start up pm2 god if (err) { console.error(err); process.exit(2); } console.log(`iguana core fork port ${_port}`); shepherd.writeLog(`iguana core fork port ${_port}`); pm2.start({ script: iguanaBin, // path to binary name: `IGUANA ${_port} ${mode} / ${coin}`, exec_mode : 'fork', args: [`-port=${_port}`], cwd: iguanaDir //set correct iguana directory }, function(err, apps) { iguanaInstanceRegistry[_port] = { 'mode': mode, 'coin': coin, 'pid': apps[0].process && apps[0].process.pid, 'pmid': apps[0].pm2_env.pm_id }; cache.setVar('iguanaInstances', iguanaInstanceRegistry); // get sync only forks info if (syncOnlyInstanceInterval === -1) { setTimeout(function() { shepherd.getSyncOnlyForksInfo(); }, 5000); setInterval(function() { shepherd.getSyncOnlyForksInfo(); }, 20000); } const successObj = { 'msg': 'success', 'result': _port }; res.end(JSON.stringify(successObj)); pm2.disconnect(); // Disconnect from PM2 if (err) { throw err; shepherd.writeLog(`iguana fork error: ${err}`); console.log(`iguana fork error: ${err}`); } }); }); }); }); /* * type: GET * params: pubkey */ shepherd.get('/cache', function(req, res, next) { cache.get(req, res, next); }); /* * type: GET * params: filename */ shepherd.get('/groom', function(req, res, next) { cache.groomGet(req, res, next); }) /* * type: DELETE * params: filename */ shepherd.delete('/groom', function(req, res, next) { cache.groomDelete(req, res, next); }); /* * type: POST * params: filename, payload */ shepherd.post('/groom', function(req, res, next) { cache.groomPost(req, res, next); }); /* * type: GET * params: userpass, pubkey, skip */ shepherd.get('/cache-all', function(req, res, next) { cache.all(req, res, next); }); /* * type: GET * params: userpass, pubkey, coin, address, skip */ shepherd.get('/cache-one', function(req, res, next) { cache.one(req, res, next); }); /* * type: GET */ shepherd.get('/mock', function(req, res, next) { mock.get(req, res, next); }); /* * type: GET * params: herd, lastLines */ shepherd.post('/debuglog', function(req, res) { let _herd = req.body.herdname, _lastNLines = req.body.lastLines, _location; if (_herd === 'iguana') { _location = iguanaDir; } else if (_herd === 'komodo') { _location = komodoDir; } shepherd.readDebugLog(`${_location}/debug.log`, _lastNLines) .then(function(result) { const _obj = { 'msg': 'success', 'result': result }; res.end(JSON.stringify(_obj)); }, function(result) { const _obj = { 'msg': 'error', 'result': result }; res.end(JSON.stringify(_obj)); }); }); /* * type: POST * params: herd */ shepherd.post('/herd', function(req, res) { console.log('======= req.body ======='); console.log(req.body); herder(req.body.herd, req.body.options); const obj = { 'msg': 'success', 'result': 'result' }; res.end(JSON.stringify(obj)); if (req.body.herd === 'komodod') { const _port = assetChainPorts[req.body.options.ac_name]; // check if komodod instance is already running setTimeout(function() { portscanner.checkPortStatus(_port, '127.0.0.1', function(error, status) { // Status is 'open' if currently in use or 'closed' if available if (status === 'closed') { shepherd.writeLog(`komodod service start error at port ${_port}, reason: port is closed`); cache.io.emit('service', { 'komodod': { 'error': 'start error' } }); } }); }, 10000); } }); /* * type: POST * params: herdname */ shepherd.post('/herdlist', function(req, res) { console.log(req.body.herdname); pm2.connect(true, function(err) { if (err) throw err; // TODO: proper error handling pm2.describe(req.body.herdname, function(err, list) { pm2.disconnect(); // disconnect after getting proc info list if (err) throw err; // TODO: proper error handling console.log(list[0].pm2_env.status) // print status of IGUANA proc console.log(list[0].pid) // print pid of IGUANA proc shepherd.writeLog(list[0].pm2_env.status); shepherd.writeLog(list[0].pid); const obj = { 'herdname': req.body.herdname, 'status': list[0].pm2_env.status, 'pid': list[0].pid }; res.end(JSON.stringify(obj)); }); }); }); /* * type: POST */ shepherd.post('/slay', function(req, res) { console.log('======= req.body ======='); console.log(req.body); slayer(req.body.slay); const obj = { 'msg': 'success', 'result': 'result' }; res.end(JSON.stringify(obj)); }); /* * type: POST */ shepherd.post('/setconf', function(req, res) { console.log('======= req.body ======='); console.log(req.body); if (os.platform() === 'win32' && req.body.chain == 'komodod') { setkomodoconf = spawn(path.join(__dirname, '../build/artifacts.supernet.org/latest/windows/genkmdconf.bat')); setkomodoconf = spawn(path.join(__dirname, '../assets/bin/win64/genkmdconf.bat')); } else { setConf(req.body.chain); } const obj = { 'msg': 'success', 'result': 'result' }; res.end(JSON.stringify(obj)); }); /* * type: POST */ shepherd.post('/getconf', function(req, res) { console.log('======= req.body ======='); console.log(req.body); const confpath = getConf(req.body.chain); console.log('got conf path is:'); console.log(confpath); shepherd.writeLog('got conf path is:'); shepherd.writeLog(confpath); const obj = { 'msg': 'success', 'result': confpath }; res.end(JSON.stringify(obj)); }); /* * type: GET * params: coin, type */ shepherd.get('/kick', function(req, res, next) { const _coin = req.query.coin; const _type = req.query.type; if (!_coin) { const errorObj = { 'msg': 'error', 'result': 'no coin name provided' }; res.end(JSON.stringify(errorObj)); } if (!_type) { const errorObj = { 'msg': 'error', 'result': 'no type provided' }; res.end(JSON.stringify(errorObj)); } const kickStartDirs = { 'soft': [ { 'name': 'DB/[coin]', 'type': 'pattern', 'match': 'balancecrc.' }, { 'name': 'DB/[coin]/utxoaddrs', 'type': 'file' }, { 'name': 'DB/[coin]/accounts', 'type': 'folder' }, { 'name': 'DB/[coin]/fastfind', 'type': 'folder' }, { 'name': 'tmp/[coin]', 'type': 'folder' } ], 'hard': [ { 'name': 'DB/[coin]', 'type': 'pattern', 'match': 'balancecrc.' }, { 'name': 'DB/[coin]/utxoaddrs', 'type': 'file' }, { 'name': 'DB/[coin]', 'type': 'pattern', 'match': 'utxoaddrs.' }, { 'name': 'DB/[coin]/accounts', 'type': 'folder' }, { 'name': 'DB/[coin]/fastfind', 'type': 'folder' }, { 'name': 'DB/[coin]/spends', 'type': 'folder' }, { 'name': 'tmp/[coin]', 'type': 'folder' } ], 'brutal': [ // delete all coin related data { 'name': 'DB/[coin]', 'type': 'folder' }, { 'name': 'DB/purgeable/[coin]', 'type': 'folder' }, { 'name': 'DB/ro/[coin]', 'type': 'folder' }, { 'name': 'tmp/[coin]', 'type': 'folder' } ] }; if (_coin && _type) { for (let i = 0; i < kickStartDirs[_type].length; i++) { let currentKickItem = kickStartDirs[_type][i]; console.log('deleting ' + currentKickItem.type + (currentKickItem.match ? ' ' + currentKickItem.match : '') + ' ' + iguanaDir + '/' + currentKickItem.name.replace('[coin]', _coin)); if (currentKickItem.type === 'folder' || currentKickItem.type === 'file') { rimraf(iguanaDir + '/' + currentKickItem.name.replace('[coin]', _coin), function(err) { if (err) { throw err; } }); } else if (currentKickItem.type === 'pattern') { let dirItems = fs.readdirSync(`${iguanaDir}/currentKickItem.name.replace('[coin]', _coin)`); if (dirItems && dirItems.length) { for (let j = 0; j < dirItems.length; j++) { if (dirItems[j].indexOf(currentKickItem.match) > -1) { rimraf(`${iguanaDir}/${currentKickItem.name.replace('[coin]', _coin)}/${dirItems[j]}`, function(err) { if (err) { throw err; } }); console.log(`deleting ${dirItems[j]}`); } } } } } const successObj = { 'msg': 'success', 'result': 'kickstart: brutal is executed' }; res.end(JSON.stringify(successObj)); } }); shepherd.readDebugLog = function(fileLocation, lastNLines) { return new Promise( function(resolve, reject) { if (lastNLines) { _fs.access(fileLocation, fs.constants.R_OK, function(err) { if (err) { console.log(`error reading ${fileLocation}`); shepherd.writeLog(`error reading ${fileLocation}`); reject(`readDebugLog error: ${err}`); } else { console.log(`reading ${fileLocation}`); _fs.readFile(fileLocation, 'utf-8', function(err, data) { if (err) throw err; const lines = data.trim().split('\n'); const lastLine = lines.slice(lines.length - lastNLines, lines.length).join('\n'); resolve(lastLine); }); } }); } else { reject('readDebugLog error: lastNLines param is not provided!'); } } ); }; function herder(flock, data) { if (data === undefined) { data = 'none'; console.log('it is undefined'); } if (flock === 'iguana') { console.log('iguana flock selected...'); console.log(`selected data: ${data}`); shepherd.writeLog('iguana flock selected...'); shepherd.writeLog(`selected data: ${data}`); // MAKE SURE IGUANA DIR IS THERE FOR USER mkdirp(iguanaDir, function(err) { if (err) console.error(err); else fs.readdir(iguanaDir, (err, files) => { files.forEach(file => { //console.log(file); }); }) }); // ADD SHEPHERD FOLDER mkdirp(`${iguanaDir}/shepherd`, function(err) { if (err) console.error(err); else fs.readdir(iguanaDir, (err, files) => { files.forEach(file => { //console.log(file); }); }) }); // COPY CONFS DIR WITH PEERS FILE TO IGUANA DIR, AND KEEP IT IN SYNC fs.copy(iguanaConfsDirSrc, iguanaConfsDir, function (err) { if (err) return console.error(err); console.log(`confs files copied successfully at: ${iguanaConfsDir}`); shepherd.writeLog(`confs files copied successfully at: ${iguanaConfsDir}`); }); pm2.connect(true,function(err) { //start up pm2 god if (err) { console.error(err); process.exit(2); } console.log(`iguana core port ${shepherd.appConfig.iguanaCorePort}`); shepherd.writeLog(`iguana core port ${shepherd.appConfig.iguanaCorePort}`); pm2.start({ script: iguanaBin, // path to binary name: 'IGUANA', exec_mode : 'fork', args: [`-port=${shepherd.appConfig.iguanaCorePort}`], cwd: iguanaDir //set correct iguana directory }, function(err, apps) { iguanaInstanceRegistry[shepherd.appConfig.iguanaCorePort] = { 'mode': 'main', 'coin': 'none', 'pid': apps[0].process.pid, 'pmid': apps[0].pm2_env.pm_id }; shepherd.writeLog(`iguana core started at port ${shepherd.appConfig.iguanaCorePort} pid ${apps[0].process.pid}`); pm2.disconnect(); // Disconnect from PM2 if (err) { throw err; shepherd.writeLog(`iguana core port ${shepherd.appConfig.iguanaCorePort}`); console.log(`iguana fork error: ${err}`); } }); }); } if (flock === 'komodod') { let kmdDebugLogLocation = (data.ac_name !== 'komodod' ? komodoDir + '/' + data.ac_name : komodoDir) + '/debug.log'; console.log('komodod flock selected...'); console.log(`selected data: ${data}`); shepherd.writeLog('komodod flock selected...'); shepherd.writeLog(`selected data: ${data}`); // truncate debug.log try { _fs.access(kmdDebugLogLocation, fs.constants.R_OK, function(err) { if (err) { console.log(`error accessing ${kmdDebugLogLocation}`); shepherd.writeLog(`error accessing ${kmdDebugLogLocation}`); } else { console.log(`truncate ${kmdDebugLogLocation}`); shepherd.writeLog(`truncate ${kmdDebugLogLocation}`); fs.unlink(kmdDebugLogLocation); } }); } catch(e) { console.log(`komodod debug.log access err: ${e}`); shepherd.writeLog(`komodod debug.log access err: ${e}`); } // get komodod instance port const _port = assetChainPorts[data.ac_name]; try { // check if komodod instance is already running portscanner.checkPortStatus(_port, '127.0.0.1', function(error, status) { // Status is 'open' if currently in use or 'closed' if available if (status === 'closed') { // start komodod via exec if (data.ac_name === 'komodod') { const _customParamDict = { 'silent': '&', 'reindex': '-reindex', 'change': '-pubkey=' }; let _customParam = ''; if (data.ac_custom_param === 'silent' || data.ac_custom_param === 'reindex') { _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}`; } console.log(`exec ${komododBin} ${data.ac_options.join(' ')}${_customParam}`); shepherd.writeLog(`exec ${komododBin} ${data.ac_options.join(' ')}${_customParam}`); exec(`${komododBin} ${data.ac_options.join(' ')}${_customParam}`, { maxBuffer: 1024 * 10000 // 10 mb }, function(error, stdout, stderr) { // console.log('stdout: ' + stdout); // console.log('stderr: ' + stderr); shepherd.writeLog(`stdout: ${stdout}`); shepherd.writeLog(`stderr: ${stderr}`); if (error !== null) { console.log(`exec error: ${error}`) shepherd.writeLog(`exec error: ${error}`); } }); } else { pm2.connect(true, function(err) { // start up pm2 god if (err) { console.error(err); process.exit(2); } pm2.start({ script: komododBin, // path to binary name: data.ac_name, // REVS, USD, EUR etc. exec_mode : 'fork', cwd: komodoDir, args: data.ac_options }, function(err, apps) { shepherd.writeLog(`komodod fork started ${data.ac_name} ${JSON.stringify(data.ac_options)}`); pm2.disconnect(); // Disconnect from PM2 if (err) throw err; }); }); } } else { console.log(`port ${_port} (${data.ac_name}) is already in use`); shepherd.writeLog(`port ${_port} (${data.ac_name}) is already in use`); } }); } catch(e) { console.log(`failed to start komodod err: ${e}`); shepherd.writeLog(`failed to start komodod err: ${e}`); } } if (flock === 'zcashd') { let kmdDebugLogLocation = `${zcashDir}/debug.log`; console.log('zcashd flock selected...'); console.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) { console.error(err); process.exit(2); } pm2.start({ script: zcashdBin, // path to binary name: data.ac_name, // REVS, USD, EUR etc. exec_mode : 'fork', cwd: 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) throw err; }); }); } // deprecated, to be removed if (flock === 'corsproxy') { console.log('corsproxy flock selected...'); console.log(`selected data: ${data}`); pm2.connect(true,function(err) { //start up pm2 god if (err) { console.error(err); process.exit(2); } pm2.start({ script: CorsProxyBin, // path to binary name: 'CORSPROXY', exec_mode : 'fork', cwd: iguanaDir }, function(err, apps) { pm2.disconnect(); // Disconnect from PM2 if (err) throw err; }); }); } } function slayer(flock) { console.log(flock); pm2.delete(flock, function(err, ret) { pm2.disconnect(); shepherd.writeLog(`deleting flock ${flock}`); shepherd.writeLog(ret); console.log(ret); }); } shepherd.setConfKMD = function() { let komodoDir; let zcashDir; if (os.platform() === 'darwin') { komodoDir = `${process.env.HOME}/Library/Application Support/Komodo`; ZcashDir = `${process.env.HOME}/Library/Application Support/Zcash`; } if (os.platform() === 'linux') { komodoDir = `${process.env.HOME}/.komodo`; ZcashDir = `${process.env.HOME}/.zcash`; } if (os.platform() === 'win32') { komodoDir = `${process.env.APPDATA}/Komodo`; ZcashDir = `${process.env.APPDATA}/Zcash`; } // check if kmd conf exists _fs.access(`${komodoDir}/komodo.conf`, fs.constants.R_OK, function(err) { if (err) { console.log('creating komodo conf'); shepherd.writeLog(`creating komodo conf in ${komodoDir}/komodo.conf`); setConf('komodod'); } else { shepherd.writeLog('komodo conf exists'); console.log('komodo conf exists'); } }); } function setConf(flock) { let komodoDir; let zcashDir; console.log(flock); shepherd.writeLog(`setconf ${flock}`); if (os.platform() === 'darwin') { komodoDir = `${process.env.HOME}/Library/Application Support/Komodo`; ZcashDir = `${process.env.HOME}/Library/Application Support/Zcash`; } if (os.platform() === 'linux') { komodoDir = `${process.env.HOME}/.komodo`; ZcashDir = `${process.env.HOME}/.zcash`; } if (os.platform() === 'win32') { komodoDir = `${process.env.APPDATA}/Komodo`; ZcashDir = `${process.env.APPDATA}/Zcash`; } let DaemonConfPath; switch (flock) { case 'komodod': DaemonConfPath = `${komodoDir}/komodo.conf`; if (os.platform() === 'win32') { DaemonConfPath = path.normalize(DaemonConfPath); } break; case 'zcashd': DaemonConfPath = `${ZcashDir}/zcash.conf`; if (os.platform() === 'win32') { DaemonConfPath = path.normalize(DaemonConfPath); } break; default: DaemonConfPath = `${komodoDir}/${flock}/${flock}.conf`; if (os.platform() === 'win32') { DaemonConfPath = path.normalize(DaemonConfPath); } } console.log(DaemonConfPath); shepherd.writeLog(`setconf ${DaemonConfPath}`); const CheckFileExists = function() { return new Promise(function(resolve, reject) { const result = 'Check Conf file exists is done' fs.ensureFile(DaemonConfPath, function(err) { console.log(err); // => null }); setTimeout(function() { console.log(result); shepherd.writeLog(`setconf ${result}`); resolve(result); }, 2000); }); } const FixFilePermissions = function() { return new Promise(function(resolve, reject) { const result = 'Conf file permissions updated to Read/Write'; fsnode.chmodSync(DaemonConfPath, '0666'); setTimeout(function() { console.log(result); shepherd.writeLog(`setconf ${result}`); resolve(result); }, 1000); }); } const RemoveLines = function() { return new Promise(function(resolve, reject) { const result = 'RemoveLines is done' fs.readFile(DaemonConfPath, 'utf8', function(err, data) { if (err) { shepherd.writeLog(`setconf error '${err}`); return console.log(err); } const rmlines = data.replace(/(?:(?:\r\n|\r|\n)\s*){2}/gm, '\n'); fs.writeFile(DaemonConfPath, rmlines, 'utf8', function(err) { if (err) return console.log(err); }); }); fsnode.chmodSync(DaemonConfPath, '0666'); setTimeout(function() { shepherd.writeLog(`setconf ${result}`); console.log(result); resolve(result); }, 2000); }); } const CheckConf = function() { return new Promise(function(resolve, reject) { const result = 'CheckConf is done'; setconf.status(DaemonConfPath, function(err, status) { const rpcuser = function() { return new Promise(function(resolve, reject) { const result = 'checking rpcuser...'; if (status[0].hasOwnProperty('rpcuser')) { console.log('rpcuser: OK'); shepherd.writeLog('rpcuser: OK'); } else { const randomstring = md5(Math.random() * Math.random() * 999); console.log('rpcuser: NOT FOUND'); shepherd.writeLog('rpcuser: NOT FOUND'); fs.appendFile(DaemonConfPath, `\nrpcuser=user${randomstring.substring(0, 16)}`, (err) => { if (err) throw err; console.log('rpcuser: ADDED'); shepherd.writeLog('rpcuser: ADDED'); }); } resolve(result); }); } const rpcpass = function() { return new Promise(function(resolve, reject) { const result = 'checking rpcpassword...'; if (status[0].hasOwnProperty('rpcpassword')) { console.log('rpcpassword: OK'); shepherd.writeLog('rpcpassword: OK'); } else { var randomstring = md5(Math.random() * Math.random() * 999); console.log('rpcpassword: NOT FOUND'); shepherd.writeLog('rpcpassword: NOT FOUND'); fs.appendFile(DaemonConfPath, `\nrpcpassword=${randomstring}`, (err) => { if (err) throw err; console.log('rpcpassword: ADDED'); shepherd.writeLog('rpcpassword: ADDED'); }); } resolve(result); }); } const server = function() { return new Promise(function(resolve, reject) { const result = 'checking server...'; if (status[0].hasOwnProperty('server')) { console.log('server: OK'); shepherd.writeLog('server: OK'); } else { console.log('server: NOT FOUND'); shepherd.writeLog('server: NOT FOUND'); fs.appendFile(DaemonConfPath, '\nserver=1', (err) => { if (err) throw err; console.log('server: ADDED'); shepherd.writeLog('server: ADDED'); }); } resolve(result); }); } const addnode = function() { return new Promise(function(resolve, reject) { const result = 'checking addnode...'; if (status[0].hasOwnProperty('addnode')) { console.log('addnode: OK'); shepherd.writeLog('addnode: OK'); } else { console.log('addnode: NOT FOUND') fs.appendFile(DaemonConfPath, '\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', (err) => { if (err) throw err; console.log('addnode: ADDED'); shepherd.writeLog('addnode: ADDED'); }); } resolve(result); }); } rpcuser() .then(function(result) { return rpcpass(); }) .then(server) .then(addnode); }); setTimeout(function() { console.log(result); shepherd.writeLog(`checkconf addnode ${result}`); resolve(result); }, 2000); }); } const MakeConfReadOnly = function() { return new Promise(function(resolve, reject) { const result = 'Conf file permissions updated to Read Only'; fsnode.chmodSync(DaemonConfPath, '0400'); setTimeout(function() { console.log(result); shepherd.writeLog(`MakeConfReadOnly ${result}`); resolve(result); }, 1000); }); } CheckFileExists() .then(function(result) { return FixFilePermissions(); }) .then(RemoveLines) .then(CheckConf) .then(MakeConfReadOnly); } function getConf(flock) { let komodoDir = '', ZcashDir = '', DaemonConfPath = ''; console.log(flock); shepherd.writeLog(`getconf flock: ${flock}`); if (os.platform() === 'darwin') { komodoDir = `${process.env.HOME}/Library/Application Support/Komodo`; ZcashDir = `${process.env.HOME}/Library/Application Support/Zcash`; } if (os.platform() === 'linux') { komodoDir = `${process.env.HOME}/.komodo`; ZcashDir = `${process.env.HOME}/.zcash`; } if (os.platform() === 'win32') { komodoDir = `${process.env.APPDATA}/Komodo`; ZcashDir = `${process.env.APPDATA}/Zcash`; } switch (flock) { case 'komodod': DaemonConfPath = komodoDir; if (os.platform() === 'win32') { DaemonConfPath = path.normalize(DaemonConfPath); console.log('===>>> SHEPHERD API OUTPUT ===>>>'); } break; case 'zcashd': DaemonConfPath = ZcashDir; if (os.platform() === 'win32') { DaemonConfPath = path.normalize(DaemonConfPath); } break; default: DaemonConfPath = `${komodoDir}/${flock}`; if (os.platform() === 'win32') { DaemonConfPath = path.normalize(DaemonConfPath); } } shepherd.writeLog(`getconf path: ${DaemonConfPath}`); console.log(DaemonConfPath); return DaemonConfPath; } function formatBytes(bytes, decimals) { if (bytes === 0) return '0 Bytes'; const k = 1000, dm = decimals + 1 || 3, sizes = [ 'Bytes', 'KB', 'MB', 'GB', 'TB', 'PB', 'EB', 'ZB', 'YB' ], i = Math.floor(Math.log(bytes) / Math.log(k)); return `${parseFloat((bytes / Math.pow(k, i)).toFixed(dm))} ${sizes[i]}`; } shepherd.SystemInfo = function() { const os_data = { 'totalmem_bytes': os.totalmem(), 'totalmem_readable': formatBytes(os.totalmem()), 'arch': os.arch(), 'cpu': os.cpus()[0].model, 'cpu_cores': os.cpus().length, 'platform': os.platform(), 'os_release': os.release(), 'os_type': os.type() }; return os_data; } shepherd.appInfo = function() { const sysInfo = shepherd.SystemInfo(); const releaseInfo = shepherd.appBasicInfo; const dirs = { iguanaDir, iguanaBin, komodoDir, komododBin, configLocation: `${iguanaDir}/config.json`, cacheLocation: `${iguanaDir}/shepherd`, }; return { sysInfo, releaseInfo, dirs, appSession: shepherd.appSessionHash }; } module.exports = shepherd;