2017-09-06 23:36:15 +02:00
|
|
|
const BN = require('ethereumjs-util').BN
|
2017-09-07 20:59:15 +02:00
|
|
|
const normalize = require('eth-sig-util').normalize
|
2017-09-06 23:36:15 +02:00
|
|
|
|
|
|
|
class PendingBalanceCalculator {
|
|
|
|
|
2018-04-19 17:38:56 +02:00
|
|
|
/**
|
|
|
|
* Used for calculating a users "pending balance": their current balance minus the total possible cost of all their
|
|
|
|
* pending transactions.
|
|
|
|
*
|
|
|
|
* @typedef {Object} PendingBalanceCalculator
|
|
|
|
* @param {Function} getBalance Returns a promise of a BN of the current balance in Wei
|
|
|
|
* @param {Function} getPendingTransactions Returns an array of TxMeta Objects, which have txParams properties,
|
|
|
|
* which include value, gasPrice, and gas, all in a base=16 hex format.
|
|
|
|
*
|
|
|
|
*/
|
2017-09-06 23:36:15 +02:00
|
|
|
constructor ({ getBalance, getPendingTransactions }) {
|
|
|
|
this.getPendingTransactions = getPendingTransactions
|
2017-09-07 21:30:25 +02:00
|
|
|
this.getNetworkBalance = getBalance
|
2017-09-06 23:36:15 +02:00
|
|
|
}
|
|
|
|
|
2018-04-19 17:38:56 +02:00
|
|
|
/**
|
|
|
|
* Returns the users "pending balance": their current balance minus the total possible cost of all their
|
|
|
|
* pending transactions.
|
|
|
|
*
|
|
|
|
* @returns {Promise<string>} Promises a base 16 hex string that contains the user's "pending balance"
|
|
|
|
*
|
|
|
|
*/
|
2017-10-21 21:06:39 +02:00
|
|
|
async getBalance () {
|
2017-09-06 23:37:46 +02:00
|
|
|
const results = await Promise.all([
|
2017-09-07 21:30:25 +02:00
|
|
|
this.getNetworkBalance(),
|
2017-09-06 23:37:46 +02:00
|
|
|
this.getPendingTransactions(),
|
|
|
|
])
|
|
|
|
|
2017-09-25 23:36:49 +02:00
|
|
|
const [ balance, pending ] = results
|
2019-12-05 21:23:43 +01:00
|
|
|
if (!balance) return undefined
|
2017-09-14 00:02:05 +02:00
|
|
|
|
2017-09-07 21:43:10 +02:00
|
|
|
const pendingValue = pending.reduce((total, tx) => {
|
2017-09-25 23:36:49 +02:00
|
|
|
return total.add(this.calculateMaxCost(tx))
|
2017-09-07 20:59:15 +02:00
|
|
|
}, new BN(0))
|
|
|
|
|
2017-09-07 21:54:28 +02:00
|
|
|
return `0x${balance.sub(pendingValue).toString(16)}`
|
2017-09-07 20:59:15 +02:00
|
|
|
}
|
|
|
|
|
2018-04-19 17:38:56 +02:00
|
|
|
/**
|
|
|
|
* Calculates the maximum possible cost of a single transaction, based on the value, gas price and gas limit.
|
|
|
|
*
|
|
|
|
* @param {object} tx Contains all that data about a transaction.
|
|
|
|
* @property {object} tx.txParams Contains data needed to calculate the maximum cost of the transaction: gas,
|
|
|
|
* gasLimit and value.
|
|
|
|
*
|
|
|
|
* @returns {string} Returns a base 16 hex string that contains the maximum possible cost of the transaction.
|
|
|
|
*/
|
2017-09-25 23:36:49 +02:00
|
|
|
calculateMaxCost (tx) {
|
2017-09-07 21:30:25 +02:00
|
|
|
const txValue = tx.txParams.value
|
2017-09-07 21:43:10 +02:00
|
|
|
const value = this.hexToBn(txValue)
|
2017-09-07 21:52:49 +02:00
|
|
|
const gasPrice = this.hexToBn(tx.txParams.gasPrice)
|
|
|
|
|
|
|
|
const gas = tx.txParams.gas
|
|
|
|
const gasLimit = tx.txParams.gasLimit
|
|
|
|
const gasLimitBn = this.hexToBn(gas || gasLimit)
|
|
|
|
|
|
|
|
const gasCost = gasPrice.mul(gasLimitBn)
|
|
|
|
return value.add(gasCost)
|
2017-09-06 23:36:15 +02:00
|
|
|
}
|
|
|
|
|
2018-04-19 17:38:56 +02:00
|
|
|
/**
|
|
|
|
* Converts a hex string to a BN object
|
|
|
|
*
|
|
|
|
* @param {string} hex A number represented as a hex string
|
|
|
|
* @returns {Object} A BN object
|
|
|
|
*
|
|
|
|
*/
|
2017-09-07 21:43:10 +02:00
|
|
|
hexToBn (hex) {
|
|
|
|
return new BN(normalize(hex).substring(2), 16)
|
|
|
|
}
|
|
|
|
|
2017-09-06 23:36:15 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = PendingBalanceCalculator
|