2017-12-18 01:36:03 +01:00
|
|
|
const ObservableStore = require('obs-store')
|
|
|
|
const extend = require('xtend')
|
2018-01-12 00:00:48 +01:00
|
|
|
const BN = require('ethereumjs-util').BN
|
2018-01-12 19:25:36 +01:00
|
|
|
const EthQuery = require('eth-query')
|
2018-04-12 23:06:59 +02:00
|
|
|
const log = require('loglevel')
|
2017-12-18 01:36:03 +01:00
|
|
|
|
|
|
|
class RecentBlocksController {
|
|
|
|
|
2018-04-19 17:38:56 +02:00
|
|
|
/**
|
|
|
|
* Controller responsible for storing, updating and managing the recent history of blocks. Blocks are back filled
|
|
|
|
* upon the controller's construction and then the list is updated when the given block tracker gets a 'block' event
|
|
|
|
* (indicating that there is a new block to process).
|
|
|
|
*
|
|
|
|
* @typedef {Object} RecentBlocksController
|
|
|
|
* @param {object} opts Contains objects necessary for tracking blocks and querying the blockchain
|
|
|
|
* @param {BlockTracker} opts.blockTracker Contains objects necessary for tracking blocks and querying the blockchain
|
|
|
|
* @param {BlockTracker} opts.provider The provider used to create a new EthQuery instance.
|
|
|
|
* @property {BlockTracker} blockTracker Points to the passed BlockTracker. On RecentBlocksController construction,
|
|
|
|
* listens for 'block' events so that new blocks can be processed and added to storage.
|
|
|
|
* @property {EthQuery} ethQuery Points to the EthQuery instance created with the passed provider
|
|
|
|
* @property {number} historyLength The maximum length of blocks to track
|
|
|
|
* @property {object} store Stores the recentBlocks
|
|
|
|
* @property {array} store.recentBlocks Contains all recent blocks, up to a total that is equal to this.historyLength
|
|
|
|
*
|
|
|
|
*/
|
2017-12-18 01:36:03 +01:00
|
|
|
constructor (opts = {}) {
|
2018-01-12 19:25:36 +01:00
|
|
|
const { blockTracker, provider } = opts
|
2017-12-18 01:36:03 +01:00
|
|
|
this.blockTracker = blockTracker
|
2018-01-12 19:25:36 +01:00
|
|
|
this.ethQuery = new EthQuery(provider)
|
2017-12-18 01:36:03 +01:00
|
|
|
this.historyLength = opts.historyLength || 40
|
|
|
|
|
|
|
|
const initState = extend({
|
|
|
|
recentBlocks: [],
|
|
|
|
}, opts.initState)
|
|
|
|
this.store = new ObservableStore(initState)
|
|
|
|
|
|
|
|
this.blockTracker.on('block', this.processBlock.bind(this))
|
2018-01-12 00:00:48 +01:00
|
|
|
this.backfill()
|
2017-12-18 01:36:03 +01:00
|
|
|
}
|
|
|
|
|
2018-04-19 17:38:56 +02:00
|
|
|
/**
|
|
|
|
* Sets store.recentBlocks to an empty array
|
|
|
|
*
|
|
|
|
*/
|
2017-12-18 01:36:03 +01:00
|
|
|
resetState () {
|
|
|
|
this.store.updateState({
|
|
|
|
recentBlocks: [],
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2018-04-19 17:38:56 +02:00
|
|
|
/**
|
|
|
|
* Receives a new block and modifies it with this.mapTransactionsToPrices. Then adds that block to the recentBlocks
|
|
|
|
* array in storage. If the recentBlocks array contains the maximum number of blocks, the oldest block is removed.
|
|
|
|
*
|
|
|
|
* @param {object} newBlock The new block to modify and add to the recentBlocks array
|
|
|
|
*
|
|
|
|
*/
|
2017-12-18 01:36:03 +01:00
|
|
|
processBlock (newBlock) {
|
2018-01-12 00:00:48 +01:00
|
|
|
const block = this.mapTransactionsToPrices(newBlock)
|
2017-12-19 21:22:48 +01:00
|
|
|
|
2017-12-18 01:36:03 +01:00
|
|
|
const state = this.store.getState()
|
2017-12-19 21:22:48 +01:00
|
|
|
state.recentBlocks.push(block)
|
2017-12-18 01:36:03 +01:00
|
|
|
|
|
|
|
while (state.recentBlocks.length > this.historyLength) {
|
|
|
|
state.recentBlocks.shift()
|
|
|
|
}
|
|
|
|
|
|
|
|
this.store.updateState(state)
|
|
|
|
}
|
2018-01-12 00:00:48 +01:00
|
|
|
|
2018-04-19 17:38:56 +02:00
|
|
|
/**
|
|
|
|
* Receives a new block and modifies it with this.mapTransactionsToPrices. Adds that block to the recentBlocks
|
|
|
|
* array in storage, but only if the recentBlocks array contains fewer than the maximum permitted.
|
|
|
|
*
|
|
|
|
* Unlike this.processBlock, backfillBlock adds the modified new block to the beginning of the recent block array.
|
|
|
|
*
|
|
|
|
* @param {object} newBlock The new block to modify and add to the beginning of the recentBlocks array
|
|
|
|
*
|
|
|
|
*/
|
2018-01-12 00:00:48 +01:00
|
|
|
backfillBlock (newBlock) {
|
|
|
|
const block = this.mapTransactionsToPrices(newBlock)
|
|
|
|
|
|
|
|
const state = this.store.getState()
|
|
|
|
|
|
|
|
if (state.recentBlocks.length < this.historyLength) {
|
|
|
|
state.recentBlocks.unshift(block)
|
|
|
|
}
|
|
|
|
|
|
|
|
this.store.updateState(state)
|
|
|
|
}
|
|
|
|
|
2018-04-19 17:38:56 +02:00
|
|
|
/**
|
|
|
|
* Receives a block and gets the gasPrice of each of its transactions. These gas prices are added to the block at a
|
|
|
|
* new property, and the block's transactions are removed.
|
|
|
|
*
|
|
|
|
* @param {object} newBlock The block to modify. It's transaction array will be replaced by a gasPrices array.
|
|
|
|
* @returns {object} The modified block.
|
|
|
|
*
|
|
|
|
*/
|
2018-01-12 00:00:48 +01:00
|
|
|
mapTransactionsToPrices (newBlock) {
|
|
|
|
const block = extend(newBlock, {
|
|
|
|
gasPrices: newBlock.transactions.map((tx) => {
|
|
|
|
return tx.gasPrice
|
|
|
|
}),
|
|
|
|
})
|
|
|
|
delete block.transactions
|
|
|
|
return block
|
|
|
|
}
|
|
|
|
|
2018-04-19 17:38:56 +02:00
|
|
|
/**
|
|
|
|
* On this.blockTracker's first 'block' event after this RecentBlocksController's instantiation, the store.recentBlocks
|
|
|
|
* array is populated with this.historyLength number of blocks. The block number of the this.blockTracker's first
|
|
|
|
* 'block' event is used to iteratively generate all the numbers of the previous blocks, which are obtained by querying
|
|
|
|
* the blockchain. These blocks are backfilled so that the recentBlocks array is ordered from oldest to newest.
|
|
|
|
*
|
|
|
|
* Each iteration over the block numbers is delayed by 100 milliseconds.
|
|
|
|
*
|
|
|
|
* @returns {Promise<void>} Promises undefined
|
|
|
|
*/
|
2018-01-12 00:00:48 +01:00
|
|
|
async backfill() {
|
|
|
|
this.blockTracker.once('block', async (block) => {
|
|
|
|
let blockNum = block.number
|
|
|
|
let recentBlocks
|
|
|
|
let state = this.store.getState()
|
|
|
|
recentBlocks = state.recentBlocks
|
|
|
|
|
|
|
|
while (recentBlocks.length < this.historyLength) {
|
|
|
|
try {
|
|
|
|
let blockNumBn = new BN(blockNum.substr(2), 16)
|
|
|
|
const newNum = blockNumBn.subn(1).toString(10)
|
|
|
|
const newBlock = await this.getBlockByNumber(newNum)
|
|
|
|
|
|
|
|
if (newBlock) {
|
|
|
|
this.backfillBlock(newBlock)
|
|
|
|
blockNum = newBlock.number
|
|
|
|
}
|
|
|
|
|
|
|
|
state = this.store.getState()
|
|
|
|
recentBlocks = state.recentBlocks
|
|
|
|
} catch (e) {
|
|
|
|
log.error(e)
|
|
|
|
}
|
|
|
|
await this.wait()
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2018-04-19 17:38:56 +02:00
|
|
|
/**
|
|
|
|
* A helper for this.backfill. Provides an easy way to ensure a 100 millisecond delay using await
|
|
|
|
*
|
|
|
|
* @returns {Promise<void>} Promises undefined
|
|
|
|
*
|
|
|
|
*/
|
2018-01-12 00:00:48 +01:00
|
|
|
async wait () {
|
|
|
|
return new Promise((resolve) => {
|
|
|
|
setTimeout(resolve, 100)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2018-04-19 17:38:56 +02:00
|
|
|
/**
|
|
|
|
* Uses EthQuery to get a block that has a given block number.
|
|
|
|
*
|
|
|
|
* @param {number} number The number of the block to get
|
|
|
|
* @returns {Promise<object>} Promises A block with the passed number
|
|
|
|
*
|
|
|
|
*/
|
2018-01-12 00:00:48 +01:00
|
|
|
async getBlockByNumber (number) {
|
|
|
|
const bn = new BN(number)
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
this.ethQuery.getBlockByNumber('0x' + bn.toString(16), true, (err, block) => {
|
|
|
|
if (err) reject(err)
|
|
|
|
resolve(block)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2017-12-18 01:36:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = RecentBlocksController
|