Files
sub2api/backend/internal/service/subscription_calculate_progress_test.go
yangjianbo bb5a5dd65e test: 完善自动化测试体系(7个模块,73个任务)
系统性地修复、补充和强化项目的自动化测试能力:

1. 测试基础设施修复
   - 修复 stubConcurrencyCache 缺失方法和构造函数参数不匹配
   - 创建 testutil 共享包(stubs.go, fixtures.go, httptest.go)
   - 为所有 Stub 添加编译期接口断言

2. 中间件测试补充
   - 新增 JWT 认证中间件测试(有效/过期/篡改/缺失 Token)
   - 补充 rate_limiter 和 recovery 中间件测试场景

3. 网关核心路径测试
   - 新增账户选择、等待队列、流式响应、并发控制、计费、Claude Code 检测测试
   - 覆盖负载均衡、粘性会话、SSE 转发、槽位管理等关键逻辑

4. 前端测试体系(11个新测试文件,163个测试用例)
   - Pinia stores: auth, app, subscriptions
   - API client: 请求拦截器、响应拦截器、401 刷新
   - Router guards: 认证重定向、管理员权限、简易模式限制
   - Composables: useForm, useTableLoader, useClipboard
   - Components: LoginForm, ApiKeyCreate, Dashboard

5. CI/CD 流水线重构
   - 重构 backend-ci.yml 为统一的 ci.yml
   - 前后端 4 个并行 Job + Postgres/Redis services
   - Race 检测、覆盖率收集与门禁、Docker 构建验证

6. E2E 自动化测试
   - e2e-test.sh 自动化脚本(Docker 启动→健康检查→测试→清理)
   - 用户注册→登录→API Key→网关调用完整链路测试
   - Mock 模式和 API Key 脱敏支持

7. 修复预存问题
   - tlsfingerprint dialer_test.go 缺失 build tag 导致集成测试编译冲突

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-02-08 12:05:39 +08:00

232 lines
6.6 KiB
Go

