3x-ui/web/controller/inbound.go

666 lines
21 KiB
Go
Raw Normal View History

2023-02-09 19:18:06 +00:00
package controller
import (
"bytes"
2023-12-08 19:08:44 +00:00
"encoding/json"
2023-02-09 19:18:06 +00:00
"fmt"
"io"
2023-02-09 19:18:06 +00:00
"strconv"
2025-09-19 08:05:43 +00:00
"github.com/mhsanaei/3x-ui/v2/database/model"
2026-01-05 21:12:53 +00:00
"github.com/mhsanaei/3x-ui/v2/logger"
2025-09-19 08:05:43 +00:00
"github.com/mhsanaei/3x-ui/v2/web/service"
"github.com/mhsanaei/3x-ui/v2/web/session"
feat: Add WebSocket support for real-time updates and enhance VLESS settings (#3605) * feat: add support for trusted X-Forwarded-For and testseed parameters in VLESS settings * chore: update Xray Core version to 25.12.8 in release workflow * chore: update Xray Core version to 25.12.8 in Docker initialization script * chore: bump version to 2.8.6 and add watcher for security changes in inbound modal * refactor: remove default and random seed buttons from outbound form * refactor: update VLESS form to rename 'Test Seed' to 'Vision Seed' and change button functionality for seed generation * refactor: enhance TLS settings form layout with improved button styling and spacing * feat: integrate WebSocket support for real-time updates on inbounds and Xray service status * chore: downgrade version to 2.8.5 * refactor: translate comments to English * fix: ensure testseed is initialized correctly for VLESS protocol and improve client handling in inbound modal * refactor: simplify VLESS divider condition by removing unnecessary flow checks * fix: add fallback date formatting for cases when IntlUtil is not available * refactor: simplify WebSocket message handling by removing batching and ensuring individual message delivery * refactor: disable WebSocket notifications in inbound and index HTML files * refactor: enhance VLESS testseed initialization and button functionality in inbound modal * fix: * refactor: ensure proper WebSocket URL construction by normalizing basePath * fix: * fix: * fix: * refactor: update testseed methods for improved reactivity and binding in VLESS form * logger info to debug --------- Co-authored-by: lolka1333 <test123@gmail.com>
2026-01-03 04:26:00 +00:00
"github.com/mhsanaei/3x-ui/v2/web/websocket"
2023-02-18 12:37:32 +00:00
"github.com/gin-gonic/gin"
2023-02-09 19:18:06 +00:00
)
2025-09-20 07:35:50 +00:00
// InboundController handles HTTP requests related to Xray inbounds management.
2023-02-09 19:18:06 +00:00
type InboundController struct {
inboundService service.InboundService
xrayService service.XrayService
}
2025-09-20 07:35:50 +00:00
// NewInboundController creates a new InboundController and sets up its routes.
2023-02-09 19:18:06 +00:00
func NewInboundController(g *gin.RouterGroup) *InboundController {
a := &InboundController{}
a.initRouter(g)
return a
}
2025-09-20 07:35:50 +00:00
// initRouter initializes the routes for inbound-related operations.
2023-02-09 19:18:06 +00:00
func (a *InboundController) initRouter(g *gin.RouterGroup) {
2025-09-08 23:22:43 +00:00
g.GET("/list", a.getInbounds)
g.GET("/get/:id", a.getInbound)
g.GET("/getClientTraffics/:email", a.getClientTraffics)
g.GET("/getClientTrafficsById/:id", a.getClientTrafficsById)
2023-02-09 19:18:06 +00:00
g.POST("/add", a.addInbound)
g.POST("/del/:id", a.delInbound)
g.POST("/update/:id", a.updateInbound)
2023-02-28 19:54:29 +00:00
g.POST("/clientIps/:email", a.getClientIps)
g.POST("/clearClientIps/:email", a.clearClientIps)
g.POST("/addClient", a.addInboundClient)
g.POST("/:id/delClient/:clientId", a.delInboundClient)
g.POST("/updateClient/:clientId", a.updateInboundClient)
2023-03-17 16:07:49 +00:00
g.POST("/:id/resetClientTraffic/:email", a.resetClientTraffic)
g.POST("/resetAllTraffics", a.resetAllTraffics)
g.POST("/resetAllClientTraffics/:id", a.resetAllClientTraffics)
g.POST("/delDepletedClients/:id", a.delDepletedClients)
2023-12-08 19:08:44 +00:00
g.POST("/import", a.importInbound)
2023-12-04 18:13:21 +00:00
g.POST("/onlines", a.onlines)
2025-09-08 23:22:43 +00:00
g.POST("/lastOnline", a.lastOnline)
g.POST("/updateClientTraffic/:email", a.updateClientTraffic)
2025-09-10 14:36:12 +00:00
g.POST("/:id/delClientByEmail/:email", a.delInboundClientByEmail)
2023-02-09 19:18:06 +00:00
}
2025-09-20 07:35:50 +00:00
// getInbounds retrieves the list of inbounds for the logged-in user.
2023-02-09 19:18:06 +00:00
func (a *InboundController) getInbounds(c *gin.Context) {
user := session.GetLoginUser(c)
inbounds, err := a.inboundService.GetInbounds(user.Id)
if err != nil {
2023-05-20 22:59:27 +00:00
jsonMsg(c, I18nWeb(c, "pages.inbounds.toasts.obtain"), err)
2023-02-09 19:18:06 +00:00
return
}
jsonObj(c, inbounds, nil)
}
2023-05-30 20:51:14 +00:00
2025-09-20 07:35:50 +00:00
// getInbound retrieves a specific inbound by its ID.
2023-02-09 19:18:06 +00:00
func (a *InboundController) getInbound(c *gin.Context) {
id, err := strconv.Atoi(c.Param("id"))
if err != nil {
2023-05-20 22:59:27 +00:00
jsonMsg(c, I18nWeb(c, "get"), err)
2023-02-09 19:18:06 +00:00
return
}
inbound, err := a.inboundService.GetInbound(id)
if err != nil {
2023-05-20 22:59:27 +00:00
jsonMsg(c, I18nWeb(c, "pages.inbounds.toasts.obtain"), err)
2023-02-09 19:18:06 +00:00
return
}
jsonObj(c, inbound, nil)
}
2025-09-20 07:35:50 +00:00
// getClientTraffics retrieves client traffic information by email.
func (a *InboundController) getClientTraffics(c *gin.Context) {
email := c.Param("email")
clientTraffics, err := a.inboundService.GetClientTrafficByEmail(email)
if err != nil {
2025-05-09 03:46:29 +00:00
jsonMsg(c, I18nWeb(c, "pages.inbounds.toasts.trafficGetError"), err)
return
}
jsonObj(c, clientTraffics, nil)
}
2025-09-20 07:35:50 +00:00
// getClientTrafficsById retrieves client traffic information by inbound ID.
func (a *InboundController) getClientTrafficsById(c *gin.Context) {
id := c.Param("id")
clientTraffics, err := a.inboundService.GetClientTrafficByID(id)
if err != nil {
2025-05-09 03:46:29 +00:00
jsonMsg(c, I18nWeb(c, "pages.inbounds.toasts.trafficGetError"), err)
return
}
jsonObj(c, clientTraffics, nil)
}
2025-09-20 07:35:50 +00:00
// addInbound creates a new inbound configuration.
2023-02-09 19:18:06 +00:00
func (a *InboundController) addInbound(c *gin.Context) {
// Try to get nodeIds from JSON body first (if Content-Type is application/json)
// This must be done BEFORE ShouldBind, which reads the body
var nodeIdsFromJSON []int
var nodeIdFromJSON *int
var hasNodeIdsInJSON, hasNodeIdInJSON bool
if c.ContentType() == "application/json" {
// Read raw body to extract nodeIds
bodyBytes, err := c.GetRawData()
if err == nil && len(bodyBytes) > 0 {
// Parse JSON to extract nodeIds
var jsonData map[string]interface{}
if err := json.Unmarshal(bodyBytes, &jsonData); err == nil {
// Check for nodeIds array
if nodeIdsVal, ok := jsonData["nodeIds"]; ok {
hasNodeIdsInJSON = true
if nodeIdsArray, ok := nodeIdsVal.([]interface{}); ok {
for _, val := range nodeIdsArray {
if num, ok := val.(float64); ok {
nodeIdsFromJSON = append(nodeIdsFromJSON, int(num))
} else if num, ok := val.(int); ok {
nodeIdsFromJSON = append(nodeIdsFromJSON, num)
}
}
} else if num, ok := nodeIdsVal.(float64); ok {
// Single number instead of array
nodeIdsFromJSON = append(nodeIdsFromJSON, int(num))
} else if num, ok := nodeIdsVal.(int); ok {
nodeIdsFromJSON = append(nodeIdsFromJSON, num)
}
}
// Check for nodeId (backward compatibility)
if nodeIdVal, ok := jsonData["nodeId"]; ok {
hasNodeIdInJSON = true
if num, ok := nodeIdVal.(float64); ok {
nodeId := int(num)
nodeIdFromJSON = &nodeId
} else if num, ok := nodeIdVal.(int); ok {
nodeIdFromJSON = &num
}
}
}
// Restore body for ShouldBind
c.Request.Body = io.NopCloser(bytes.NewBuffer(bodyBytes))
}
}
2023-02-09 19:18:06 +00:00
inbound := &model.Inbound{}
err := c.ShouldBind(inbound)
if err != nil {
2026-01-05 21:12:53 +00:00
logger.Errorf("Failed to bind inbound data: %v", err)
2025-05-09 03:46:29 +00:00
jsonMsg(c, I18nWeb(c, "pages.inbounds.toasts.inboundCreateSuccess"), err)
2023-02-09 19:18:06 +00:00
return
}
2026-01-05 21:12:53 +00:00
2023-02-09 19:18:06 +00:00
user := session.GetLoginUser(c)
inbound.UserId = user.Id
if inbound.Listen == "" || inbound.Listen == "0.0.0.0" || inbound.Listen == "::" || inbound.Listen == "::0" {
2024-02-21 18:50:51 +00:00
inbound.Tag = fmt.Sprintf("inbound-%v", inbound.Port)
} else {
inbound.Tag = fmt.Sprintf("inbound-%v:%v", inbound.Listen, inbound.Port)
}
inbound, needRestart, err := a.inboundService.AddInbound(inbound)
if err != nil {
2026-01-05 21:12:53 +00:00
logger.Errorf("Failed to add inbound: %v", err)
jsonMsg(c, I18nWeb(c, "somethingWentWrong"), err)
return
}
2026-01-05 21:12:53 +00:00
// Handle node assignment in multi-node mode
nodeService := service.NodeService{}
// Get nodeIds from form (for form-encoded requests)
2026-01-05 21:12:53 +00:00
nodeIdsStr := c.PostFormArray("nodeIds")
logger.Debugf("Received nodeIds from form: %v", nodeIdsStr)
// Check if nodeIds array was provided (even if empty)
nodeIdStr := c.PostForm("nodeId")
// Determine which source to use: JSON takes precedence over form data
useJSON := hasNodeIdsInJSON || hasNodeIdInJSON
useForm := (len(nodeIdsStr) > 0 || nodeIdStr != "") && !useJSON
if useJSON || useForm {
var nodeIds []int
var nodeId *int
if useJSON {
// Use data from JSON
nodeIds = nodeIdsFromJSON
nodeId = nodeIdFromJSON
} else {
// Parse nodeIds array from form
for _, idStr := range nodeIdsStr {
if idStr != "" {
if id, err := strconv.Atoi(idStr); err == nil && id > 0 {
nodeIds = append(nodeIds, id)
}
}
}
// Parse single nodeId from form
if nodeIdStr != "" && nodeIdStr != "null" {
if parsedId, err := strconv.Atoi(nodeIdStr); err == nil && parsedId > 0 {
nodeId = &parsedId
2026-01-05 21:12:53 +00:00
}
}
}
if len(nodeIds) > 0 {
// Assign to multiple nodes
if err := nodeService.AssignInboundToNodes(inbound.Id, nodeIds); err != nil {
logger.Errorf("Failed to assign inbound %d to nodes %v: %v", inbound.Id, nodeIds, err)
jsonMsg(c, I18nWeb(c, "somethingWentWrong"), err)
return
}
} else if nodeId != nil && *nodeId > 0 {
2026-01-05 21:12:53 +00:00
// Backward compatibility: single nodeId
if err := nodeService.AssignInboundToNode(inbound.Id, *nodeId); err != nil {
logger.Warningf("Failed to assign inbound %d to node %d: %v", inbound.Id, *nodeId, err)
2026-01-05 21:12:53 +00:00
}
}
}
2025-08-04 16:01:32 +00:00
jsonMsgObj(c, I18nWeb(c, "pages.inbounds.toasts.inboundCreateSuccess"), inbound, nil)
if needRestart {
2023-02-09 19:18:06 +00:00
a.xrayService.SetToNeedRestart()
}
feat: Add WebSocket support for real-time updates and enhance VLESS settings (#3605) * feat: add support for trusted X-Forwarded-For and testseed parameters in VLESS settings * chore: update Xray Core version to 25.12.8 in release workflow * chore: update Xray Core version to 25.12.8 in Docker initialization script * chore: bump version to 2.8.6 and add watcher for security changes in inbound modal * refactor: remove default and random seed buttons from outbound form * refactor: update VLESS form to rename 'Test Seed' to 'Vision Seed' and change button functionality for seed generation * refactor: enhance TLS settings form layout with improved button styling and spacing * feat: integrate WebSocket support for real-time updates on inbounds and Xray service status * chore: downgrade version to 2.8.5 * refactor: translate comments to English * fix: ensure testseed is initialized correctly for VLESS protocol and improve client handling in inbound modal * refactor: simplify VLESS divider condition by removing unnecessary flow checks * fix: add fallback date formatting for cases when IntlUtil is not available * refactor: simplify WebSocket message handling by removing batching and ensuring individual message delivery * refactor: disable WebSocket notifications in inbound and index HTML files * refactor: enhance VLESS testseed initialization and button functionality in inbound modal * fix: * refactor: ensure proper WebSocket URL construction by normalizing basePath * fix: * fix: * fix: * refactor: update testseed methods for improved reactivity and binding in VLESS form * logger info to debug --------- Co-authored-by: lolka1333 <test123@gmail.com>
2026-01-03 04:26:00 +00:00
// Broadcast inbounds update via WebSocket
inbounds, _ := a.inboundService.GetInbounds(user.Id)
websocket.BroadcastInbounds(inbounds)
2023-02-09 19:18:06 +00:00
}
2025-09-20 07:35:50 +00:00
// delInbound deletes an inbound configuration by its ID.
2023-02-09 19:18:06 +00:00
func (a *InboundController) delInbound(c *gin.Context) {
id, err := strconv.Atoi(c.Param("id"))
if err != nil {
2025-05-09 03:46:29 +00:00
jsonMsg(c, I18nWeb(c, "pages.inbounds.toasts.inboundDeleteSuccess"), err)
2023-02-09 19:18:06 +00:00
return
}
needRestart, err := a.inboundService.DelInbound(id)
if err != nil {
jsonMsg(c, I18nWeb(c, "somethingWentWrong"), err)
return
}
2025-08-04 16:01:32 +00:00
jsonMsgObj(c, I18nWeb(c, "pages.inbounds.toasts.inboundDeleteSuccess"), id, nil)
if needRestart {
2023-02-09 19:18:06 +00:00
a.xrayService.SetToNeedRestart()
}
feat: Add WebSocket support for real-time updates and enhance VLESS settings (#3605) * feat: add support for trusted X-Forwarded-For and testseed parameters in VLESS settings * chore: update Xray Core version to 25.12.8 in release workflow * chore: update Xray Core version to 25.12.8 in Docker initialization script * chore: bump version to 2.8.6 and add watcher for security changes in inbound modal * refactor: remove default and random seed buttons from outbound form * refactor: update VLESS form to rename 'Test Seed' to 'Vision Seed' and change button functionality for seed generation * refactor: enhance TLS settings form layout with improved button styling and spacing * feat: integrate WebSocket support for real-time updates on inbounds and Xray service status * chore: downgrade version to 2.8.5 * refactor: translate comments to English * fix: ensure testseed is initialized correctly for VLESS protocol and improve client handling in inbound modal * refactor: simplify VLESS divider condition by removing unnecessary flow checks * fix: add fallback date formatting for cases when IntlUtil is not available * refactor: simplify WebSocket message handling by removing batching and ensuring individual message delivery * refactor: disable WebSocket notifications in inbound and index HTML files * refactor: enhance VLESS testseed initialization and button functionality in inbound modal * fix: * refactor: ensure proper WebSocket URL construction by normalizing basePath * fix: * fix: * fix: * refactor: update testseed methods for improved reactivity and binding in VLESS form * logger info to debug --------- Co-authored-by: lolka1333 <test123@gmail.com>
2026-01-03 04:26:00 +00:00
// Broadcast inbounds update via WebSocket
user := session.GetLoginUser(c)
inbounds, _ := a.inboundService.GetInbounds(user.Id)
websocket.BroadcastInbounds(inbounds)
2023-02-09 19:18:06 +00:00
}
2025-09-20 07:35:50 +00:00
// updateInbound updates an existing inbound configuration.
2023-02-09 19:18:06 +00:00
func (a *InboundController) updateInbound(c *gin.Context) {
id, err := strconv.Atoi(c.Param("id"))
if err != nil {
2025-05-09 03:46:29 +00:00
jsonMsg(c, I18nWeb(c, "pages.inbounds.toasts.inboundUpdateSuccess"), err)
2023-02-09 19:18:06 +00:00
return
}
2026-01-05 21:12:53 +00:00
// Try to get nodeIds from JSON body first (if Content-Type is application/json)
var nodeIdsFromJSON []int
var nodeIdFromJSON *int
var hasNodeIdsInJSON, hasNodeIdInJSON bool
if c.ContentType() == "application/json" {
// Read raw body to extract nodeIds
bodyBytes, err := c.GetRawData()
if err == nil && len(bodyBytes) > 0 {
// Parse JSON to extract nodeIds
var jsonData map[string]interface{}
if err := json.Unmarshal(bodyBytes, &jsonData); err == nil {
// Check for nodeIds array
if nodeIdsVal, ok := jsonData["nodeIds"]; ok {
hasNodeIdsInJSON = true
if nodeIdsArray, ok := nodeIdsVal.([]interface{}); ok {
for _, val := range nodeIdsArray {
if num, ok := val.(float64); ok {
nodeIdsFromJSON = append(nodeIdsFromJSON, int(num))
} else if num, ok := val.(int); ok {
nodeIdsFromJSON = append(nodeIdsFromJSON, num)
}
}
} else if num, ok := nodeIdsVal.(float64); ok {
// Single number instead of array
nodeIdsFromJSON = append(nodeIdsFromJSON, int(num))
} else if num, ok := nodeIdsVal.(int); ok {
nodeIdsFromJSON = append(nodeIdsFromJSON, num)
}
}
// Check for nodeId (backward compatibility)
if nodeIdVal, ok := jsonData["nodeId"]; ok {
hasNodeIdInJSON = true
if num, ok := nodeIdVal.(float64); ok {
nodeId := int(num)
nodeIdFromJSON = &nodeId
} else if num, ok := nodeIdVal.(int); ok {
nodeIdFromJSON = &num
}
}
}
// Restore body for ShouldBind
c.Request.Body = io.NopCloser(bytes.NewBuffer(bodyBytes))
}
}
// Get nodeIds from form (for form-encoded requests)
2026-01-05 21:12:53 +00:00
nodeIdsStr := c.PostFormArray("nodeIds")
logger.Debugf("Received nodeIds from form: %v (count: %d)", nodeIdsStr, len(nodeIdsStr))
// Check if nodeIds array was provided
nodeIdStr := c.PostForm("nodeId")
logger.Debugf("Received nodeId from form: %s", nodeIdStr)
// Check if nodeIds or nodeId was explicitly provided in the form
_, hasNodeIds := c.GetPostForm("nodeIds")
_, hasNodeId := c.GetPostForm("nodeId")
logger.Debugf("Form has nodeIds: %v, has nodeId: %v", hasNodeIds, hasNodeId)
logger.Debugf("JSON has nodeIds: %v (values: %v), has nodeId: %v (value: %v)", hasNodeIdsInJSON, nodeIdsFromJSON, hasNodeIdInJSON, nodeIdFromJSON)
2026-01-05 21:12:53 +00:00
2023-02-09 19:18:06 +00:00
inbound := &model.Inbound{
Id: id,
}
2026-01-05 21:12:53 +00:00
// Bind inbound data (nodeIds will be ignored since we handle it separately)
2023-02-09 19:18:06 +00:00
err = c.ShouldBind(inbound)
if err != nil {
2026-01-05 21:12:53 +00:00
logger.Errorf("Failed to bind inbound data: %v", err)
2025-05-09 03:46:29 +00:00
jsonMsg(c, I18nWeb(c, "pages.inbounds.toasts.inboundUpdateSuccess"), err)
2023-02-09 19:18:06 +00:00
return
}
inbound, needRestart, err := a.inboundService.UpdateInbound(inbound)
if err != nil {
2026-01-05 21:12:53 +00:00
logger.Errorf("Failed to update inbound: %v", err)
jsonMsg(c, I18nWeb(c, "somethingWentWrong"), err)
return
}
2026-01-05 21:12:53 +00:00
// Handle node assignment in multi-node mode
nodeService := service.NodeService{}
// Determine which source to use: JSON takes precedence over form data
useJSON := hasNodeIdsInJSON || hasNodeIdInJSON
useForm := (hasNodeIds || hasNodeId) && !useJSON
if useJSON || useForm {
var nodeIds []int
var nodeId *int
var hasNodeIdsFlag bool
if useJSON {
// Use data from JSON
nodeIds = nodeIdsFromJSON
nodeId = nodeIdFromJSON
hasNodeIdsFlag = hasNodeIdsInJSON
} else {
// Use data from form
hasNodeIdsFlag = hasNodeIds
// Parse nodeIds array from form
for _, idStr := range nodeIdsStr {
if idStr != "" {
if id, err := strconv.Atoi(idStr); err == nil && id > 0 {
nodeIds = append(nodeIds, id)
} else {
logger.Warningf("Invalid nodeId in array: %s (error: %v)", idStr, err)
}
}
}
// Parse single nodeId from form
if nodeIdStr != "" && nodeIdStr != "null" {
if parsedId, err := strconv.Atoi(nodeIdStr); err == nil && parsedId > 0 {
nodeId = &parsedId
2026-01-05 21:12:53 +00:00
}
}
}
logger.Debugf("Parsed nodeIds: %v, nodeId: %v", nodeIds, nodeId)
2026-01-05 21:12:53 +00:00
if len(nodeIds) > 0 {
// Assign to multiple nodes
if err := nodeService.AssignInboundToNodes(inbound.Id, nodeIds); err != nil {
logger.Errorf("Failed to assign inbound %d to nodes %v: %v", inbound.Id, nodeIds, err)
jsonMsg(c, I18nWeb(c, "somethingWentWrong"), err)
return
}
logger.Debugf("Successfully assigned inbound %d to nodes %v", inbound.Id, nodeIds)
} else if nodeId != nil && *nodeId > 0 {
2026-01-05 21:12:53 +00:00
// Backward compatibility: single nodeId
if err := nodeService.AssignInboundToNode(inbound.Id, *nodeId); err != nil {
logger.Errorf("Failed to assign inbound %d to node %d: %v", inbound.Id, *nodeId, err)
jsonMsg(c, I18nWeb(c, "somethingWentWrong"), err)
return
2026-01-05 21:12:53 +00:00
}
logger.Debugf("Successfully assigned inbound %d to node %d", inbound.Id, *nodeId)
} else if hasNodeIdsFlag {
2026-01-05 21:12:53 +00:00
// nodeIds was explicitly provided but is empty - unassign all
if err := nodeService.UnassignInboundFromNode(inbound.Id); err != nil {
logger.Warningf("Failed to unassign inbound %d from nodes: %v", inbound.Id, err)
} else {
logger.Debugf("Successfully unassigned inbound %d from all nodes", inbound.Id)
}
}
// If neither nodeIds nor nodeId was provided, don't change assignments
}
2025-08-04 16:01:32 +00:00
jsonMsgObj(c, I18nWeb(c, "pages.inbounds.toasts.inboundUpdateSuccess"), inbound, nil)
if needRestart {
2023-02-09 19:18:06 +00:00
a.xrayService.SetToNeedRestart()
}
feat: Add WebSocket support for real-time updates and enhance VLESS settings (#3605) * feat: add support for trusted X-Forwarded-For and testseed parameters in VLESS settings * chore: update Xray Core version to 25.12.8 in release workflow * chore: update Xray Core version to 25.12.8 in Docker initialization script * chore: bump version to 2.8.6 and add watcher for security changes in inbound modal * refactor: remove default and random seed buttons from outbound form * refactor: update VLESS form to rename 'Test Seed' to 'Vision Seed' and change button functionality for seed generation * refactor: enhance TLS settings form layout with improved button styling and spacing * feat: integrate WebSocket support for real-time updates on inbounds and Xray service status * chore: downgrade version to 2.8.5 * refactor: translate comments to English * fix: ensure testseed is initialized correctly for VLESS protocol and improve client handling in inbound modal * refactor: simplify VLESS divider condition by removing unnecessary flow checks * fix: add fallback date formatting for cases when IntlUtil is not available * refactor: simplify WebSocket message handling by removing batching and ensuring individual message delivery * refactor: disable WebSocket notifications in inbound and index HTML files * refactor: enhance VLESS testseed initialization and button functionality in inbound modal * fix: * refactor: ensure proper WebSocket URL construction by normalizing basePath * fix: * fix: * fix: * refactor: update testseed methods for improved reactivity and binding in VLESS form * logger info to debug --------- Co-authored-by: lolka1333 <test123@gmail.com>
2026-01-03 04:26:00 +00:00
// Broadcast inbounds update via WebSocket
user := session.GetLoginUser(c)
inbounds, _ := a.inboundService.GetInbounds(user.Id)
websocket.BroadcastInbounds(inbounds)
2023-02-09 19:18:06 +00:00
}
2023-03-17 16:07:49 +00:00
2025-09-20 07:35:50 +00:00
// getClientIps retrieves the IP addresses associated with a client by email.
2023-02-28 19:54:29 +00:00
func (a *InboundController) getClientIps(c *gin.Context) {
email := c.Param("email")
2023-02-09 19:18:06 +00:00
ips, err := a.inboundService.GetInboundClientIps(email)
2023-06-14 16:20:19 +00:00
if err != nil || ips == "" {
2023-02-28 19:54:29 +00:00
jsonObj(c, "No IP Record", nil)
return
}
2023-02-28 19:54:29 +00:00
jsonObj(c, ips, nil)
}
2025-09-20 07:35:50 +00:00
// clearClientIps clears the IP addresses for a client by email.
2023-02-28 19:54:29 +00:00
func (a *InboundController) clearClientIps(c *gin.Context) {
email := c.Param("email")
err := a.inboundService.ClearClientIps(email)
if err != nil {
2025-05-09 03:46:29 +00:00
jsonMsg(c, I18nWeb(c, "pages.inbounds.toasts.updateSuccess"), err)
2023-02-28 19:54:29 +00:00
return
}
2025-05-09 03:46:29 +00:00
jsonMsg(c, I18nWeb(c, "pages.inbounds.toasts.logCleanSuccess"), nil)
2023-02-28 19:54:29 +00:00
}
2023-05-30 20:51:14 +00:00
2025-09-20 07:35:50 +00:00
// addInboundClient adds a new client to an existing inbound.
2023-03-17 16:07:49 +00:00
func (a *InboundController) addInboundClient(c *gin.Context) {
2024-01-23 10:00:21 +00:00
data := &model.Inbound{}
err := c.ShouldBind(data)
if err != nil {
2025-05-09 03:46:29 +00:00
jsonMsg(c, I18nWeb(c, "pages.inbounds.toasts.inboundUpdateSuccess"), err)
return
}
needRestart, err := a.inboundService.AddInboundClient(data)
if err != nil {
2025-05-09 03:46:29 +00:00
jsonMsg(c, I18nWeb(c, "somethingWentWrong"), err)
return
}
2025-05-09 03:46:29 +00:00
jsonMsg(c, I18nWeb(c, "pages.inbounds.toasts.inboundClientAddSuccess"), nil)
if needRestart {
a.xrayService.SetToNeedRestart()
}
}
2025-09-20 07:35:50 +00:00
// delInboundClient deletes a client from an inbound by inbound ID and client ID.
2023-03-17 16:07:49 +00:00
func (a *InboundController) delInboundClient(c *gin.Context) {
id, err := strconv.Atoi(c.Param("id"))
2023-03-17 16:07:49 +00:00
if err != nil {
2025-05-09 03:46:29 +00:00
jsonMsg(c, I18nWeb(c, "pages.inbounds.toasts.inboundUpdateSuccess"), err)
2023-03-17 16:07:49 +00:00
return
}
clientId := c.Param("clientId")
2023-03-17 16:07:49 +00:00
needRestart, err := a.inboundService.DelInboundClient(id, clientId)
2023-03-17 16:07:49 +00:00
if err != nil {
2025-05-09 03:46:29 +00:00
jsonMsg(c, I18nWeb(c, "somethingWentWrong"), err)
2023-03-17 16:07:49 +00:00
return
}
2025-05-09 03:46:29 +00:00
jsonMsg(c, I18nWeb(c, "pages.inbounds.toasts.inboundClientDeleteSuccess"), nil)
if needRestart {
2023-03-17 16:07:49 +00:00
a.xrayService.SetToNeedRestart()
}
}
2025-09-20 07:35:50 +00:00
// updateInboundClient updates a client's configuration in an inbound.
2023-03-17 16:07:49 +00:00
func (a *InboundController) updateInboundClient(c *gin.Context) {
clientId := c.Param("clientId")
2023-03-17 16:07:49 +00:00
inbound := &model.Inbound{}
err := c.ShouldBind(inbound)
2023-03-17 16:07:49 +00:00
if err != nil {
2025-05-09 03:46:29 +00:00
jsonMsg(c, I18nWeb(c, "pages.inbounds.toasts.inboundUpdateSuccess"), err)
2023-03-17 16:07:49 +00:00
return
}
needRestart, err := a.inboundService.UpdateInboundClient(inbound, clientId)
2023-03-17 16:07:49 +00:00
if err != nil {
2025-05-09 03:46:29 +00:00
jsonMsg(c, I18nWeb(c, "somethingWentWrong"), err)
2023-03-17 16:07:49 +00:00
return
}
2025-05-09 03:46:29 +00:00
jsonMsg(c, I18nWeb(c, "pages.inbounds.toasts.inboundClientUpdateSuccess"), nil)
if needRestart {
2023-03-17 16:07:49 +00:00
a.xrayService.SetToNeedRestart()
}
}
2025-09-20 07:35:50 +00:00
// resetClientTraffic resets the traffic counter for a specific client in an inbound.
2023-02-09 19:18:06 +00:00
func (a *InboundController) resetClientTraffic(c *gin.Context) {
2023-03-17 16:07:49 +00:00
id, err := strconv.Atoi(c.Param("id"))
if err != nil {
2025-05-09 03:46:29 +00:00
jsonMsg(c, I18nWeb(c, "pages.inbounds.toasts.inboundUpdateSuccess"), err)
2023-03-17 16:07:49 +00:00
return
}
2023-02-09 19:18:06 +00:00
email := c.Param("email")
2024-07-07 09:55:59 +00:00
needRestart, err := a.inboundService.ResetClientTraffic(id, email)
2023-02-09 19:18:06 +00:00
if err != nil {
2025-05-09 03:46:29 +00:00
jsonMsg(c, I18nWeb(c, "somethingWentWrong"), err)
2023-02-09 19:18:06 +00:00
return
}
2025-05-09 03:46:29 +00:00
jsonMsg(c, I18nWeb(c, "pages.inbounds.toasts.resetInboundClientTrafficSuccess"), nil)
if needRestart {
2023-03-17 16:07:49 +00:00
a.xrayService.SetToNeedRestart()
}
2023-02-09 19:18:06 +00:00
}
2025-09-20 07:35:50 +00:00
// resetAllTraffics resets all traffic counters across all inbounds.
func (a *InboundController) resetAllTraffics(c *gin.Context) {
err := a.inboundService.ResetAllTraffics()
if err != nil {
2025-05-09 03:46:29 +00:00
jsonMsg(c, I18nWeb(c, "somethingWentWrong"), err)
return
} else {
a.xrayService.SetToNeedRestart()
}
2025-05-09 03:46:29 +00:00
jsonMsg(c, I18nWeb(c, "pages.inbounds.toasts.resetAllTrafficSuccess"), nil)
}
2025-09-20 07:35:50 +00:00
// resetAllClientTraffics resets traffic counters for all clients in a specific inbound.
func (a *InboundController) resetAllClientTraffics(c *gin.Context) {
id, err := strconv.Atoi(c.Param("id"))
if err != nil {
2025-05-09 03:46:29 +00:00
jsonMsg(c, I18nWeb(c, "pages.inbounds.toasts.inboundUpdateSuccess"), err)
return
}
err = a.inboundService.ResetAllClientTraffics(id)
if err != nil {
2025-05-09 03:46:29 +00:00
jsonMsg(c, I18nWeb(c, "somethingWentWrong"), err)
return
} else {
a.xrayService.SetToNeedRestart()
}
2025-05-09 03:46:29 +00:00
jsonMsg(c, I18nWeb(c, "pages.inbounds.toasts.resetAllClientTrafficSuccess"), nil)
}
2025-09-20 07:35:50 +00:00
// importInbound imports an inbound configuration from provided data.
2023-12-08 19:08:44 +00:00
func (a *InboundController) importInbound(c *gin.Context) {
inbound := &model.Inbound{}
err := json.Unmarshal([]byte(c.PostForm("data")), inbound)
if err != nil {
2025-05-09 03:46:29 +00:00
jsonMsg(c, I18nWeb(c, "somethingWentWrong"), err)
2023-12-08 19:08:44 +00:00
return
}
user := session.GetLoginUser(c)
inbound.Id = 0
inbound.UserId = user.Id
if inbound.Listen == "" || inbound.Listen == "0.0.0.0" || inbound.Listen == "::" || inbound.Listen == "::0" {
2024-02-21 18:50:51 +00:00
inbound.Tag = fmt.Sprintf("inbound-%v", inbound.Port)
} else {
inbound.Tag = fmt.Sprintf("inbound-%v:%v", inbound.Listen, inbound.Port)
}
2023-12-08 19:08:44 +00:00
for index := range inbound.ClientStats {
inbound.ClientStats[index].Id = 0
inbound.ClientStats[index].Enable = true
}
needRestart := false
inbound, needRestart, err = a.inboundService.AddInbound(inbound)
2025-05-09 03:46:29 +00:00
jsonMsgObj(c, I18nWeb(c, "pages.inbounds.toasts.inboundCreateSuccess"), inbound, err)
2023-12-08 19:08:44 +00:00
if err == nil && needRestart {
a.xrayService.SetToNeedRestart()
}
}
2025-09-20 07:35:50 +00:00
// delDepletedClients deletes clients in an inbound who have exhausted their traffic limits.
func (a *InboundController) delDepletedClients(c *gin.Context) {
id, err := strconv.Atoi(c.Param("id"))
if err != nil {
2025-05-09 03:46:29 +00:00
jsonMsg(c, I18nWeb(c, "pages.inbounds.toasts.inboundUpdateSuccess"), err)
return
}
err = a.inboundService.DelDepletedClients(id)
if err != nil {
2025-05-09 03:46:29 +00:00
jsonMsg(c, I18nWeb(c, "somethingWentWrong"), err)
return
}
2025-05-09 03:46:29 +00:00
jsonMsg(c, I18nWeb(c, "pages.inbounds.toasts.delDepletedClientsSuccess"), nil)
}
2023-12-04 18:13:21 +00:00
2025-09-20 07:35:50 +00:00
// onlines retrieves the list of currently online clients.
2023-12-04 18:13:21 +00:00
func (a *InboundController) onlines(c *gin.Context) {
2026-01-05 21:12:53 +00:00
clients := a.inboundService.GetOnlineClients()
jsonObj(c, clients, nil)
2023-12-04 18:13:21 +00:00
}
2025-09-20 07:35:50 +00:00
// lastOnline retrieves the last online timestamps for clients.
func (a *InboundController) lastOnline(c *gin.Context) {
data, err := a.inboundService.GetClientsLastOnline()
jsonObj(c, data, err)
}
2025-09-20 07:35:50 +00:00
// updateClientTraffic updates the traffic statistics for a client by email.
func (a *InboundController) updateClientTraffic(c *gin.Context) {
email := c.Param("email")
2025-08-04 16:01:32 +00:00
// Define the request structure for traffic update
type TrafficUpdateRequest struct {
Upload int64 `json:"upload"`
Download int64 `json:"download"`
}
2025-08-04 16:01:32 +00:00
var request TrafficUpdateRequest
err := c.ShouldBindJSON(&request)
if err != nil {
jsonMsg(c, I18nWeb(c, "pages.inbounds.toasts.inboundUpdateSuccess"), err)
return
}
2025-08-04 16:01:32 +00:00
err = a.inboundService.UpdateClientTrafficByEmail(email, request.Upload, request.Download)
if err != nil {
jsonMsg(c, I18nWeb(c, "somethingWentWrong"), err)
return
}
2025-08-04 16:01:32 +00:00
jsonMsg(c, I18nWeb(c, "pages.inbounds.toasts.inboundClientUpdateSuccess"), nil)
}
2025-09-10 14:36:12 +00:00
2025-09-20 07:35:50 +00:00
// delInboundClientByEmail deletes a client from an inbound by email address.
2025-09-10 14:36:12 +00:00
func (a *InboundController) delInboundClientByEmail(c *gin.Context) {
inboundId, err := strconv.Atoi(c.Param("id"))
if err != nil {
jsonMsg(c, "Invalid inbound ID", err)
return
}
email := c.Param("email")
needRestart, err := a.inboundService.DelInboundClientByEmail(inboundId, email)
if err != nil {
jsonMsg(c, "Failed to delete client by email", err)
return
}
jsonMsg(c, "Client deleted successfully", nil)
if needRestart {
a.xrayService.SetToNeedRestart()
}
}