mirror of
https://gitee.com/wanwujie/sub2api
synced 2026-04-03 06:52:13 +08:00
- 后端 handler:ResetSubscriptionQuotaRequest 新增 Monthly 字段, 验证逻辑扩展为 daily/weekly/monthly 至少一项为 true - 后端 service:AdminResetQuota 新增 resetMonthly 参数, 调用 ResetMonthlyUsage;重置后追加 subCacheL1.Wait(), 保证 ristretto Del() 的异步删除立即生效,消除重置后 /v1/usage 返回旧用量数据的竞态窗口 - 后端测试:更新存量测试用例匹配新签名,补充 TestAdminResetQuota_ResetMonthlyOnly / TestAdminResetQuota_ResetMonthlyUsageError 两个新用例 - 前端 API:resetQuota options 类型新增 monthly: boolean - 前端视图:confirmResetQuota 改为同时重置 daily/weekly/monthly - i18n:中英文确认提示文案更新,提及每月配额 Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
208 lines
6.3 KiB
Go
208 lines
6.3 KiB
Go
//go:build unit
|
||
|
||
package service
|
||
|
||
import (
|
||
"context"
|
||
"errors"
|
||
"testing"
|
||
"time"
|
||
|
||
"github.com/stretchr/testify/require"
|
||
)
|
||
|
||
// resetQuotaUserSubRepoStub 支持 GetByID、ResetDailyUsage、ResetWeeklyUsage、ResetMonthlyUsage,
|
||
// 其余方法继承 userSubRepoNoop(panic)。
|
||
type resetQuotaUserSubRepoStub struct {
|
||
userSubRepoNoop
|
||
|
||
sub *UserSubscription
|
||
|
||
resetDailyCalled bool
|
||
resetWeeklyCalled bool
|
||
resetMonthlyCalled bool
|
||
resetDailyErr error
|
||
resetWeeklyErr error
|
||
resetMonthlyErr error
|
||
}
|
||
|
||
func (r *resetQuotaUserSubRepoStub) GetByID(_ context.Context, id int64) (*UserSubscription, error) {
|
||
if r.sub == nil || r.sub.ID != id {
|
||
return nil, ErrSubscriptionNotFound
|
||
}
|
||
cp := *r.sub
|
||
return &cp, nil
|
||
}
|
||
|
||
func (r *resetQuotaUserSubRepoStub) ResetDailyUsage(_ context.Context, _ int64, windowStart time.Time) error {
|
||
r.resetDailyCalled = true
|
||
if r.resetDailyErr == nil && r.sub != nil {
|
||
r.sub.DailyUsageUSD = 0
|
||
r.sub.DailyWindowStart = &windowStart
|
||
}
|
||
return r.resetDailyErr
|
||
}
|
||
|
||
func (r *resetQuotaUserSubRepoStub) ResetWeeklyUsage(_ context.Context, _ int64, _ time.Time) error {
|
||
r.resetWeeklyCalled = true
|
||
return r.resetWeeklyErr
|
||
}
|
||
|
||
func (r *resetQuotaUserSubRepoStub) ResetMonthlyUsage(_ context.Context, _ int64, _ time.Time) error {
|
||
r.resetMonthlyCalled = true
|
||
return r.resetMonthlyErr
|
||
}
|
||
|
||
func newResetQuotaSvc(stub *resetQuotaUserSubRepoStub) *SubscriptionService {
|
||
return NewSubscriptionService(groupRepoNoop{}, stub, nil, nil, nil)
|
||
}
|
||
|
||
func TestAdminResetQuota_ResetBoth(t *testing.T) {
|
||
stub := &resetQuotaUserSubRepoStub{
|
||
sub: &UserSubscription{ID: 1, UserID: 10, GroupID: 20},
|
||
}
|
||
svc := newResetQuotaSvc(stub)
|
||
|
||
result, err := svc.AdminResetQuota(context.Background(), 1, true, true, false)
|
||
|
||
require.NoError(t, err)
|
||
require.NotNil(t, result)
|
||
require.True(t, stub.resetDailyCalled, "应调用 ResetDailyUsage")
|
||
require.True(t, stub.resetWeeklyCalled, "应调用 ResetWeeklyUsage")
|
||
require.False(t, stub.resetMonthlyCalled, "不应调用 ResetMonthlyUsage")
|
||
}
|
||
|
||
func TestAdminResetQuota_ResetDailyOnly(t *testing.T) {
|
||
stub := &resetQuotaUserSubRepoStub{
|
||
sub: &UserSubscription{ID: 2, UserID: 10, GroupID: 20},
|
||
}
|
||
svc := newResetQuotaSvc(stub)
|
||
|
||
result, err := svc.AdminResetQuota(context.Background(), 2, true, false, false)
|
||
|
||
require.NoError(t, err)
|
||
require.NotNil(t, result)
|
||
require.True(t, stub.resetDailyCalled, "应调用 ResetDailyUsage")
|
||
require.False(t, stub.resetWeeklyCalled, "不应调用 ResetWeeklyUsage")
|
||
require.False(t, stub.resetMonthlyCalled, "不应调用 ResetMonthlyUsage")
|
||
}
|
||
|
||
func TestAdminResetQuota_ResetWeeklyOnly(t *testing.T) {
|
||
stub := &resetQuotaUserSubRepoStub{
|
||
sub: &UserSubscription{ID: 3, UserID: 10, GroupID: 20},
|
||
}
|
||
svc := newResetQuotaSvc(stub)
|
||
|
||
result, err := svc.AdminResetQuota(context.Background(), 3, false, true, false)
|
||
|
||
require.NoError(t, err)
|
||
require.NotNil(t, result)
|
||
require.False(t, stub.resetDailyCalled, "不应调用 ResetDailyUsage")
|
||
require.True(t, stub.resetWeeklyCalled, "应调用 ResetWeeklyUsage")
|
||
require.False(t, stub.resetMonthlyCalled, "不应调用 ResetMonthlyUsage")
|
||
}
|
||
|
||
func TestAdminResetQuota_BothFalseReturnsError(t *testing.T) {
|
||
stub := &resetQuotaUserSubRepoStub{
|
||
sub: &UserSubscription{ID: 7, UserID: 10, GroupID: 20},
|
||
}
|
||
svc := newResetQuotaSvc(stub)
|
||
|
||
_, err := svc.AdminResetQuota(context.Background(), 7, false, false, false)
|
||
|
||
require.ErrorIs(t, err, ErrInvalidInput)
|
||
require.False(t, stub.resetDailyCalled)
|
||
require.False(t, stub.resetWeeklyCalled)
|
||
require.False(t, stub.resetMonthlyCalled)
|
||
}
|
||
|
||
func TestAdminResetQuota_SubscriptionNotFound(t *testing.T) {
|
||
stub := &resetQuotaUserSubRepoStub{sub: nil}
|
||
svc := newResetQuotaSvc(stub)
|
||
|
||
_, err := svc.AdminResetQuota(context.Background(), 999, true, true, true)
|
||
|
||
require.ErrorIs(t, err, ErrSubscriptionNotFound)
|
||
require.False(t, stub.resetDailyCalled)
|
||
require.False(t, stub.resetWeeklyCalled)
|
||
require.False(t, stub.resetMonthlyCalled)
|
||
}
|
||
|
||
func TestAdminResetQuota_ResetDailyUsageError(t *testing.T) {
|
||
dbErr := errors.New("db error")
|
||
stub := &resetQuotaUserSubRepoStub{
|
||
sub: &UserSubscription{ID: 4, UserID: 10, GroupID: 20},
|
||
resetDailyErr: dbErr,
|
||
}
|
||
svc := newResetQuotaSvc(stub)
|
||
|
||
_, err := svc.AdminResetQuota(context.Background(), 4, true, true, false)
|
||
|
||
require.ErrorIs(t, err, dbErr)
|
||
require.True(t, stub.resetDailyCalled)
|
||
require.False(t, stub.resetWeeklyCalled, "daily 失败后不应继续调用 weekly")
|
||
}
|
||
|
||
func TestAdminResetQuota_ResetWeeklyUsageError(t *testing.T) {
|
||
dbErr := errors.New("db error")
|
||
stub := &resetQuotaUserSubRepoStub{
|
||
sub: &UserSubscription{ID: 5, UserID: 10, GroupID: 20},
|
||
resetWeeklyErr: dbErr,
|
||
}
|
||
svc := newResetQuotaSvc(stub)
|
||
|
||
_, err := svc.AdminResetQuota(context.Background(), 5, false, true, false)
|
||
|
||
require.ErrorIs(t, err, dbErr)
|
||
require.True(t, stub.resetWeeklyCalled)
|
||
}
|
||
|
||
func TestAdminResetQuota_ResetMonthlyOnly(t *testing.T) {
|
||
stub := &resetQuotaUserSubRepoStub{
|
||
sub: &UserSubscription{ID: 8, UserID: 10, GroupID: 20},
|
||
}
|
||
svc := newResetQuotaSvc(stub)
|
||
|
||
result, err := svc.AdminResetQuota(context.Background(), 8, false, false, true)
|
||
|
||
require.NoError(t, err)
|
||
require.NotNil(t, result)
|
||
require.False(t, stub.resetDailyCalled, "不应调用 ResetDailyUsage")
|
||
require.False(t, stub.resetWeeklyCalled, "不应调用 ResetWeeklyUsage")
|
||
require.True(t, stub.resetMonthlyCalled, "应调用 ResetMonthlyUsage")
|
||
}
|
||
|
||
func TestAdminResetQuota_ResetMonthlyUsageError(t *testing.T) {
|
||
dbErr := errors.New("db error")
|
||
stub := &resetQuotaUserSubRepoStub{
|
||
sub: &UserSubscription{ID: 9, UserID: 10, GroupID: 20},
|
||
resetMonthlyErr: dbErr,
|
||
}
|
||
svc := newResetQuotaSvc(stub)
|
||
|
||
_, err := svc.AdminResetQuota(context.Background(), 9, false, false, true)
|
||
|
||
require.ErrorIs(t, err, dbErr)
|
||
require.True(t, stub.resetMonthlyCalled)
|
||
}
|
||
|
||
func TestAdminResetQuota_ReturnsRefreshedSub(t *testing.T) {
|
||
stub := &resetQuotaUserSubRepoStub{
|
||
sub: &UserSubscription{
|
||
ID: 6,
|
||
UserID: 10,
|
||
GroupID: 20,
|
||
DailyUsageUSD: 99.9,
|
||
},
|
||
}
|
||
|
||
svc := newResetQuotaSvc(stub)
|
||
result, err := svc.AdminResetQuota(context.Background(), 6, true, false, false)
|
||
|
||
require.NoError(t, err)
|
||
// ResetDailyUsage stub 会将 sub.DailyUsageUSD 归零,
|
||
// 服务应返回第二次 GetByID 的刷新值而非初始的 99.9
|
||
require.Equal(t, float64(0), result.DailyUsageUSD, "返回的订阅应反映已归零的用量")
|
||
require.True(t, stub.resetDailyCalled)
|
||
}
|