https://github.com/MetaMask/eth-block-tracker

A JS module for keeping track of the latest Ethereum block by polling an ethereum provider.

通过拉取以太坊的provider来跟踪最新的以太坊区块

eth-block-tracker

This module walks the Ethereum blockchain, keeping track of the latest block. It uses a web3 provider as a data source and will continuously poll for the next block.

这个模块沿着以太坊区块链去跟踪最新的区块。它使用web3 provider作为其数据源,并不停地从最新的区块中拉取数据

const HttpProvider = require('ethjs-provider-http')
const PollingBlockTracker = require('eth-block-tracker') const provider = new HttpProvider('https://mainnet.infura.io')//设置provider
const blockTracker = new PollingBlockTracker({ provider })
blockTracker.on('latest', console.log)
 

methods

new PollingBlockTracker({ provider, pollingInterval, retryTimeout, keepEventLoopActive })

creates a new block tracker with provider as a data source and pollingInterval (ms) timeout between polling for the latest block. If an Error is encountered when fetching blocks, it will wait retryTimeout (ms) before attempting again. If keepEventLoopActive is false, in Node.js it will unref the polling timeout, allowing the process to exit during the polling interval. defaults to true, meaning the process will be kept alive.

创建了一个指定provider数据源的区块追踪器,并在pollingInterval的间隔时间内拉取最新区块。当获取区块的时候如果出现了错误,那么她就会等待retryTimeout (ms)再重新获取。如果keepEventLoopActive是false,在nodejs中它会取消轮询超时,即在拉取的间隔中允许进程退出。默认是true,意味着进程始终保持活跃

getCurrentBlock()

synchronous returns the current block. may be null.

同步等到当前的区块

console.log(blockTracker.getCurrentBlock())
async getLatestBlock()

Asynchronously returns the latest block. if not immediately available, it will fetch one.

异步得到最新区块

async checkForLatestBlock()

Tells the block tracker to ask for a new block immediately, in addition to its normal polling interval. Useful if you received a hint of a new block (e.g. via tx.blockNumber from getTransactionByHash). Will resolve to the new latest block when its done polling.

要求区块追踪器马上去取新的区块,除非现在正处于正常的polling间隔。如果你收到了新的区块的暗示时这个功能就十分有用(via tx.blockNumber from getTransactionByHash)。当其拉取完后,将释放最新的区块

EVENTS

latest

The latest event is emitted for whenever a new latest block is detected. This may mean skipping blocks if there were two created since the last polling period.

blockTracker.on('latest', (newBlock) => console.log(newBlock))

当新的区块被检测到时,这个事件就会被触发。如果在上一个拉取周期中生成了两个区块,那么将会跳过,不会触发事件

sync

The sync event is emitted the same as "latest" but includes the previous block.

blockTracker.on('sync', ({ newBlock, oldBlock }) => console.log(newBlock, oldBlock))

它跟"latest"是相同的,不同在与它还返回了之前的一个区块

error

The error event means an error occurred while polling for the latest block.

blockTracker.on('error', (err) => console.error(err))

当拉取最新的区块的时候出现错误时触发

代码:

eth-block-tracker/package.json

  "main": "src/polling.js",
"scripts": {
"test": "npm run build && node test/index.js",
"prepublish": "npm run build",
"build": "mkdir -p ./dist && npm run bundle",
"bundle": "babel src -d dist/es5/ && npm run bundle-polling && npm run bundle-base",
"bundle-polling": "browserify -s PollingBlockTracker -e src/polling.js -t [ babelify --presets [ es2015 ] ] > dist/PollingBlockTracker.js",
"bundle-base": "browserify -s BaseBlockTracker -e src/base.js -t [ babelify --presets [ es2015 ] ] > dist/BaseBlockTracker.js"
},

eth-block-tracker/example.js

const createInfuraProvider = require('eth-json-rpc-infura/src/createProvider')
const PollingBlockTracker = require('./src/polling') const provider = createInfuraProvider({ network: 'mainnet' })//使用infura的mainnet网生成provider
const blockTracker = new PollingBlockTracker({ provider }) //生成blockTracker blockTracker.on('sync', ({ newBlock, oldBlock }) => {//监听sync事件
if (oldBlock) {
console.log(`sync #${Number(oldBlock)} -> #${Number(newBlock)}`)
} else {
console.log(`first sync #${Number(newBlock)}`)
}
})

eth-block-tracker/src/polling.js

