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.

258 lines
6.4 KiB

6 years ago
/*!
* 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)
}
}
6 years ago
/**
* 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'}`
6 years ago
// Return transaction from cache when possible
if (this.txCache.has(keyCache))
return this.txCache.get(keyCache)
6 years ago
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)
}
}
6 years ago
/**
* 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: []
}
6 years ago
if (!ret.vsize)
delete ret.vsize
6 years ago
if (tx.time)
ret.created = tx.time
6 years ago
// Process block informations
if (tx.blockhash && tx.confirmations && tx.blocktime) {
ret.block = {
height: rpcLatestBlock.height - tx.confirmations + 1,
hash: tx.blockhash,
time: tx.blocktime
6 years ago
}
}
6 years ago
let inAmount = 0
let outAmount = 0
6 years ago
// Process the inputs
ret.inputs = await this._getInputs(tx, fees)
inAmount = ret.inputs.reduce((prev, cur) => prev + cur.outpoint.value, 0)
6 years ago
// Process the outputs
ret.outputs = await this._getOutputs(tx)
outAmount = ret.outputs.reduce((prev, cur) => prev + cur.value, 0)
6 years ago
// 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)
6 years ago
}
return ret
6 years ago
}
6 years ago
/**
* 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()