db.go 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190
  1. package database
  2. import (
  3. "bytes"
  4. "io"
  5. "io/fs"
  6. "log"
  7. "os"
  8. "path"
  9. "slices"
  10. "github.com/mhsanaei/3x-ui/config"
  11. "github.com/mhsanaei/3x-ui/database/model"
  12. "github.com/mhsanaei/3x-ui/util/crypto"
  13. "github.com/mhsanaei/3x-ui/xray"
  14. "gorm.io/driver/sqlite"
  15. "gorm.io/gorm"
  16. "gorm.io/gorm/logger"
  17. )
  18. var db *gorm.DB
  19. const (
  20. defaultUsername = "admin"
  21. defaultPassword = "admin"
  22. )
  23. func initModels() error {
  24. models := []any{
  25. &model.User{},
  26. &model.Inbound{},
  27. &model.OutboundTraffics{},
  28. &model.Setting{},
  29. &model.InboundClientIps{},
  30. &xray.ClientTraffic{},
  31. &model.HistoryOfSeeders{},
  32. }
  33. for _, model := range models {
  34. if err := db.AutoMigrate(model); err != nil {
  35. log.Printf("Error auto migrating model: %v", err)
  36. return err
  37. }
  38. }
  39. return nil
  40. }
  41. func initUser() error {
  42. empty, err := isTableEmpty("users")
  43. if err != nil {
  44. log.Printf("Error checking if users table is empty: %v", err)
  45. return err
  46. }
  47. if empty {
  48. hashedPassword, err := crypto.HashPasswordAsBcrypt(defaultPassword)
  49. if err != nil {
  50. log.Printf("Error hashing default password: %v", err)
  51. return err
  52. }
  53. user := &model.User{
  54. Username: defaultUsername,
  55. Password: hashedPassword,
  56. }
  57. return db.Create(user).Error
  58. }
  59. return nil
  60. }
  61. func runSeeders(isUsersEmpty bool) error {
  62. empty, err := isTableEmpty("history_of_seeders")
  63. if err != nil {
  64. log.Printf("Error checking if users table is empty: %v", err)
  65. return err
  66. }
  67. if empty && isUsersEmpty {
  68. hashSeeder := &model.HistoryOfSeeders{
  69. SeederName: "UserPasswordHash",
  70. }
  71. return db.Create(hashSeeder).Error
  72. } else {
  73. var seedersHistory []string
  74. db.Model(&model.HistoryOfSeeders{}).Pluck("seeder_name", &seedersHistory)
  75. if !slices.Contains(seedersHistory, "UserPasswordHash") && !isUsersEmpty {
  76. var users []model.User
  77. db.Find(&users)
  78. for _, user := range users {
  79. hashedPassword, err := crypto.HashPasswordAsBcrypt(user.Password)
  80. if err != nil {
  81. log.Printf("Error hashing password for user '%s': %v", user.Username, err)
  82. return err
  83. }
  84. db.Model(&user).Update("password", hashedPassword)
  85. }
  86. hashSeeder := &model.HistoryOfSeeders{
  87. SeederName: "UserPasswordHash",
  88. }
  89. return db.Create(hashSeeder).Error
  90. }
  91. }
  92. return nil
  93. }
  94. func isTableEmpty(tableName string) (bool, error) {
  95. var count int64
  96. err := db.Table(tableName).Count(&count).Error
  97. return count == 0, err
  98. }
  99. func InitDB(dbPath string) error {
  100. dir := path.Dir(dbPath)
  101. err := os.MkdirAll(dir, fs.ModePerm)
  102. if err != nil {
  103. return err
  104. }
  105. var gormLogger logger.Interface
  106. if config.IsDebug() {
  107. gormLogger = logger.Default
  108. } else {
  109. gormLogger = logger.Discard
  110. }
  111. c := &gorm.Config{
  112. Logger: gormLogger,
  113. }
  114. db, err = gorm.Open(sqlite.Open(dbPath), c)
  115. if err != nil {
  116. return err
  117. }
  118. if err := initModels(); err != nil {
  119. return err
  120. }
  121. isUsersEmpty, err := isTableEmpty("users")
  122. if err != nil {
  123. return err
  124. }
  125. if err := initUser(); err != nil {
  126. return err
  127. }
  128. return runSeeders(isUsersEmpty)
  129. }
  130. func CloseDB() error {
  131. if db != nil {
  132. sqlDB, err := db.DB()
  133. if err != nil {
  134. return err
  135. }
  136. return sqlDB.Close()
  137. }
  138. return nil
  139. }
  140. func GetDB() *gorm.DB {
  141. return db
  142. }
  143. func IsNotFound(err error) bool {
  144. return err == gorm.ErrRecordNotFound
  145. }
  146. func IsSQLiteDB(file io.ReaderAt) (bool, error) {
  147. signature := []byte("SQLite format 3\x00")
  148. buf := make([]byte, len(signature))
  149. _, err := file.ReadAt(buf, 0)
  150. if err != nil {
  151. return false, err
  152. }
  153. return bytes.Equal(buf, signature), nil
  154. }
  155. func Checkpoint() error {
  156. // Update WAL
  157. err := db.Exec("PRAGMA wal_checkpoint;").Error
  158. if err != nil {
  159. return err
  160. }
  161. return nil
  162. }