main.go 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306
  1. package model
  2. import (
  3. "log"
  4. "one-api/common"
  5. "one-api/constant"
  6. "os"
  7. "strings"
  8. "sync"
  9. "time"
  10. "github.com/glebarez/sqlite"
  11. "gorm.io/driver/mysql"
  12. "gorm.io/driver/postgres"
  13. "gorm.io/gorm"
  14. )
  15. var groupCol string
  16. var keyCol string
  17. func initCol() {
  18. if common.UsingPostgreSQL {
  19. groupCol = `"group"`
  20. keyCol = `"key"`
  21. } else {
  22. groupCol = "`group`"
  23. keyCol = "`key`"
  24. }
  25. }
  26. var DB *gorm.DB
  27. var LOG_DB *gorm.DB
  28. func createRootAccountIfNeed() error {
  29. var user User
  30. //if user.Status != common.UserStatusEnabled {
  31. if err := DB.First(&user).Error; err != nil {
  32. common.SysLog("no user exists, create a root user for you: username is root, password is 123456")
  33. hashedPassword, err := common.Password2Hash("123456")
  34. if err != nil {
  35. return err
  36. }
  37. rootUser := User{
  38. Username: "root",
  39. Password: hashedPassword,
  40. Role: common.RoleRootUser,
  41. Status: common.UserStatusEnabled,
  42. DisplayName: "Root User",
  43. AccessToken: nil,
  44. Quota: 100000000,
  45. }
  46. DB.Create(&rootUser)
  47. }
  48. return nil
  49. }
  50. func CheckSetup() {
  51. setup := GetSetup()
  52. if setup == nil {
  53. // No setup record exists, check if we have a root user
  54. if RootUserExists() {
  55. common.SysLog("system is not initialized, but root user exists")
  56. // Create setup record
  57. newSetup := Setup{
  58. Version: common.Version,
  59. InitializedAt: time.Now().Unix(),
  60. }
  61. err := DB.Create(&newSetup).Error
  62. if err != nil {
  63. common.SysLog("failed to create setup record: " + err.Error())
  64. }
  65. constant.Setup = true
  66. } else {
  67. common.SysLog("system is not initialized and no root user exists")
  68. constant.Setup = false
  69. }
  70. } else {
  71. // Setup record exists, system is initialized
  72. common.SysLog("system is already initialized at: " + time.Unix(setup.InitializedAt, 0).String())
  73. constant.Setup = true
  74. }
  75. }
  76. func chooseDB(envName string) (*gorm.DB, error) {
  77. defer func() {
  78. initCol()
  79. }()
  80. dsn := os.Getenv(envName)
  81. if dsn != "" {
  82. if strings.HasPrefix(dsn, "postgres://") || strings.HasPrefix(dsn, "postgresql://") {
  83. // Use PostgreSQL
  84. common.SysLog("using PostgreSQL as database")
  85. common.UsingPostgreSQL = true
  86. return gorm.Open(postgres.New(postgres.Config{
  87. DSN: dsn,
  88. PreferSimpleProtocol: true, // disables implicit prepared statement usage
  89. }), &gorm.Config{
  90. PrepareStmt: true, // precompile SQL
  91. })
  92. }
  93. if strings.HasPrefix(dsn, "local") {
  94. common.SysLog("SQL_DSN not set, using SQLite as database")
  95. common.UsingSQLite = true
  96. return gorm.Open(sqlite.Open(common.SQLitePath), &gorm.Config{
  97. PrepareStmt: true, // precompile SQL
  98. })
  99. }
  100. // Use MySQL
  101. common.SysLog("using MySQL as database")
  102. // check parseTime
  103. if !strings.Contains(dsn, "parseTime") {
  104. if strings.Contains(dsn, "?") {
  105. dsn += "&parseTime=true"
  106. } else {
  107. dsn += "?parseTime=true"
  108. }
  109. }
  110. common.UsingMySQL = true
  111. return gorm.Open(mysql.Open(dsn), &gorm.Config{
  112. PrepareStmt: true, // precompile SQL
  113. })
  114. }
  115. // Use SQLite
  116. common.SysLog("SQL_DSN not set, using SQLite as database")
  117. common.UsingSQLite = true
  118. return gorm.Open(sqlite.Open(common.SQLitePath), &gorm.Config{
  119. PrepareStmt: true, // precompile SQL
  120. })
  121. }
  122. func InitDB() (err error) {
  123. db, err := chooseDB("SQL_DSN")
  124. if err == nil {
  125. if common.DebugEnabled {
  126. db = db.Debug()
  127. }
  128. DB = db
  129. sqlDB, err := DB.DB()
  130. if err != nil {
  131. return err
  132. }
  133. sqlDB.SetMaxIdleConns(common.GetEnvOrDefault("SQL_MAX_IDLE_CONNS", 100))
  134. sqlDB.SetMaxOpenConns(common.GetEnvOrDefault("SQL_MAX_OPEN_CONNS", 1000))
  135. sqlDB.SetConnMaxLifetime(time.Second * time.Duration(common.GetEnvOrDefault("SQL_MAX_LIFETIME", 60)))
  136. if !common.IsMasterNode {
  137. return nil
  138. }
  139. if common.UsingMySQL {
  140. _, _ = sqlDB.Exec("ALTER TABLE channels MODIFY model_mapping TEXT;") // TODO: delete this line when most users have upgraded
  141. }
  142. common.SysLog("database migration started")
  143. err = migrateDB()
  144. return err
  145. } else {
  146. common.FatalLog(err)
  147. }
  148. return err
  149. }
  150. func InitLogDB() (err error) {
  151. if os.Getenv("LOG_SQL_DSN") == "" {
  152. LOG_DB = DB
  153. return
  154. }
  155. db, err := chooseDB("LOG_SQL_DSN")
  156. if err == nil {
  157. if common.DebugEnabled {
  158. db = db.Debug()
  159. }
  160. LOG_DB = db
  161. sqlDB, err := LOG_DB.DB()
  162. if err != nil {
  163. return err
  164. }
  165. sqlDB.SetMaxIdleConns(common.GetEnvOrDefault("SQL_MAX_IDLE_CONNS", 100))
  166. sqlDB.SetMaxOpenConns(common.GetEnvOrDefault("SQL_MAX_OPEN_CONNS", 1000))
  167. sqlDB.SetConnMaxLifetime(time.Second * time.Duration(common.GetEnvOrDefault("SQL_MAX_LIFETIME", 60)))
  168. if !common.IsMasterNode {
  169. return nil
  170. }
  171. //if common.UsingMySQL {
  172. // _, _ = sqlDB.Exec("DROP INDEX idx_channels_key ON channels;") // TODO: delete this line when most users have upgraded
  173. // _, _ = sqlDB.Exec("ALTER TABLE midjourneys MODIFY action VARCHAR(40);") // TODO: delete this line when most users have upgraded
  174. // _, _ = sqlDB.Exec("ALTER TABLE midjourneys MODIFY progress VARCHAR(30);") // TODO: delete this line when most users have upgraded
  175. // _, _ = sqlDB.Exec("ALTER TABLE midjourneys MODIFY status VARCHAR(20);") // TODO: delete this line when most users have upgraded
  176. //}
  177. common.SysLog("database migration started")
  178. err = migrateLOGDB()
  179. return err
  180. } else {
  181. common.FatalLog(err)
  182. }
  183. return err
  184. }
  185. func migrateDB() error {
  186. err := DB.AutoMigrate(&Channel{})
  187. if err != nil {
  188. return err
  189. }
  190. err = DB.AutoMigrate(&Token{})
  191. if err != nil {
  192. return err
  193. }
  194. err = DB.AutoMigrate(&User{})
  195. if err != nil {
  196. return err
  197. }
  198. err = DB.AutoMigrate(&Option{})
  199. if err != nil {
  200. return err
  201. }
  202. err = DB.AutoMigrate(&Redemption{})
  203. if err != nil {
  204. return err
  205. }
  206. err = DB.AutoMigrate(&Ability{})
  207. if err != nil {
  208. return err
  209. }
  210. err = DB.AutoMigrate(&Log{})
  211. if err != nil {
  212. return err
  213. }
  214. err = DB.AutoMigrate(&Midjourney{})
  215. if err != nil {
  216. return err
  217. }
  218. err = DB.AutoMigrate(&TopUp{})
  219. if err != nil {
  220. return err
  221. }
  222. err = DB.AutoMigrate(&QuotaData{})
  223. if err != nil {
  224. return err
  225. }
  226. err = DB.AutoMigrate(&Task{})
  227. if err != nil {
  228. return err
  229. }
  230. err = DB.AutoMigrate(&Setup{})
  231. common.SysLog("database migrated")
  232. //err = createRootAccountIfNeed()
  233. return err
  234. }
  235. func migrateLOGDB() error {
  236. var err error
  237. if err = LOG_DB.AutoMigrate(&Log{}); err != nil {
  238. return err
  239. }
  240. return nil
  241. }
  242. func closeDB(db *gorm.DB) error {
  243. sqlDB, err := db.DB()
  244. if err != nil {
  245. return err
  246. }
  247. err = sqlDB.Close()
  248. return err
  249. }
  250. func CloseDB() error {
  251. if LOG_DB != DB {
  252. err := closeDB(LOG_DB)
  253. if err != nil {
  254. return err
  255. }
  256. }
  257. return closeDB(DB)
  258. }
  259. var (
  260. lastPingTime time.Time
  261. pingMutex sync.Mutex
  262. )
  263. func PingDB() error {
  264. pingMutex.Lock()
  265. defer pingMutex.Unlock()
  266. if time.Since(lastPingTime) < time.Second*10 {
  267. return nil
  268. }
  269. sqlDB, err := DB.DB()
  270. if err != nil {
  271. log.Printf("Error getting sql.DB from GORM: %v", err)
  272. return err
  273. }
  274. err = sqlDB.Ping()
  275. if err != nil {
  276. log.Printf("Error pinging DB: %v", err)
  277. return err
  278. }
  279. lastPingTime = time.Now()
  280. common.SysLog("Database pinged successfully")
  281. return nil
  282. }