2023-02-09 19:18:06 +00:00
|
|
|
package controller
|
|
|
|
|
|
|
|
|
|
import (
|
2026-01-07 19:05:04 +00:00
|
|
|
"bytes"
|
2023-12-08 19:08:44 +00:00
|
|
|
"encoding/json"
|
2023-02-09 19:18:06 +00:00
|
|
|
"fmt"
|
2026-01-07 19:05:04 +00:00
|
|
|
"io"
|
2023-02-09 19:18:06 +00:00
|
|
|
"strconv"
|
2025-02-04 10:27:58 +00:00
|
|
|
|
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"
|
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)
|
2023-04-18 18:04:06 +00:00
|
|
|
g.POST("/addClient", a.addInboundClient)
|
2023-04-24 11:37:11 +00:00
|
|
|
g.POST("/:id/delClient/:clientId", a.delInboundClient)
|
2023-04-25 11:08:35 +00:00
|
|
|
g.POST("/updateClient/:clientId", a.updateInboundClient)
|
2023-03-17 16:07:49 +00:00
|
|
|
g.POST("/:id/resetClientTraffic/:email", a.resetClientTraffic)
|
2023-04-09 19:43:18 +00:00
|
|
|
g.POST("/resetAllTraffics", a.resetAllTraffics)
|
|
|
|
|
g.POST("/resetAllClientTraffics/:id", a.resetAllClientTraffics)
|
2023-04-25 15:13:37 +00:00
|
|
|
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)
|
|
|
|
|
}
|
2023-07-01 12:26:43 +00:00
|
|
|
|
2025-09-20 07:35:50 +00:00
|
|
|
// getClientTraffics retrieves client traffic information by email.
|
2023-04-21 15:36:59 +00:00
|
|
|
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)
|
2023-04-21 15:36:59 +00:00
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
jsonObj(c, clientTraffics, nil)
|
2024-07-23 09:28:28 +00:00
|
|
|
}
|
|
|
|
|
|
2025-09-20 07:35:50 +00:00
|
|
|
// getClientTrafficsById retrieves client traffic information by inbound ID.
|
2024-07-23 09:28:28 +00:00
|
|
|
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)
|
2024-07-23 09:28:28 +00:00
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
jsonObj(c, clientTraffics, nil)
|
2023-04-21 15:36:59 +00:00
|
|
|
}
|
|
|
|
|
|
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) {
|
2026-01-07 19:05:04 +00:00
|
|
|
// 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
|
2024-01-17 12:51:28 +00:00
|
|
|
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)
|
2024-01-17 12:51:28 +00:00
|
|
|
} else {
|
|
|
|
|
inbound.Tag = fmt.Sprintf("inbound-%v:%v", inbound.Listen, inbound.Port)
|
|
|
|
|
}
|
2023-07-17 23:10:22 +00:00
|
|
|
|
2025-09-19 08:47:28 +00:00
|
|
|
inbound, needRestart, err := a.inboundService.AddInbound(inbound)
|
2025-05-16 16:56:56 +00:00
|
|
|
if err != nil {
|
2026-01-05 21:12:53 +00:00
|
|
|
logger.Errorf("Failed to add inbound: %v", err)
|
2025-05-16 16:56:56 +00:00
|
|
|
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{}
|
|
|
|
|
|
2026-01-07 19:05:04 +00:00
|
|
|
// 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")
|
2026-01-07 19:05:04 +00:00
|
|
|
|
|
|
|
|
// 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
|
|
|
|
|
}
|
2026-01-07 19:05:04 +00:00
|
|
|
} else if nodeId != nil && *nodeId > 0 {
|
2026-01-05 21:12:53 +00:00
|
|
|
// Backward compatibility: single nodeId
|
2026-01-07 19:05:04 +00:00
|
|
|
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()
|
|
|
|
|
}
|
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
|
|
|
|
|
}
|
2025-09-19 08:47:28 +00:00
|
|
|
needRestart, err := a.inboundService.DelInbound(id)
|
2025-05-16 16:56:56 +00:00
|
|
|
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()
|
|
|
|
|
}
|
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
|
|
|
|
2026-01-07 19:05:04 +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)
|
2026-01-07 19:05:04 +00:00
|
|
|
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
|
|
|
|
|
}
|
2025-09-19 08:47:28 +00:00
|
|
|
inbound, needRestart, err := a.inboundService.UpdateInbound(inbound)
|
2025-05-16 16:56:56 +00:00
|
|
|
if err != nil {
|
2026-01-05 21:12:53 +00:00
|
|
|
logger.Errorf("Failed to update inbound: %v", err)
|
2025-05-16 16:56:56 +00:00
|
|
|
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{}
|
|
|
|
|
|
2026-01-07 19:05:04 +00:00
|
|
|
// 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
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2026-01-07 19:05:04 +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)
|
2026-01-07 19:05:04 +00:00
|
|
|
} else if nodeId != nil && *nodeId > 0 {
|
2026-01-05 21:12:53 +00:00
|
|
|
// Backward compatibility: single nodeId
|
2026-01-07 19:05:04 +00:00
|
|
|
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
|
|
|
}
|
2026-01-07 19:05:04 +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()
|
|
|
|
|
}
|
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
|
|
|
|
2023-04-09 19:43:18 +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-05-22 14:01:41 +00:00
|
|
|
|
2023-02-28 19:54:29 +00:00
|
|
|
jsonObj(c, ips, nil)
|
|
|
|
|
}
|
2023-05-22 14:01:41 +00:00
|
|
|
|
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{}
|
2024-01-26 18:37:15 +00:00
|
|
|
err := c.ShouldBind(data)
|
|
|
|
|
if err != nil {
|
2025-05-09 03:46:29 +00:00
|
|
|
jsonMsg(c, I18nWeb(c, "pages.inbounds.toasts.inboundUpdateSuccess"), err)
|
2024-01-26 18:37:15 +00:00
|
|
|
return
|
|
|
|
|
}
|
2024-01-22 11:36:01 +00:00
|
|
|
|
2025-09-19 08:47:28 +00:00
|
|
|
needRestart, err := a.inboundService.AddInboundClient(data)
|
2024-01-26 18:37:15 +00:00
|
|
|
if err != nil {
|
2025-05-09 03:46:29 +00:00
|
|
|
jsonMsg(c, I18nWeb(c, "somethingWentWrong"), err)
|
2024-01-26 18:37:15 +00:00
|
|
|
return
|
|
|
|
|
}
|
2025-05-09 03:46:29 +00:00
|
|
|
jsonMsg(c, I18nWeb(c, "pages.inbounds.toasts.inboundClientAddSuccess"), nil)
|
2024-03-11 08:16:54 +00:00
|
|
|
if needRestart {
|
2024-01-26 18:37:15 +00:00
|
|
|
a.xrayService.SetToNeedRestart()
|
|
|
|
|
}
|
2024-01-22 11:36:01 +00:00
|
|
|
}
|
|
|
|
|
|
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) {
|
2023-04-24 11:37:11 +00:00
|
|
|
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
|
|
|
|
|
}
|
2023-04-24 11:37:11 +00:00
|
|
|
clientId := c.Param("clientId")
|
2023-03-17 16:07:49 +00:00
|
|
|
|
2025-09-19 08:47:28 +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)
|
2024-03-11 08:16:54 +00:00
|
|
|
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) {
|
2023-04-25 11:08:35 +00:00
|
|
|
clientId := c.Param("clientId")
|
2023-03-17 16:07:49 +00:00
|
|
|
|
|
|
|
|
inbound := &model.Inbound{}
|
2023-04-25 11:08:35 +00:00
|
|
|
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
|
|
|
|
|
}
|
|
|
|
|
|
2025-09-19 08:47:28 +00:00
|
|
|
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)
|
2024-03-11 08:16:54 +00:00
|
|
|
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)
|
2024-03-11 08:16:54 +00:00
|
|
|
if needRestart {
|
2023-03-17 16:07:49 +00:00
|
|
|
a.xrayService.SetToNeedRestart()
|
|
|
|
|
}
|
2023-02-09 19:18:06 +00:00
|
|
|
}
|
2023-04-09 19:43:18 +00:00
|
|
|
|
2025-09-20 07:35:50 +00:00
|
|
|
// resetAllTraffics resets all traffic counters across all inbounds.
|
2023-04-09 19:43:18 +00:00
|
|
|
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)
|
2023-04-09 19:43:18 +00:00
|
|
|
return
|
2023-06-04 21:02:19 +00:00
|
|
|
} else {
|
|
|
|
|
a.xrayService.SetToNeedRestart()
|
2023-04-09 19:43:18 +00:00
|
|
|
}
|
2025-05-09 03:46:29 +00:00
|
|
|
jsonMsg(c, I18nWeb(c, "pages.inbounds.toasts.resetAllTrafficSuccess"), nil)
|
2023-04-09 19:43:18 +00:00
|
|
|
}
|
|
|
|
|
|
2025-09-20 07:35:50 +00:00
|
|
|
// resetAllClientTraffics resets traffic counters for all clients in a specific inbound.
|
2023-04-09 19:43:18 +00:00
|
|
|
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)
|
2023-04-09 19:43:18 +00:00
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = a.inboundService.ResetAllClientTraffics(id)
|
|
|
|
|
if err != nil {
|
2025-05-09 03:46:29 +00:00
|
|
|
jsonMsg(c, I18nWeb(c, "somethingWentWrong"), err)
|
2023-04-09 19:43:18 +00:00
|
|
|
return
|
2023-06-04 21:02:19 +00:00
|
|
|
} else {
|
|
|
|
|
a.xrayService.SetToNeedRestart()
|
2023-04-09 19:43:18 +00:00
|
|
|
}
|
2025-05-09 03:46:29 +00:00
|
|
|
jsonMsg(c, I18nWeb(c, "pages.inbounds.toasts.resetAllClientTrafficSuccess"), nil)
|
2023-04-09 19:43:18 +00:00
|
|
|
}
|
2023-04-25 15:13:37 +00:00
|
|
|
|
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
|
2024-01-17 12:51:28 +00:00
|
|
|
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)
|
2024-01-17 12:51:28 +00:00
|
|
|
} 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.
|
2023-04-25 15:13:37 +00:00
|
|
|
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)
|
2023-04-25 15:13:37 +00:00
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
err = a.inboundService.DelDepletedClients(id)
|
|
|
|
|
if err != nil {
|
2025-05-09 03:46:29 +00:00
|
|
|
jsonMsg(c, I18nWeb(c, "somethingWentWrong"), err)
|
2023-04-25 15:13:37 +00:00
|
|
|
return
|
|
|
|
|
}
|
2025-05-09 03:46:29 +00:00
|
|
|
jsonMsg(c, I18nWeb(c, "pages.inbounds.toasts.delDepletedClientsSuccess"), nil)
|
2023-04-25 15:13:37 +00:00
|
|
|
}
|
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-07-22 21:43:48 +00:00
|
|
|
|
2025-09-20 07:35:50 +00:00
|
|
|
// lastOnline retrieves the last online timestamps for clients.
|
2025-08-31 16:33:50 +00:00
|
|
|
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.
|
2025-07-22 21:43:48 +00:00
|
|
|
func (a *InboundController) updateClientTraffic(c *gin.Context) {
|
|
|
|
|
email := c.Param("email")
|
2025-08-04 16:01:32 +00:00
|
|
|
|
2025-07-22 21:43:48 +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
|
|
|
|
2025-07-22 21:43:48 +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
|
|
|
|
2025-07-22 21:43:48 +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
|
|
|
|
2025-07-22 21:43:48 +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()
|
|
|
|
|
}
|
|
|
|
|
}
|