180 lines
5.7 KiB
Go
180 lines
5.7 KiB
Go
|
|
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
|
|||
|
|
}
|