【问题标题】:Websocket connection drops using Golang's gorilla/websocket package使用 Golang 的 gorilla/websocket 包的 Websocket 连接断开
【发布时间】:2018-08-28 15:21:16
【问题描述】:

我正在尝试使用gorilla/websocket 包将建立并维护与第三方服务器的 Websocket 连接的 NodeJS 脚本移植到 Go。在Node script 中,在 ping 之后收到一个 pong,并且连接无限期地保持活动状态。在Go script 中,ping/pong 有效,但连接在大约 30 秒后被服务器断开。

我怀疑使用 Go websocket 包发送的 ping 格式不正确,但我无法查明原因。在运行这些程序时比较捕获的加密网络流量表明 TCP 请求和响应的响应长度没有差异,因此这可能不是问题。任何帮助将不胜感激!

websocket.js

#!/usr/bin/env node

// npm install websocket@1.0.25 --save
const WebSocketClient = require('websocket').client;
const client = new WebSocketClient();

let lastPing = new Date().getTime();

client.on('connectFailed', function(error) {
    console.log('Connect Error: ' + error.toString());
});

client.on('connect', function(connection) {
    console.log('Connected to Server...');
    connection.on('error', function(error) {
        console.log("Connection Error: " + error.toString());
    });
    connection.on('close', function() {
        console.log('Connection Closed');
    });
    connection.on('message', function(message) {
      if (message.type === 'utf8') {
        console.log(message.utf8Data);
      }
    });
    connection.on('pong', function(){
      console.log('[pingpong] response took', (new Date().getTime() - lastPing) + 'ms');
    })

    function send(message) {
      if (connection.connected) {
          connection.sendUTF(message);
      }
    }

    // Send a ping every 10s
    // to keep the connection live
    setInterval(function(){
      lastPing = new Date().getTime();
      connection.ping();
    }, 10000);
});

client.connect('wss://ws.radarrelay.com/0x/v0/ws');

websocket.go

package main

import (
    "flag"
    "log"
    "os"
    "os/signal"
    "time"

    "github.com/gorilla/websocket"
)

var addr = "wss://api.radarrelay.com/0x/v0/ws"

func main() {
    flag.Parse()
    log.SetFlags(0)

    timeoutDuration := 2 * time.Minute

    interrupt := make(chan os.Signal, 1)
    signal.Notify(interrupt, os.Interrupt)

    c, _, err := websocket.DefaultDialer.Dial(addr, nil)
    if err != nil {
        log.Fatal("dial:", err)
    } else {
        log.Println("Connected to server")
    }

    c.SetPongHandler(func(str string) error {
        log.Println("pong received", str)
        return nil
    })

    defer c.Close()

    done := make(chan struct{})

    go func() {
        defer c.Close()
        defer close(done)
        for {
            c.SetReadDeadline(time.Now().Add(timeoutDuration))
            _, message, err := c.ReadMessage()
            if err != nil {
                log.Println("read:", err)
                return
            }
            if len(message) >= 2 {
                message = message[2:]
            }
            log.Printf("recv: %s", message)
        }
    }()

    ticker := time.NewTicker(10 * time.Second)
    defer ticker.Stop()

    for {
        select {
        case _ = <-ticker.C:
            err := c.WriteMessage(websocket.PingMessage, []byte{})
            if err != nil {
                log.Println("write:", err)
                return
            } else {
                log.Println("ping sent")
            }
        case <-interrupt:
            log.Println("interrupt")
            // To cleanly close a connection, a client should send a close
            // frame and wait for the server to close the connection.
            err := c.WriteMessage(
                websocket.CloseMessage,
                websocket.FormatCloseMessage(websocket.CloseNormalClosure, ""))
            if err != nil {
                log.Println("write close:", err)
                return
            }
            select {
            case <-done:
            case <-time.After(time.Second):
            }
            c.Close()
            return
        }
    }
}

【问题讨论】:

  • 让我们看一些代码。
  • @Peter 查看链接的要点。
  • @MaxGillett:始终在您的帖子中包含minimal reproducible example 的代码(请参阅How to Ask)。
  • 在此处粘贴该代码。网站规则规定代码必须包含在问题中,而不仅仅是链接到。
  • 因为你报告客户端收到pongs,看来pings必须格式正确。

标签: node.js go websocket


【解决方案1】:

两者的地址不同。节点代码正在与

wss://ws.radarrelay.com/0x/v0/ws

Go 代码正在与:

wss://api.radarrelay.com/0x/v0/ws

看起来它们解析到不同的 IP,因此它们可能具有不同的入口基础设施,并且 ws 子域上不存在的 api 子域上存在超时。

【讨论】:

  • 如果这是真正的问题,那么很好 :-)
猜你喜欢
  • 2015-05-03
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2011-05-20
  • 1970-01-01
  • 2017-08-30
  • 1970-01-01
相关资源
最近更新 更多