package handler import ( "context" "testing" "time" "github.com/Wei-Shaw/sub2api/internal/service" "github.com/stretchr/testify/require" ) // --------------------------------------------------------------------------- // Mock // --------------------------------------------------------------------------- // mockTempUnscheduler 记录 TempUnscheduleRetryableError 的调用信息。 type mockTempUnscheduler struct { calls []tempUnscheduleCall } type tempUnscheduleCall struct { accountID int64 failoverErr *service.UpstreamFailoverError } func (m *mockTempUnscheduler) TempUnscheduleRetryableError(_ context.Context, accountID int64, failoverErr *service.UpstreamFailoverError) { m.calls = append(m.calls, tempUnscheduleCall{accountID: accountID, failoverErr: failoverErr}) } // --------------------------------------------------------------------------- // Helper // --------------------------------------------------------------------------- func newTestFailoverErr(statusCode int, retryable, forceBilling bool) *service.UpstreamFailoverError { return &service.UpstreamFailoverError{ StatusCode: statusCode, RetryableOnSameAccount: retryable, ForceCacheBilling: forceBilling, } } // --------------------------------------------------------------------------- // NewFailoverState 测试 // --------------------------------------------------------------------------- func TestNewFailoverState(t *testing.T) { t.Run("初始化字段正确", func(t *testing.T) { fs := NewFailoverState(5, true) require.Equal(t, 5, fs.MaxSwitches) require.Equal(t, 0, fs.SwitchCount) require.NotNil(t, fs.FailedAccountIDs) require.Empty(t, fs.FailedAccountIDs) require.NotNil(t, fs.SameAccountRetryCount) require.Empty(t, fs.SameAccountRetryCount) require.Nil(t, fs.LastFailoverErr) require.False(t, fs.ForceCacheBilling) require.True(t, fs.hasBoundSession) }) t.Run("无绑定会话", func(t *testing.T) { fs := NewFailoverState(3, false) require.Equal(t, 3, fs.MaxSwitches) require.False(t, fs.hasBoundSession) }) t.Run("零最大切换次数", func(t *testing.T) { fs := NewFailoverState(0, false) require.Equal(t, 0, fs.MaxSwitches) }) } // --------------------------------------------------------------------------- // sleepWithContext 测试 // --------------------------------------------------------------------------- func TestSleepWithContext(t *testing.T) { t.Run("零时长立即返回true", func(t *testing.T) { start := time.Now() ok := sleepWithContext(context.Background(), 0) require.True(t, ok) require.Less(t, time.Since(start), 50*time.Millisecond) }) t.Run("负时长立即返回true", func(t *testing.T) { start := time.Now() ok := sleepWithContext(context.Background(), -1*time.Second) require.True(t, ok) require.Less(t, time.Since(start), 50*time.Millisecond) }) t.Run("正常等待后返回true", func(t *testing.T) { start := time.Now() ok := sleepWithContext(context.Background(), 50*time.Millisecond) elapsed := time.Since(start) require.True(t, ok) require.GreaterOrEqual(t, elapsed, 40*time.Millisecond) require.Less(t, elapsed, 500*time.Millisecond) }) t.Run("已取消context立即返回false", func(t *testing.T) { ctx, cancel := context.WithCancel(context.Background()) cancel() start := time.Now() ok := sleepWithContext(ctx, 5*time.Second) require.False(t, ok) require.Less(t, time.Since(start), 50*time.Millisecond) }) t.Run("等待期间context取消返回false", func(t *testing.T) { ctx, cancel := context.WithCancel(context.Background()) go func() { time.Sleep(30 * time.Millisecond) cancel() }() start := time.Now() ok := sleepWithContext(ctx, 5*time.Second) elapsed := time.Since(start) require.False(t, ok) require.Less(t, elapsed, 500*time.Millisecond) }) } // --------------------------------------------------------------------------- // HandleFailoverError — 基本切换流程 // --------------------------------------------------------------------------- func TestHandleFailoverError_BasicSwitch(t *testing.T) { t.Run("非重试错误_非Antigravity_直接切换", func(t *testing.T) { mock := &mockTempUnscheduler{} fs := NewFailoverState(3, false) err := newTestFailoverErr(500, false, false) action := fs.HandleFailoverError(context.Background(), mock, 100, "openai", err) require.Equal(t, FailoverContinue, action) require.Equal(t, 1, fs.SwitchCount) require.Contains(t, fs.FailedAccountIDs, int64(100)) require.Equal(t, err, fs.LastFailoverErr) require.False(t, fs.ForceCacheBilling) require.Empty(t, mock.calls, "不应调用 TempUnschedule") }) t.Run("非重试错误_Antigravity_第一次切换无延迟", func(t *testing.T) { // switchCount 从 0→1 时,sleepFailoverDelay(ctx, 1) 的延时 = (1-1)*1s = 0 mock := &mockTempUnscheduler{} fs := NewFailoverState(3, false) err := newTestFailoverErr(500, false, false) start := time.Now() action := fs.HandleFailoverError(context.Background(), mock, 100, service.PlatformAntigravity, err) elapsed := time.Since(start) require.Equal(t, FailoverContinue, action) require.Equal(t, 1, fs.SwitchCount) require.Less(t, elapsed, 200*time.Millisecond, "第一次切换延迟应为 0") }) t.Run("非重试错误_Antigravity_第二次切换有1秒延迟", func(t *testing.T) { // switchCount 从 1→2 时,sleepFailoverDelay(ctx, 2) 的延时 = (2-1)*1s = 1s mock := &mockTempUnscheduler{} fs := NewFailoverState(3, false) fs.SwitchCount = 1 // 模拟已切换一次 err := newTestFailoverErr(500, false, false) start := time.Now() action := fs.HandleFailoverError(context.Background(), mock, 200, service.PlatformAntigravity, err) elapsed := time.Since(start) require.Equal(t, FailoverContinue, action) require.Equal(t, 2, fs.SwitchCount) require.GreaterOrEqual(t, elapsed, 800*time.Millisecond, "第二次切换延迟应约 1s") require.Less(t, elapsed, 3*time.Second) }) t.Run("连续切换直到耗尽", func(t *testing.T) { mock := &mockTempUnscheduler{} fs := NewFailoverState(2, false) // 第一次切换:0→1 err1 := newTestFailoverErr(500, false, false) action := fs.HandleFailoverError(context.Background(), mock, 100, "openai", err1) require.Equal(t, FailoverContinue, action) require.Equal(t, 1, fs.SwitchCount) // 第二次切换:1→2 err2 := newTestFailoverErr(502, false, false) action = fs.HandleFailoverError(context.Background(), mock, 200, "openai", err2) require.Equal(t, FailoverContinue, action) require.Equal(t, 2, fs.SwitchCount) // 第三次已耗尽:SwitchCount(2) >= MaxSwitches(2) err3 := newTestFailoverErr(503, false, false) action = fs.HandleFailoverError(context.Background(), mock, 300, "openai", err3) require.Equal(t, FailoverExhausted, action) require.Equal(t, 2, fs.SwitchCount, "耗尽时不应继续递增") // 验证失败账号列表 require.Len(t, fs.FailedAccountIDs, 3) require.Contains(t, fs.FailedAccountIDs, int64(100)) require.Contains(t, fs.FailedAccountIDs, int64(200)) require.Contains(t, fs.FailedAccountIDs, int64(300)) // LastFailoverErr 应为最后一次的错误 require.Equal(t, err3, fs.LastFailoverErr) }) t.Run("MaxSwitches为0时首次即耗尽", func(t *testing.T) { mock := &mockTempUnscheduler{} fs := NewFailoverState(0, false) err := newTestFailoverErr(500, false, false) action := fs.HandleFailoverError(context.Background(), mock, 100, "openai", err) require.Equal(t, FailoverExhausted, action) require.Equal(t, 0, fs.SwitchCount) require.Contains(t, fs.FailedAccountIDs, int64(100)) }) } // --------------------------------------------------------------------------- // HandleFailoverError — 缓存计费 (ForceCacheBilling) // --------------------------------------------------------------------------- func TestHandleFailoverError_CacheBilling(t *testing.T) { t.Run("hasBoundSession为true时设置ForceCacheBilling", func(t *testing.T) { mock := &mockTempUnscheduler{} fs := NewFailoverState(3, true) // hasBoundSession=true err := newTestFailoverErr(500, false, false) fs.HandleFailoverError(context.Background(), mock, 100, "openai", err) require.True(t, fs.ForceCacheBilling) }) t.Run("failoverErr.ForceCacheBilling为true时设置", func(t *testing.T) { mock := &mockTempUnscheduler{} fs := NewFailoverState(3, false) err := newTestFailoverErr(500, false, true) // ForceCacheBilling=true fs.HandleFailoverError(context.Background(), mock, 100, "openai", err) require.True(t, fs.ForceCacheBilling) }) t.Run("两者均为false时不设置", func(t *testing.T) { mock := &mockTempUnscheduler{} fs := NewFailoverState(3, false) err := newTestFailoverErr(500, false, false) fs.HandleFailoverError(context.Background(), mock, 100, "openai", err) require.False(t, fs.ForceCacheBilling) }) t.Run("一旦设置不会被后续错误重置", func(t *testing.T) { mock := &mockTempUnscheduler{} fs := NewFailoverState(3, false) // 第一次:ForceCacheBilling=true → 设置 err1 := newTestFailoverErr(500, false, true) fs.HandleFailoverError(context.Background(), mock, 100, "openai", err1) require.True(t, fs.ForceCacheBilling) // 第二次:ForceCacheBilling=false → 仍然保持 true err2 := newTestFailoverErr(502, false, false) fs.HandleFailoverError(context.Background(), mock, 200, "openai", err2) require.True(t, fs.ForceCacheBilling, "ForceCacheBilling 一旦设置不应被重置") }) } // --------------------------------------------------------------------------- // HandleFailoverError — 同账号重试 (RetryableOnSameAccount) // --------------------------------------------------------------------------- func TestHandleFailoverError_SameAccountRetry(t *testing.T) { t.Run("第一次重试返回FailoverContinue", func(t *testing.T) { mock := &mockTempUnscheduler{} fs := NewFailoverState(3, false) err := newTestFailoverErr(400, true, false) start := time.Now() action := fs.HandleFailoverError(context.Background(), mock, 100, "openai", err) elapsed := time.Since(start) require.Equal(t, FailoverContinue, action) require.Equal(t, 1, fs.SameAccountRetryCount[100]) require.Equal(t, 0, fs.SwitchCount, "同账号重试不应增加切换计数") require.NotContains(t, fs.FailedAccountIDs, int64(100), "同账号重试不应加入失败列表") require.Empty(t, mock.calls, "同账号重试期间不应调用 TempUnschedule") // 验证等待了 sameAccountRetryDelay (500ms) require.GreaterOrEqual(t, elapsed, 400*time.Millisecond) require.Less(t, elapsed, 2*time.Second) }) t.Run("第二次重试仍返回FailoverContinue", func(t *testing.T) { mock := &mockTempUnscheduler{} fs := NewFailoverState(3, false) err := newTestFailoverErr(400, true, false) // 第一次 action := fs.HandleFailoverError(context.Background(), mock, 100, "openai", err) require.Equal(t, FailoverContinue, action) require.Equal(t, 1, fs.SameAccountRetryCount[100]) // 第二次 action = fs.HandleFailoverError(context.Background(), mock, 100, "openai", err) require.Equal(t, FailoverContinue, action) require.Equal(t, 2, fs.SameAccountRetryCount[100]) require.Empty(t, mock.calls, "两次重试期间均不应调用 TempUnschedule") }) t.Run("第三次重试耗尽_触发TempUnschedule并切换", func(t *testing.T) { mock := &mockTempUnscheduler{} fs := NewFailoverState(3, false) err := newTestFailoverErr(400, true, false) // 第一次、第二次重试 fs.HandleFailoverError(context.Background(), mock, 100, "openai", err) fs.HandleFailoverError(context.Background(), mock, 100, "openai", err) require.Equal(t, 2, fs.SameAccountRetryCount[100]) // 第三次:重试已达到 maxSameAccountRetries(2),应切换账号 action := fs.HandleFailoverError(context.Background(), mock, 100, "openai", err) require.Equal(t, FailoverContinue, action) require.Equal(t, 1, fs.SwitchCount) require.Contains(t, fs.FailedAccountIDs, int64(100)) // 验证 TempUnschedule 被调用 require.Len(t, mock.calls, 1) require.Equal(t, int64(100), mock.calls[0].accountID) require.Equal(t, err, mock.calls[0].failoverErr) }) t.Run("不同账号独立跟踪重试次数", func(t *testing.T) { mock := &mockTempUnscheduler{} fs := NewFailoverState(5, false) err := newTestFailoverErr(400, true, false) // 账号 100 第一次重试 action := fs.HandleFailoverError(context.Background(), mock, 100, "openai", err) require.Equal(t, FailoverContinue, action) require.Equal(t, 1, fs.SameAccountRetryCount[100]) // 账号 200 第一次重试(独立计数) action = fs.HandleFailoverError(context.Background(), mock, 200, "openai", err) require.Equal(t, FailoverContinue, action) require.Equal(t, 1, fs.SameAccountRetryCount[200]) require.Equal(t, 1, fs.SameAccountRetryCount[100], "账号 100 的计数不应受影响") }) t.Run("重试耗尽后再次遇到同账号_直接切换", func(t *testing.T) { mock := &mockTempUnscheduler{} fs := NewFailoverState(5, false) err := newTestFailoverErr(400, true, false) // 耗尽账号 100 的重试 fs.HandleFailoverError(context.Background(), mock, 100, "openai", err) fs.HandleFailoverError(context.Background(), mock, 100, "openai", err) // 第三次: 重试耗尽 → 切换 action := fs.HandleFailoverError(context.Background(), mock, 100, "openai", err) require.Equal(t, FailoverContinue, action) // 再次遇到账号 100,计数仍为 2,条件不满足 → 直接切换 action = fs.HandleFailoverError(context.Background(), mock, 100, "openai", err) require.Equal(t, FailoverContinue, action) require.Len(t, mock.calls, 2, "第二次耗尽也应调用 TempUnschedule") }) } // --------------------------------------------------------------------------- // HandleFailoverError — TempUnschedule 调用验证 // --------------------------------------------------------------------------- func TestHandleFailoverError_TempUnschedule(t *testing.T) { t.Run("非重试错误不调用TempUnschedule", func(t *testing.T) { mock := &mockTempUnscheduler{} fs := NewFailoverState(3, false) err := newTestFailoverErr(500, false, false) // RetryableOnSameAccount=false fs.HandleFailoverError(context.Background(), mock, 100, "openai", err) require.Empty(t, mock.calls) }) t.Run("重试错误耗尽后调用TempUnschedule_传入正确参数", func(t *testing.T) { mock := &mockTempUnscheduler{} fs := NewFailoverState(3, false) err := newTestFailoverErr(502, true, false) // 耗尽重试 fs.HandleFailoverError(context.Background(), mock, 42, "openai", err) fs.HandleFailoverError(context.Background(), mock, 42, "openai", err) fs.HandleFailoverError(context.Background(), mock, 42, "openai", err) require.Len(t, mock.calls, 1) require.Equal(t, int64(42), mock.calls[0].accountID) require.Equal(t, 502, mock.calls[0].failoverErr.StatusCode) require.True(t, mock.calls[0].failoverErr.RetryableOnSameAccount) }) } // --------------------------------------------------------------------------- // HandleFailoverError — Context 取消 // --------------------------------------------------------------------------- func TestHandleFailoverError_ContextCanceled(t *testing.T) { t.Run("同账号重试sleep期间context取消", func(t *testing.T) { mock := &mockTempUnscheduler{} fs := NewFailoverState(3, false) err := newTestFailoverErr(400, true, false) ctx, cancel := context.WithCancel(context.Background()) cancel() // 立即取消 start := time.Now() action := fs.HandleFailoverError(ctx, mock, 100, "openai", err) elapsed := time.Since(start) require.Equal(t, FailoverCanceled, action) require.Less(t, elapsed, 100*time.Millisecond, "应立即返回") // 重试计数仍应递增 require.Equal(t, 1, fs.SameAccountRetryCount[100]) }) t.Run("Antigravity延迟期间context取消", func(t *testing.T) { mock := &mockTempUnscheduler{} fs := NewFailoverState(3, false) fs.SwitchCount = 1 // 下一次 switchCount=2 → delay = 1s err := newTestFailoverErr(500, false, false) ctx, cancel := context.WithCancel(context.Background()) cancel() // 立即取消 start := time.Now() action := fs.HandleFailoverError(ctx, mock, 100, service.PlatformAntigravity, err) elapsed := time.Since(start) require.Equal(t, FailoverCanceled, action) require.Less(t, elapsed, 100*time.Millisecond, "应立即返回而非等待 1s") }) } // --------------------------------------------------------------------------- // HandleFailoverError — FailedAccountIDs 跟踪 // --------------------------------------------------------------------------- func TestHandleFailoverError_FailedAccountIDs(t *testing.T) { t.Run("切换时添加到失败列表", func(t *testing.T) { mock := &mockTempUnscheduler{} fs := NewFailoverState(3, false) fs.HandleFailoverError(context.Background(), mock, 100, "openai", newTestFailoverErr(500, false, false)) require.Contains(t, fs.FailedAccountIDs, int64(100)) fs.HandleFailoverError(context.Background(), mock, 200, "openai", newTestFailoverErr(502, false, false)) require.Contains(t, fs.FailedAccountIDs, int64(200)) require.Len(t, fs.FailedAccountIDs, 2) }) t.Run("耗尽时也添加到失败列表", func(t *testing.T) { mock := &mockTempUnscheduler{} fs := NewFailoverState(0, false) action := fs.HandleFailoverError(context.Background(), mock, 100, "openai", newTestFailoverErr(500, false, false)) require.Equal(t, FailoverExhausted, action) require.Contains(t, fs.FailedAccountIDs, int64(100)) }) t.Run("同账号重试期间不添加到失败列表", func(t *testing.T) { mock := &mockTempUnscheduler{} fs := NewFailoverState(3, false) action := fs.HandleFailoverError(context.Background(), mock, 100, "openai", newTestFailoverErr(400, true, false)) require.Equal(t, FailoverContinue, action) require.NotContains(t, fs.FailedAccountIDs, int64(100)) }) t.Run("同一账号多次切换不重复添加", func(t *testing.T) { mock := &mockTempUnscheduler{} fs := NewFailoverState(5, false) fs.HandleFailoverError(context.Background(), mock, 100, "openai", newTestFailoverErr(500, false, false)) fs.HandleFailoverError(context.Background(), mock, 100, "openai", newTestFailoverErr(500, false, false)) require.Len(t, fs.FailedAccountIDs, 1, "map 天然去重") }) } // --------------------------------------------------------------------------- // HandleFailoverError — LastFailoverErr 更新 // --------------------------------------------------------------------------- func TestHandleFailoverError_LastFailoverErr(t *testing.T) { t.Run("每次调用都更新LastFailoverErr", func(t *testing.T) { mock := &mockTempUnscheduler{} fs := NewFailoverState(3, false) err1 := newTestFailoverErr(500, false, false) fs.HandleFailoverError(context.Background(), mock, 100, "openai", err1) require.Equal(t, err1, fs.LastFailoverErr) err2 := newTestFailoverErr(502, false, false) fs.HandleFailoverError(context.Background(), mock, 200, "openai", err2) require.Equal(t, err2, fs.LastFailoverErr) }) t.Run("同账号重试时也更新LastFailoverErr", func(t *testing.T) { mock := &mockTempUnscheduler{} fs := NewFailoverState(3, false) err := newTestFailoverErr(400, true, false) fs.HandleFailoverError(context.Background(), mock, 100, "openai", err) require.Equal(t, err, fs.LastFailoverErr) }) } // --------------------------------------------------------------------------- // HandleFailoverError — 综合集成场景 // --------------------------------------------------------------------------- func TestHandleFailoverError_IntegrationScenario(t *testing.T) { t.Run("模拟完整failover流程_多账号混合重试与切换", func(t *testing.T) { mock := &mockTempUnscheduler{} fs := NewFailoverState(3, true) // hasBoundSession=true // 1. 账号 100 遇到可重试错误,同账号重试 2 次 retryErr := newTestFailoverErr(400, true, false) action := fs.HandleFailoverError(context.Background(), mock, 100, "openai", retryErr) require.Equal(t, FailoverContinue, action) require.True(t, fs.ForceCacheBilling, "hasBoundSession=true 应设置 ForceCacheBilling") action = fs.HandleFailoverError(context.Background(), mock, 100, "openai", retryErr) require.Equal(t, FailoverContinue, action) // 2. 账号 100 重试耗尽 → TempUnschedule + 切换 action = fs.HandleFailoverError(context.Background(), mock, 100, "openai", retryErr) require.Equal(t, FailoverContinue, action) require.Equal(t, 1, fs.SwitchCount) require.Len(t, mock.calls, 1) // 3. 账号 200 遇到不可重试错误 → 直接切换 switchErr := newTestFailoverErr(500, false, false) action = fs.HandleFailoverError(context.Background(), mock, 200, "openai", switchErr) require.Equal(t, FailoverContinue, action) require.Equal(t, 2, fs.SwitchCount) // 4. 账号 300 遇到不可重试错误 → 再切换 action = fs.HandleFailoverError(context.Background(), mock, 300, "openai", switchErr) require.Equal(t, FailoverContinue, action) require.Equal(t, 3, fs.SwitchCount) // 5. 账号 400 → 已耗尽 (SwitchCount=3 >= MaxSwitches=3) action = fs.HandleFailoverError(context.Background(), mock, 400, "openai", switchErr) require.Equal(t, FailoverExhausted, action) // 最终状态验证 require.Equal(t, 3, fs.SwitchCount, "耗尽时不再递增") require.Len(t, fs.FailedAccountIDs, 4, "4个不同账号都在失败列表中") require.True(t, fs.ForceCacheBilling) require.Len(t, mock.calls, 1, "只有账号 100 触发了 TempUnschedule") }) t.Run("模拟Antigravity平台完整流程", func(t *testing.T) { mock := &mockTempUnscheduler{} fs := NewFailoverState(2, false) err := newTestFailoverErr(500, false, false) // 第一次切换:delay = 0s start := time.Now() action := fs.HandleFailoverError(context.Background(), mock, 100, service.PlatformAntigravity, err) elapsed := time.Since(start) require.Equal(t, FailoverContinue, action) require.Less(t, elapsed, 200*time.Millisecond, "第一次切换延迟为 0") // 第二次切换:delay = 1s start = time.Now() action = fs.HandleFailoverError(context.Background(), mock, 200, service.PlatformAntigravity, err) elapsed = time.Since(start) require.Equal(t, FailoverContinue, action) require.GreaterOrEqual(t, elapsed, 800*time.Millisecond, "第二次切换延迟约 1s") // 第三次:耗尽(无延迟,因为在检查延迟之前就返回了) start = time.Now() action = fs.HandleFailoverError(context.Background(), mock, 300, service.PlatformAntigravity, err) elapsed = time.Since(start) require.Equal(t, FailoverExhausted, action) require.Less(t, elapsed, 200*time.Millisecond, "耗尽时不应有延迟") }) t.Run("ForceCacheBilling通过错误标志设置", func(t *testing.T) { mock := &mockTempUnscheduler{} fs := NewFailoverState(3, false) // hasBoundSession=false // 第一次:ForceCacheBilling=false err1 := newTestFailoverErr(500, false, false) fs.HandleFailoverError(context.Background(), mock, 100, "openai", err1) require.False(t, fs.ForceCacheBilling) // 第二次:ForceCacheBilling=true(Antigravity 粘性会话切换) err2 := newTestFailoverErr(500, false, true) fs.HandleFailoverError(context.Background(), mock, 200, "openai", err2) require.True(t, fs.ForceCacheBilling, "错误标志应触发 ForceCacheBilling") // 第三次:ForceCacheBilling=false,但状态仍保持 true err3 := newTestFailoverErr(500, false, false) fs.HandleFailoverError(context.Background(), mock, 300, "openai", err3) require.True(t, fs.ForceCacheBilling, "不应重置") }) } // --------------------------------------------------------------------------- // HandleFailoverError — 边界条件 // --------------------------------------------------------------------------- func TestHandleFailoverError_EdgeCases(t *testing.T) { t.Run("StatusCode为0的错误也能正常处理", func(t *testing.T) { mock := &mockTempUnscheduler{} fs := NewFailoverState(3, false) err := newTestFailoverErr(0, false, false) action := fs.HandleFailoverError(context.Background(), mock, 100, "openai", err) require.Equal(t, FailoverContinue, action) }) t.Run("AccountID为0也能正常跟踪", func(t *testing.T) { mock := &mockTempUnscheduler{} fs := NewFailoverState(3, false) err := newTestFailoverErr(500, true, false) action := fs.HandleFailoverError(context.Background(), mock, 0, "openai", err) require.Equal(t, FailoverContinue, action) require.Equal(t, 1, fs.SameAccountRetryCount[0]) }) t.Run("负AccountID也能正常跟踪", func(t *testing.T) { mock := &mockTempUnscheduler{} fs := NewFailoverState(3, false) err := newTestFailoverErr(500, true, false) action := fs.HandleFailoverError(context.Background(), mock, -1, "openai", err) require.Equal(t, FailoverContinue, action) require.Equal(t, 1, fs.SameAccountRetryCount[-1]) }) t.Run("空平台名称不触发Antigravity延迟", func(t *testing.T) { mock := &mockTempUnscheduler{} fs := NewFailoverState(3, false) fs.SwitchCount = 1 err := newTestFailoverErr(500, false, false) start := time.Now() action := fs.HandleFailoverError(context.Background(), mock, 100, "", err) elapsed := time.Since(start) require.Equal(t, FailoverContinue, action) require.Less(t, elapsed, 200*time.Millisecond, "空平台不应触发 Antigravity 延迟") }) } // --------------------------------------------------------------------------- // HandleSelectionExhausted 测试 // --------------------------------------------------------------------------- func TestHandleSelectionExhausted(t *testing.T) { t.Run("无LastFailoverErr时返回Exhausted", func(t *testing.T) { fs := NewFailoverState(3, false) // LastFailoverErr 为 nil action := fs.HandleSelectionExhausted(context.Background()) require.Equal(t, FailoverExhausted, action) }) t.Run("非503错误返回Exhausted", func(t *testing.T) { fs := NewFailoverState(3, false) fs.LastFailoverErr = newTestFailoverErr(500, false, false) action := fs.HandleSelectionExhausted(context.Background()) require.Equal(t, FailoverExhausted, action) }) t.Run("503且未耗尽_等待后返回Continue并清除失败列表", func(t *testing.T) { fs := NewFailoverState(3, false) fs.LastFailoverErr = newTestFailoverErr(503, false, false) fs.FailedAccountIDs[100] = struct{}{} fs.SwitchCount = 1 start := time.Now() action := fs.HandleSelectionExhausted(context.Background()) elapsed := time.Since(start) require.Equal(t, FailoverContinue, action) require.Empty(t, fs.FailedAccountIDs, "应清除失败账号列表") require.GreaterOrEqual(t, elapsed, 1500*time.Millisecond, "应等待约 2s") require.Less(t, elapsed, 5*time.Second) }) t.Run("503但SwitchCount已超过MaxSwitches_返回Exhausted", func(t *testing.T) { fs := NewFailoverState(2, false) fs.LastFailoverErr = newTestFailoverErr(503, false, false) fs.SwitchCount = 3 // > MaxSwitches(2) start := time.Now() action := fs.HandleSelectionExhausted(context.Background()) elapsed := time.Since(start) require.Equal(t, FailoverExhausted, action) require.Less(t, elapsed, 100*time.Millisecond, "不应等待") }) t.Run("503但context已取消_返回Canceled", func(t *testing.T) { fs := NewFailoverState(3, false) fs.LastFailoverErr = newTestFailoverErr(503, false, false) ctx, cancel := context.WithCancel(context.Background()) cancel() start := time.Now() action := fs.HandleSelectionExhausted(ctx) elapsed := time.Since(start) require.Equal(t, FailoverCanceled, action) require.Less(t, elapsed, 100*time.Millisecond, "应立即返回") }) t.Run("503且SwitchCount等于MaxSwitches_仍可重试", func(t *testing.T) { fs := NewFailoverState(2, false) fs.LastFailoverErr = newTestFailoverErr(503, false, false) fs.SwitchCount = 2 // == MaxSwitches,条件是 <=,仍可重试 action := fs.HandleSelectionExhausted(context.Background()) require.Equal(t, FailoverContinue, action) }) }