mirror of
https://github.com/MHSanaei/3x-ui.git
synced 2026-04-22 15:35:52 +00:00
* feat: implement real-time traffic monitoring and UI updates using a high-performance WebSocket hub and background job system * feat: add bulk client management support and improve inbound data handling * Fix bug * **Fixes & Changes:** 1. **Fixed XPadding Placement Dropdown**: - Added the missing `cookie` and `query` options to `xPaddingPlacement` (`stream_xhttp.html`). - *Why:* Previously, users wanting `cookie` obfuscation were forced to use the `header` placement string. This caused Xray-core to blindly intercept the entire monolithic HTTP Cookie header, failing internal padding-length validations and causing the inbound to silently drop the connection. 2. **Fixed Uplink Data Placement Validation**: - Replaced the unsupported `query` option with `cookie` in `uplinkDataPlacement`. - *Why:* Xray-core's `transport_internet.go` explicitly forbids `query` as an uplink placement option. Selecting it from the UI previously sent a payload that would cause Xray-core to instantly throw an `unsupported uplink data placement: query` panic. Adding `cookie` perfectly aligns the UI with Xray-core restrictions. ### Related Issues - Resolves #3992 * This commit fixes structural payload issues preventing XHTTP from functioning correctly and eliminates WebSocket log spam. - **[Fix X-Padding UI]** Added missing `cookie` and `query` options to X-Padding Placement. Fixes the issue where using Cookie fallback triggers whole HTTP Cookie header interception and silent drop in Xray-core. (Resolves [#3992](https://github.com/MHSanaei/3x-ui/issues/3992)) - **[Fix Uplink Data Options]** Replaced the invalid `query` option with `cookie` in Uplink Data Placement dropdown to prevent Xray-core backend panic `unsupported uplink data placement: query`. - **[Fix WebSockets Spam]** Boosted `maxMessageSize` boundary to 100MB and gracefully handled fallback fetch signals via `broadcastInvalidate` to avoid buffer dropping spam. (Resolves [#3984](https://github.com/MHSanaei/3x-ui/issues/3984)) * Fix * gofmt * fix(websocket): resolve channel race condition and graceful shutdown deadlock * Fix: inbounds switch * Change max quantity from 10000 to 500 * fix
103 lines
2.7 KiB
Go
103 lines
2.7 KiB
Go
// Package websocket provides WebSocket hub for real-time updates and notifications.
|
|
package websocket
|
|
|
|
import (
|
|
"github.com/mhsanaei/3x-ui/v2/logger"
|
|
"github.com/mhsanaei/3x-ui/v2/web/global"
|
|
)
|
|
|
|
// GetHub returns the global WebSocket hub instance
|
|
func GetHub() *Hub {
|
|
webServer := global.GetWebServer()
|
|
if webServer == nil {
|
|
return nil
|
|
}
|
|
hub := webServer.GetWSHub()
|
|
if hub == nil {
|
|
return nil
|
|
}
|
|
wsHub, ok := hub.(*Hub)
|
|
if !ok {
|
|
logger.Warning("WebSocket hub type assertion failed")
|
|
return nil
|
|
}
|
|
return wsHub
|
|
}
|
|
|
|
// HasClients returns true if there are any WebSocket clients connected.
|
|
// Use this to skip expensive work (DB queries, serialization) when no browser is open.
|
|
func HasClients() bool {
|
|
hub := GetHub()
|
|
if hub == nil {
|
|
return false
|
|
}
|
|
return hub.GetClientCount() > 0
|
|
}
|
|
|
|
// BroadcastStatus broadcasts server status update to all connected clients
|
|
func BroadcastStatus(status any) {
|
|
hub := GetHub()
|
|
if hub != nil {
|
|
hub.Broadcast(MessageTypeStatus, status)
|
|
}
|
|
}
|
|
|
|
// BroadcastTraffic broadcasts traffic statistics update to all connected clients
|
|
func BroadcastTraffic(traffic any) {
|
|
hub := GetHub()
|
|
if hub != nil {
|
|
hub.Broadcast(MessageTypeTraffic, traffic)
|
|
}
|
|
}
|
|
|
|
// BroadcastInbounds broadcasts inbounds list update to all connected clients
|
|
func BroadcastInbounds(inbounds any) {
|
|
hub := GetHub()
|
|
if hub != nil {
|
|
hub.Broadcast(MessageTypeInbounds, inbounds)
|
|
}
|
|
}
|
|
|
|
// BroadcastOutbounds broadcasts outbounds list update to all connected clients
|
|
func BroadcastOutbounds(outbounds any) {
|
|
hub := GetHub()
|
|
if hub != nil {
|
|
hub.Broadcast(MessageTypeOutbounds, outbounds)
|
|
}
|
|
}
|
|
|
|
// BroadcastNotification broadcasts a system notification to all connected clients
|
|
func BroadcastNotification(title, message, level string) {
|
|
hub := GetHub()
|
|
if hub != nil {
|
|
notification := map[string]string{
|
|
"title": title,
|
|
"message": message,
|
|
"level": level, // info, warning, error, success
|
|
}
|
|
hub.Broadcast(MessageTypeNotification, notification)
|
|
}
|
|
}
|
|
|
|
// BroadcastXrayState broadcasts Xray state change to all connected clients
|
|
func BroadcastXrayState(state string, errorMsg string) {
|
|
hub := GetHub()
|
|
if hub != nil {
|
|
stateUpdate := map[string]string{
|
|
"state": state,
|
|
"errorMsg": errorMsg,
|
|
}
|
|
hub.Broadcast(MessageTypeXrayState, stateUpdate)
|
|
}
|
|
}
|
|
|
|
// BroadcastInvalidate sends a lightweight invalidate signal for the given data type,
|
|
// telling connected frontends to re-fetch data via REST API.
|
|
// Use this instead of BroadcastInbounds/BroadcastOutbounds when you know the payload
|
|
// will be too large, to avoid wasting resources on serialization.
|
|
func BroadcastInvalidate(dataType MessageType) {
|
|
hub := GetHub()
|
|
if hub != nil {
|
|
hub.broadcastInvalidate(dataType)
|
|
}
|
|
}
|