|
@@ -5,25 +5,15 @@ import (
|
|
|
"net/http"
|
|
"net/http"
|
|
|
"net/url"
|
|
"net/url"
|
|
|
"strings"
|
|
"strings"
|
|
|
- "time"
|
|
|
|
|
|
|
|
|
|
- "github.com/google/uuid"
|
|
|
|
|
"github.com/mhsanaei/3x-ui/v2/logger"
|
|
"github.com/mhsanaei/3x-ui/v2/logger"
|
|
|
- "github.com/mhsanaei/3x-ui/v2/util/common"
|
|
|
|
|
|
|
+ "github.com/mhsanaei/3x-ui/v2/web/service"
|
|
|
"github.com/mhsanaei/3x-ui/v2/web/session"
|
|
"github.com/mhsanaei/3x-ui/v2/web/session"
|
|
|
- "github.com/mhsanaei/3x-ui/v2/web/websocket"
|
|
|
|
|
|
|
|
|
|
"github.com/gin-gonic/gin"
|
|
"github.com/gin-gonic/gin"
|
|
|
ws "github.com/gorilla/websocket"
|
|
ws "github.com/gorilla/websocket"
|
|
|
)
|
|
)
|
|
|
|
|
|
|
|
-const (
|
|
|
|
|
- writeWait = 10 * time.Second
|
|
|
|
|
- pongWait = 60 * time.Second
|
|
|
|
|
- pingPeriod = (pongWait * 9) / 10
|
|
|
|
|
- clientReadLimit = 512
|
|
|
|
|
-)
|
|
|
|
|
-
|
|
|
|
|
var upgrader = ws.Upgrader{
|
|
var upgrader = ws.Upgrader{
|
|
|
ReadBufferSize: 32768,
|
|
ReadBufferSize: 32768,
|
|
|
WriteBufferSize: 32768,
|
|
WriteBufferSize: 32768,
|
|
@@ -57,18 +47,21 @@ func checkSameOrigin(r *http.Request) bool {
|
|
|
return strings.EqualFold(u.Hostname(), host)
|
|
return strings.EqualFold(u.Hostname(), host)
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
-// WebSocketController handles WebSocket connections for real-time updates.
|
|
|
|
|
|
|
+// WebSocketController handles the HTTP→WebSocket upgrade for real-time updates.
|
|
|
|
|
+// All per-connection lifecycle (pumps, hub registration) lives in
|
|
|
|
|
+// service.WebSocketService — this controller is HTTP-layer only.
|
|
|
type WebSocketController struct {
|
|
type WebSocketController struct {
|
|
|
BaseController
|
|
BaseController
|
|
|
- hub *websocket.Hub
|
|
|
|
|
|
|
+ service *service.WebSocketService
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
-// NewWebSocketController creates a new WebSocket controller.
|
|
|
|
|
-func NewWebSocketController(hub *websocket.Hub) *WebSocketController {
|
|
|
|
|
- return &WebSocketController{hub: hub}
|
|
|
|
|
|
|
+// NewWebSocketController creates a controller wired to the given service.
|
|
|
|
|
+func NewWebSocketController(svc *service.WebSocketService) *WebSocketController {
|
|
|
|
|
+ return &WebSocketController{service: svc}
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
-// HandleWebSocket upgrades the HTTP connection and starts the read/write pumps.
|
|
|
|
|
|
|
+// HandleWebSocket authenticates the request, upgrades the HTTP connection, and
|
|
|
|
|
+// hands ownership of the connection off to the service.
|
|
|
func (w *WebSocketController) HandleWebSocket(c *gin.Context) {
|
|
func (w *WebSocketController) HandleWebSocket(c *gin.Context) {
|
|
|
if !session.IsLogin(c) {
|
|
if !session.IsLogin(c) {
|
|
|
logger.Warningf("Unauthorized WebSocket connection attempt from %s", getRemoteIp(c))
|
|
logger.Warningf("Unauthorized WebSocket connection attempt from %s", getRemoteIp(c))
|
|
@@ -82,71 +75,5 @@ func (w *WebSocketController) HandleWebSocket(c *gin.Context) {
|
|
|
return
|
|
return
|
|
|
}
|
|
}
|
|
|
|
|
|
|
|
- client := websocket.NewClient(uuid.New().String())
|
|
|
|
|
- w.hub.Register(client)
|
|
|
|
|
- logger.Debugf("WebSocket client %s registered from %s", client.ID, getRemoteIp(c))
|
|
|
|
|
-
|
|
|
|
|
- go w.writePump(client, conn)
|
|
|
|
|
- go w.readPump(client, conn)
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-// readPump consumes inbound frames so the gorilla deadline/pong machinery keeps
|
|
|
|
|
-// running. Clients send no commands today; frames are discarded.
|
|
|
|
|
-func (w *WebSocketController) readPump(client *websocket.Client, conn *ws.Conn) {
|
|
|
|
|
- defer func() {
|
|
|
|
|
- if r := common.Recover("WebSocket readPump panic"); r != nil {
|
|
|
|
|
- logger.Error("WebSocket readPump panic recovered:", r)
|
|
|
|
|
- }
|
|
|
|
|
- w.hub.Unregister(client)
|
|
|
|
|
- conn.Close()
|
|
|
|
|
- }()
|
|
|
|
|
-
|
|
|
|
|
- conn.SetReadLimit(clientReadLimit)
|
|
|
|
|
- conn.SetReadDeadline(time.Now().Add(pongWait))
|
|
|
|
|
- conn.SetPongHandler(func(string) error {
|
|
|
|
|
- return conn.SetReadDeadline(time.Now().Add(pongWait))
|
|
|
|
|
- })
|
|
|
|
|
-
|
|
|
|
|
- for {
|
|
|
|
|
- if _, _, err := conn.ReadMessage(); err != nil {
|
|
|
|
|
- if ws.IsUnexpectedCloseError(err, ws.CloseGoingAway, ws.CloseAbnormalClosure) {
|
|
|
|
|
- logger.Debugf("WebSocket read error for client %s: %v", client.ID, err)
|
|
|
|
|
- }
|
|
|
|
|
- return
|
|
|
|
|
- }
|
|
|
|
|
- }
|
|
|
|
|
-}
|
|
|
|
|
-
|
|
|
|
|
-// writePump pushes hub messages to the connection and emits keepalive pings.
|
|
|
|
|
-func (w *WebSocketController) writePump(client *websocket.Client, conn *ws.Conn) {
|
|
|
|
|
- ticker := time.NewTicker(pingPeriod)
|
|
|
|
|
- defer func() {
|
|
|
|
|
- if r := common.Recover("WebSocket writePump panic"); r != nil {
|
|
|
|
|
- logger.Error("WebSocket writePump panic recovered:", r)
|
|
|
|
|
- }
|
|
|
|
|
- ticker.Stop()
|
|
|
|
|
- conn.Close()
|
|
|
|
|
- }()
|
|
|
|
|
-
|
|
|
|
|
- for {
|
|
|
|
|
- select {
|
|
|
|
|
- case msg, ok := <-client.Send:
|
|
|
|
|
- conn.SetWriteDeadline(time.Now().Add(writeWait))
|
|
|
|
|
- if !ok {
|
|
|
|
|
- conn.WriteMessage(ws.CloseMessage, []byte{})
|
|
|
|
|
- return
|
|
|
|
|
- }
|
|
|
|
|
- if err := conn.WriteMessage(ws.TextMessage, msg); err != nil {
|
|
|
|
|
- logger.Debugf("WebSocket write error for client %s: %v", client.ID, err)
|
|
|
|
|
- return
|
|
|
|
|
- }
|
|
|
|
|
-
|
|
|
|
|
- case <-ticker.C:
|
|
|
|
|
- conn.SetWriteDeadline(time.Now().Add(writeWait))
|
|
|
|
|
- if err := conn.WriteMessage(ws.PingMessage, nil); err != nil {
|
|
|
|
|
- logger.Debugf("WebSocket ping error for client %s: %v", client.ID, err)
|
|
|
|
|
- return
|
|
|
|
|
- }
|
|
|
|
|
- }
|
|
|
|
|
- }
|
|
|
|
|
|
|
+ w.service.HandleConnection(conn, getRemoteIp(c))
|
|
|
}
|
|
}
|