redemption.go 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185
  1. package model
  2. import (
  3. "errors"
  4. "fmt"
  5. "one-api/common"
  6. "strconv"
  7. "gorm.io/gorm"
  8. )
  9. type Redemption struct {
  10. Id int `json:"id"`
  11. UserId int `json:"user_id"`
  12. Key string `json:"key" gorm:"type:char(32);uniqueIndex"`
  13. Status int `json:"status" gorm:"default:1"`
  14. Name string `json:"name" gorm:"index"`
  15. Quota int `json:"quota" gorm:"default:100"`
  16. CreatedTime int64 `json:"created_time" gorm:"bigint"`
  17. RedeemedTime int64 `json:"redeemed_time" gorm:"bigint"`
  18. Count int `json:"count" gorm:"-:all"` // only for api request
  19. UsedUserId int `json:"used_user_id"`
  20. DeletedAt gorm.DeletedAt `gorm:"index"`
  21. }
  22. func GetAllRedemptions(startIdx int, num int) (redemptions []*Redemption, total int64, err error) {
  23. // 开始事务
  24. tx := DB.Begin()
  25. if tx.Error != nil {
  26. return nil, 0, tx.Error
  27. }
  28. defer func() {
  29. if r := recover(); r != nil {
  30. tx.Rollback()
  31. }
  32. }()
  33. // 获取总数
  34. err = tx.Model(&Redemption{}).Count(&total).Error
  35. if err != nil {
  36. tx.Rollback()
  37. return nil, 0, err
  38. }
  39. // 获取分页数据
  40. err = tx.Order("id desc").Limit(num).Offset(startIdx).Find(&redemptions).Error
  41. if err != nil {
  42. tx.Rollback()
  43. return nil, 0, err
  44. }
  45. // 提交事务
  46. if err = tx.Commit().Error; err != nil {
  47. return nil, 0, err
  48. }
  49. return redemptions, total, nil
  50. }
  51. func SearchRedemptions(keyword string, startIdx int, num int) (redemptions []*Redemption, total int64, err error) {
  52. tx := DB.Begin()
  53. if tx.Error != nil {
  54. return nil, 0, tx.Error
  55. }
  56. defer func() {
  57. if r := recover(); r != nil {
  58. tx.Rollback()
  59. }
  60. }()
  61. // Build query based on keyword type
  62. query := tx.Model(&Redemption{})
  63. // Only try to convert to ID if the string represents a valid integer
  64. if id, err := strconv.Atoi(keyword); err == nil {
  65. query = query.Where("id = ? OR name LIKE ?", id, keyword+"%")
  66. } else {
  67. query = query.Where("name LIKE ?", keyword+"%")
  68. }
  69. // Get total count
  70. err = query.Count(&total).Error
  71. if err != nil {
  72. tx.Rollback()
  73. return nil, 0, err
  74. }
  75. // Get paginated data
  76. err = query.Order("id desc").Limit(num).Offset(startIdx).Find(&redemptions).Error
  77. if err != nil {
  78. tx.Rollback()
  79. return nil, 0, err
  80. }
  81. if err = tx.Commit().Error; err != nil {
  82. return nil, 0, err
  83. }
  84. return redemptions, total, nil
  85. }
  86. func GetRedemptionById(id int) (*Redemption, error) {
  87. if id == 0 {
  88. return nil, errors.New("id 为空!")
  89. }
  90. redemption := Redemption{Id: id}
  91. var err error = nil
  92. err = DB.First(&redemption, "id = ?", id).Error
  93. return &redemption, err
  94. }
  95. func Redeem(key string, userId int) (quota int, err error) {
  96. if key == "" {
  97. return 0, errors.New("未提供兑换码")
  98. }
  99. if userId == 0 {
  100. return 0, errors.New("无效的 user id")
  101. }
  102. redemption := &Redemption{}
  103. keyCol := "`key`"
  104. if common.UsingPostgreSQL {
  105. keyCol = `"key"`
  106. }
  107. common.RandomSleep()
  108. err = DB.Transaction(func(tx *gorm.DB) error {
  109. err := tx.Set("gorm:query_option", "FOR UPDATE").Where(keyCol+" = ?", key).First(redemption).Error
  110. if err != nil {
  111. return errors.New("无效的兑换码")
  112. }
  113. if redemption.Status != common.RedemptionCodeStatusEnabled {
  114. return errors.New("该兑换码已被使用")
  115. }
  116. err = tx.Model(&User{}).Where("id = ?", userId).Update("quota", gorm.Expr("quota + ?", redemption.Quota)).Error
  117. if err != nil {
  118. return err
  119. }
  120. redemption.RedeemedTime = common.GetTimestamp()
  121. redemption.Status = common.RedemptionCodeStatusUsed
  122. redemption.UsedUserId = userId
  123. err = tx.Save(redemption).Error
  124. return err
  125. })
  126. if err != nil {
  127. return 0, errors.New("兑换失败," + err.Error())
  128. }
  129. RecordLog(userId, LogTypeTopup, fmt.Sprintf("通过兑换码充值 %s,兑换码ID %d", common.LogQuota(redemption.Quota), redemption.Id))
  130. return redemption.Quota, nil
  131. }
  132. func (redemption *Redemption) Insert() error {
  133. var err error
  134. err = DB.Create(redemption).Error
  135. return err
  136. }
  137. func (redemption *Redemption) SelectUpdate() error {
  138. // This can update zero values
  139. return DB.Model(redemption).Select("redeemed_time", "status").Updates(redemption).Error
  140. }
  141. // Update Make sure your token's fields is completed, because this will update non-zero values
  142. func (redemption *Redemption) Update() error {
  143. var err error
  144. err = DB.Model(redemption).Select("name", "status", "quota", "redeemed_time").Updates(redemption).Error
  145. return err
  146. }
  147. func (redemption *Redemption) Delete() error {
  148. var err error
  149. err = DB.Delete(redemption).Error
  150. return err
  151. }
  152. func DeleteRedemptionById(id int) (err error) {
  153. if id == 0 {
  154. return errors.New("id 为空!")
  155. }
  156. redemption := Redemption{Id: id}
  157. err = DB.Where(redemption).First(&redemption).Error
  158. if err != nil {
  159. return err
  160. }
  161. return redemption.Delete()
  162. }