const EthQuery = require('eth-query')
const EventEmitter = require('events')
const pify = require('pify')
const BaseBlockTracker = require('./base') const sec =
const min = * sec class PollingBlockTracker extends BaseBlockTracker {//1 继承自BaseBlockTracker constructor(opts = {}) {
// parse + validate args
if (!opts.provider) throw new Error('PollingBlockTracker - no provider specified.')//2 必须要传入provider
const pollingInterval = opts.pollingInterval || * sec //3 设置poll间隔时间
const retryTimeout = opts.retryTimeout || pollingInterval / 10 //4 设置出错后的重试时间
const keepEventLoopActive = opts.keepEventLoopActive !== undefined ? opts.keepEventLoopActive : true //5进程是否保持活跃
// BaseBlockTracker constructor
super(Object.assign({//6 将opts复制到前一个参数数组中,并作为BaseBlockTracker参数传给其构造函数
blockResetDuration: pollingInterval,
}, opts))
// config,配置相应的信息
this._provider = opts.provider
this._pollingInterval = pollingInterval
this._retryTimeout = retryTimeout
this._keepEventLoopActive = keepEventLoopActive
// util
this._query = new EthQuery(this._provider)
} //
// public
//
//公有函数
// trigger block polling
async checkForLatestBlock() {
await this._updateLatestBlock()
return await this.getLatestBlock() //调用BaseBlockTracker的函数getLatestBlock
} //
// private
//
//下面为私有函数
_start() {
this._performSync().catch(err => this.emit('error', err))
} async _performSync () {
while (this._isRunning) {
try {
await this._updateLatestBlock()
await timeout(this._pollingInterval, !this._keepEventLoopActive)
} catch (err) {
const newErr = new Error(`PollingBlockTracker - encountered an error while attempting to update latest block:\n${err.stack}`)
try {
this.emit('error', newErr)
} catch (emitErr) {
console.error(newErr)
}
await timeout(this._retryTimeout, !this._keepEventLoopActive)
}
}
} async _updateLatestBlock () {
// fetch + set latest block
const latestBlock = await this._fetchLatestBlock()
this._newPotentialLatest(latestBlock)
} async _fetchLatestBlock () {
return await pify(this._query.blockNumber).call(this._query)
} } module.exports = PollingBlockTracker function timeout (duration, unref) {
return new Promise(resolve => {
const timoutRef = setTimeout(resolve, duration)
// don't keep process open
if (timoutRef.unref && unref) {
timoutRef.unref()
}
})
}

eth-block-tracker/src/base.js

const EthQuery = require('eth-query')
const pify = require('pify')
const SafeEventEmitter = require('safe-event-emitter') const sec = const calculateSum = (accumulator, currentValue) => accumulator + currentValue
const blockTrackerEvents = ['sync', 'latest'] class BaseBlockTracker extends SafeEventEmitter { //
// public
// constructor(opts = {}) {//6
super()
// config
this._blockResetDuration = opts.blockResetDuration || * sec //拉取的间隔时间
// state
this._blockResetTimeout
this._currentBlock = null
this._isRunning = false
// bind functions for internal use
this._onNewListener = this._onNewListener.bind(this)
this._onRemoveListener = this._onRemoveListener.bind(this)
this._resetCurrentBlock = this._resetCurrentBlock.bind(this)
// listen for handler changes
this._setupInternalEvents()
} isRunning() {
return this._isRunning
} getCurrentBlock () {
return this._currentBlock
} async getLatestBlock () {
// return if available
if (this._currentBlock) return this._currentBlock
// wait for a new latest block
const latestBlock = await new Promise(resolve => this.once('latest', resolve))
// return newly set current block
return latestBlock
} // dont allow module consumer to remove our internal event listeners
removeAllListeners(eventName) {
// perform default behavior, preserve fn arity
if (eventName) {
super.removeAllListeners(eventName)
} else {
super.removeAllListeners()
}
// re-add internal events
this._setupInternalEvents()
// trigger stop check just in case
this._onRemoveListener()
} //
// to be implemented in subclass
// _start () {
// default behavior is noop
} _end () {
// default behavior is noop
} //
// private
// _setupInternalEvents () {
// first remove listeners for idempotence
this.removeListener('newListener', this._onNewListener)
this.removeListener('removeListener', this._onRemoveListener)
// then add them
this.on('newListener', this._onNewListener)
this.on('removeListener', this._onRemoveListener)
} _onNewListener (eventName, handler) {
// `newListener` is called *before* the listener is added
if (!blockTrackerEvents.includes(eventName)) return
this._maybeStart()
} _onRemoveListener (eventName, handler) {
// `removeListener` is called *after* the listener is removed
if (this._getBlockTrackerEventCount() > ) return
this._maybeEnd()
} _maybeStart () {
if (this._isRunning) return
this._isRunning = true
// cancel setting latest block to stale
this._cancelBlockResetTimeout()
this._start()
} _maybeEnd () {
if (!this._isRunning) return
this._isRunning = false
this._setupBlockResetTimeout()
this._end()
} _getBlockTrackerEventCount () {
return blockTrackerEvents
.map(eventName => this.listenerCount(eventName))
.reduce(calculateSum)
} _newPotentialLatest (newBlock) {
const currentBlock = this._currentBlock
// only update if blok number is higher
if (currentBlock && (hexToInt(newBlock) <= hexToInt(currentBlock))) return
this._setCurrentBlock(newBlock)
} _setCurrentBlock (newBlock) {
const oldBlock = this._currentBlock
this._currentBlock = newBlock
this.emit('latest', newBlock)
this.emit('sync', { oldBlock, newBlock })
} _setupBlockResetTimeout() {
// clear any existing timeout
this._cancelBlockResetTimeout()
// clear latest block when stale
this._blockResetTimeout = setTimeout(this._resetCurrentBlock, this._blockResetDuration)
// nodejs - dont hold process open
if (this._blockResetTimeout.unref) {
this._blockResetTimeout.unref()
}
} _cancelBlockResetTimeout() {
clearTimeout(this._blockResetTimeout)
} _resetCurrentBlock () {
this._currentBlock = null
} } module.exports = BaseBlockTracker function hexToInt(hexInt) {
return Number.parseInt(hexInt, )
}
 

