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

578 lines
17 KiB
Go
Raw Normal View History

2023-02-09 19:18:06 +00:00
package controller
import (
2023-12-08 19:08:44 +00:00
"encoding/json"
2026-04-06 14:12:38 +00:00
"errors"
2023-02-09 19:18:06 +00:00
"fmt"
"strconv"
2026-02-11 21:21:09 +00:00
"time"
2025-09-19 08:05:43 +00:00
"github.com/mhsanaei/3x-ui/v2/database/model"
"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"
2026-04-06 14:12:38 +00:00
"gorm.io/gorm"
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
settingService service.SettingService
2023-02-09 19:18:06 +00:00
}
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)
g.POST("/batchUpdateClients", a.batchUpdateInboundClients)
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
}
2026-04-06 14:12:38 +00:00
user := session.GetLoginUser(c)
inbound, err := a.inboundService.GetInboundForUser(user.Id, user.Role == "admin", id)
2023-02-09 19:18:06 +00:00
if err != nil {
2026-04-06 14:12:38 +00:00
if errors.Is(err, gorm.ErrRecordNotFound) {
jsonMsg(c, I18nWeb(c, "pages.inbounds.toasts.obtain"), errors.New("inbound not found"))
return
}
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) {
inbound := &model.Inbound{}
err := c.ShouldBind(inbound)
if err != nil {
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
}
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 {
jsonMsg(c, I18nWeb(c, "somethingWentWrong"), err)
return
}
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
}
2026-04-06 14:12:38 +00:00
user := session.GetLoginUser(c)
needRestart, err := a.inboundService.DelInboundForUser(user.Id, user.Role == "admin", 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
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
}
inbound := &model.Inbound{
Id: id,
}
err = c.ShouldBind(inbound)
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-04-06 14:12:38 +00:00
user := session.GetLoginUser(c)
inbound, needRestart, err := a.inboundService.UpdateInboundForUser(user.Id, user.Role == "admin", inbound)
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.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
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
}
2026-02-11 21:21:09 +00:00
// Prefer returning a normalized string list for consistent UI rendering
type ipWithTimestamp struct {
IP string `json:"ip"`
Timestamp int64 `json:"timestamp"`
}
var ipsWithTime []ipWithTimestamp
if err := json.Unmarshal([]byte(ips), &ipsWithTime); err == nil && len(ipsWithTime) > 0 {
formatted := make([]string, 0, len(ipsWithTime))
for _, item := range ipsWithTime {
if item.IP == "" {
continue
}
if item.Timestamp > 0 {
ts := time.Unix(item.Timestamp, 0).Local().Format("2006-01-02 15:04:05")
formatted = append(formatted, fmt.Sprintf("%s (%s)", item.IP, ts))
continue
}
formatted = append(formatted, item.IP)
}
jsonObj(c, formatted, nil)
return
}
var oldIps []string
if err := json.Unmarshal([]byte(ips), &oldIps); err == nil && len(oldIps) > 0 {
jsonObj(c, oldIps, nil)
return
}
// If parsing fails, return as string
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
}
2026-04-06 14:12:38 +00:00
user := session.GetLoginUser(c)
needRestart, err := a.inboundService.AddInboundClientForUser(user.Id, user.Role == "admin", 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
2026-04-06 14:12:38 +00:00
user := session.GetLoginUser(c)
needRestart, err := a.inboundService.DelInboundClientForUser(user.Id, user.Role == "admin", 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
}
2026-04-06 14:12:38 +00:00
user := session.GetLoginUser(c)
needRestart, err := a.inboundService.UpdateInboundClientForUser(user.Id, user.Role == "admin", 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")
2026-04-06 14:12:38 +00:00
user := session.GetLoginUser(c)
needRestart, err := a.inboundService.ResetClientTrafficForUser(user.Id, user.Role == "admin", 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
}
2026-04-06 14:12:38 +00:00
user := session.GetLoginUser(c)
err = a.inboundService.ResetAllClientTrafficsForUser(user.Id, user.Role == "admin", 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
}
2026-04-06 14:12:38 +00:00
user := session.GetLoginUser(c)
err = a.inboundService.DelDepletedClientsForUser(user.Id, user.Role == "admin", 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) {
2024-07-07 09:55:59 +00:00
jsonObj(c, a.inboundService.GetOnlineClients(), 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")
2026-04-06 14:12:38 +00:00
user := session.GetLoginUser(c)
needRestart, err := a.inboundService.DelInboundClientByEmailForUser(user.Id, user.Role == "admin", inboundId, email)
2025-09-10 14:36:12 +00:00
if err != nil {
jsonMsg(c, "Failed to delete client by email", err)
return
}
jsonMsg(c, "Client deleted successfully", nil)
if needRestart {
a.xrayService.SetToNeedRestart()
}
}
// batchUpdateInboundClients updates multiple clients in an inbound with the same field changes.
func (a *InboundController) batchUpdateInboundClients(c *gin.Context) {
var request struct {
InboundID int `json:"inboundId"`
ClientIDs []string `json:"clientIds"`
UpdateFields string `json:"updateFields"`
}
if err := c.ShouldBindJSON(&request); err != nil {
jsonMsg(c, I18nWeb(c, "pages.inbounds.toasts.inboundUpdateSuccess"), err)
return
}
user := session.GetLoginUser(c)
needRestart, err := a.inboundService.BatchUpdateInboundClientsForUser(
user.Id, user.Role == "admin", request.InboundID, request.ClientIDs, request.UpdateFields,
)
if err != nil {
jsonMsg(c, I18nWeb(c, "somethingWentWrong"), err)
return
}
jsonMsg(c, I18nWeb(c, "pages.inbounds.toasts.inboundClientUpdateSuccess"), nil)
if needRestart {
a.xrayService.SetToNeedRestart()
}
}
// getUserInfo returns client traffic information for the logged-in user.
func (a *InboundController) getUserInfo(c *gin.Context) {
user := session.GetLoginUser(c)
traffic, err := a.inboundService.GetClientTrafficByEmail(user.Username)
if err != nil {
jsonMsg(c, I18nWeb(c, "pages.inbounds.toasts.trafficGetError"), err)
return
}
jsonObj(c, traffic, nil)
}
// getUserSubscriptions returns subscription URLs for the logged-in user.
func (a *InboundController) getUserSubscriptions(c *gin.Context) {
user := session.GetLoginUser(c)
traffic, err := a.inboundService.GetClientTrafficByEmail(user.Username)
if err != nil || traffic == nil {
jsonObj(c, gin.H{"subClashEnable": false}, nil)
return
}
subId := traffic.SubId
if subId == "" {
jsonObj(c, gin.H{"subClashEnable": false}, nil)
return
}
settingsAny, err := a.settingService.GetDefaultSettings(c.Request.Host)
if err != nil {
jsonObj(c, gin.H{"subClashEnable": false}, nil)
return
}
settings, ok := settingsAny.(map[string]any)
if !ok {
jsonObj(c, gin.H{"subClashEnable": false}, nil)
return
}
subEnable := false
if v, ok := settings["subEnable"]; ok {
if b, ok2 := v.(bool); ok2 {
subEnable = b
}
}
subUrl := ""
if subEnable {
if uri, ok := settings["subURI"]; ok {
if s, ok2 := uri.(string); ok2 && s != "" {
subUrl = s + subId
}
}
}
subClashEnable := false
if v, ok := settings["subClashEnable"]; ok {
if b, ok2 := v.(bool); ok2 {
subClashEnable = b
}
}
subClashUrl := ""
if subClashEnable {
if uri, ok := settings["subClashURI"]; ok {
if s, ok2 := uri.(string); ok2 && s != "" {
subClashUrl = s + subId
}
}
}
jsonObj(c, gin.H{
"subId": subId,
"subEnable": subEnable,
"subUrl": subUrl,
"subClashEnable": subClashEnable,
"subClashUrl": subClashUrl,
}, nil)
}