使用websocket实现casper节点通信

Posted 复杂美干货日记

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了使用websocket实现casper节点通信相关的知识,希望对你有一定的参考价值。

  • 使用websocket实现casper节点通信

    • websocket 相比与 http 协议的优点

    • capser-go 项目中节点 websocket 通信的具体实现

      • server端

      • client端

在casper-go项目中,各个casper节点需要两两互相通信。这可以通过websocket实现。WebSocket是一种在单个TCP连接上进行全双工通讯的协议。websocket属于应用层协议。它基于TCP传输协议,并复用HTTP的握手通道。WebSocket通讯协定于2011年被IETF定为标准RFC 6455,并由RFC7936补充规范。WebSocket API也被W3C定为标准。

WebSocket使得客户端和服务器之间的数据交换变得更加简单,允许服务端主动向客户端推送数据。在WebSocket API中,浏览器和服务器只需要完成一次握手,两者之间就直接可以建立持久性的连接,并进行双向数据传输。

Websocket 相较于 http 协议的优点

较少的控制开销。在连接建立后,服务器和客户端之间交换数据时,用于协议控制的数据包头部相对较小。在不包含扩展的情况下,对于服务器到客户端的内容,此头部大小只有2至10字节(和数据包长度有关);对于客户端到服务器的内容,此头部还需要加上额外的4字节的掩码。相对于HTTP请求每次都要携带完整的头部,此项开销显著减少了。

更强的实时性。由于协议是全双工的,所以服务器可以随时主动给客户端下发数据。相对于HTTP请求需要等待客户端发起请求服务端才能响应,延迟明显更少;即使是和Comet等类似的长轮询⽐较,其也能在短时间内更多次地传递数据。

保持连接状态。于HTTP不同的是,Websocket需要先建立连接,这就使得其成为一种有状态的协议,之后通信时可以省略部分状态信息。而HTTP请求可能需要在每个请求都携带状态信息(如身份认证等)。更好的二进制支持。Websocket定义了二进制帧,相对HTTP,可以更轻松地处理二进制内容。

可以支持扩展。Websocket定义了扩展,用户可以扩展协议、实现部分自定义的子协议。如部分浏览器支持压缩等。

更好的压缩效果。相对于HTTP压缩,Websocket在适当的扩展支持下,可以沿用之前内容的上下文,在传递类似的数据时,可以显著地提高压缩率。

capser-go项目中节点websocket通信的具体实现

Go官方的标准包里面提供了一个WebSocket的包golang.org/x/net/websocket,但也说的很明确,这个包里面并没有实现WebSocket协议规定的一些特性,而推荐使用github.com/gorilla/websocket这个包。所以实际中,我们如果使用到了Go的WebSocket也一般使用后者,项目也使用后者。这个包的内容不是很多,而且README里面对于这个包的使用也说的比较清楚和详细,所以这里就不再做翻译工作了。

项目每个节点既要发消息,又要收消息。因此每个节点既是server端,又是client端。给其它节点发消息通过client端实现,收消息通过server端实现。

//TODO 其实原先的设想是不分server端、client端,收发消息都在一个端完成。或者收发消息只在server端或client端完成。由于没有比较简单的实现方式,因此最终选择上面那种⽐较简单粗暴的方式实现。如果有哪位大神知道的话,指点一波。

server端

http.HandleFunc("/", receive) log.Fatal(http.ListenAndServe("localhost:3001", nil))

如上代码所示,使当前的server监听本机的3001端口,此处端口可自行设置。当client端发出请求时,server端定义了相应的处理函数receive,以下是receive函数。

func receive(w http.ResponseWriter, r *http.Request) {    conn, err := upgrader.Upgrade(w, r, nil)    /* server端收到client端的请求后,使用此函数(receive函数)处理该
     请求。通过upgrade得到长连接conn。这个长连接conn是属于server端的,
     主要是用来收其它节点的消息的。发消息请见下面的client端。*/
   if err != nil {        return    }    defer conn.Close()    for {        _, message, err := conn.ReadMessage()        if err != nil {            break        }    //消息处理代码, 省略    } }

client端

显然,每个节点既要发消息,又要收消息。在高超的实现中,收消息是通过server端实现,而发消息是通过client端实现。每个节点既是client端,又是server端。

// 开一个goroutine去连接其他节点并发送消息go func() {    // 持续发起连接,直到连接成功    for {        f = false        for i := 0; i < num; i++ {            if i != num_index {                conn[i], _, err[i] = websocket.DefaultDialer.Dial(addr[i], nil)            }        }        // 判断是否与其他节点都连上,省略    }    if connected == true { // 成功建立连接        // 发消息逻辑,省略    } }()


以上是关于使用websocket实现casper节点通信的主要内容,如果未能解决你的问题,请参考以下文章

使用WebSocket帮助应用程序群集节点间通信

以太坊性能优化:分片技术雷电网络Casper-下一代以太坊共识协议

Netty中使用WebSocket实现服务端与客户端的长连接通信发送消息

python实现建立websocket通信

websocket通信 实现java模拟一个client与webclient通信

使用node+vue实现简单的WebSocket聊天功能