【问题标题】:How to make send message to particular URI after successfull webscoket connection?成功 websocket 连接后如何向特定 URL 发送消息?
【发布时间】:2020-02-10 13:20:08
【问题描述】:

我有一个运行在 localhost:443/server-demo 上的安全 websocket 服务器(jetty websocket 服务器)。 现在我正在编写一个可以与 websocket 服务器通信的 go 客户端。我能够使用正确的证书连接到 websocket 服务器。这是示例代码。

package main
import (
    "crypto/tls"
    "crypto/x509"
    "fmt"
    "io"
    "log"
)
func main() {
    cert, err := tls.LoadX509KeyPair("nifi-1.10.0-bin/nifi-1.10.0/extras/gen-certs/certs/admin.pem", "nifi-1.10.0-bin/nifi-1.10.0/extras/gen-certs/certs/admin-key.pem")
    if err != nil {
        log.Fatalf("server: loadkeys: %s", err)
    }
    config := tls.Config{Certificates: []tls.Certificate{cert}, InsecureSkipVerify: true}
    conn, err := tls.Dial("tcp", "127.0.0.1:443", &config)
    if err != nil {
        log.Fatalf("client: dial: %s", err)
    }
    defer conn.Close()
    log.Println("client: connected to: ", conn.RemoteAddr())
    state := conn.ConnectionState() 
    for _, v := range state.PeerCertificates {
        fmt.Println(x509.MarshalPKIXPublicKey(v.PublicKey))
        fmt.Println(v.Subject)
    }
    log.Println("client: handshake: ", state.HandshakeComplete)
    log.Println("client: mutual: ", state.NegotiatedProtocolIsMutual)
    message := "Hello\n"
    n, err := io.WriteString(conn, message)
    if err != nil {
        log.Fatalf("client: write: %s", err)
    }
    log.Printf("client: wrote %q (%d bytes)", message, n)
    reply := make([]byte, 256)
    n, err = conn.Read(reply)
    log.Printf("client: read %q (%d bytes)", string(reply[:n]), n)
    log.Print("client: exiting")
}

上面的代码抛出这个错误:

 "HTTP/1.1 400 No URI\r\nContent-Type: text/html;charset=iso-8859-1\r\nContent-Length: 49\r\nConnection: close\r\nServer: Jetty(9.4.19.v20190610)\r\n\r\n<h1>Bad Message 400</h1><pre>reason: No URI</pre>" (188 bytes)

我的问题是建立连接后如何向特定 URI 发送消息?即我想向wss://localhost:443/server-demo发送消息。

【问题讨论】:

    标签: go websocket ssl-certificate client-server gorilla


    【解决方案1】:

    问题中的代码没有建立到服务器的 WebSocket 连接。

    要建立 WebSocket 连接,应用程序必须向conn 写入 WebSocket 握手并接收握手响应。详情请见RFC

    大多数应用程序使用 websocket 包来处理所有这些细节。 gorilla/websocket 包是一个受欢迎的选择。

    这段代码应该让你开始使用大猩猩:

    cert, err := tls.LoadX509KeyPair("nifi-1.10.0-bin/nifi-1.10.0/extras/gen-certs/certs/admin.pem", "nifi-1.10.0-bin/nifi-1.10.0/extras/gen-certs/certs/admin-key.pem")
    if err != nil {
        log.Fatalf("server: loadkeys: %s", err)
    }
    config := tls.Config{Certificates: []tls.Certificate{cert}, InsecureSkipVerify: true}
    d := websocket.Dialer{
        TLSClientConfig: &config,
    }
    c, _, err := d.Dial("wss://localhost:443/server-demo", nil)
    if err != nil {
        log.Fatal(err)
    }
    
    defer c.Close()
    // Use `c` to send and receive messages
    

    【讨论】:

    • 必须重复...InsecureSkipVerify true 禁用 tls 检查,因此如果您认为您的代码是安全的,请在投入生产之前删除该属性...用类似 my_dialer.TLSConfig = &amp;tls.Config{ RootCAs: caCertPool, ServerName: domain_name, } 的东西替换它
    猜你喜欢
    • 2013-04-23
    • 1970-01-01
    • 2016-02-16
    • 1970-01-01
    • 1970-01-01
    • 2016-04-28
    • 1970-01-01
    • 2014-07-27
    • 1970-01-01
    相关资源
    最近更新 更多