db.go 3.8 KB

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