setting.go 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155
  1. package controller
  2. import (
  3. "errors"
  4. "time"
  5. "github.com/mhsanaei/3x-ui/v3/util/crypto"
  6. "github.com/mhsanaei/3x-ui/v3/web/entity"
  7. "github.com/mhsanaei/3x-ui/v3/web/service"
  8. "github.com/mhsanaei/3x-ui/v3/web/session"
  9. "github.com/gin-gonic/gin"
  10. )
  11. // updateUserForm represents the form for updating user credentials.
  12. type updateUserForm struct {
  13. OldUsername string `json:"oldUsername" form:"oldUsername"`
  14. OldPassword string `json:"oldPassword" form:"oldPassword"`
  15. NewUsername string `json:"newUsername" form:"newUsername"`
  16. NewPassword string `json:"newPassword" form:"newPassword"`
  17. }
  18. // SettingController handles settings and user management operations.
  19. type SettingController struct {
  20. settingService service.SettingService
  21. userService service.UserService
  22. panelService service.PanelService
  23. }
  24. // NewSettingController creates a new SettingController and initializes its routes.
  25. func NewSettingController(g *gin.RouterGroup) *SettingController {
  26. a := &SettingController{}
  27. a.initRouter(g)
  28. return a
  29. }
  30. // initRouter sets up the routes for settings management.
  31. func (a *SettingController) initRouter(g *gin.RouterGroup) {
  32. g = g.Group("/setting")
  33. g.POST("/all", a.getAllSetting)
  34. g.POST("/defaultSettings", a.getDefaultSettings)
  35. g.POST("/update", a.updateSetting)
  36. g.POST("/updateUser", a.updateUser)
  37. g.POST("/restartPanel", a.restartPanel)
  38. g.GET("/getDefaultJsonConfig", a.getDefaultXrayConfig)
  39. g.GET("/getApiToken", a.getApiToken)
  40. g.POST("/regenerateApiToken", a.regenerateApiToken)
  41. }
  42. // getAllSetting retrieves all current settings.
  43. func (a *SettingController) getAllSetting(c *gin.Context) {
  44. allSetting, err := a.settingService.GetAllSetting()
  45. if err != nil {
  46. jsonMsg(c, I18nWeb(c, "pages.settings.toasts.getSettings"), err)
  47. return
  48. }
  49. jsonObj(c, allSetting, nil)
  50. }
  51. // getDefaultSettings retrieves the default settings based on the host.
  52. func (a *SettingController) getDefaultSettings(c *gin.Context) {
  53. result, err := a.settingService.GetDefaultSettings(c.Request.Host)
  54. if err != nil {
  55. jsonMsg(c, I18nWeb(c, "pages.settings.toasts.getSettings"), err)
  56. return
  57. }
  58. jsonObj(c, result, nil)
  59. }
  60. // updateSetting updates all settings with the provided data.
  61. func (a *SettingController) updateSetting(c *gin.Context) {
  62. allSetting := &entity.AllSetting{}
  63. err := c.ShouldBind(allSetting)
  64. if err != nil {
  65. jsonMsg(c, I18nWeb(c, "pages.settings.toasts.modifySettings"), err)
  66. return
  67. }
  68. oldTwoFactor, twoFactorErr := a.settingService.GetTwoFactorEnable()
  69. err = a.settingService.UpdateAllSetting(allSetting)
  70. if err == nil && twoFactorErr == nil && !oldTwoFactor && allSetting.TwoFactorEnable {
  71. if bumpErr := a.userService.BumpLoginEpoch(); bumpErr != nil {
  72. err = bumpErr
  73. }
  74. }
  75. jsonMsg(c, I18nWeb(c, "pages.settings.toasts.modifySettings"), err)
  76. }
  77. // updateUser updates the current user's username and password.
  78. func (a *SettingController) updateUser(c *gin.Context) {
  79. form := &updateUserForm{}
  80. err := c.ShouldBind(form)
  81. if err != nil {
  82. jsonMsg(c, I18nWeb(c, "pages.settings.toasts.modifySettings"), err)
  83. return
  84. }
  85. user := session.GetLoginUser(c)
  86. if user.Username != form.OldUsername || !crypto.CheckPasswordHash(user.Password, form.OldPassword) {
  87. jsonMsg(c, I18nWeb(c, "pages.settings.toasts.modifyUserError"), errors.New(I18nWeb(c, "pages.settings.toasts.originalUserPassIncorrect")))
  88. return
  89. }
  90. if form.NewUsername == "" || form.NewPassword == "" {
  91. jsonMsg(c, I18nWeb(c, "pages.settings.toasts.modifyUserError"), errors.New(I18nWeb(c, "pages.settings.toasts.userPassMustBeNotEmpty")))
  92. return
  93. }
  94. err = a.userService.UpdateUser(user.Id, form.NewUsername, form.NewPassword)
  95. if err == nil {
  96. user.Username = form.NewUsername
  97. user.Password, _ = crypto.HashPasswordAsBcrypt(form.NewPassword)
  98. if saveErr := session.SetLoginUser(c, user); saveErr != nil {
  99. err = saveErr
  100. }
  101. }
  102. jsonMsg(c, I18nWeb(c, "pages.settings.toasts.modifyUser"), err)
  103. }
  104. // restartPanel restarts the panel service after a delay.
  105. func (a *SettingController) restartPanel(c *gin.Context) {
  106. err := a.panelService.RestartPanel(time.Second * 3)
  107. jsonMsg(c, I18nWeb(c, "pages.settings.restartPanelSuccess"), err)
  108. }
  109. // getDefaultXrayConfig retrieves the default Xray configuration.
  110. func (a *SettingController) getDefaultXrayConfig(c *gin.Context) {
  111. defaultJsonConfig, err := a.settingService.GetDefaultXrayConfig()
  112. if err != nil {
  113. jsonMsg(c, I18nWeb(c, "pages.settings.toasts.getSettings"), err)
  114. return
  115. }
  116. jsonObj(c, defaultJsonConfig, nil)
  117. }
  118. // getApiToken returns the panel's API token used by remote central
  119. // panels to authenticate as Bearer tokens. The token is auto-generated
  120. // on first read so existing installs upgrade transparently.
  121. func (a *SettingController) getApiToken(c *gin.Context) {
  122. tok, err := a.settingService.GetApiToken()
  123. if err != nil {
  124. jsonMsg(c, I18nWeb(c, "pages.settings.toasts.getSettings"), err)
  125. return
  126. }
  127. jsonObj(c, tok, nil)
  128. }
  129. // regenerateApiToken rotates the API token. Any central panel that had
  130. // the old value cached will start failing heartbeats until it is updated
  131. // with the new token — that's intentional, it's the whole point of rotation.
  132. func (a *SettingController) regenerateApiToken(c *gin.Context) {
  133. tok, err := a.settingService.RegenerateApiToken()
  134. if err != nil {
  135. jsonMsg(c, I18nWeb(c, "pages.settings.toasts.modifySettings"), err)
  136. return
  137. }
  138. jsonObj(c, tok, nil)
  139. }