https://github.com/MetaMask/json-rpc-middleware-stream/blob/master/test/index.js#L20

A small toolset for streaming json rpc and matching requests and responses. Made to be used with json-rpc-engine.

可以用来与json-rpc-engine结合使用的,对输入的json rpc进行读入、写出处理

json-rpc-middleware-stream/index.js

const SafeEventEmitter = require('safe-event-emitter')
const DuplexStream = require('readable-stream').Duplex //即一个可读且可写的流 module.exports = createStreamMiddleware function createStreamMiddleware() {
const idMap = {}
const stream = new DuplexStream({
objectMode: true, //输入的是任意形式的数据,而不非是字符串和 Buffer(或 Uint8Array
read: readNoop, //下面定义的返回false的函数,会覆写_read(),当要从别地流处读出时调用
write: processMessage,//会覆写_write(),当要写入别的流时调用
}) const events = new SafeEventEmitter() const middleware = (req, res, next, end) => {
// write req to stream
stream.push(req)//从req流中读出,就会去调用readNoop函数
// register request on id map
idMap[req.id] = { req, res, next, end }//并在数组中将req根据其id记录下来
} return { events, middleware, stream } function readNoop () {
return false
} function processMessage (res, encoding, cb) {//当要写出时调用
let err
try {
const isNotification = !res.id //如果res.id有值,则isNotification为false;否则为true
if (isNotification) {//res.id没值或为0
processNotification(res)//触发事件'notification'
} else {
processResponse(res)//res.id有值
}
} catch (_err) {
err = _err
}
// continue processing stream
cb(err)
} function processResponse(res) {//将流中内容写出到res流
const context = idMap[res.id] //查看有没有与相应的res.id相同的ID的流之前读入过
if (!context) throw new Error(`StreamMiddleware - Unknown response id ${res.id}`) //如果context为空,则说明相应的id的流并没有读入过,这样写出的时候就不知道要怎么写出了,无end,所以会出错
delete idMap[res.id] //如果有读入过,则写出前先清空idMap中的相应内容
// copy whole res onto original res
Object.assign(context.res, res) //然后将现在要写出到的res流覆写context.res,并返回context.res
// run callback on empty stack,
// prevent internal stream-handler from catching errors
setTimeout(context.end) //调用之前读入时写好的end函数来结束写出操作
} function processNotification(res) {//该事件的监听会在inpage-provider处设置
events.emit('notification', res)
} }
Object.assign(target, ...sources):用于将所有可枚举属性的值从一个或多个源对象复制到目标对象。它将返回目标对象。

json-rpc-middleware-stream/engineStream.js

const DuplexStream = require('readable-stream').Duplex

module.exports = createEngineStream

function createEngineStream({ engine }) {//engine即RpcEngine
if (!engine) throw new Error('Missing engine parameter!')
const stream = new DuplexStream({ objectMode: true, read, write })
// forward notifications
if (engine.on) {
engine.on('notification', (message) => {//监听'notification'事件
stream.push(message) //事件被触发的话就将message数据读入stream,调用read函数
})
}
return stream function read () {
return false
}
function write (req, encoding, cb) {//当写出时调用该函数
engine.handle(req, (err, res) => {
this.push(res)
})
cb()
}
}

测试:

json-rpc-middleware-stream/test/index.js

const test = require('tape')
const RpcEngine = require('json-rpc-engine')
const createJsonRpcStream = require('../index')
const createEngineStream = require('../engineStream') test('middleware - raw test', (t) => { const jsonRpcConnection = createJsonRpcStream()
const req = { id: , jsonrpc: '2.0', method: 'test' }
const initRes = { id: , jsonrpc: '2.0' }
const res = { id: , jsonrpc: '2.0', result: 'test' } // listen for incomming requests
jsonRpcConnection.stream.on('data', (_req) => {//监听data事件
t.equal(req, _req, 'got the expected request')//说明触发data事件传来的
jsonRpcConnection.stream.write(res)//将流中的与res.id相同的数据写出
}) // run middleware, expect end fn to be called
jsonRpcConnection.middleware(req, initRes, () => {//将req流写入createJsonRpcStream
t.fail('should not call next')
}, (err) => {
t.notOk(err, 'should not error')
t.deepEqual(initRes, res, 'got the expected response')
t.end()
}) }) test('engine to stream - raw test', (t) => { const engine = new RpcEngine()
engine.push((req, res, next, end) => {
res.result = 'test'
end()
}) const stream = createEngineStream({ engine })
const req = { id: , jsonrpc: '2.0', method: 'test' }
const res = { id: , jsonrpc: '2.0', result: 'test' } // listen for incomming requests
stream.on('data', (_res) => {
t.deepEqual(res, _res, 'got the expected response')
t.end()
}) stream.on('error', (err) => {
t.fail(error.message)
}) stream.write(req) }) test('middleware and engine to stream', (t) => {//上面两者的结合 // create guest
const engineA = new RpcEngine()
const jsonRpcConnection = createJsonRpcStream()
engineA.push(jsonRpcConnection.middleware) // create host
const engineB = new RpcEngine()
engineB.push((req, res, next, end) => {
res.result = 'test'
end()
}) // connect both
const clientSideStream = jsonRpcConnection.stream
const hostSideStream = createEngineStream({ engine: engineB })
clientSideStream
.pipe(hostSideStream)
.pipe(clientSideStream) // request and expected result
const req = { id: , jsonrpc: '2.0', method: 'test' }
const res = { id: , jsonrpc: '2.0', result: 'test' } engineA.handle(req, (err, _res) => {//req调用jsonRpcConnection.middleware读入clientSideStream,然后hostSideStream从clientSideStream中读入req数据,然后调用engineB的方法写出,所以得到的result: 'test'
t.notOk(err, 'does not error')
t.deepEqual(res, _res, 'got the expected response')
t.end()
}) }) test('server notification', (t) => {
t.plan() const jsonRpcConnection = createJsonRpcStream()
const notif = { jsonrpc: '2.0', method: 'test_notif' }//这里没有设置id,所以调用write所以会触发processNotification函数,触发'notification'事件 jsonRpcConnection.events.once('notification', (message) => {
t.equals(message.method, notif.method)
t.end()
}) // receive notification
jsonRpcConnection.stream.write(notif)
}) test('server notification in stream', (t) => {
const engine = new RpcEngine() const stream = createEngineStream({ engine })
const notif = { jsonrpc: '2.0', method: 'test_notif' } // listen for incomming requests
stream.once('data', (_notif) => {
t.deepEqual(notif, _notif, 'got the expected notification')
t.end()
}) stream.on('error', (err) => {
t.fail(error.message)
}) engine.emit('notification', notif)//将触发createEngineStream中的'notification'事件,notif将被读入stream,将触发data事件
})

