go - 成功的websocket连接后如何使发送消息到特定的URL?

标签 go websocket ssl-certificate client-server gorilla

我有一个在本地主机上运行的安全websocket服务器:443/server-demo(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发送消息。

最佳答案

问题中的代码未建立与服务器的WebSocket连接。

要建立WebSocket连接,应用程序必须将WebSocket握手写入conn并接收握手响应。有关详细信息,请参见RFC

大多数应用程序都使用websocket软件包,而不是处理所有这些详细信息。 gorilla/websocket软件包是一个流行的选择。

这段代码应该让您开始使用 gorilla :

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

关于go - 成功的websocket连接后如何使发送消息到特定的URL?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/60144327/

相关文章:

php - 简单的 php websocket 示例

ssl - 如何在服务器上配置 Elixir、NGINX、Websockets

java - JAVA中Keystore文件的位置在哪里?

c# - 如何获得微软证书(*.cer & .pfx)?

api - 无法使用管理 REST API 升级 Azure 部署(SSL 证书问题)

ssh - 通过 shell 与 ssh 连接?

Qt websocket 服务器示例不起作用

go - 来自 xx.xx.xx.xx :14333: EOF 的 TLS 握手错误

go - 如何在golang中写一个带有文件的目录?

Golang 避免代码重复