mirror of https://github.com/lukechilds/Agama.git
pbca26
7 years ago
committed by
GitHub
49 changed files with 5374 additions and 2998 deletions
@ -1 +1 @@ |
|||||
Subproject commit 0092ac50871b6964e6462f3ba68ac7d1cd8c1421 |
Subproject commit 54f9dfc6e7b519a5a2295b49813cfd99be131cae |
@ -1,126 +0,0 @@ |
|||||
/* |
|
||||
* Copyright (c) 2015 Satinderjit Singh |
|
||||
* |
|
||||
* Permission is hereby granted, free of charge, to any person obtaining a copy |
|
||||
* of this software and associated documentation files (the "Software"), to deal |
|
||||
* in the Software without restriction, including without limitation the rights |
|
||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell |
|
||||
* copies of the Software, and to permit persons to whom the Software is |
|
||||
* furnished to do so, subject to the following conditions: |
|
||||
* |
|
||||
* The above copyright notice and this permission notice shall be included in all |
|
||||
* copies or substantial portions of the Software. |
|
||||
* |
|
||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR |
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |
|
||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE |
|
||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER |
|
||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, |
|
||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE |
|
||||
* SOFTWARE. |
|
||||
* |
|
||||
*/ |
|
||||
|
|
||||
/* |
|
||||
* Agama komodo-cli paths |
|
||||
* |
|
||||
*/ |
|
||||
|
|
||||
var child_process = require('child_process'), |
|
||||
path = require('path'), |
|
||||
os = require('os'); |
|
||||
|
|
||||
if (os.platform() === 'darwin') { |
|
||||
var komodocliBin = path.join(__dirname, '../assets/bin/osx/komodo-cli'), |
|
||||
zcashcliBin = '/Applications/ZCashSwingWalletUI.app/Contents/MacOS/zcash-cli'; |
|
||||
} |
|
||||
|
|
||||
if (os.platform() === 'linux') { |
|
||||
var komodocliBin = path.join(__dirname, '../assets/bin/linux64/komodo-cli'); |
|
||||
} |
|
||||
|
|
||||
if (os.platform() === 'win32') { |
|
||||
var komodocliBin = path.join(__dirname, '../assets/bin/win64/komodo-cli.exe'), |
|
||||
komodocliBin = path.normalize(komodocliBin); |
|
||||
} |
|
||||
|
|
||||
console.log(komodocliBin) |
|
||||
|
|
||||
/** |
|
||||
* The **komodo-cli** command is used to get komodo api calls answer. |
|
||||
* |
|
||||
* @private |
|
||||
* @category kmdcli |
|
||||
* |
|
||||
*/ |
|
||||
var kmdcli = module.exports = { |
|
||||
exec: child_process.exec, |
|
||||
command: command |
|
||||
}; |
|
||||
|
|
||||
/** |
|
||||
* Parses komodo-cli commands. |
|
||||
* |
|
||||
* @private |
|
||||
* @static |
|
||||
* @category kmdcli |
|
||||
* @param {function} callback The callback function. |
|
||||
* |
|
||||
*/ |
|
||||
function parse_kmdcli_commands(callback) { |
|
||||
return function(error, stdout, stderr) { |
|
||||
if (error) callback(error, stderr); |
|
||||
else callback(error, stdout); |
|
||||
//console.log(stdout)
|
|
||||
}; |
|
||||
} |
|
||||
|
|
||||
/** |
|
||||
* Parses komodo-cli commands. |
|
||||
* |
|
||||
* @private |
|
||||
* @static |
|
||||
* @category kmdcli |
|
||||
* @param {function} callback The callback function. |
|
||||
* @example |
|
||||
* |
|
||||
* var kmdcli = require('./kmdcli'); |
|
||||
* |
|
||||
* kmdcli.command('getinfo', function(err, command) { |
|
||||
* console.log(command); |
|
||||
* }); |
|
||||
* |
|
||||
* // =>
|
|
||||
* { |
|
||||
* "version" : 1000550, |
|
||||
* "protocolversion" : 170002, |
|
||||
* "notarized" : 254740, |
|
||||
* "notarizedhash" : "01f4f1c46662ccca2e7fa9e7e38d4d2e4ced4402fa0f4fc116b8f004bb8cf272", |
|
||||
* "notarizedtxid" : "2b16e47a176f8c1886ca0268243f9b96f8b2db466ea26ae99873d5224bbf80b6", |
|
||||
* "walletversion" : 60000, |
|
||||
* "balance" : 32632.46167742, |
|
||||
* "interest" : 0.00478671, |
|
||||
* "blocks" : 254791, |
|
||||
* "longestchain" : 254791, |
|
||||
* "timeoffset" : 0, |
|
||||
* "tiptime" : 1490815616, |
|
||||
* "connections" : 8, |
|
||||
* "proxy" : "", |
|
||||
* "difficulty" : 707836.56791394, |
|
||||
* "testnet" : false, |
|
||||
* "keypoololdest" : 1482746526, |
|
||||
* "keypoolsize" : 101, |
|
||||
* "paytxfee" : 0.00000000, |
|
||||
* "relayfee" : 0.00001000, |
|
||||
* "errors" : "WARNING: check your network connection, 157 blocks received in the last 4 hours (240 expected)", |
|
||||
* "notaryid" : -1, |
|
||||
* "pubkey" : "000000000000000000000000000000000000000000000000000000000000000000" |
|
||||
* } |
|
||||
* |
|
||||
*/ |
|
||||
function command(kmd_command, callback) { |
|
||||
if (callback) { |
|
||||
return this.exec(komodocliBin + " " + kmd_command, |
|
||||
parse_kmdcli_commands(callback)); |
|
||||
} |
|
||||
} |
|
@ -0,0 +1,120 @@ |
|||||
|
/* |
||||
|
MIT License |
||||
|
|
||||
|
Copyright (c) 2017 Yuki Akiyama, SuperNET |
||||
|
|
||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy |
||||
|
of this software and associated documentation files (the "Software"), to deal |
||||
|
in the Software without restriction, including without limitation the rights |
||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell |
||||
|
copies of the Software, and to permit persons to whom the Software is |
||||
|
furnished to do so, subject to the following conditions: |
||||
|
|
||||
|
The above copyright notice and this permission notice shall be included in all |
||||
|
copies or substantial portions of the Software. |
||||
|
|
||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR |
||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |
||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE |
||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER |
||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, |
||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE |
||||
|
SOFTWARE. |
||||
|
*/ |
||||
|
|
||||
|
var bitcoin = require('bitcoinjs-lib-zcash'); |
||||
|
|
||||
|
var decodeFormat = function(tx) { |
||||
|
var result = { |
||||
|
txid: tx.getId(), |
||||
|
version: tx.version, |
||||
|
locktime: tx.locktime, |
||||
|
}; |
||||
|
|
||||
|
return result; |
||||
|
} |
||||
|
|
||||
|
var decodeInput = function(tx) { |
||||
|
var result = []; |
||||
|
|
||||
|
tx.ins.forEach(function(input, n) { |
||||
|
var vin = { |
||||
|
txid: input.hash.reverse().toString('hex'), |
||||
|
n: input.index, |
||||
|
script: bitcoin.script.toASM(input.script), |
||||
|
sequence: input.sequence, |
||||
|
}; |
||||
|
|
||||
|
result.push(vin); |
||||
|
}); |
||||
|
|
||||
|
return result; |
||||
|
} |
||||
|
|
||||
|
var decodeOutput = function(tx, network) { |
||||
|
var format = function(out, n, network) { |
||||
|
var vout = { |
||||
|
satoshi: out.value, |
||||
|
value: (1e-8 * out.value).toFixed(8), |
||||
|
n: n, |
||||
|
scriptPubKey: { |
||||
|
asm: bitcoin.script.toASM(out.script), |
||||
|
hex: out.script.toString('hex'), |
||||
|
type: bitcoin.script.classifyOutput(out.script), |
||||
|
addresses: [], |
||||
|
}, |
||||
|
}; |
||||
|
|
||||
|
switch(vout.scriptPubKey.type) { |
||||
|
case 'pubkeyhash': |
||||
|
vout.scriptPubKey.addresses.push(bitcoin.address.fromOutputScript(out.script, network)); |
||||
|
break; |
||||
|
case 'pubkey': |
||||
|
const pubKeyBuffer = new Buffer(vout.scriptPubKey.asm.split(' ')[0], 'hex'); |
||||
|
vout.scriptPubKey.addresses.push(bitcoin.ECPair.fromPublicKeyBuffer(pubKeyBuffer, network).getAddress()); |
||||
|
break; |
||||
|
case 'scripthash': |
||||
|
vout.scriptPubKey.addresses.push(bitcoin.address.fromOutputScript(out.script, network)); |
||||
|
break; |
||||
|
} |
||||
|
|
||||
|
return vout; |
||||
|
} |
||||
|
|
||||
|
var result = []; |
||||
|
|
||||
|
tx.outs.forEach(function(out, n) { |
||||
|
result.push(format(out, n, network)); |
||||
|
}); |
||||
|
|
||||
|
return result; |
||||
|
} |
||||
|
|
||||
|
var TxDecoder = module.exports = function(rawtx, network) { |
||||
|
try { |
||||
|
const _tx = bitcoin.Transaction.fromHex(rawtx); |
||||
|
|
||||
|
return { |
||||
|
tx: _tx, |
||||
|
network: network, |
||||
|
format: decodeFormat(_tx), |
||||
|
inputs: decodeInput(_tx), |
||||
|
outputs: decodeOutput(_tx, network), |
||||
|
}; |
||||
|
} catch (e) { |
||||
|
return false; |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
TxDecoder.prototype.decode = function() { |
||||
|
var result = {}; |
||||
|
var self = this; |
||||
|
|
||||
|
Object.keys(self.format).forEach(function(key) { |
||||
|
result[key] = self.format[key]; |
||||
|
}); |
||||
|
|
||||
|
result.outputs = self.outputs; |
||||
|
|
||||
|
return result; |
||||
|
} |
@ -0,0 +1,123 @@ |
|||||
|
/* |
||||
|
MIT License |
||||
|
|
||||
|
Copyright (c) 2017 Yuki Akiyama, SuperNET |
||||
|
|
||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy |
||||
|
of this software and associated documentation files (the "Software"), to deal |
||||
|
in the Software without restriction, including without limitation the rights |
||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell |
||||
|
copies of the Software, and to permit persons to whom the Software is |
||||
|
furnished to do so, subject to the following conditions: |
||||
|
|
||||
|
The above copyright notice and this permission notice shall be included in all |
||||
|
copies or substantial portions of the Software. |
||||
|
|
||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR |
||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |
||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE |
||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER |
||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, |
||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE |
||||
|
SOFTWARE. |
||||
|
*/ |
||||
|
|
||||
|
var bitcoin = require('bitcoinjs-lib-pos'); |
||||
|
var script = require('bitcoinjs-lib-pos/src/script'); |
||||
|
var address = require('bitcoinjs-lib-pos/src/address'); |
||||
|
var bitcoinJS = require('bitcoinjs-lib'); |
||||
|
|
||||
|
var decodeFormat = function(tx) { |
||||
|
var result = { |
||||
|
txid: tx.getId(), |
||||
|
version: tx.version, |
||||
|
locktime: tx.locktime, |
||||
|
}; |
||||
|
|
||||
|
return result; |
||||
|
} |
||||
|
|
||||
|
var decodeInput = function(tx) { |
||||
|
var result = []; |
||||
|
|
||||
|
tx.ins.forEach(function(input, n) { |
||||
|
var vin = { |
||||
|
txid: input.hash.reverse().toString('hex'), |
||||
|
n: input.index, |
||||
|
script: script.fromHex(input.hash), |
||||
|
sequence: input.sequence, |
||||
|
}; |
||||
|
|
||||
|
result.push(vin); |
||||
|
}); |
||||
|
|
||||
|
return result; |
||||
|
} |
||||
|
|
||||
|
var decodeOutput = function(tx, network) { |
||||
|
var format = function(out, n, network) { |
||||
|
var vout = { |
||||
|
satoshi: out.value, |
||||
|
value: (1e-8 * out.value).toFixed(8), |
||||
|
n: n, |
||||
|
scriptPubKey: { |
||||
|
asm: bitcoinJS.script.toASM(out.script.chunks), |
||||
|
hex: out.script.toHex(), |
||||
|
type: bitcoin.scripts.classifyOutput(out.script), |
||||
|
addresses: [], |
||||
|
}, |
||||
|
}; |
||||
|
|
||||
|
switch(vout.scriptPubKey.type) { |
||||
|
case 'pubkeyhash': |
||||
|
vout.scriptPubKey.addresses.push(address.fromOutputScript(out.script, network)); |
||||
|
break; |
||||
|
case 'pubkey': |
||||
|
const pubKeyBuffer = new Buffer(vout.scriptPubKey.asm.split(' ')[0], 'hex'); |
||||
|
vout.scriptPubKey.addresses.push(bitcoin.ECPair.fromPublicKeyBuffer(pubKeyBuffer, network).getAddress()); |
||||
|
break; |
||||
|
case 'scripthash': |
||||
|
vout.scriptPubKey.addresses.push(address.fromOutputScript(out.script, network)); |
||||
|
break; |
||||
|
} |
||||
|
|
||||
|
return vout; |
||||
|
} |
||||
|
|
||||
|
var result = []; |
||||
|
|
||||
|
tx.outs.forEach(function(out, n) { |
||||
|
result.push(format(out, n, network)); |
||||
|
}); |
||||
|
|
||||
|
return result; |
||||
|
} |
||||
|
|
||||
|
var TxDecoder = module.exports = function(rawtx, network) { |
||||
|
try { |
||||
|
const _tx = bitcoin.Transaction.fromHex(rawtx, network); |
||||
|
|
||||
|
return { |
||||
|
tx: _tx, |
||||
|
network: network, |
||||
|
format: decodeFormat(_tx), |
||||
|
inputs: decodeInput(_tx), |
||||
|
outputs: decodeOutput(_tx, network), |
||||
|
}; |
||||
|
} catch (e) { |
||||
|
return false; |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
TxDecoder.prototype.decode = function() { |
||||
|
var result = {}; |
||||
|
var self = this; |
||||
|
|
||||
|
Object.keys(self.format).forEach(function(key) { |
||||
|
result[key] = self.format[key]; |
||||
|
}); |
||||
|
|
||||
|
result.outputs = self.outputs; |
||||
|
|
||||
|
return result; |
||||
|
} |
@ -1,239 +0,0 @@ |
|||||
module.exports = (shepherd) => { |
|
||||
shepherd.testClearAll = () => { |
|
||||
return new shepherd.Promise((resolve, reject) => { |
|
||||
shepherd.fs.removeSync(`${iguanaTestDir}`); |
|
||||
resolve('done'); |
|
||||
}); |
|
||||
} |
|
||||
|
|
||||
shepherd.testBins = (daemonName) => { |
|
||||
return new shepherd.Promise((resolve, reject) => { |
|
||||
const _bins = { |
|
||||
komodod: shepherd.komododBin, |
|
||||
komodoCli: shepherd.komodocliBin, |
|
||||
}; |
|
||||
const _arg = null; |
|
||||
let _pid; |
|
||||
|
|
||||
shepherd.log('testBins exec ' + _bins[daemonName]); |
|
||||
|
|
||||
if (!shepherd.fs.existsSync(shepherd.agamaTestDir)) { |
|
||||
shepherd.fs.mkdirSync(shepherd.agamaTestDir); |
|
||||
} |
|
||||
|
|
||||
try { |
|
||||
shepherd._fs.access(`${shepherd.agamaTestDir}/${daemonName}Test.log`, shepherd.fs.constants.R_OK, (err) => { |
|
||||
if (!err) { |
|
||||
try { |
|
||||
shepherd._fs.unlinkSync(`${shepherd.agamaTestDir}/${daemonName}Test.log`); |
|
||||
} catch (e) {} |
|
||||
} else { |
|
||||
shepherd.log(`path ${shepherd.agamaTestDir}/${daemonName}Test.log doesnt exist`); |
|
||||
} |
|
||||
}); |
|
||||
} catch (e) {} |
|
||||
|
|
||||
if (daemonName === 'komodod') { |
|
||||
try { |
|
||||
shepherd._fs.access(`${iguanaTestDir}/debug.log`, shepherd.fs.constants.R_OK, (err) => { |
|
||||
if (!err) { |
|
||||
shepherd._fs.unlinkSync(`${iguanaTestDir}/db.log`); |
|
||||
shepherd._fs.unlinkSync(`${iguanaTestDir}/debug.log`); |
|
||||
shepherd._fs.unlinkSync(`${iguanaTestDir}/komodo.conf`); |
|
||||
shepherd._fs.unlinkSync(`${iguanaTestDir}/komodod.pid`); |
|
||||
shepherd._fs.unlinkSync(`${iguanaTestDir}/komodostate`); |
|
||||
shepherd._fs.unlinkSync(`${iguanaTestDir}/realtime`); |
|
||||
shepherd._fs.unlinkSync(`${iguanaTestDir}/wallet.dat`); |
|
||||
shepherd._fs.unlinkSync(`${iguanaTestDir}/.lock`); |
|
||||
shepherd.fs.removeSync(`${iguanaTestDir}/blocks`); |
|
||||
shepherd.fs.removeSync(`${iguanaTestDir}/chainstate`); |
|
||||
shepherd.fs.removeSync(`${iguanaTestDir}/database`); |
|
||||
execKomodod(); |
|
||||
} else { |
|
||||
shepherd.log(`test: nothing to remove in ${iguanaTestDir}`); |
|
||||
execKomodod(); |
|
||||
} |
|
||||
}); |
|
||||
} catch (e) {} |
|
||||
|
|
||||
const execKomodod = () => { |
|
||||
let _komododTest = { |
|
||||
port: 'unknown', |
|
||||
start: 'unknown', |
|
||||
getinfo: 'unknown', |
|
||||
errors: { |
|
||||
assertFailed: false, |
|
||||
zcashParams: false, |
|
||||
}, |
|
||||
}; |
|
||||
const _komodoConf = 'rpcuser=user83f3afba8d714993\n' + |
|
||||
'rpcpassword=0d4430ca1543833e35bce5a0cc9e16b3\n' + |
|
||||
'server=1\n' + |
|
||||
'addnode=78.47.196.146\n' + |
|
||||
'addnode=5.9.102.210\n' + |
|
||||
'addnode=178.63.69.164\n' + |
|
||||
'addnode=88.198.65.74\n' + |
|
||||
'addnode=5.9.122.241\n' + |
|
||||
'addnode=144.76.94.3\n' + |
|
||||
'addnode=144.76.94.38\n' + |
|
||||
'addnode=89.248.166.91\n' + |
|
||||
'addnode=148.251.57.148\n' + |
|
||||
'addnode=149.56.28.84\n' + |
|
||||
'addnode=176.9.26.39\n' + |
|
||||
'addnode=94.102.63.199\n' + |
|
||||
'addnode=94.102.63.200\n' + |
|
||||
'addnode=104.255.64.3\n' + |
|
||||
'addnode=221.121.144.140\n' + |
|
||||
'addnode=103.18.58.150\n' + |
|
||||
'addnode=103.18.58.146\n' + |
|
||||
'addnode=213.202.253.10\n' + |
|
||||
'addnode=185.106.121.32\n' + |
|
||||
'addnode=27.100.36.201\n'; |
|
||||
|
|
||||
shepherd.fs.writeFile(`${iguanaTestDir}/komodo.conf`, _komodoConf, (err) => { |
|
||||
if (err) { |
|
||||
shepherd.log(`test: error writing komodo conf in ${iguanaTestDir}`); |
|
||||
} |
|
||||
}); |
|
||||
|
|
||||
shepherd.portscanner.checkPortStatus('7771', '127.0.0.1', (error, status) => { |
|
||||
// Status is 'open' if currently in use or 'closed' if available
|
|
||||
if (status === 'closed') { |
|
||||
_komododTest.port = 'passed'; |
|
||||
} else { |
|
||||
_komododTest.port = 'failed'; |
|
||||
} |
|
||||
}); |
|
||||
|
|
||||
/*pm2.connect(true,function(err) { //start up pm2 god |
|
||||
if (err) { |
|
||||
shepherd.error(err); |
|
||||
process.exit(2); |
|
||||
} |
|
||||
|
|
||||
pm2.start({ |
|
||||
script: shepherd.komododBin, // path to binary
|
|
||||
name: 'komodod', |
|
||||
exec_mode : 'fork', |
|
||||
args: [ |
|
||||
'-daemon=0', |
|
||||
'-addnode=78.47.196.146', |
|
||||
`-datadir=${iguanaTestDir}/` |
|
||||
], |
|
||||
output: `${iguanaTestDir}/komododTest.log`, |
|
||||
mergeLogs: true, |
|
||||
}, function(err, apps) { |
|
||||
if (apps[0] && |
|
||||
apps[0].process && |
|
||||
apps[0].process.pid) { |
|
||||
_komododTest.start = 'success'; |
|
||||
shepherd.log(`test: got komodod instance pid = ${apps[0].process.pid}`); |
|
||||
shepherd.writeLog(`test: komodod started with pid ${apps[0].process.pid}`); |
|
||||
} else { |
|
||||
_komododTest.start = 'failed'; |
|
||||
shepherd.log(`unable to start komodod`); |
|
||||
} |
|
||||
|
|
||||
pm2.disconnect(); // Disconnect from PM2
|
|
||||
if (err) { |
|
||||
shepherd.writeLog(`test: error starting komodod`); |
|
||||
shepherd.log(`komodod fork err: ${err}`); |
|
||||
// throw err;
|
|
||||
} |
|
||||
}); |
|
||||
});*/ |
|
||||
|
|
||||
setTimeout(() => { |
|
||||
const options = { |
|
||||
url: `http://localhost:7771`, |
|
||||
method: 'POST', |
|
||||
auth: { |
|
||||
user: 'user83f3afba8d714993', |
|
||||
pass: '0d4430ca1543833e35bce5a0cc9e16b3', |
|
||||
}, |
|
||||
body: JSON.stringify({ |
|
||||
agent: 'bitcoinrpc', |
|
||||
method: 'getinfo', |
|
||||
}), |
|
||||
}; |
|
||||
|
|
||||
shepherd.request(options, (error, response, body) => { |
|
||||
if (response && |
|
||||
response.statusCode && |
|
||||
response.statusCode === 200) { |
|
||||
// res.end(body);
|
|
||||
shepherd.log(JSON.stringify(body, null, '\t')); |
|
||||
} else { |
|
||||
// res.end(body);
|
|
||||
shepherd.log(JSON.stringify(body, null, '\t')); |
|
||||
} |
|
||||
}); |
|
||||
}, 10000); |
|
||||
|
|
||||
setTimeout(() => { |
|
||||
pm2.delete('komodod'); |
|
||||
resolve(_komododTest); |
|
||||
}, 20000); |
|
||||
} |
|
||||
// komodod debug.log hooks
|
|
||||
|
|
||||
//"{\"result\":{\"version\":1000850,\"protocolversion\":170002,\"KMDversion\":\"0.1.1\",\"notarized\":0,\"notarizedhash\":\"0000000000000000000000000000000000000000000000000000000000000000\",\"notarizedtxid\":\"0000000000000000000000000000000000000000000000000000000000000000\",\"notarizedtxid_height\":\"mempool\",\"notarized_confirms\":0,\"walletversion\":60000,\"balance\":0.00000000,\"interest\":0.00000000,\"blocks\":128,\"longestchain\":472331,\"timeoffset\":0,\"tiptime\":1473827710,\"connections\":1,\"proxy\":\"\",\"difficulty\":1,\"testnet\":false,\"keypoololdest\":1504118047,\"keypoolsize\":101,\"paytxfee\":0.00000000,\"relayfee\":0.00000100,\"errors\":\"\"},\"error\":null,\"id\":null}\n"
|
|
||||
|
|
||||
//2017-08-30 17:51:33 Error: Cannot find the Zcash network parameters in the following directory:
|
|
||||
//"/home/pbca/.zcash-params"
|
|
||||
//Please run 'zcash-fetch-params' or './zcutil/fetch-params.sh' and then restart.
|
|
||||
//EXCEPTION: St13runtime_error
|
|
||||
//Assertion failed.
|
|
||||
//2017-08-30 17:51:14 Using config file /home/pbca/.iguana/test/komodo.conf
|
|
||||
//2017-08-30 18:23:43 UpdateTip: new best=0a47c1323f393650f7221c217d19d149d002d35444f47fde61be2dd90fbde8e6 height=1 log2_work=5.0874628 tx=2 date=2016-09-13 19:04:01 progress=0.000001 cache=0.0MiB(1tx)
|
|
||||
//2017-08-30 18:23:43 UpdateTip: new best=05076a4e1fc9af0f5fda690257b17ae20c12d4796dfba1624804d012c9ec00be height=2 log2_work=5.6724253 tx=3 date=2016-09-13 19:05:28 progress=0.000001 cache=0.0MiB(2tx)
|
|
||||
|
|
||||
/*shepherd.execFile(`${shepherd.komododBin}`, _arg, { |
|
||||
maxBuffer: 1024 * 10000 // 10 mb
|
|
||||
}, function(error, stdout, stderr) { |
|
||||
shepherd.writeLog(`stdout: ${stdout}`); |
|
||||
shepherd.writeLog(`stderr: ${stderr}`); |
|
||||
|
|
||||
if (error !== null) { |
|
||||
console.log(`exec error: ${error}`); |
|
||||
shepherd.writeLog(`exec error: ${error}`); |
|
||||
|
|
||||
if (error.toString().indexOf('using -reindex') > -1) { |
|
||||
shepherd.io.emit('service', { |
|
||||
komodod: { |
|
||||
error: 'run -reindex', |
|
||||
} |
|
||||
}); |
|
||||
} |
|
||||
} |
|
||||
});*/ |
|
||||
} |
|
||||
}); |
|
||||
} |
|
||||
|
|
||||
// komodod datadir location test
|
|
||||
shepherd.testLocation = (path) => { |
|
||||
return new shepherd.Promise((resolve, reject) => { |
|
||||
if (path.indexOf(' ') > -1) { |
|
||||
shepherd.log(`error testing path ${path}`); |
|
||||
resolve(-1); |
|
||||
} else { |
|
||||
shepherd.fs.lstat(path, (err, stats) => { |
|
||||
if (err) { |
|
||||
shepherd.log(`error testing path ${path}`); |
|
||||
resolve(-1); |
|
||||
} else { |
|
||||
if (stats.isDirectory()) { |
|
||||
resolve(true); |
|
||||
} else { |
|
||||
shepherd.log(`error testing path ${path} not a folder`); |
|
||||
resolve(false); |
|
||||
} |
|
||||
} |
|
||||
}); |
|
||||
} |
|
||||
}); |
|
||||
} |
|
||||
|
|
||||
return shepherd; |
|
||||
}; |
|
@ -0,0 +1,372 @@ |
|||||
|
const bs58check = require('bs58check'); |
||||
|
const bitcoin = require('bitcoinjs-lib'); |
||||
|
|
||||
|
module.exports = (shepherd) => { |
||||
|
shepherd.elections = {}; |
||||
|
|
||||
|
shepherd.hex2str = (hexx) => { |
||||
|
const hex = hexx.toString(); // force conversion
|
||||
|
let str = ''; |
||||
|
|
||||
|
for (let i = 0; i < hex.length; i += 2) { |
||||
|
str += String.fromCharCode(parseInt(hex.substr(i, 2), 16)); |
||||
|
} |
||||
|
|
||||
|
return str; |
||||
|
}; |
||||
|
|
||||
|
shepherd.post('/elections/status', (req, res, next) => { |
||||
|
if (shepherd.checkToken(req.body.token)) { |
||||
|
const successObj = { |
||||
|
msg: 'success', |
||||
|
result: shepherd.elections.pub ? shepherd.elections.pub : 'unauth', |
||||
|
}; |
||||
|
|
||||
|
res.end(JSON.stringify(successObj)); |
||||
|
} else { |
||||
|
const errorObj = { |
||||
|
msg: 'error', |
||||
|
result: 'unauthorized access', |
||||
|
}; |
||||
|
|
||||
|
res.end(JSON.stringify(errorObj)); |
||||
|
} |
||||
|
}); |
||||
|
|
||||
|
shepherd.post('/elections/login', (req, res, next) => { |
||||
|
if (shepherd.checkToken(req.body.token)) { |
||||
|
const _seed = req.body.seed; |
||||
|
const _network = req.body.network; |
||||
|
let keys; |
||||
|
let isWif = false; |
||||
|
|
||||
|
if (_seed.match('^[a-zA-Z0-9]{34}$')) { |
||||
|
shepherd.log('watchonly elections pub addr'); |
||||
|
shepherd.elections = { |
||||
|
priv: _seed, |
||||
|
pub: _seed, |
||||
|
}; |
||||
|
} else { |
||||
|
try { |
||||
|
bs58check.decode(_seed); |
||||
|
isWif = true; |
||||
|
} catch (e) {} |
||||
|
|
||||
|
if (isWif) { |
||||
|
try { |
||||
|
let key = bitcoin.ECPair.fromWIF(_seed, shepherd.getNetworkData(_network.toLowerCase()), true); |
||||
|
keys = { |
||||
|
priv: key.toWIF(), |
||||
|
pub: key.getAddress(), |
||||
|
}; |
||||
|
} catch (e) { |
||||
|
_wifError = true; |
||||
|
} |
||||
|
} else { |
||||
|
keys = shepherd.seedToWif(_seed, _network, req.body.iguana); |
||||
|
} |
||||
|
|
||||
|
shepherd.elections = { |
||||
|
priv: keys.priv, |
||||
|
pub: keys.pub, |
||||
|
}; |
||||
|
} |
||||
|
|
||||
|
const successObj = { |
||||
|
msg: 'success', |
||||
|
result: shepherd.elections.pub, |
||||
|
}; |
||||
|
|
||||
|
res.end(JSON.stringify(successObj)); |
||||
|
} else { |
||||
|
const errorObj = { |
||||
|
msg: 'error', |
||||
|
result: 'unauthorized access', |
||||
|
}; |
||||
|
|
||||
|
res.end(JSON.stringify(errorObj)); |
||||
|
} |
||||
|
}); |
||||
|
|
||||
|
shepherd.post('/elections/logout', (req, res, next) => { |
||||
|
if (shepherd.checkToken(req.body.token)) { |
||||
|
shepherd.elections = {}; |
||||
|
|
||||
|
const successObj = { |
||||
|
msg: 'success', |
||||
|
result: true, |
||||
|
}; |
||||
|
|
||||
|
res.end(JSON.stringify(successObj)); |
||||
|
} else { |
||||
|
const errorObj = { |
||||
|
msg: 'error', |
||||
|
result: 'unauthorized access', |
||||
|
}; |
||||
|
|
||||
|
res.end(JSON.stringify(errorObj)); |
||||
|
} |
||||
|
}); |
||||
|
|
||||
|
shepherd.electionsDecodeTx = (decodedTx, ecl, network, _network, transaction, blockInfo, address) => { |
||||
|
let txInputs = []; |
||||
|
|
||||
|
return new shepherd.Promise((resolve, reject) => { |
||||
|
if (decodedTx && |
||||
|
decodedTx.inputs) { |
||||
|
shepherd.Promise.all(decodedTx.inputs.map((_decodedInput, index) => { |
||||
|
return new shepherd.Promise((_resolve, _reject) => { |
||||
|
if (_decodedInput.txid !== '0000000000000000000000000000000000000000000000000000000000000000') { |
||||
|
ecl.blockchainTransactionGet(_decodedInput.txid) |
||||
|
.then((rawInput) => { |
||||
|
const decodedVinVout = shepherd.electrumJSTxDecoder(rawInput, network, _network); |
||||
|
|
||||
|
shepherd.log('electrum raw input tx ==>', true); |
||||
|
|
||||
|
if (decodedVinVout) { |
||||
|
shepherd.log(decodedVinVout.outputs[_decodedInput.n], true); |
||||
|
txInputs.push(decodedVinVout.outputs[_decodedInput.n]); |
||||
|
_resolve(true); |
||||
|
} else { |
||||
|
_resolve(true); |
||||
|
} |
||||
|
}); |
||||
|
} else { |
||||
|
_resolve(true); |
||||
|
} |
||||
|
}); |
||||
|
})) |
||||
|
.then(promiseResult => { |
||||
|
const _parsedTx = { |
||||
|
network: decodedTx.network, |
||||
|
format: decodedTx.format, |
||||
|
inputs: txInputs, |
||||
|
outputs: decodedTx.outputs, |
||||
|
height: transaction.height, |
||||
|
timestamp: Number(transaction.height) === 0 ? Math.floor(Date.now() / 1000) : blockInfo.timestamp, |
||||
|
}; |
||||
|
|
||||
|
resolve(shepherd.parseTransactionAddresses(_parsedTx, address, network, true)); |
||||
|
}); |
||||
|
} else { |
||||
|
const _parsedTx = { |
||||
|
network: decodedTx.network, |
||||
|
format: 'cant parse', |
||||
|
inputs: 'cant parse', |
||||
|
outputs: 'cant parse', |
||||
|
height: transaction.height, |
||||
|
timestamp: Number(transaction.height) === 0 ? Math.floor(Date.now() / 1000) : blockInfo.timestamp, |
||||
|
}; |
||||
|
|
||||
|
resolve(shepherd.parseTransactionAddresses(_parsedTx, address, network)); |
||||
|
} |
||||
|
}); |
||||
|
}; |
||||
|
|
||||
|
shepherd.get('/elections/listtransactions', (req, res, next) => { |
||||
|
if (shepherd.checkToken(req.query.token)) { |
||||
|
const network = req.query.network || shepherd.findNetworkObj(req.query.coin); |
||||
|
const ecl = new shepherd.electrumJSCore(shepherd.electrumServers[network].port, shepherd.electrumServers[network].address, shepherd.electrumServers[network].proto); // tcp or tls
|
||||
|
const type = req.query.type; |
||||
|
const _address = req.query.address; |
||||
|
|
||||
|
shepherd.log('electrum elections listtransactions ==>', true); |
||||
|
|
||||
|
const MAX_TX = req.query.maxlength || 10; |
||||
|
ecl.connect(); |
||||
|
|
||||
|
ecl.blockchainAddressGetHistory(_address) |
||||
|
.then((json) => { |
||||
|
if (json && |
||||
|
json.length) { |
||||
|
let _rawtx = []; |
||||
|
|
||||
|
json = shepherd.sortTransactions(json); |
||||
|
// json = json.length > MAX_TX ? json.slice(0, MAX_TX) : json;
|
||||
|
|
||||
|
shepherd.log(json.length, true); |
||||
|
|
||||
|
shepherd.Promise.all(json.map((transaction, index) => { |
||||
|
return new shepherd.Promise((resolve, reject) => { |
||||
|
ecl.blockchainBlockGetHeader(transaction.height) |
||||
|
.then((blockInfo) => { |
||||
|
if (blockInfo && |
||||
|
blockInfo.timestamp) { |
||||
|
ecl.blockchainTransactionGet(transaction['tx_hash']) |
||||
|
.then((_rawtxJSON) => { |
||||
|
//shepherd.log('electrum gettransaction ==>', true);
|
||||
|
//shepherd.log((index + ' | ' + (_rawtxJSON.length - 1)), true);
|
||||
|
//shepherd.log(_rawtxJSON, true);
|
||||
|
|
||||
|
// decode tx
|
||||
|
const _network = shepherd.getNetworkData(network); |
||||
|
const decodedTx = shepherd.electrumJSTxDecoder(_rawtxJSON, network, _network); |
||||
|
let _res = {}; |
||||
|
let _opreturnFound = false; |
||||
|
let _region; |
||||
|
|
||||
|
if (decodedTx && |
||||
|
decodedTx.outputs && |
||||
|
decodedTx.outputs.length) { |
||||
|
for (let i = 0; i < decodedTx.outputs.length; i++) { |
||||
|
if (decodedTx.outputs[i].scriptPubKey.asm.indexOf('OP_RETURN') > -1) { |
||||
|
_opreturnFound = true; |
||||
|
_region = shepherd.hex2str(decodedTx.outputs[i].scriptPubKey.hex.substr(4, decodedTx.outputs[i].scriptPubKey.hex.length)); |
||||
|
shepherd.log(`found opreturn tag ${_region}`); |
||||
|
break; |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
if (_opreturnFound) { |
||||
|
let _candidate = {}; |
||||
|
|
||||
|
for (let i = 0; i < decodedTx.outputs.length; i++) { |
||||
|
if (type === 'voter' && |
||||
|
decodedTx.outputs[i].scriptPubKey.addresses && |
||||
|
decodedTx.outputs[i].scriptPubKey.addresses[0] && |
||||
|
decodedTx.outputs[i].scriptPubKey.addresses[0] !== _address) { |
||||
|
if (_region === 'ne2k18-na-1-eu-2-ae-3-sh-4') { |
||||
|
const _regionsLookup = [ |
||||
|
'ne2k18-na', |
||||
|
'ne2k18-eu', |
||||
|
'ne2k18-ae', |
||||
|
'ne2k18-sh' |
||||
|
]; |
||||
|
|
||||
|
shepherd.log(`i voted ${decodedTx.outputs[i].value} for ${decodedTx.outputs[i].scriptPubKey.addresses[0]}`); |
||||
|
_rawtx.push({ |
||||
|
address: decodedTx.outputs[i].scriptPubKey.addresses[0], |
||||
|
amount: decodedTx.outputs[i].value, |
||||
|
region: _regionsLookup[i], |
||||
|
timestamp: blockInfo.timestamp, |
||||
|
}); |
||||
|
resolve(true); |
||||
|
} else { |
||||
|
shepherd.log(`i voted ${decodedTx.outputs[i].value} for ${decodedTx.outputs[i].scriptPubKey.addresses[0]}`); |
||||
|
_rawtx.push({ |
||||
|
address: decodedTx.outputs[i].scriptPubKey.addresses[0], |
||||
|
amount: decodedTx.outputs[i].value, |
||||
|
region: _region, |
||||
|
timestamp: blockInfo.timestamp, |
||||
|
}); |
||||
|
resolve(true); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
if (type === 'candidate') { |
||||
|
if (_region === 'ne2k18-na-1-eu-2-ae-3-sh-4') { |
||||
|
if (decodedTx.outputs[i].scriptPubKey.addresses[0] === _address && decodedTx.outputs[i].scriptPubKey.asm.indexOf('OP_RETURN') === -1) { |
||||
|
const _regionsLookup = [ |
||||
|
'ne2k18-na', |
||||
|
'ne2k18-eu', |
||||
|
'ne2k18-ae', |
||||
|
'ne2k18-sh' |
||||
|
]; |
||||
|
|
||||
|
shepherd.electionsDecodeTx(decodedTx, ecl, network, _network, transaction, blockInfo, _address) |
||||
|
.then((res) => { |
||||
|
shepherd.log(`i received ${decodedTx.outputs[i].value} from ${res.outputAddresses[0]} out ${i} region ${_regionsLookup[i]}`); |
||||
|
_rawtx.push({ |
||||
|
address: res.outputAddresses[0], |
||||
|
timestamp: blockInfo.timestamp, |
||||
|
amount: decodedTx.outputs[i].value, |
||||
|
region: _regionsLookup[i], |
||||
|
}); |
||||
|
resolve(true); |
||||
|
}); |
||||
|
} |
||||
|
} else { |
||||
|
shepherd.electionsDecodeTx(decodedTx, ecl, network, _network, transaction, blockInfo, _address) |
||||
|
.then((res) => { |
||||
|
if (decodedTx.outputs[i].scriptPubKey.addresses[0] === _address) { |
||||
|
_candidate.amount = decodedTx.outputs[i].value; |
||||
|
} else if (decodedTx.outputs[i].scriptPubKey.addresses[0] !== _address && decodedTx.outputs[i].scriptPubKey.asm.indexOf('OP_RETURN') === -1) { |
||||
|
_candidate.address = decodedTx.outputs[i].scriptPubKey.addresses[0]; |
||||
|
_candidate.region = _region; |
||||
|
_candidate.timestamp = blockInfo.timestamp; |
||||
|
} |
||||
|
|
||||
|
if (i === decodedTx.outputs.length - 1) { |
||||
|
shepherd.log(`i received ${_candidate.amount} from ${_candidate.address} region ${_region}`); |
||||
|
_rawtx.push(_candidate); |
||||
|
resolve(true); |
||||
|
} |
||||
|
}); |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
} else { |
||||
|
shepherd.log('elections regular tx', true); |
||||
|
shepherd.electionsDecodeTx(decodedTx, ecl, network, _network, transaction, blockInfo, _address) |
||||
|
.then((_regularTx) => { |
||||
|
if (_regularTx[0] && |
||||
|
_regularTx[1]) { |
||||
|
_rawtx.push({ |
||||
|
address: _regularTx[type === 'voter' ? 0 : 1].address || 'self', |
||||
|
timestamp: _regularTx[type === 'voter' ? 0 : 1].timestamp, |
||||
|
amount: _regularTx[type === 'voter' ? 0 : 1].amount, |
||||
|
region: 'unknown', |
||||
|
regularTx: true, |
||||
|
hash: transaction['tx_hash'], |
||||
|
}); |
||||
|
} else { |
||||
|
if ((type === 'voter' && _regularTx.type !== 'received') && (type === 'candidate' && _regularTx.type !== 'sent')) { |
||||
|
_rawtx.push({ |
||||
|
address: _regularTx.address || 'self', |
||||
|
timestamp: _regularTx.timestamp, |
||||
|
amount: _regularTx.amount, |
||||
|
region: 'unknown', |
||||
|
regularTx: true, |
||||
|
hash: transaction['tx_hash'], |
||||
|
}); |
||||
|
} |
||||
|
} |
||||
|
resolve(true); |
||||
|
}); |
||||
|
} |
||||
|
}); |
||||
|
} else { |
||||
|
_rawtx.push({ |
||||
|
address: 'unknown', |
||||
|
timestamp: 'cant get block info', |
||||
|
amount: 'unknown', |
||||
|
region: 'unknown', |
||||
|
regularTx: true, |
||||
|
}); |
||||
|
resolve(false); |
||||
|
} |
||||
|
}); |
||||
|
}); |
||||
|
})) |
||||
|
.then(promiseResult => { |
||||
|
ecl.close(); |
||||
|
|
||||
|
const successObj = { |
||||
|
msg: 'success', |
||||
|
result: _rawtx, |
||||
|
}; |
||||
|
|
||||
|
res.end(JSON.stringify(successObj)); |
||||
|
}); |
||||
|
} else { |
||||
|
const successObj = { |
||||
|
msg: 'success', |
||||
|
result: [], |
||||
|
}; |
||||
|
|
||||
|
res.end(JSON.stringify(successObj)); |
||||
|
} |
||||
|
}); |
||||
|
} else { |
||||
|
const errorObj = { |
||||
|
msg: 'error', |
||||
|
result: 'unauthorized access', |
||||
|
}; |
||||
|
|
||||
|
res.end(JSON.stringify(errorObj)); |
||||
|
} |
||||
|
}); |
||||
|
|
||||
|
return shepherd; |
||||
|
} |
@ -0,0 +1,500 @@ |
|||||
|
const bitcoinJSForks = require('bitcoinforksjs-lib'); |
||||
|
const bitcoinZcash = require('bitcoinjs-lib-zcash'); |
||||
|
const bitcoinPos = require('bitcoinjs-lib-pos'); |
||||
|
|
||||
|
// not prod ready, only for voting!
|
||||
|
// needs a fix
|
||||
|
|
||||
|
module.exports = (shepherd) => { |
||||
|
shepherd.post('/electrum/createrawtx-multiout', (req, res, next) => { |
||||
|
if (shepherd.checkToken(req.body.token)) { |
||||
|
// TODO: 1) unconf output(s) error message
|
||||
|
// 2) check targets integrity
|
||||
|
const network = req.body.network || shepherd.findNetworkObj(req.body.coin); |
||||
|
const ecl = new shepherd.electrumJSCore(shepherd.electrumServers[network].port, shepherd.electrumServers[network].address, shepherd.electrumServers[network].proto); // tcp or tls
|
||||
|
const initTargets = JSON.parse(JSON.stringify(req.body.targets)); |
||||
|
let targets = req.body.targets; |
||||
|
const changeAddress = req.body.change; |
||||
|
const push = req.body.push; |
||||
|
const opreturn = req.body.opreturn; |
||||
|
const btcFee = req.body.btcfee ? Number(req.body.btcfee) : null; |
||||
|
let fee = shepherd.electrumServers[network].txfee; |
||||
|
let wif = req.body.wif; |
||||
|
|
||||
|
if (req.body.gui) { |
||||
|
wif = shepherd.electrumKeys[req.body.coin].priv; |
||||
|
} |
||||
|
|
||||
|
if (req.body.vote) { |
||||
|
wif = shepherd.elections.priv; |
||||
|
} |
||||
|
|
||||
|
if (btcFee) { |
||||
|
fee = 0; |
||||
|
} |
||||
|
|
||||
|
shepherd.log('electrum createrawtx =>', true); |
||||
|
|
||||
|
ecl.connect(); |
||||
|
shepherd.listunspent(ecl, changeAddress, network, true, req.body.verify === 'true' ? true : null) |
||||
|
.then((utxoList) => { |
||||
|
ecl.close(); |
||||
|
|
||||
|
if (utxoList && |
||||
|
utxoList.length && |
||||
|
utxoList[0] && |
||||
|
utxoList[0].txid) { |
||||
|
let utxoListFormatted = []; |
||||
|
let totalInterest = 0; |
||||
|
let totalInterestUTXOCount = 0; |
||||
|
let interestClaimThreshold = 200; |
||||
|
let utxoVerified = true; |
||||
|
|
||||
|
for (let i = 0; i < utxoList.length; i++) { |
||||
|
if (network === 'komodo') { |
||||
|
utxoListFormatted.push({ |
||||
|
txid: utxoList[i].txid, |
||||
|
vout: utxoList[i].vout, |
||||
|
value: Number(utxoList[i].amountSats), |
||||
|
interestSats: Number(utxoList[i].interestSats), |
||||
|
verified: utxoList[i].verified ? utxoList[i].verified : false, |
||||
|
}); |
||||
|
} else { |
||||
|
utxoListFormatted.push({ |
||||
|
txid: utxoList[i].txid, |
||||
|
vout: utxoList[i].vout, |
||||
|
value: Number(utxoList[i].amountSats), |
||||
|
verified: utxoList[i].verified ? utxoList[i].verified : false, |
||||
|
}); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
shepherd.log('electrum listunspent unformatted ==>', true); |
||||
|
shepherd.log(utxoList, true); |
||||
|
|
||||
|
shepherd.log('electrum listunspent formatted ==>', true); |
||||
|
shepherd.log(utxoListFormatted, true); |
||||
|
|
||||
|
const _maxSpendBalance = Number(shepherd.maxSpendBalance(utxoListFormatted)); |
||||
|
/*let targets = [{ |
||||
|
address: outputAddress, |
||||
|
value: value > _maxSpendBalance ? _maxSpendBalance : value, |
||||
|
}];*/ |
||||
|
shepherd.log('targets =>', true); |
||||
|
shepherd.log(targets, true); |
||||
|
|
||||
|
targets[0].value = targets[0].value + fee; |
||||
|
|
||||
|
shepherd.log(`default fee ${fee}`, true); |
||||
|
shepherd.log(`targets ==>`, true); |
||||
|
shepherd.log(targets, true); |
||||
|
|
||||
|
// default coin selection algo blackjack with fallback to accumulative
|
||||
|
// make a first run, calc approx tx fee
|
||||
|
// if ins and outs are empty reduce max spend by txfee
|
||||
|
const firstRun = shepherd.coinSelect(utxoListFormatted, targets, btcFee ? btcFee : 0); |
||||
|
let inputs = firstRun.inputs; |
||||
|
let outputs = firstRun.outputs; |
||||
|
|
||||
|
if (btcFee) { |
||||
|
shepherd.log(`btc fee per byte ${btcFee}`, true); |
||||
|
fee = firstRun.fee; |
||||
|
} |
||||
|
|
||||
|
shepherd.log('coinselect res =>', true); |
||||
|
shepherd.log('coinselect inputs =>', true); |
||||
|
shepherd.log(inputs, true); |
||||
|
shepherd.log('coinselect outputs =>', true); |
||||
|
shepherd.log(outputs, true); |
||||
|
shepherd.log('coinselect calculated fee =>', true); |
||||
|
shepherd.log(fee, true); |
||||
|
|
||||
|
if (!outputs) { |
||||
|
targets[0].value = targets[0].value - fee; |
||||
|
shepherd.log('second run', true); |
||||
|
shepherd.log('coinselect adjusted targets =>', true); |
||||
|
shepherd.log(targets, true); |
||||
|
|
||||
|
const secondRun = shepherd.coinSelect(utxoListFormatted, targets, 0); |
||||
|
inputs = secondRun.inputs; |
||||
|
outputs = secondRun.outputs; |
||||
|
fee = fee ? fee : secondRun.fee; |
||||
|
|
||||
|
shepherd.log('second run coinselect inputs =>', true); |
||||
|
shepherd.log(inputs, true); |
||||
|
shepherd.log('second run coinselect outputs =>', true); |
||||
|
shepherd.log(outputs, true); |
||||
|
shepherd.log('second run coinselect fee =>', true); |
||||
|
shepherd.log(fee, true); |
||||
|
} |
||||
|
|
||||
|
let _change = 0; |
||||
|
|
||||
|
if (outputs && |
||||
|
outputs.length > 1) { |
||||
|
_change = outputs[outputs.length - 1].value - fee; |
||||
|
} |
||||
|
|
||||
|
if (!btcFee && |
||||
|
_change === 0) { |
||||
|
outputs[0].value = outputs[0].value - fee; |
||||
|
} |
||||
|
|
||||
|
shepherd.log('adjusted outputs'); |
||||
|
shepherd.log(outputs, true); |
||||
|
|
||||
|
shepherd.log('init targets', true); |
||||
|
shepherd.log(initTargets, true); |
||||
|
|
||||
|
if (initTargets[0].value < targets[0].value) { |
||||
|
targets[0].value = initTargets[0].value; |
||||
|
} |
||||
|
|
||||
|
let _targetsSum = 0; |
||||
|
|
||||
|
for (let i = 0; i < targets.length; i++) { |
||||
|
_targetsSum += Number(targets[i].value); |
||||
|
} |
||||
|
|
||||
|
shepherd.log(`total targets sum ${_targetsSum}`); |
||||
|
|
||||
|
/*if (btcFee) { |
||||
|
value = _targetsSum; |
||||
|
} else { |
||||
|
if (_change > 0) { |
||||
|
value = _targetsSum - fee; |
||||
|
} |
||||
|
}*/ |
||||
|
value = _targetsSum; |
||||
|
|
||||
|
shepherd.log('adjusted outputs, value - default fee =>', true); |
||||
|
shepherd.log(outputs, true); |
||||
|
|
||||
|
// check if any outputs are unverified
|
||||
|
if (inputs && |
||||
|
inputs.length) { |
||||
|
for (let i = 0; i < inputs.length; i++) { |
||||
|
if (!inputs[i].verified) { |
||||
|
utxoVerified = false; |
||||
|
break; |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
for (let i = 0; i < inputs.length; i++) { |
||||
|
if (Number(inputs[i].interestSats) > interestClaimThreshold) { |
||||
|
totalInterest += Number(inputs[i].interestSats); |
||||
|
totalInterestUTXOCount++; |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
const _maxSpend = shepherd.maxSpendBalance(utxoListFormatted); |
||||
|
|
||||
|
if (value > _maxSpend) { |
||||
|
const successObj = { |
||||
|
msg: 'error', |
||||
|
result: `Spend value is too large. Max available amount is ${Number((_maxSpend * 0.00000001.toFixed(8)))}`, |
||||
|
}; |
||||
|
|
||||
|
res.end(JSON.stringify(successObj)); |
||||
|
} else { |
||||
|
shepherd.log(`maxspend ${_maxSpend} (${_maxSpend * 0.00000001})`, true); |
||||
|
shepherd.log(`value ${value}`, true); |
||||
|
// shepherd.log(`sendto ${outputAddress} amount ${value} (${value * 0.00000001})`, true);
|
||||
|
shepherd.log(`changeto ${changeAddress} amount ${_change} (${_change * 0.00000001})`, true); |
||||
|
|
||||
|
// account for KMD interest
|
||||
|
if (network === 'komodo' && |
||||
|
totalInterest > 0) { |
||||
|
// account for extra vout
|
||||
|
// const _feeOverhead = outputs.length === 1 ? shepherd.estimateTxSize(0, 1) * feeRate : 0;
|
||||
|
const _feeOverhead = 0; |
||||
|
|
||||
|
shepherd.log(`max interest to claim ${totalInterest} (${totalInterest * 0.00000001})`, true); |
||||
|
shepherd.log(`estimated fee overhead ${_feeOverhead}`, true); |
||||
|
shepherd.log(`current change amount ${_change} (${_change * 0.00000001}), boosted change amount ${_change + (totalInterest - _feeOverhead)} (${(_change + (totalInterest - _feeOverhead)) * 0.00000001})`, true); |
||||
|
|
||||
|
if (_maxSpend - fee === value) { |
||||
|
_change = totalInterest - _change - _feeOverhead; |
||||
|
|
||||
|
if (outputAddress === changeAddress) { |
||||
|
value += _change; |
||||
|
_change = 0; |
||||
|
shepherd.log(`send to self ${outputAddress} = ${changeAddress}`, true); |
||||
|
shepherd.log(`send to self old val ${value}, new val ${value + _change}`, true); |
||||
|
} |
||||
|
} else { |
||||
|
_change = _change + (totalInterest - _feeOverhead); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
if (!inputs && |
||||
|
!outputs) { |
||||
|
const successObj = { |
||||
|
msg: 'error', |
||||
|
result: 'Can\'t find best fit utxo. Try lower amount.', |
||||
|
}; |
||||
|
|
||||
|
res.end(JSON.stringify(successObj)); |
||||
|
} else { |
||||
|
let vinSum = 0; |
||||
|
|
||||
|
for (let i = 0; i < inputs.length; i++) { |
||||
|
vinSum += inputs[i].value; |
||||
|
} |
||||
|
|
||||
|
let voutSum = 0; |
||||
|
|
||||
|
for (let i = 0; i < outputs.length; i++) { |
||||
|
voutSum += outputs[i].value; |
||||
|
} |
||||
|
|
||||
|
const _estimatedFee = vinSum - voutSum; |
||||
|
|
||||
|
shepherd.log(`vin sum ${vinSum} (${vinSum * 0.00000001})`, true); |
||||
|
shepherd.log(`vout sum ${voutSum} (${voutSum * 0.00000001})`, true); |
||||
|
shepherd.log(`estimatedFee ${_estimatedFee} (${_estimatedFee * 0.00000001})`, true); |
||||
|
// double check no extra fee is applied
|
||||
|
shepherd.log(`vin - vout - change ${vinSum - value - _change}`); |
||||
|
|
||||
|
if ((vinSum - value - _change) > fee) { |
||||
|
_change += fee; |
||||
|
shepherd.log(`double fee, increase change by ${fee}`); |
||||
|
shepherd.log(`adjusted vin - vout - change ${vinSum - value - _change}`); |
||||
|
} |
||||
|
|
||||
|
// TODO: use individual dust thresholds
|
||||
|
if (_change > 0 && |
||||
|
_change <= 1000) { |
||||
|
shepherd.log(`change is < 1000 sats, donate ${_change} sats to miners`, true); |
||||
|
_change = 0; |
||||
|
} |
||||
|
|
||||
|
outputAddress = outputs; |
||||
|
|
||||
|
if (outputAddress.length > 1) { |
||||
|
outputAddress.pop(); |
||||
|
} |
||||
|
|
||||
|
let _rawtx; |
||||
|
|
||||
|
if (network === 'btg' || |
||||
|
network === 'bch') { |
||||
|
/*_rawtx = shepherd.buildSignedTxForks( |
||||
|
outputAddress, |
||||
|
changeAddress, |
||||
|
wif, |
||||
|
network, |
||||
|
inputs, |
||||
|
_change, |
||||
|
value |
||||
|
);*/ |
||||
|
} else { |
||||
|
_rawtx = shepherd.buildSignedTxMulti( |
||||
|
outputAddress, |
||||
|
changeAddress, |
||||
|
wif, |
||||
|
network, |
||||
|
inputs, |
||||
|
_change, |
||||
|
value, |
||||
|
opreturn |
||||
|
); |
||||
|
} |
||||
|
|
||||
|
if (!push || |
||||
|
push === 'false') { |
||||
|
const successObj = { |
||||
|
msg: 'success', |
||||
|
result: { |
||||
|
utxoSet: inputs, |
||||
|
change: _change, |
||||
|
changeAdjusted: _change, |
||||
|
totalInterest, |
||||
|
// wif,
|
||||
|
fee, |
||||
|
value, |
||||
|
outputAddress, |
||||
|
changeAddress, |
||||
|
network, |
||||
|
rawtx: _rawtx, |
||||
|
utxoVerified, |
||||
|
}, |
||||
|
}; |
||||
|
|
||||
|
res.end(JSON.stringify(successObj)); |
||||
|
} else { |
||||
|
const ecl = new shepherd.electrumJSCore(shepherd.electrumServers[network].port, shepherd.electrumServers[network].address, shepherd.electrumServers[network].proto); // tcp or tls
|
||||
|
|
||||
|
ecl.connect(); |
||||
|
ecl.blockchainTransactionBroadcast(_rawtx) |
||||
|
.then((txid) => { |
||||
|
ecl.close(); |
||||
|
|
||||
|
const _rawObj = { |
||||
|
utxoSet: inputs, |
||||
|
change: _change, |
||||
|
changeAdjusted: _change, |
||||
|
totalInterest, |
||||
|
fee, |
||||
|
value, |
||||
|
outputAddress, |
||||
|
changeAddress, |
||||
|
network, |
||||
|
rawtx: _rawtx, |
||||
|
txid, |
||||
|
utxoVerified, |
||||
|
}; |
||||
|
|
||||
|
if (txid && |
||||
|
txid.indexOf('bad-txns-inputs-spent') > -1) { |
||||
|
const successObj = { |
||||
|
msg: 'error', |
||||
|
result: 'Bad transaction inputs spent', |
||||
|
raw: _rawObj, |
||||
|
}; |
||||
|
|
||||
|
res.end(JSON.stringify(successObj)); |
||||
|
} else { |
||||
|
if (txid && |
||||
|
txid.length === 64) { |
||||
|
if (txid.indexOf('bad-txns-in-belowout') > -1) { |
||||
|
const successObj = { |
||||
|
msg: 'error', |
||||
|
result: 'Bad transaction inputs spent', |
||||
|
raw: _rawObj, |
||||
|
}; |
||||
|
|
||||
|
res.end(JSON.stringify(successObj)); |
||||
|
} else { |
||||
|
const successObj = { |
||||
|
msg: 'success', |
||||
|
result: _rawObj, |
||||
|
}; |
||||
|
|
||||
|
res.end(JSON.stringify(successObj)); |
||||
|
} |
||||
|
} else { |
||||
|
if (txid && |
||||
|
txid.indexOf('bad-txns-in-belowout') > -1) { |
||||
|
const successObj = { |
||||
|
msg: 'error', |
||||
|
result: 'Bad transaction inputs spent', |
||||
|
raw: _rawObj, |
||||
|
}; |
||||
|
|
||||
|
res.end(JSON.stringify(successObj)); |
||||
|
} else { |
||||
|
const successObj = { |
||||
|
msg: 'error', |
||||
|
result: 'Can\'t broadcast transaction', |
||||
|
raw: _rawObj, |
||||
|
}; |
||||
|
|
||||
|
res.end(JSON.stringify(successObj)); |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
}); |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
} else { |
||||
|
const successObj = { |
||||
|
msg: 'error', |
||||
|
result: utxoList, |
||||
|
}; |
||||
|
|
||||
|
res.end(JSON.stringify(successObj)); |
||||
|
} |
||||
|
}); |
||||
|
} else { |
||||
|
const errorObj = { |
||||
|
msg: 'error', |
||||
|
result: 'unauthorized access', |
||||
|
}; |
||||
|
|
||||
|
res.end(JSON.stringify(errorObj)); |
||||
|
} |
||||
|
}); |
||||
|
|
||||
|
// single sig
|
||||
|
shepherd.buildSignedTxMulti = (sendTo, changeAddress, wif, network, utxo, changeValue, spendValue, opreturn) => { |
||||
|
let key = shepherd.isZcash(network) ? bitcoinZcash.ECPair.fromWIF(wif, shepherd.getNetworkData(network)) : shepherd.bitcoinJS.ECPair.fromWIF(wif, shepherd.getNetworkData(network)); |
||||
|
let tx; |
||||
|
|
||||
|
if (shepherd.isZcash(network)) { |
||||
|
tx = new bitcoinZcash.TransactionBuilder(shepherd.getNetworkData(network)); |
||||
|
} else if (shepherd.isPos(network)) { |
||||
|
tx = new bitcoinPos.TransactionBuilder(shepherd.getNetworkData(network)); |
||||
|
} else { |
||||
|
tx = new shepherd.bitcoinJS.TransactionBuilder(shepherd.getNetworkData(network)); |
||||
|
} |
||||
|
|
||||
|
shepherd.log('buildSignedTx', true); |
||||
|
// console.log(`buildSignedTx priv key ${wif}`);
|
||||
|
shepherd.log(`buildSignedTx pub key ${key.getAddress().toString()}`, true); |
||||
|
// console.log('buildSignedTx std tx fee ' + shepherd.electrumServers[network].txfee);
|
||||
|
|
||||
|
for (let i = 0; i < utxo.length; i++) { |
||||
|
tx.addInput(utxo[i].txid, utxo[i].vout); |
||||
|
} |
||||
|
|
||||
|
for (let i = 0; i < sendTo.length; i++) { |
||||
|
if (shepherd.isPos(network)) { |
||||
|
tx.addOutput(sendTo[i].address, Number(sendTo[i].value), shepherd.getNetworkData(network)); |
||||
|
} else { |
||||
|
tx.addOutput(sendTo[i].address, Number(sendTo[i].value)); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
if (changeValue > 0) { |
||||
|
if (shepherd.isPos(network)) { |
||||
|
tx.addOutput(changeAddress, Number(changeValue), shepherd.getNetworkData(network)); |
||||
|
} else { |
||||
|
tx.addOutput(changeAddress, Number(changeValue)); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
if (opreturn && |
||||
|
opreturn.length) { |
||||
|
for (let i = 0; i < opreturn.length; i++) { |
||||
|
shepherd.log(`opreturn ${i} ${opreturn[i]}`); |
||||
|
const data = Buffer.from(opreturn[i], 'utf8'); |
||||
|
const dataScript = shepherd.bitcoinJS.script.nullData.output.encode(data); |
||||
|
tx.addOutput(dataScript, 1000); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
if (network === 'komodo' || |
||||
|
network === 'KMD') { |
||||
|
const _locktime = Math.floor(Date.now() / 1000) - 777; |
||||
|
tx.setLockTime(_locktime); |
||||
|
shepherd.log(`kmd tx locktime set to ${_locktime}`, true); |
||||
|
} |
||||
|
|
||||
|
shepherd.log('buildSignedTx unsigned tx data vin', true); |
||||
|
shepherd.log(tx.tx.ins, true); |
||||
|
shepherd.log('buildSignedTx unsigned tx data vout', true); |
||||
|
shepherd.log(tx.tx.outs, true); |
||||
|
shepherd.log('buildSignedTx unsigned tx data', true); |
||||
|
shepherd.log(tx, true); |
||||
|
|
||||
|
for (let i = 0; i < utxo.length; i++) { |
||||
|
if (shepherd.isPos(network)) { |
||||
|
tx.sign(shepherd.getNetworkData(network), i, key); |
||||
|
} else { |
||||
|
tx.sign(i, key); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
const rawtx = tx.build().toHex(); |
||||
|
|
||||
|
shepherd.log('buildSignedTx signed tx hex', true); |
||||
|
shepherd.log(rawtx, true); |
||||
|
|
||||
|
return rawtx; |
||||
|
} |
||||
|
|
||||
|
return shepherd; |
||||
|
}; |
@ -0,0 +1,86 @@ |
|||||
|
const bitcoinJSForks = require('bitcoinforksjs-lib'); |
||||
|
const bitcoinZcash = require('bitcoinjs-lib-zcash'); |
||||
|
const bitcoinPos = require('bitcoinjs-lib-pos'); |
||||
|
|
||||
|
module.exports = (shepherd) => { |
||||
|
// utxo split 1 -> 1, multiple outputs
|
||||
|
shepherd.post('/electrum/createrawtx-split', (req, res, next) => { |
||||
|
if (shepherd.checkToken(req.body.token)) { |
||||
|
const wif = req.body.payload.wif; |
||||
|
const utxo = req.body.payload.utxo; |
||||
|
const targets = req.body.payload.targets; |
||||
|
const network = req.body.payload.network; |
||||
|
const change = req.body.payload.change; |
||||
|
const outputAddress = req.body.payload.outputAddress; |
||||
|
const changeAddress = req.body.payload.changeAddress; |
||||
|
|
||||
|
let key = shepherd.isZcash(network) ? bitcoinZcash.ECPair.fromWIF(wif, shepherd.getNetworkData(network)) : shepherd.bitcoinJS.ECPair.fromWIF(wif, shepherd.getNetworkData(network)); |
||||
|
let tx; |
||||
|
|
||||
|
if (shepherd.isZcash(network)) { |
||||
|
tx = new bitcoinZcash.TransactionBuilder(shepherd.getNetworkData(network)); |
||||
|
} else if (shepherd.isPos(network)) { |
||||
|
tx = new bitcoinPos.TransactionBuilder(shepherd.getNetworkData(network)); |
||||
|
} else { |
||||
|
tx = new shepherd.bitcoinJS.TransactionBuilder(shepherd.getNetworkData(network)); |
||||
|
} |
||||
|
|
||||
|
shepherd.log('buildSignedTx', true); |
||||
|
shepherd.log(`buildSignedTx pub key ${key.getAddress().toString()}`, true); |
||||
|
|
||||
|
for (let i = 0; i < utxo.length; i++) { |
||||
|
tx.addInput(utxo[i].txid, utxo[i].vout); |
||||
|
} |
||||
|
|
||||
|
for (let i = 0; i < targets.length; i++) { |
||||
|
if (shepherd.isPos(network)) { |
||||
|
tx.addOutput(outputAddress, Number(targets[i]), shepherd.getNetworkData(network)); |
||||
|
} else { |
||||
|
tx.addOutput(outputAddress, Number(targets[i])); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
if (Number(change) > 0) { |
||||
|
if (shepherd.isPos(network)) { |
||||
|
tx.addOutput(changeAddress, Number(change), shepherd.getNetworkData(network)); |
||||
|
} else { |
||||
|
shepherd.log(`change ${change}`, true); |
||||
|
tx.addOutput(changeAddress, Number(change)); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
if (network === 'komodo' || |
||||
|
network === 'KMD') { |
||||
|
const _locktime = Math.floor(Date.now() / 1000) - 777; |
||||
|
tx.setLockTime(_locktime); |
||||
|
shepherd.log(`kmd tx locktime set to ${_locktime}`, true); |
||||
|
} |
||||
|
|
||||
|
for (let i = 0; i < utxo.length; i++) { |
||||
|
if (shepherd.isPos(network)) { |
||||
|
tx.sign(shepherd.getNetworkData(network), i, key); |
||||
|
} else { |
||||
|
tx.sign(i, key); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
const rawtx = tx.build().toHex(); |
||||
|
|
||||
|
const successObj = { |
||||
|
msg: 'success', |
||||
|
result: rawtx, |
||||
|
}; |
||||
|
|
||||
|
res.end(JSON.stringify(successObj)); |
||||
|
} else { |
||||
|
const errorObj = { |
||||
|
msg: 'error', |
||||
|
result: 'unauthorized access', |
||||
|
}; |
||||
|
|
||||
|
res.end(JSON.stringify(errorObj)); |
||||
|
} |
||||
|
}); |
||||
|
|
||||
|
return shepherd; |
||||
|
}; |
@ -1,3 +1,3 @@ |
|||||
version=0.2.0.25e |
version=0.2.0.29c |
||||
type=e-beta |
type=c-beta |
||||
minversion=0.2.0.2 |
minversion=0.2.0.29 |
@ -1 +1 @@ |
|||||
0.2.0.25e-beta |
0.2.0.29c-beta |
Loading…
Reference in new issue