db.go 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228
  1. // Package database provides database initialization, migration, and management utilities
  2. // for the 3x-ui panel using GORM with SQLite.
  3. package database
  4. import (
  5. "bytes"
  6. "errors"
  7. "io"
  8. "io/fs"
  9. "log"
  10. "os"
  11. "path"
  12. "slices"
  13. "github.com/mhsanaei/3x-ui/v2/config"
  14. "github.com/mhsanaei/3x-ui/v2/database/model"
  15. "github.com/mhsanaei/3x-ui/v2/util/crypto"
  16. "github.com/mhsanaei/3x-ui/v2/xray"
  17. "gorm.io/driver/sqlite"
  18. "gorm.io/gorm"
  19. "gorm.io/gorm/logger"
  20. )
  21. var db *gorm.DB
  22. const (
  23. defaultUsername = "admin"
  24. defaultPassword = "admin"
  25. )
  26. func initModels() error {
  27. models := []any{
  28. &model.User{},
  29. &model.Inbound{},
  30. &model.OutboundTraffics{},
  31. &model.Setting{},
  32. &model.InboundClientIps{},
  33. &xray.ClientTraffic{},
  34. &model.HistoryOfSeeders{},
  35. }
  36. for _, model := range models {
  37. if err := db.AutoMigrate(model); err != nil {
  38. log.Printf("Error auto migrating model: %v", err)
  39. return err
  40. }
  41. }
  42. return nil
  43. }
  44. // initUser creates a default admin user if the users table is empty.
  45. func initUser() error {
  46. empty, err := isTableEmpty("users")
  47. if err != nil {
  48. log.Printf("Error checking if users table is empty: %v", err)
  49. return err
  50. }
  51. if empty {
  52. hashedPassword, err := crypto.HashPasswordAsBcrypt(defaultPassword)
  53. if err != nil {
  54. log.Printf("Error hashing default password: %v", err)
  55. return err
  56. }
  57. user := &model.User{
  58. Username: defaultUsername,
  59. Password: hashedPassword,
  60. }
  61. return db.Create(user).Error
  62. }
  63. return nil
  64. }
  65. // runSeeders migrates user passwords to bcrypt and records seeder execution to prevent re-running.
  66. func runSeeders(isUsersEmpty bool) error {
  67. empty, err := isTableEmpty("history_of_seeders")
  68. if err != nil {
  69. log.Printf("Error checking if users table is empty: %v", err)
  70. return err
  71. }
  72. if empty && isUsersEmpty {
  73. hashSeeder := &model.HistoryOfSeeders{
  74. SeederName: "UserPasswordHash",
  75. }
  76. return db.Create(hashSeeder).Error
  77. } else {
  78. var seedersHistory []string
  79. db.Model(&model.HistoryOfSeeders{}).Pluck("seeder_name", &seedersHistory)
  80. if !slices.Contains(seedersHistory, "UserPasswordHash") && !isUsersEmpty {
  81. var users []model.User
  82. db.Find(&users)
  83. for _, user := range users {
  84. hashedPassword, err := crypto.HashPasswordAsBcrypt(user.Password)
  85. if err != nil {
  86. log.Printf("Error hashing password for user '%s': %v", user.Username, err)
  87. return err
  88. }
  89. db.Model(&user).Update("password", hashedPassword)
  90. }
  91. hashSeeder := &model.HistoryOfSeeders{
  92. SeederName: "UserPasswordHash",
  93. }
  94. return db.Create(hashSeeder).Error
  95. }
  96. }
  97. return nil
  98. }
  99. // isTableEmpty returns true if the named table contains zero rows.
  100. func isTableEmpty(tableName string) (bool, error) {
  101. var count int64
  102. err := db.Table(tableName).Count(&count).Error
  103. return count == 0, err
  104. }
  105. // InitDB sets up the database connection, migrates models, and runs seeders.
  106. func InitDB(dbPath string) error {
  107. dir := path.Dir(dbPath)
  108. err := os.MkdirAll(dir, fs.ModePerm)
  109. if err != nil {
  110. return err
  111. }
  112. var gormLogger logger.Interface
  113. if config.IsDebug() {
  114. gormLogger = logger.Default
  115. } else {
  116. gormLogger = logger.Discard
  117. }
  118. c := &gorm.Config{
  119. Logger: gormLogger,
  120. }
  121. db, err = gorm.Open(sqlite.Open(dbPath), c)
  122. if err != nil {
  123. return err
  124. }
  125. if err := initModels(); err != nil {
  126. return err
  127. }
  128. isUsersEmpty, err := isTableEmpty("users")
  129. if err != nil {
  130. return err
  131. }
  132. if err := initUser(); err != nil {
  133. return err
  134. }
  135. return runSeeders(isUsersEmpty)
  136. }
  137. // CloseDB closes the database connection if it exists.
  138. func CloseDB() error {
  139. if db != nil {
  140. sqlDB, err := db.DB()
  141. if err != nil {
  142. return err
  143. }
  144. return sqlDB.Close()
  145. }
  146. return nil
  147. }
  148. // GetDB returns the global GORM database instance.
  149. func GetDB() *gorm.DB {
  150. return db
  151. }
  152. // IsNotFound checks if the given error is a GORM record not found error.
  153. func IsNotFound(err error) bool {
  154. return err == gorm.ErrRecordNotFound
  155. }
  156. // IsSQLiteDB checks if the given file is a valid SQLite database by reading its signature.
  157. func IsSQLiteDB(file io.ReaderAt) (bool, error) {
  158. signature := []byte("SQLite format 3\x00")
  159. buf := make([]byte, len(signature))
  160. _, err := file.ReadAt(buf, 0)
  161. if err != nil {
  162. return false, err
  163. }
  164. return bytes.Equal(buf, signature), nil
  165. }
  166. // Checkpoint performs a WAL checkpoint on the SQLite database to ensure data consistency.
  167. func Checkpoint() error {
  168. // Update WAL
  169. err := db.Exec("PRAGMA wal_checkpoint;").Error
  170. if err != nil {
  171. return err
  172. }
  173. return nil
  174. }
  175. // ValidateSQLiteDB opens the provided sqlite DB path with a throw-away connection
  176. // and runs a PRAGMA integrity_check to ensure the file is structurally sound.
  177. // It does not mutate global state or run migrations.
  178. func ValidateSQLiteDB(dbPath string) error {
  179. if _, err := os.Stat(dbPath); err != nil { // file must exist
  180. return err
  181. }
  182. gdb, err := gorm.Open(sqlite.Open(dbPath), &gorm.Config{Logger: logger.Discard})
  183. if err != nil {
  184. return err
  185. }
  186. sqlDB, err := gdb.DB()
  187. if err != nil {
  188. return err
  189. }
  190. defer sqlDB.Close()
  191. var res string
  192. if err := gdb.Raw("PRAGMA integrity_check;").Scan(&res).Error; err != nil {
  193. return err
  194. }
  195. if res != "ok" {
  196. return errors.New("sqlite integrity check failed: " + res)
  197. }
  198. return nil
  199. }