user.go 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168
  1. package panel
  2. import (
  3. "errors"
  4. "github.com/mhsanaei/3x-ui/v3/internal/database"
  5. "github.com/mhsanaei/3x-ui/v3/internal/database/model"
  6. "github.com/mhsanaei/3x-ui/v3/internal/logger"
  7. "github.com/mhsanaei/3x-ui/v3/internal/util/crypto"
  8. ldaputil "github.com/mhsanaei/3x-ui/v3/internal/util/ldap"
  9. "github.com/mhsanaei/3x-ui/v3/internal/web/service"
  10. "github.com/xlzd/gotp"
  11. "gorm.io/gorm"
  12. )
  13. // UserService provides business logic for user management and authentication.
  14. // It handles user creation, login, password management, and 2FA operations.
  15. type UserService struct {
  16. settingService service.SettingService
  17. }
  18. // GetFirstUser retrieves the first user from the database.
  19. // This is typically used for initial setup or when there's only one admin user.
  20. func (s *UserService) GetFirstUser() (*model.User, error) {
  21. db := database.GetDB()
  22. user := &model.User{}
  23. err := db.Model(model.User{}).
  24. First(user).
  25. Error
  26. if err != nil {
  27. return nil, err
  28. }
  29. return user, nil
  30. }
  31. func (s *UserService) CheckUser(username string, password string, twoFactorCode string) (*model.User, error) {
  32. db := database.GetDB()
  33. user := &model.User{}
  34. err := db.Model(model.User{}).
  35. Where("username = ?", username).
  36. First(user).
  37. Error
  38. if err == gorm.ErrRecordNotFound {
  39. return nil, errors.New("invalid credentials")
  40. } else if err != nil {
  41. logger.Warning("check user err:", err)
  42. return nil, err
  43. }
  44. if !crypto.CheckPasswordHash(user.Password, password) {
  45. ldapEnabled, _ := s.settingService.GetLdapEnable()
  46. if !ldapEnabled {
  47. return nil, errors.New("invalid credentials")
  48. }
  49. host, _ := s.settingService.GetLdapHost()
  50. port, _ := s.settingService.GetLdapPort()
  51. useTLS, _ := s.settingService.GetLdapUseTLS()
  52. bindDN, _ := s.settingService.GetLdapBindDN()
  53. ldapPass, _ := s.settingService.GetLdapPassword()
  54. baseDN, _ := s.settingService.GetLdapBaseDN()
  55. userFilter, _ := s.settingService.GetLdapUserFilter()
  56. userAttr, _ := s.settingService.GetLdapUserAttr()
  57. cfg := ldaputil.Config{
  58. Host: host,
  59. Port: port,
  60. UseTLS: useTLS,
  61. BindDN: bindDN,
  62. Password: ldapPass,
  63. BaseDN: baseDN,
  64. UserFilter: userFilter,
  65. UserAttr: userAttr,
  66. }
  67. ok, err := ldaputil.AuthenticateUser(cfg, username, password)
  68. if err != nil || !ok {
  69. return nil, errors.New("invalid credentials")
  70. }
  71. }
  72. twoFactorEnable, err := s.settingService.GetTwoFactorEnable()
  73. if err != nil {
  74. logger.Warning("check two factor err:", err)
  75. return nil, err
  76. }
  77. if twoFactorEnable {
  78. twoFactorToken, err := s.settingService.GetTwoFactorToken()
  79. if err != nil {
  80. logger.Warning("check two factor token err:", err)
  81. return nil, err
  82. }
  83. if gotp.NewDefaultTOTP(twoFactorToken).Now() != twoFactorCode {
  84. return nil, errors.New("invalid 2fa code")
  85. }
  86. }
  87. return user, nil
  88. }
  89. func (s *UserService) BumpLoginEpoch() error {
  90. db := database.GetDB()
  91. return db.Model(model.User{}).
  92. Where("1 = 1").
  93. Update("login_epoch", gorm.Expr("login_epoch + 1")).
  94. Error
  95. }
  96. func (s *UserService) UpdateUser(id int, username string, password string) error {
  97. db := database.GetDB()
  98. hashedPassword, err := crypto.HashPasswordAsBcrypt(password)
  99. if err != nil {
  100. return err
  101. }
  102. twoFactorEnable, err := s.settingService.GetTwoFactorEnable()
  103. if err != nil {
  104. return err
  105. }
  106. if twoFactorEnable {
  107. s.settingService.SetTwoFactorEnable(false)
  108. s.settingService.SetTwoFactorToken("")
  109. }
  110. return db.Model(model.User{}).
  111. Where("id = ?", id).
  112. Updates(map[string]any{
  113. "username": username,
  114. "password": hashedPassword,
  115. "login_epoch": gorm.Expr("login_epoch + 1"),
  116. }).
  117. Error
  118. }
  119. func (s *UserService) UpdateFirstUser(username string, password string) error {
  120. if username == "" {
  121. return errors.New("username can not be empty")
  122. } else if password == "" {
  123. return errors.New("password can not be empty")
  124. }
  125. hashedPassword, er := crypto.HashPasswordAsBcrypt(password)
  126. if er != nil {
  127. return er
  128. }
  129. db := database.GetDB()
  130. user := &model.User{}
  131. err := db.Model(model.User{}).First(user).Error
  132. if database.IsNotFound(err) {
  133. user.Username = username
  134. user.Password = hashedPassword
  135. return db.Model(model.User{}).Create(user).Error
  136. } else if err != nil {
  137. return err
  138. }
  139. user.Username = username
  140. user.Password = hashedPassword
  141. user.LoginEpoch++
  142. return db.Save(user).Error
  143. }