go websocket jsonrpc 浏览器连接关闭
Go websocket jsonrpc browser connection close
我有一个 WebSocket JSON-RPC 服务器示例,我想在浏览器中使用它。当你访问 URL "http:localhost:8080" 时,浏览器会正常打开 WebSocket 连接。但是当浏览器发送WebSocket请求时,服务器关闭了WebSocket连接。我什至看不到在服务器上调用 RPC 方法的痕迹。
但是,从 Go 客户端调用服务器工作得很好。
server.go
package main
import (
"log"
"net/http"
"net/rpc"
"net/rpc/jsonrpc"
"github.com/gorilla/websocket"
)
type Service struct{}
func (t *Service) Echo(req *string, res *string) error {
log.Printf("Service.Echo")
*res = *req
log.Printf("Service.Echo req:%s res:%s", *req, *res)
return nil
}
var upgrader = websocket.Upgrader{
ReadBufferSize: 1024,
WriteBufferSize: 1024,
}
func serveWS(w http.ResponseWriter, r *http.Request) {
ws, err := upgrader.Upgrade(w, r, nil)
defer ws.Close()
if err != nil {
log.Println(err)
return
}
jsonrpc.ServeConn(ws.UnderlyingConn())
}
func main() {
rpc.Register(new(Service))
http.Handle("/", http.FileServer(http.Dir("web")))
http.HandleFunc("/ws", serveWS)
http.ListenAndServe(":8080", nil)
}
web/index.html
<!DOCTYPE html>
<html lang="en">
<head>
<script type="text/javascript">
var ws = new WebSocket("ws://localhost:8080/ws");
ws.onopen = function(ev){
alert("open");
}
ws.onmessage = function(ev){
alert("message");
}
ws.onclose = function(ev){
alert("close");
}
function send() {
msg = {
method: "Service.Echo",
params: "hello",
id: 0
};
var s = JSON.stringify(msg);
alert(s);
ws.send(s);
}
</script>
</head>
<body>
<button onclick='send()'>Send</button>
</body>
</html>
client.go
package main
import (
"log"
"net/rpc/jsonrpc"
"github.com/gorilla/websocket"
)
func main() {
ws, _, err := websocket.DefaultDialer.Dial("ws://localhost:8080/ws", nil)
if err != nil {
log.Fatal("dial:", err)
}
defer ws.Close()
client := jsonrpc.NewClient(ws.UnderlyingConn())
req := "hello"
var res string
err = client.Call("Service.Echo", &req, &res)
if err != nil {
log.Fatal("Service.Echo error:", err)
}
log.Printf("Service.Echo: req:%s res:%s", req, res)
}
你知道问题出在哪里吗?
非常感谢。
干杯
当服务器应用程序调用 jsonrpc.ServeConn(ws.UnderlyingConn()) }
时,服务器正在从 WebSocket 协议切换到 JSON-RPC protocol. The browser continues to use the WebSocket 协议。连接已关闭,因为其中一个对等方在读取它不期望的协议时出错。
Go 客户端应用程序不会出现此问题,因为此应用程序也从 WebSocket 协议切换到 JSON-RPC 协议。
无法从浏览器应用程序访问基础网络连接。
可以使用带有 Codec 的 net/rpc 包来使用 WebSocket 协议。另一种选择是编写一个适配器,将基于消息的 WebSocket API 转换为 net/rpc/jsonrpc 服务器期望的流。
我有一个 WebSocket JSON-RPC 服务器示例,我想在浏览器中使用它。当你访问 URL "http:localhost:8080" 时,浏览器会正常打开 WebSocket 连接。但是当浏览器发送WebSocket请求时,服务器关闭了WebSocket连接。我什至看不到在服务器上调用 RPC 方法的痕迹。
但是,从 Go 客户端调用服务器工作得很好。
server.go
package main
import (
"log"
"net/http"
"net/rpc"
"net/rpc/jsonrpc"
"github.com/gorilla/websocket"
)
type Service struct{}
func (t *Service) Echo(req *string, res *string) error {
log.Printf("Service.Echo")
*res = *req
log.Printf("Service.Echo req:%s res:%s", *req, *res)
return nil
}
var upgrader = websocket.Upgrader{
ReadBufferSize: 1024,
WriteBufferSize: 1024,
}
func serveWS(w http.ResponseWriter, r *http.Request) {
ws, err := upgrader.Upgrade(w, r, nil)
defer ws.Close()
if err != nil {
log.Println(err)
return
}
jsonrpc.ServeConn(ws.UnderlyingConn())
}
func main() {
rpc.Register(new(Service))
http.Handle("/", http.FileServer(http.Dir("web")))
http.HandleFunc("/ws", serveWS)
http.ListenAndServe(":8080", nil)
}
web/index.html
<!DOCTYPE html>
<html lang="en">
<head>
<script type="text/javascript">
var ws = new WebSocket("ws://localhost:8080/ws");
ws.onopen = function(ev){
alert("open");
}
ws.onmessage = function(ev){
alert("message");
}
ws.onclose = function(ev){
alert("close");
}
function send() {
msg = {
method: "Service.Echo",
params: "hello",
id: 0
};
var s = JSON.stringify(msg);
alert(s);
ws.send(s);
}
</script>
</head>
<body>
<button onclick='send()'>Send</button>
</body>
</html>
client.go
package main
import (
"log"
"net/rpc/jsonrpc"
"github.com/gorilla/websocket"
)
func main() {
ws, _, err := websocket.DefaultDialer.Dial("ws://localhost:8080/ws", nil)
if err != nil {
log.Fatal("dial:", err)
}
defer ws.Close()
client := jsonrpc.NewClient(ws.UnderlyingConn())
req := "hello"
var res string
err = client.Call("Service.Echo", &req, &res)
if err != nil {
log.Fatal("Service.Echo error:", err)
}
log.Printf("Service.Echo: req:%s res:%s", req, res)
}
你知道问题出在哪里吗?
非常感谢。
干杯
当服务器应用程序调用 jsonrpc.ServeConn(ws.UnderlyingConn()) }
时,服务器正在从 WebSocket 协议切换到 JSON-RPC protocol. The browser continues to use the WebSocket 协议。连接已关闭,因为其中一个对等方在读取它不期望的协议时出错。
Go 客户端应用程序不会出现此问题,因为此应用程序也从 WebSocket 协议切换到 JSON-RPC 协议。
无法从浏览器应用程序访问基础网络连接。
可以使用带有 Codec 的 net/rpc 包来使用 WebSocket 协议。另一种选择是编写一个适配器,将基于消息的 WebSocket API 转换为 net/rpc/jsonrpc 服务器期望的流。