mirror of https://github.com/lukechilds/Agama.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
175 lines
4.9 KiB
175 lines
4.9 KiB
/*
|
|
MIT License
|
|
|
|
Copyright (c) 2017 Yuki Akiyama
|
|
Copyright (c) 2017 - 2018 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');
|
|
// zcash fallback
|
|
const Buffer = require('safe-buffer').Buffer;
|
|
const {
|
|
readSlice,
|
|
readInt32,
|
|
readUInt32,
|
|
} = require('tx-decoder/src/buffer-utils');
|
|
const {
|
|
compose,
|
|
addProp,
|
|
} = require('tx-decoder/src/compose');
|
|
const {
|
|
readInputs,
|
|
readInput,
|
|
readOutput,
|
|
} = require('tx-decoder/src/tx-decoder');
|
|
const crypto = require('crypto');
|
|
const _sha256 = (data) => {
|
|
return crypto.createHash('sha256').update(data).digest();
|
|
};
|
|
|
|
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 ? input.hash : 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) {
|
|
if (network.isZcash) {
|
|
console.log('z tx decode fallback');
|
|
|
|
const buffer = Buffer.from(rawtx, 'hex');
|
|
|
|
const decodeTx = buffer => (
|
|
compose([
|
|
addProp('version', readInt32), // 4 bytes
|
|
addProp('ins', readInputs(readInput)), // 1-9 bytes (VarInt), Input counter; Variable, Inputs
|
|
addProp('outs', readInputs(readOutput)), // 1-9 bytes (VarInt), Output counter; Variable, Outputs
|
|
addProp('locktime', readUInt32) // 4 bytes
|
|
])({}, buffer)
|
|
);
|
|
|
|
const readHash = buffer => {
|
|
const [res, bufferLeft] = readSlice(32)(_sha256(_sha256(buffer)));
|
|
const hash = Buffer.from(res, 'hex').reverse().toString('hex');
|
|
return [hash, bufferLeft];
|
|
};
|
|
|
|
let decodedtx = decodeTx(buffer);
|
|
decodedtx[0].getId = () => {
|
|
return readHash(buffer)[0];
|
|
};
|
|
|
|
return {
|
|
tx: decodedtx[0],
|
|
network: network,
|
|
format: decodeFormat(decodedtx[0]),
|
|
inputs: !decodedtx[0].ins.length ? [{ txid: '0000000000000000000000000000000000000000000000000000000000000000' }] : decodeInput(decodedtx[0]),
|
|
outputs: decodeOutput(decodedtx[0], network),
|
|
};
|
|
} else {
|
|
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;
|
|
}
|