-
Notifications
You must be signed in to change notification settings - Fork 9
/
Copy pathcomm_ws.go
143 lines (121 loc) · 3.66 KB
/
comm_ws.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
// OneBot Connect - 通信方式 - 正向 WebSocket
// https://12.onebot.dev/connect/communication/websocket/
package libonebot
import (
"context"
"fmt"
"net/http"
"sync"
"github.com/gorilla/websocket"
"github.com/tevino/abool/v2"
)
type wsCommCommon struct {
ob *OneBot
}
func (comm *wsCommCommon) handleRequest(conn *websocket.Conn, connWriteLock *sync.Mutex, messageBytes []byte, messageType int, reqComm RequestComm) {
isBinary := messageType == websocket.BinaryMessage
resp := comm.ob.decodeAndHandleRequest(messageBytes, isBinary, reqComm)
respBytes, _ := comm.ob.encodeResponse(resp, isBinary)
connWriteLock.Lock()
conn.WriteMessage(messageType, respBytes)
connWriteLock.Unlock()
}
func (comm *wsCommCommon) pushEvent(conn *websocket.Conn, connWriteLock *sync.Mutex, event marshaledEvent) {
connWriteLock.Lock()
conn.WriteMessage(websocket.TextMessage, event.bytes)
connWriteLock.Unlock()
}
type wsComm struct {
wsCommCommon
config ConfigCommWS
addr string
authorizer *httpAuthorizer
}
var wsUpgrader = websocket.Upgrader{
CheckOrigin: func(r *http.Request) bool {
return true
},
}
func (comm *wsComm) handle(w http.ResponseWriter, r *http.Request) {
comm.ob.Logger.Debugf("收到来自 %v 的 WebSocket (%v) 连接请求", r.RemoteAddr, comm.addr)
// authorization
if !comm.authorizer.authorize(r) {
comm.ob.Logger.Errorf("请求鉴权失败")
w.WriteHeader(http.StatusUnauthorized)
return
}
conn, err := wsUpgrader.Upgrade(w, r, nil)
if err != nil {
comm.ob.Logger.Errorf("WebSocket (%v) 连接失败, 错误: %v", comm.addr, err)
return
}
comm.ob.Logger.Infof("WebSocket (%v) 连接成功", comm.addr)
defer conn.Close()
// protect concurrent writes to the same connection
connWriteLock := &sync.Mutex{}
isClosed := abool.New()
checkError := func(err error) bool {
if err != nil {
if isClosed.IsNotSet() {
if websocket.IsCloseError(err, websocket.CloseNormalClosure) {
comm.ob.Logger.Infof("WebSocket (%v) 连接断开", comm.addr)
} else {
comm.ob.Logger.Errorf("WebSocket (%v) 连接异常断开, 错误: %v", comm.addr, err)
}
}
isClosed.Set()
return true
}
return false
}
eventChan := comm.ob.openEventListenChan()
defer comm.ob.closeEventListenChan(eventChan)
go func() {
// keep pushing events throught the connection
for event := range eventChan {
comm.ob.Logger.Debugf("通过 WebSocket (%v) 推送事件 `%v`", comm.addr, event.name)
go comm.pushEvent(conn, connWriteLock, event)
}
}()
for {
// this is the only one place we read from the connection, no need to lock
messageType, messageBytes, err := conn.ReadMessage()
if checkError(err) {
break
}
go comm.handleRequest(conn, connWriteLock, messageBytes, messageType, RequestComm{
Method: CommMethodWS,
Config: comm.config,
})
}
}
func commRunWS(c ConfigCommWS, ob *OneBot, ctx context.Context, wg *sync.WaitGroup) {
defer wg.Done()
addr := fmt.Sprintf("%s:%d", c.Host, c.Port)
ob.Logger.Infof("正在启动 WebSocket (%v)...", addr)
comm := &wsComm{
wsCommCommon: wsCommCommon{ob: ob},
config: c,
addr: addr,
authorizer: &httpAuthorizer{
accessToken: c.AccessToken,
},
}
mux := http.NewServeMux()
mux.HandleFunc("/", comm.handle)
server := &http.Server{
Addr: addr,
Handler: mux,
}
go func() {
if err := server.ListenAndServe(); err != nil && err != http.ErrServerClosed {
ob.Logger.Errorf("WebSocket (%v) 启动失败, 错误: %v", addr, err)
}
}()
<-ctx.Done()
if err := server.Shutdown(context.TODO()); err != nil {
ob.Logger.Errorf("WebSocket (%v) 关闭失败, 错误: %v", addr, err)
} else {
ob.Logger.Infof("WebSocket (%v) 已关闭", addr)
}
}