| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179 |
- package model
- import (
- "errors"
- "math/rand"
- "time"
- "github.com/QuantumNous/new-api/common"
- "github.com/QuantumNous/new-api/setting/operation_setting"
- "gorm.io/gorm"
- )
- // Checkin 签到记录
- type Checkin struct {
- Id int `json:"id" gorm:"primaryKey;autoIncrement"`
- UserId int `json:"user_id" gorm:"not null;uniqueIndex:idx_user_checkin_date"`
- CheckinDate string `json:"checkin_date" gorm:"type:varchar(10);not null;uniqueIndex:idx_user_checkin_date"` // 格式: YYYY-MM-DD
- QuotaAwarded int `json:"quota_awarded" gorm:"not null"`
- CreatedAt int64 `json:"created_at" gorm:"bigint"`
- }
- // CheckinRecord 用于API返回的签到记录(不包含敏感字段)
- type CheckinRecord struct {
- CheckinDate string `json:"checkin_date"`
- QuotaAwarded int `json:"quota_awarded"`
- }
- func (Checkin) TableName() string {
- return "checkins"
- }
- // GetUserCheckinRecords 获取用户在指定日期范围内的签到记录
- func GetUserCheckinRecords(userId int, startDate, endDate string) ([]Checkin, error) {
- var records []Checkin
- err := DB.Where("user_id = ? AND checkin_date >= ? AND checkin_date <= ?",
- userId, startDate, endDate).
- Order("checkin_date DESC").
- Find(&records).Error
- return records, err
- }
- // HasCheckedInToday 检查用户今天是否已签到
- func HasCheckedInToday(userId int) (bool, error) {
- today := time.Now().Format("2006-01-02")
- var count int64
- err := DB.Model(&Checkin{}).
- Where("user_id = ? AND checkin_date = ?", userId, today).
- Count(&count).Error
- return count > 0, err
- }
- // UserCheckin 执行用户签到
- // MySQL 和 PostgreSQL 使用事务保证原子性
- // SQLite 不支持嵌套事务,使用顺序操作 + 手动回滚
- func UserCheckin(userId int) (*Checkin, error) {
- setting := operation_setting.GetCheckinSetting()
- if !setting.Enabled {
- return nil, errors.New("签到功能未启用")
- }
- // 检查今天是否已签到
- hasChecked, err := HasCheckedInToday(userId)
- if err != nil {
- return nil, err
- }
- if hasChecked {
- return nil, errors.New("今日已签到")
- }
- // 计算随机额度奖励
- quotaAwarded := setting.MinQuota
- if setting.MaxQuota > setting.MinQuota {
- quotaAwarded = setting.MinQuota + rand.Intn(setting.MaxQuota-setting.MinQuota+1)
- }
- today := time.Now().Format("2006-01-02")
- checkin := &Checkin{
- UserId: userId,
- CheckinDate: today,
- QuotaAwarded: quotaAwarded,
- CreatedAt: time.Now().Unix(),
- }
- // 根据数据库类型选择不同的策略
- if common.UsingSQLite {
- // SQLite 不支持嵌套事务,使用顺序操作 + 手动回滚
- return userCheckinWithoutTransaction(checkin, userId, quotaAwarded)
- }
- // MySQL 和 PostgreSQL 支持事务,使用事务保证原子性
- return userCheckinWithTransaction(checkin, userId, quotaAwarded)
- }
- // userCheckinWithTransaction 使用事务执行签到(适用于 MySQL 和 PostgreSQL)
- func userCheckinWithTransaction(checkin *Checkin, userId int, quotaAwarded int) (*Checkin, error) {
- err := DB.Transaction(func(tx *gorm.DB) error {
- // 步骤1: 创建签到记录
- // 数据库有唯一约束 (user_id, checkin_date),可以防止并发重复签到
- if err := tx.Create(checkin).Error; err != nil {
- return errors.New("签到失败,请稍后重试")
- }
- // 步骤2: 在事务中增加用户额度
- if err := tx.Model(&User{}).Where("id = ?", userId).
- Update("quota", gorm.Expr("quota + ?", quotaAwarded)).Error; err != nil {
- return errors.New("签到失败:更新额度出错")
- }
- return nil
- })
- if err != nil {
- return nil, err
- }
- // 事务成功后,异步更新缓存
- go func() {
- _ = cacheIncrUserQuota(userId, int64(quotaAwarded))
- }()
- return checkin, nil
- }
- // userCheckinWithoutTransaction 不使用事务执行签到(适用于 SQLite)
- func userCheckinWithoutTransaction(checkin *Checkin, userId int, quotaAwarded int) (*Checkin, error) {
- // 步骤1: 创建签到记录
- // 数据库有唯一约束 (user_id, checkin_date),可以防止并发重复签到
- if err := DB.Create(checkin).Error; err != nil {
- return nil, errors.New("签到失败,请稍后重试")
- }
- // 步骤2: 增加用户额度
- // 使用 db=true 强制直接写入数据库,不使用批量更新
- if err := IncreaseUserQuota(userId, quotaAwarded, true); err != nil {
- // 如果增加额度失败,需要回滚签到记录
- DB.Delete(checkin)
- return nil, errors.New("签到失败:更新额度出错")
- }
- return checkin, nil
- }
- // GetUserCheckinStats 获取用户签到统计信息
- func GetUserCheckinStats(userId int, month string) (map[string]interface{}, error) {
- // 获取指定月份的所有签到记录
- startDate := month + "-01"
- endDate := month + "-31"
- records, err := GetUserCheckinRecords(userId, startDate, endDate)
- if err != nil {
- return nil, err
- }
- // 转换为不包含敏感字段的记录
- checkinRecords := make([]CheckinRecord, len(records))
- for i, r := range records {
- checkinRecords[i] = CheckinRecord{
- CheckinDate: r.CheckinDate,
- QuotaAwarded: r.QuotaAwarded,
- }
- }
- // 检查今天是否已签到
- hasCheckedToday, _ := HasCheckedInToday(userId)
- // 获取用户所有时间的签到统计
- var totalCheckins int64
- var totalQuota int64
- DB.Model(&Checkin{}).Where("user_id = ?", userId).Count(&totalCheckins)
- DB.Model(&Checkin{}).Where("user_id = ?", userId).Select("COALESCE(SUM(quota_awarded), 0)").Scan(&totalQuota)
- return map[string]interface{}{
- "total_quota": totalQuota, // 所有时间累计获得的额度
- "total_checkins": totalCheckins, // 所有时间累计签到次数
- "checkin_count": len(records), // 本月签到次数
- "checked_in_today": hasCheckedToday, // 今天是否已签到
- "records": checkinRecords, // 本月签到记录详情(不含id和user_id)
- }, nil
- }
|