| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259 |
- package controller
- import (
- "errors"
- "strconv"
- "time"
- "github.com/mhsanaei/3x-ui/v3/internal/logger"
- "github.com/mhsanaei/3x-ui/v3/internal/util/crypto"
- "github.com/mhsanaei/3x-ui/v3/internal/web/entity"
- "github.com/mhsanaei/3x-ui/v3/internal/web/middleware"
- "github.com/mhsanaei/3x-ui/v3/internal/web/service"
- "github.com/mhsanaei/3x-ui/v3/internal/web/service/email"
- "github.com/mhsanaei/3x-ui/v3/internal/web/service/panel"
- "github.com/mhsanaei/3x-ui/v3/internal/web/session"
- "github.com/gin-gonic/gin"
- )
- // updateUserForm represents the form for updating user credentials.
- type updateUserForm struct {
- OldUsername string `json:"oldUsername" form:"oldUsername"`
- OldPassword string `json:"oldPassword" form:"oldPassword"`
- NewUsername string `json:"newUsername" form:"newUsername"`
- NewPassword string `json:"newPassword" form:"newPassword"`
- }
- // SettingController handles settings and user management operations.
- type SettingController struct {
- settingService service.SettingService
- userService panel.UserService
- panelService panel.PanelService
- apiTokenService panel.ApiTokenService
- xrayService service.XrayService
- }
- // NewSettingController creates a new SettingController and initializes its routes.
- func NewSettingController(g *gin.RouterGroup) *SettingController {
- a := &SettingController{}
- a.initRouter(g)
- return a
- }
- // initRouter sets up the routes for settings management.
- func (a *SettingController) initRouter(g *gin.RouterGroup) {
- g = g.Group("/setting")
- g.POST("/all", a.getAllSetting)
- g.POST("/defaultSettings", a.getDefaultSettings)
- g.POST("/update", a.updateSetting)
- g.POST("/updateUser", a.updateUser)
- g.POST("/restartPanel", a.restartPanel)
- g.GET("/getDefaultJsonConfig", a.getDefaultXrayConfig)
- g.GET("/apiTokens", a.listApiTokens)
- g.POST("/apiTokens/create", a.createApiToken)
- g.POST("/apiTokens/delete/:id", a.deleteApiToken)
- g.POST("/apiTokens/setEnabled/:id", a.setApiTokenEnabled)
- g.POST("/testSmtp", a.testSmtp)
- g.POST("/testTgBot", a.testTgBot)
- }
- // getAllSetting retrieves all current settings as the browser-safe view:
- // secret values are redacted and surfaced as has* presence flags instead.
- func (a *SettingController) getAllSetting(c *gin.Context) {
- allSetting, err := a.settingService.GetAllSettingView()
- if err != nil {
- jsonMsg(c, I18nWeb(c, "pages.settings.toasts.getSettings"), err)
- return
- }
- jsonObj(c, allSetting, nil)
- }
- // getDefaultSettings retrieves the default settings based on the host.
- func (a *SettingController) getDefaultSettings(c *gin.Context) {
- result, err := a.settingService.GetDefaultSettings(c.Request.Host)
- if err != nil {
- jsonMsg(c, I18nWeb(c, "pages.settings.toasts.getSettings"), err)
- return
- }
- jsonObj(c, result, nil)
- }
- // updateSetting updates all settings with the provided data.
- func (a *SettingController) updateSetting(c *gin.Context) {
- allSetting, ok := middleware.BindAndValidate[entity.AllSetting](c)
- if !ok {
- return
- }
- oldTwoFactor, twoFactorErr := a.settingService.GetTwoFactorEnable()
- oldPanelOutbound, _ := a.settingService.GetPanelOutbound()
- err := a.settingService.UpdateAllSetting(allSetting)
- if err == nil && twoFactorErr == nil && !oldTwoFactor && allSetting.TwoFactorEnable {
- if bumpErr := a.userService.BumpLoginEpoch(); bumpErr != nil {
- err = bumpErr
- }
- }
- if err == nil && allSetting.PanelOutbound != oldPanelOutbound {
- // The egress bridge lives in the generated config; reconcile the
- // running core. One SOCKS inbound plus one routing rule — both
- // hot-appliable, so this normally does not restart Xray.
- if applyErr := a.xrayService.RestartXray(false); applyErr != nil {
- logger.Warning("apply panel outbound change failed:", applyErr)
- }
- }
- jsonMsg(c, I18nWeb(c, "pages.settings.toasts.modifySettings"), err)
- }
- // updateUser updates the current user's username and password.
- func (a *SettingController) updateUser(c *gin.Context) {
- form := &updateUserForm{}
- err := c.ShouldBind(form)
- if err != nil {
- jsonMsg(c, I18nWeb(c, "pages.settings.toasts.modifySettings"), err)
- return
- }
- user := session.GetLoginUser(c)
- if user.Username != form.OldUsername || !crypto.CheckPasswordHash(user.Password, form.OldPassword) {
- jsonMsg(c, I18nWeb(c, "pages.settings.toasts.modifyUserError"), errors.New(I18nWeb(c, "pages.settings.toasts.originalUserPassIncorrect")))
- return
- }
- if form.NewUsername == "" || form.NewPassword == "" {
- jsonMsg(c, I18nWeb(c, "pages.settings.toasts.modifyUserError"), errors.New(I18nWeb(c, "pages.settings.toasts.userPassMustBeNotEmpty")))
- return
- }
- err = a.userService.UpdateUser(user.Id, form.NewUsername, form.NewPassword)
- if err == nil {
- user.Username = form.NewUsername
- user.Password, _ = crypto.HashPasswordAsBcrypt(form.NewPassword)
- if saveErr := session.SetLoginUser(c, user); saveErr != nil {
- err = saveErr
- }
- }
- jsonMsg(c, I18nWeb(c, "pages.settings.toasts.modifyUser"), err)
- }
- // restartPanel restarts the panel service after a delay.
- func (a *SettingController) restartPanel(c *gin.Context) {
- err := a.panelService.RestartPanel(time.Second * 3)
- jsonMsg(c, I18nWeb(c, "pages.settings.restartPanelSuccess"), err)
- }
- // getDefaultXrayConfig retrieves the default Xray configuration.
- func (a *SettingController) getDefaultXrayConfig(c *gin.Context) {
- defaultJsonConfig, err := a.settingService.GetDefaultXrayConfig()
- if err != nil {
- jsonMsg(c, I18nWeb(c, "pages.settings.toasts.getSettings"), err)
- return
- }
- jsonObj(c, defaultJsonConfig, nil)
- }
- type apiTokenCreateForm struct {
- Name string `json:"name" form:"name"`
- }
- type apiTokenEnabledForm struct {
- Enabled bool `json:"enabled" form:"enabled"`
- }
- func (a *SettingController) listApiTokens(c *gin.Context) {
- rows, err := a.apiTokenService.List()
- if err != nil {
- jsonMsg(c, I18nWeb(c, "pages.settings.toasts.getSettings"), err)
- return
- }
- jsonObj(c, rows, nil)
- }
- func (a *SettingController) createApiToken(c *gin.Context) {
- form := &apiTokenCreateForm{}
- if err := c.ShouldBind(form); err != nil {
- jsonMsg(c, I18nWeb(c, "pages.settings.toasts.modifySettings"), err)
- return
- }
- row, err := a.apiTokenService.Create(form.Name)
- if err != nil {
- jsonMsg(c, I18nWeb(c, "pages.settings.toasts.modifySettings"), err)
- return
- }
- jsonObj(c, row, nil)
- }
- func (a *SettingController) deleteApiToken(c *gin.Context) {
- id, err := strconv.Atoi(c.Param("id"))
- if err != nil {
- jsonMsg(c, I18nWeb(c, "pages.settings.toasts.modifySettings"), err)
- return
- }
- jsonMsg(c, I18nWeb(c, "pages.settings.toasts.modifySettings"), a.apiTokenService.Delete(id))
- }
- func (a *SettingController) setApiTokenEnabled(c *gin.Context) {
- id, err := strconv.Atoi(c.Param("id"))
- if err != nil {
- jsonMsg(c, I18nWeb(c, "pages.settings.toasts.modifySettings"), err)
- return
- }
- form := &apiTokenEnabledForm{}
- if bindErr := c.ShouldBind(form); bindErr != nil {
- jsonMsg(c, I18nWeb(c, "pages.settings.toasts.modifySettings"), bindErr)
- return
- }
- jsonMsg(c, I18nWeb(c, "pages.settings.toasts.modifySettings"), a.apiTokenService.SetEnabled(id, form.Enabled))
- }
- func (a *SettingController) testSmtp(c *gin.Context) {
- if emailService == nil {
- jsonMsg(c, I18nWeb(c, "pages.settings.smtpNotInitialized"), errors.New("email service not available"))
- return
- }
- logger.Info("SMTP test: starting...")
- result := emailService.TestConnection()
- if !result.Success {
- logger.Warning("SMTP test failed at", result.Stage+":", result.Message)
- c.JSON(200, gin.H{
- "success": false,
- "stage": result.Stage,
- "msg": result.Message,
- })
- return
- }
- logger.Info("SMTP test: success")
- c.JSON(200, gin.H{
- "success": true,
- "stage": result.Stage,
- "msg": result.Message,
- })
- }
- func (a *SettingController) testTgBot(c *gin.Context) {
- enabled, err := a.settingService.GetTgbotEnabled()
- if err != nil || !enabled {
- jsonMsg(c, I18nWeb(c, "pages.settings.tgBotNotEnabled"), errors.New("telegram bot disabled"))
- return
- }
- // Import tgbot package would create a circular dependency, so we call
- // the test through the global function registered at startup.
- if testTgFunc != nil {
- if err := testTgFunc(); err != nil {
- jsonMsg(c, I18nWeb(c, "pages.settings.tgTestFailed")+": "+err.Error(), err)
- return
- }
- jsonMsg(c, I18nWeb(c, "pages.settings.tgTestSuccess"), nil)
- return
- }
- jsonMsg(c, I18nWeb(c, "pages.settings.tgBotNotRunning"), errors.New("bot not started"))
- }
- // testTgFunc is set from web layer to test Telegram sending without circular imports.
- var testTgFunc func() error
- // SetTestTgFunc registers the function used to test Telegram sending.
- func SetTestTgFunc(fn func() error) { testTgFunc = fn }
- // emailService is set from web layer.
- var emailService *email.EmailService
- // SetEmailService registers the email service for test endpoints.
- func SetEmailService(s *email.EmailService) { emailService = s }
|