MetaMask/json-rpc-middleware-stream的更多相关文章

  1. 測试JSON RPC远程调用(JSONclient)

    #include <string> #include <iostream> #include <curl/curl.h> /* 标题:JSonclient Auth ...

  2. MetaMask/metamask-inpage-provider

    https://github.com/MetaMask/metamask-inpage-provider Used to initialize the inpage ethereum provider ...

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

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

  4. metamask源码学习-inpage.js

    The most confusing part about porting MetaMask to a new platform is the way we provide the Web3 API ...

  5. Guide to Porting MetaMask to a New Environment

    https://github.com/MetaMask/metamask-extension/blob/develop/docs/porting_to_new_environment.md MetaM ...

  6. 【.NET Core项目实战-统一认证平台】第十六章 网关篇-Ocelot集成RPC服务

    [.NET Core项目实战-统一认证平台]开篇及目录索引 一.什么是RPC RPC是"远程调用(Remote Procedure Call)"的一个名称的缩写,并不是任何规范化的 ...

  7. metamask源码学习-contentscript.js

    When a new site is visited, the WebExtension creates a new ContentScript in that page's context, whi ...

  8. MetaMask/json-rpc-engine

    https://github.com/MetaMask/json-rpc-engine RpcEngine——MetaMask/json-rpc-engine https://github.com/M ...

  9. 以太坊RPC机制与API实例

    上一篇文章介绍了以太坊的基础知识,我们了解了web3.js的调用方式是通过以太坊RPC技术,本篇文章旨在研究如何开发.编译.运行与使用以太坊RPC接口. 关键字:以太坊,RPC,JSON-RPC,cl ...

随机推荐

  1. Extjs 项目中常用的小技巧,也许你用得着(2)

    接着来,也是刚刚遇到的 panel怎么进行收缩 collapsible: true, 这会panel就会出现这个 点这个就可以收缩了 panel怎么随便拉伸,也就是让那个小黑三角出现 split: t ...

  2. 解决默写浏览器中点击input输入框时,placeholder的值不消失的方法

    html中,placeholder作为input的一个属性,起到了在输入框中占位并提示的作用. 但是有一些浏览器,如chrome,当鼠标点击输入框时,placeholder的值不消失,只有输入数据才消 ...

  3. Java基础——Oracle(六)

    一.数据字典和动态性能视图 数据字典: oracle中的重要组成部分,提供了数据库的一些系统信息,记录了数据库的系统信息,它是只读表和视图的集合,数据字典的所有者为 sys 用户.用户只能在数据字典上 ...

  4. Cylinder Candy(积分+体积+表面积+旋转体)

    Cylinder Candy Time Limit: 2 Seconds Memory Limit: 65536 KB Special Judge Edward the confectioner is ...

  5. 【作业三】结队任务二-----CourseManagement

    031302517 031302319 ps:共同完成一篇随笔,文章中的第一人称我(517),队友(319) 一.功能分析+实现思路+结队讨论 这里我将功能分析和实现思路还有结对过程中的一些讨论结合在 ...

  6. jQ效果:jQuery时间轴插件jQuery Timelinr

    前言 这是一款可用于展示历史和计划的时间轴插件,尤其比较适合一些网站展示发展历程.大事件等场景.该插件基于jQuery,可以滑动切换.水平和垂直滚动.支持键盘方向键.经过扩展后可以支持鼠标滚轮事件. ...

  7. CSS总结div中的内容垂直居中的五种方法

    一.行高(line-height)法 如果要垂直居中的只有一行或几个文字,那它的制作最为简单,只要让文字的行高和容器的高度相同即可,比如: p { height:30px; line-height:3 ...

  8. HTML常用标签及属性

    标签格式 格式: 双边:<标签名 属性1="值1" 属性2='值2' 属性3=值3>内容</标签名> 单边:<标签名 属性1="值1&quo ...

  9. js发送请求

    1.Chrome控制台中 net::ERR_CONNECTION_REFUSED js频繁发送请求,有可能连接被拒绝,可用setTimeout,过几秒发送,给个缓冲时间 var overlayAnal ...

  10. http-server服务跨域设置

    http-server --cors -p 9999 ------------------------------------------------------------------------- ...