mirror of
https://github.com/nestriness/nestri.git
synced 2025-12-13 01:05:37 +02:00
This is a second attempt to add protobuf to Nestri, after the first one failed --------- Co-authored-by: Philipp Neumann <3daquawolf@gmail.com> Co-authored-by: DatCaptainHorse <DatCaptainHorse@users.noreply.github.com>
119 lines
3.1 KiB
Go
119 lines
3.1 KiB
Go
package relay
|
|
|
|
import (
|
|
"encoding/json"
|
|
"github.com/gorilla/websocket"
|
|
"log"
|
|
"sync"
|
|
)
|
|
|
|
// SafeWebSocket is a websocket with a mutex
|
|
type SafeWebSocket struct {
|
|
*websocket.Conn
|
|
sync.Mutex
|
|
closeCallback func() // OnClose callback
|
|
callbacks map[string]OnMessageCallback // MessageBase type -> callback
|
|
}
|
|
|
|
// NewSafeWebSocket creates a new SafeWebSocket from *websocket.Conn
|
|
func NewSafeWebSocket(conn *websocket.Conn) *SafeWebSocket {
|
|
ws := &SafeWebSocket{
|
|
Conn: conn,
|
|
closeCallback: nil,
|
|
callbacks: make(map[string]OnMessageCallback),
|
|
}
|
|
|
|
// Launch a goroutine to handle messages
|
|
go func() {
|
|
for {
|
|
// Read message
|
|
kind, data, err := ws.Conn.ReadMessage()
|
|
if websocket.IsUnexpectedCloseError(err, websocket.CloseGoingAway, websocket.CloseAbnormalClosure, websocket.CloseNoStatusReceived) {
|
|
// If unexpected close error, break
|
|
if GetFlags().Verbose {
|
|
log.Printf("Unexpected WebSocket close error, reason: %s\n", err)
|
|
}
|
|
break
|
|
} else if websocket.IsCloseError(err, websocket.CloseNormalClosure, websocket.CloseGoingAway, websocket.CloseAbnormalClosure, websocket.CloseNoStatusReceived) {
|
|
break
|
|
} else if err != nil {
|
|
log.Printf("Failed to read WebSocket message, reason: %s\n", err)
|
|
break
|
|
}
|
|
|
|
switch kind {
|
|
case websocket.TextMessage:
|
|
// Decode message
|
|
var msg MessageBase
|
|
if err = json.Unmarshal(data, &msg); err != nil {
|
|
log.Printf("Failed to decode text WebSocket message, reason: %s\n", err)
|
|
continue
|
|
}
|
|
|
|
// Handle message type callback
|
|
if callback, ok := ws.callbacks[msg.PayloadType]; ok {
|
|
callback(data)
|
|
} // TODO: Log unknown message type?
|
|
break
|
|
case websocket.BinaryMessage:
|
|
break
|
|
default:
|
|
log.Printf("Unknown WebSocket message type: %d\n", kind)
|
|
break
|
|
}
|
|
}
|
|
|
|
// Call close callback
|
|
if ws.closeCallback != nil {
|
|
ws.closeCallback()
|
|
}
|
|
}()
|
|
|
|
return ws
|
|
}
|
|
|
|
// SendJSON writes JSON to a websocket with a mutex
|
|
func (ws *SafeWebSocket) SendJSON(v interface{}) error {
|
|
ws.Lock()
|
|
defer ws.Unlock()
|
|
return ws.Conn.WriteJSON(v)
|
|
}
|
|
|
|
// SendBinary writes binary to a websocket with a mutex
|
|
func (ws *SafeWebSocket) SendBinary(data []byte) error {
|
|
ws.Lock()
|
|
defer ws.Unlock()
|
|
return ws.Conn.WriteMessage(websocket.BinaryMessage, data)
|
|
}
|
|
|
|
// RegisterMessageCallback sets the callback for binary message of given type
|
|
func (ws *SafeWebSocket) RegisterMessageCallback(msgType string, callback OnMessageCallback) {
|
|
ws.Lock()
|
|
defer ws.Unlock()
|
|
if ws.callbacks == nil {
|
|
ws.callbacks = make(map[string]OnMessageCallback)
|
|
}
|
|
ws.callbacks[msgType] = callback
|
|
}
|
|
|
|
// UnregisterMessageCallback removes the callback for binary message of given type
|
|
func (ws *SafeWebSocket) UnregisterMessageCallback(msgType string) {
|
|
ws.Lock()
|
|
defer ws.Unlock()
|
|
if ws.callbacks != nil {
|
|
delete(ws.callbacks, msgType)
|
|
}
|
|
}
|
|
|
|
// RegisterOnClose sets the callback for websocket closing
|
|
func (ws *SafeWebSocket) RegisterOnClose(callback func()) {
|
|
ws.closeCallback = func() {
|
|
// Clear our callbacks
|
|
ws.Lock()
|
|
ws.callbacks = nil
|
|
ws.Unlock()
|
|
// Call the callback
|
|
callback()
|
|
}
|
|
}
|