MetaMask/eth-block-tracker的更多相关文章

  1. MetaMask/provider-engine-2-代码

    package.json "main": "index.js", "scripts": { "test": " ...

  2. metamask源码学习-controllers-network

    https://github.com/MetaMask/metamask-extension/tree/master/app/scripts/controllers/network metamask- ...

  3. Spark Streaming 架构

    图 1   Spark Streaming 架构图 组件介绍:‰ Network Input Tracker : 通 过 接 收 器 接 收 流 数 据, 并 将 流 数 据 映 射 为 输 入DSt ...

  4. Spark应用程序的运行框架

    几个基本概念: (1)job:包含多个task组成的并行计算,往往由action催生. (2)stage:job的调度单位. (3)task:被送到某个executor上的工作单元. (4)taskS ...

  5. spark基础知识介绍2

    dataframe以RDD为基础的分布式数据集,与RDD的区别是,带有Schema元数据,即DF所表示的二维表数据集的每一列带有名称和类型,好处:精简代码:提升执行效率:减少数据读取; 如果不配置sp ...

  6. Spark应用程序的运行架构几种说

    (1)简单的说: 由driver向集群申请资源,集群分配资源,启动executor.driver将spark应用程序的代码和文件传送给executor.executor上运行task,运行完之后将结果 ...

  7. metamask源码学习-controller-transaction

    ()metamask-extension/app/scripts/controllers/transactions Transaction Controller is an aggregate of ...

  8. ganache与metamask

    1.其实ganache其实就相当于一个私有链ganache安装,这个是图形化界面的: 2.(testRpc跟他其实是一个用处,有一个即可,只不过testRpc是非图形化界面.要注意两者都仅运行在內存中 ...

  9. metamask源码学习-metamask-controller.js

    The MetaMask Controller——The central metamask controller. Aggregates other controllers and exports a ...

随机推荐

  1. 使用commons-pool2实现FTP连接池

    ​ GitHub : https://github.com/jayknoxqu/ftp-pool 一. 连接池概述 ​ 频繁的建立和关闭连接,会极大的降低系统的性能,而连接池会在初始化的时候会创建一定 ...

  2. Java面试中遇到的坑【填坑篇】

    看到大家对上篇<Java面试中遇到的坑>一文表现出强力的关注度,说明大家确实在面试中遇到了类似的难题.大家在文章留言处积极留言探讨面试中遇到的问题,其中几位同学还提出了自己的见解,我感到非 ...

  3. DOM相关

    归纳一下, 不管是DOM Core还是HTML-DOM,我们在使用JavaScript的时候要注意浏览器之间的兼容性,因为不同的浏览器对这两类方法和属性的支持可能不一样,一般推荐使用DOM Core方 ...

  4. linux系统编程:read,write与lseek的综合应用

    这个实例根据命令行参数进行相应的读学操作: 用法: usage:./io file {r<length>|R<length>|w<string>|s<offs ...

  5. python学习之老男孩python全栈第九期_day009知识点总结

    '''# len# 计算字符串的长度# s = '金老板小护士'# len(s)# 不能用 len 怎么办#low一点的方法# count = 0# for i in s:# count += 1# ...

  6. webpack打包小图片时进行Base64转码

    关于base64 优点: base64就是一串字符串码表示的图片,在加载页面和js时一块加载出来,减少了加载图片时的http请求.加载一张图片时会发起一次http请求,http请求每次建立都会需要一定 ...

  7. HttpClient的用法

    客户端模拟http请求工具 Postmen(谷歌插件).RestClient 服务器模拟http请求工具 httpclient.HttpURLConnection httpCient请求代码 /** ...

  8. kvm 创建新虚拟机命virt-install 使用说明

    virt-install 命令说明 1.命令作用      建立(provision)新虚拟机   2.语法   virt-install [选项]... 3.说明(DESCRIPTION)   vi ...

  9. Python 自定义线程池

    """思路1,将任务放在队列 1)创建队列:(初始化) 2)设置大小,线程池的最大容量 3)真实创建的线程 列表 4)空闲的线程数量 2,着手开始处理任务 1)创建线程 ...

  10. Linux awk命令常见使用方法介绍

    Linux awk命令常见使用方法介绍 By:授客 QQ:1033553122   awk运行方式有三种,其中常用的为命令行方式 awk [-F  field_separator]  '{patter ...