package service
import (
"testing"
"time"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
)
// --- Task 5: 验证 calculateProgress 纯函数行为正确 ---
func newTestSubscriptionService() *SubscriptionService {
return &SubscriptionService{}
}
func ptrFloat64(v float64) *float64 { return &v }
func ptrTime(t time.Time) *time.Time { return &t }
func TestCalculateProgress_BasicFields(t *testing.T) {
svc := newTestSubscriptionService()
now := time.Now()
sub := &UserSubscription{
ID: 100,
ExpiresAt: now.Add(30 * 24 * time.Hour),
}
group := &Group{
Name: "Premium",
}
progress := svc.calculateProgress(sub, group)
assert.Equal(t, int64(100), progress.ID)
assert.Equal(t, "Premium", progress.GroupName)
assert.Equal(t, sub.ExpiresAt, progress.ExpiresAt)
assert.Equal(t, 29, progress.ExpiresInDays) // 约 30 天
assert.Nil(t, progress.Daily, "无日限额时 Daily 应为 nil")
assert.Nil(t, progress.Weekly, "无周限额时 Weekly 应为 nil")
assert.Nil(t, progress.Monthly, "无月限额时 Monthly 应为 nil")
}
func TestCalculateProgress_DailyUsage(t *testing.T) {
svc := newTestSubscriptionService()
now := time.Now()
dailyStart := now.Add(-12 * time.Hour)
sub := &UserSubscription{
ID: 1,
ExpiresAt: now.Add(10 * 24 * time.Hour),
DailyUsageUSD: 3.0,
DailyWindowStart: ptrTime(dailyStart),
}
group := &Group{
Name: "Pro",
DailyLimitUSD: ptrFloat64(10.0),
}
progress := svc.calculateProgress(sub, group)
require.NotNil(t, progress.Daily, "有日限额和窗口时 Daily 不应为 nil")
assert.Equal(t, 10.0, progress.Daily.LimitUSD)
assert.Equal(t, 3.0, progress.Daily.UsedUSD)
assert.Equal(t, 7.0, progress.Daily.RemainingUSD)
assert.Equal(t, 30.0, progress.Daily.Percentage)
assert.Equal(t, dailyStart, progress.Daily.WindowStart)
}
func TestCalculateProgress_WeeklyUsage(t *testing.T) {
svc := newTestSubscriptionService()
now := time.Now()
weeklyStart := now.Add(-3 * 24 * time.Hour)
sub := &UserSubscription{
ID: 1,
ExpiresAt: now.Add(10 * 24 * time.Hour),
WeeklyUsageUSD: 25.0,
WeeklyWindowStart: ptrTime(weeklyStart),
}
group := &Group{
Name: "Pro",
WeeklyLimitUSD: ptrFloat64(50.0),
}
progress := svc.calculateProgress(sub, group)
require.NotNil(t, progress.Weekly, "有周限额和窗口时 Weekly 不应为 nil")
assert.Equal(t, 50.0, progress.Weekly.LimitUSD)
assert.Equal(t, 25.0, progress.Weekly.UsedUSD)
assert.Equal(t, 25.0, progress.Weekly.RemainingUSD)
assert.Equal(t, 50.0, progress.Weekly.Percentage)
}
func TestCalculateProgress_MonthlyUsage(t *testing.T) {
svc := newTestSubscriptionService()
now := time.Now()
monthlyStart := now.Add(-15 * 24 * time.Hour)
sub := &UserSubscription{
ID: 1,
ExpiresAt: now.Add(10 * 24 * time.Hour),
MonthlyUsageUSD: 80.0,
MonthlyWindowStart: ptrTime(monthlyStart),
}
group := &Group{
Name: "Enterprise",
MonthlyLimitUSD: ptrFloat64(100.0),
}
progress := svc.calculateProgress(sub, group)
require.NotNil(t, progress.Monthly, "有月限额和窗口时 Monthly 不应为 nil")
assert.Equal(t, 100.0, progress.Monthly.LimitUSD)
assert.Equal(t, 80.0, progress.Monthly.UsedUSD)
assert.Equal(t, 20.0, progress.Monthly.RemainingUSD)
assert.Equal(t, 80.0, progress.Monthly.Percentage)
}
func TestCalculateProgress_OverLimit_ClampedTo100Percent(t *testing.T) {
svc := newTestSubscriptionService()
now := time.Now()
sub := &UserSubscription{
ID: 1,
ExpiresAt: now.Add(10 * 24 * time.Hour),
DailyUsageUSD: 15.0, // 超过限额
DailyWindowStart: ptrTime(now.Add(-1 * time.Hour)),
}
group := &Group{
Name: "Pro",
DailyLimitUSD: ptrFloat64(10.0),
}
progress := svc.calculateProgress(sub, group)
require.NotNil(t, progress.Daily)
assert.Equal(t, 100.0, progress.Daily.Percentage, "超额使用应被截断为 100%")
assert.Equal(t, 0.0, progress.Daily.RemainingUSD, "超额使用时剩余应为 0")
}
func TestCalculateProgress_NoWindowStart_NoProgress(t *testing.T) {
svc := newTestSubscriptionService()
now := time.Now()
// 有限额但无窗口起始时间(订阅未激活)
sub := &UserSubscription{
ID: 1,
ExpiresAt: now.Add(10 * 24 * time.Hour),
DailyUsageUSD: 0,
WeeklyUsageUSD: 0,
}
group := &Group{
Name: "Pro",
DailyLimitUSD: ptrFloat64(10.0),
WeeklyLimitUSD: ptrFloat64(50.0),
}
progress := svc.calculateProgress(sub, group)
assert.Nil(t, progress.Daily, "无 DailyWindowStart 时 Daily 应为 nil")
assert.Nil(t, progress.Weekly, "无 WeeklyWindowStart 时 Weekly 应为 nil")
}
func TestCalculateProgress_AllLimits(t *testing.T) {
svc := newTestSubscriptionService()
now := time.Now()
sub := &UserSubscription{
ID: 1,
ExpiresAt: now.Add(10 * 24 * time.Hour),
DailyUsageUSD: 5.0,
WeeklyUsageUSD: 20.0,
MonthlyUsageUSD: 60.0,
DailyWindowStart: ptrTime(now.Add(-6 * time.Hour)),
WeeklyWindowStart: ptrTime(now.Add(-3 * 24 * time.Hour)),
MonthlyWindowStart: ptrTime(now.Add(-15 * 24 * time.Hour)),
}
group := &Group{
Name: "Full",
DailyLimitUSD: ptrFloat64(10.0),
WeeklyLimitUSD: ptrFloat64(50.0),
MonthlyLimitUSD: ptrFloat64(100.0),
}
progress := svc.calculateProgress(sub, group)
require.NotNil(t, progress.Daily)
require.NotNil(t, progress.Weekly)
require.NotNil(t, progress.Monthly)
assert.Equal(t, 50.0, progress.Daily.Percentage)
assert.Equal(t, 40.0, progress.Weekly.Percentage)
assert.Equal(t, 60.0, progress.Monthly.Percentage)
}
func TestCalculateProgress_ExpiredSubscription(t *testing.T) {
svc := newTestSubscriptionService()
sub := &UserSubscription{
ID: 1,
ExpiresAt: time.Now().Add(-24 * time.Hour), // 已过期
}
group := &Group{Name: "Expired"}
progress := svc.calculateProgress(sub, group)
assert.Equal(t, 0, progress.ExpiresInDays, "过期订阅的剩余天数应为 0")
}
func TestCalculateProgress_ResetsInSeconds_NotNegative(t *testing.T) {
svc := newTestSubscriptionService()
// 使用过去的窗口起始时间,使得重置时间已过
pastStart := time.Now().Add(-48 * time.Hour)
sub := &UserSubscription{
ID: 1,
ExpiresAt: time.Now().Add(10 * 24 * time.Hour),
DailyUsageUSD: 1.0,
DailyWindowStart: ptrTime(pastStart),
}
group := &Group{
Name: "Test",
DailyLimitUSD: ptrFloat64(10.0),
}
progress := svc.calculateProgress(sub, group)
require.NotNil(t, progress.Daily)
assert.GreaterOrEqual(t, progress.Daily.ResetsInSeconds, int64(0),
"ResetsInSeconds 不应为负数")
}