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.
257 lines
6.4 KiB
257 lines
6.4 KiB
/*!
|
|
* lib/bitcoind-rpc/transactions.js
|
|
* Copyright © 2019 – Katana Cryptographic Ltd. All Rights Reserved.
|
|
*/
|
|
'use strict'
|
|
|
|
const _ = require('lodash')
|
|
const LRU = require('lru-cache')
|
|
const errors = require('../errors')
|
|
const Logger = require('../logger')
|
|
const util = require('../util')
|
|
const RpcClient = require('./rpc-client')
|
|
const rpcLatestBlock = require('./latest-block')
|
|
|
|
|
|
/**
|
|
* A singleton providing information about transactions
|
|
*/
|
|
class Transactions {
|
|
|
|
/**
|
|
* Constructor
|
|
*/
|
|
constructor() {
|
|
// Caches
|
|
this.txCache = LRU({
|
|
// Maximum number of transactions to store
|
|
max: 10000,
|
|
// Function used to compute length of item
|
|
length: (n, key) => 1,
|
|
// Maximum age for items in the cache. Items do not expire
|
|
maxAge: Infinity
|
|
})
|
|
|
|
this.prevCache = LRU({
|
|
// Maximum number of transactions to store
|
|
max: 100000,
|
|
// Function used to compute length of item
|
|
length: (n, key) => 1,
|
|
// Maximum age for items in the cache. Items do not expire
|
|
maxAge: Infinity
|
|
})
|
|
|
|
|
|
// Initialize the rpc client
|
|
this.rpcClient = new RpcClient()
|
|
}
|
|
|
|
/**
|
|
* Get the transactions for a given array of txids
|
|
* @param {string[]} txids - txids of the transaction to be retrieved
|
|
* @param {boolean} fees - true if fees must be computed, false otherwise
|
|
* @returns {Promise} return an array of transactions (object[])
|
|
*/
|
|
async getTransactions(txids, fees) {
|
|
try {
|
|
const rpcCalls = txids.map(txid => {
|
|
return {
|
|
'method': 'getrawtransaction',
|
|
'params': [txid, true]
|
|
}
|
|
})
|
|
|
|
const txs = await this.rpcClient.batch(rpcCalls)
|
|
|
|
return await util.seriesCall(txs, async tx => {
|
|
if (tx.result == null) {
|
|
Logger.info(` got null for ${txids[tx.id]}`)
|
|
return null
|
|
} else {
|
|
return this._prepareTxResult(tx.result, fees)
|
|
}
|
|
})
|
|
|
|
} catch(e) {
|
|
Logger.error(e, 'Transaction.getTransactions()')
|
|
return Promise.reject(errors.generic.GEN)
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Get the transaction for a given txid
|
|
* @param {string} txid - txid of the transaction to be retrieved
|
|
* @param {boolean} fees - true if fees must be computed, false otherwise
|
|
* @returns {Promise}
|
|
*/
|
|
async getTransaction(txid, fees) {
|
|
const keyCache = `${txid}-${fees ? '1' : '0'}`
|
|
|
|
// Return transaction from cache when possible
|
|
if (this.txCache.has(keyCache))
|
|
return this.txCache.get(keyCache)
|
|
|
|
try {
|
|
const tx = await this.rpcClient.getrawtransaction(txid, true)
|
|
const ret = await this._prepareTxResult(tx)
|
|
// Store the result in cache
|
|
if (ret.block && ret.block.hash)
|
|
this.txCache.set(txid, ret)
|
|
return ret
|
|
} catch(e) {
|
|
Logger.error(e, 'Transaction.getTransaction()')
|
|
return Promise.reject(errors.generic.GEN)
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Formats a transaction object returned by the RPC API
|
|
* @param {object} tx - transaction
|
|
* @param {boolean} fees - true if fees must be computed, false otherwise
|
|
* @returns {Promise} return an array of inputs (object[])
|
|
*/
|
|
async _prepareTxResult(tx, fees) {
|
|
const ret = {
|
|
txid: tx.txid,
|
|
size: tx.size,
|
|
vsize: tx.vsize,
|
|
version: tx.version,
|
|
locktime: tx.locktime,
|
|
inputs: [],
|
|
outputs: []
|
|
}
|
|
|
|
if (!ret.vsize)
|
|
delete ret.vsize
|
|
|
|
if (tx.time)
|
|
ret.created = tx.time
|
|
|
|
// Process block informations
|
|
if (tx.blockhash && tx.confirmations && tx.blocktime) {
|
|
ret.block = {
|
|
height: rpcLatestBlock.height - tx.confirmations + 1,
|
|
hash: tx.blockhash,
|
|
time: tx.blocktime
|
|
}
|
|
}
|
|
|
|
let inAmount = 0
|
|
let outAmount = 0
|
|
|
|
// Process the inputs
|
|
ret.inputs = await this._getInputs(tx, fees)
|
|
inAmount = ret.inputs.reduce((prev, cur) => prev + cur.outpoint.value, 0)
|
|
|
|
// Process the outputs
|
|
ret.outputs = await this._getOutputs(tx)
|
|
outAmount = ret.outputs.reduce((prev, cur) => prev + cur.value, 0)
|
|
|
|
// Process the fees (if needed)
|
|
if (fees) {
|
|
ret.fees = inAmount - outAmount
|
|
if (ret.fees > 0 && ret.size)
|
|
ret.feerate = Math.round(ret.fees / ret.size)
|
|
if (ret.fees > 0 && ret.vsize)
|
|
ret.vfeerate = Math.round(ret.fees / ret.vsize)
|
|
}
|
|
|
|
return ret
|
|
}
|
|
|
|
|
|
/**
|
|
* Extract information about the inputs of a transaction
|
|
* @param {object} tx - transaction
|
|
* @param {boolean} fees - true if fees must be computed, false otherwise
|
|
* @returns {Promise} return an array of inputs (object[])
|
|
*/
|
|
async _getInputs(tx, fees) {
|
|
const inputs = []
|
|
let n = 0
|
|
|
|
await util.seriesCall(tx.vin, async input => {
|
|
const txin = {
|
|
n,
|
|
seq: input.sequence,
|
|
}
|
|
|
|
if (input.coinbase) {
|
|
txin.coinbase = input.coinbase
|
|
} else {
|
|
txin.outpoint = {
|
|
txid: input.txid,
|
|
vout: input.vout
|
|
}
|
|
txin.sig = input.scriptSig.hex
|
|
}
|
|
|
|
if (input.txinwitness)
|
|
txin.witness = input.txinwitness
|
|
|
|
if (fees && txin.outpoint) {
|
|
const inTxid = txin.outpoint.txid
|
|
let ptx
|
|
|
|
if (this.prevCache.has(inTxid)) {
|
|
ptx = this.prevCache.get(inTxid)
|
|
} else {
|
|
ptx = await this.rpcClient.getrawtransaction(inTxid, true)
|
|
if (ptx.blockhash && ptx.confirmations && ptx.blocktime) {
|
|
ptx.height = rpcLatestBlock.height - ptx.confirmations + 1
|
|
this.prevCache.set(inTxid, ptx)
|
|
}
|
|
}
|
|
|
|
const outpoint = ptx.vout[txin.outpoint.vout]
|
|
txin.outpoint.value = Math.round(outpoint.value * 1e8)
|
|
txin.outpoint.scriptpubkey = outpoint.scriptPubKey.hex
|
|
inputs.push(txin)
|
|
n++
|
|
|
|
} else {
|
|
inputs.push(txin)
|
|
n++
|
|
}
|
|
})
|
|
|
|
return inputs
|
|
}
|
|
|
|
/**
|
|
* Extract information about the outputs of a transaction
|
|
* @param {object} tx - transaction
|
|
* @returns {Promise} return an array of outputs (object[])
|
|
*/
|
|
async _getOutputs(tx) {
|
|
const outputs = []
|
|
let n = 0
|
|
|
|
for (let output of tx.vout) {
|
|
const pk = output.scriptPubKey
|
|
const amount = Math.round(output.value * 1e8)
|
|
|
|
let o = {
|
|
n,
|
|
value: amount,
|
|
scriptpubkey: pk.hex,
|
|
type: pk.type
|
|
}
|
|
|
|
if (pk.addresses) {
|
|
if (pk.addresses.length == 1)
|
|
o.address = pk.addresses[0]
|
|
else
|
|
o.addresses = pk.addresses
|
|
}
|
|
|
|
outputs.push(o)
|
|
n++
|
|
}
|
|
|
|
return outputs
|
|
}
|
|
|
|
}
|
|
|
|
module.exports = new Transactions()
|
|
|