121 lines
4.0 KiB
Go
121 lines
4.0 KiB
Go
//go:build unit
|
||
|
||
package service
|
||
|
||
import (
|
||
"context"
|
||
"errors"
|
||
"testing"
|
||
|
||
"github.com/stretchr/testify/require"
|
||
)
|
||
|
||
// TestDecrementWaitCount_NilCache 确保 nil cache 不会 panic
|
||
func TestDecrementWaitCount_NilCache(t *testing.T) {
|
||
svc := &ConcurrencyService{cache: nil}
|
||
// 不应 panic
|
||
svc.DecrementWaitCount(context.Background(), 1)
|
||
}
|
||
|
||
// TestDecrementWaitCount_CacheError 确保 cache 错误不会传播
|
||
func TestDecrementWaitCount_CacheError(t *testing.T) {
|
||
cache := &stubConcurrencyCacheForTest{}
|
||
svc := NewConcurrencyService(cache)
|
||
// DecrementWaitCount 使用 background context,错误只记录日志不传播
|
||
svc.DecrementWaitCount(context.Background(), 1)
|
||
}
|
||
|
||
// TestDecrementAccountWaitCount_NilCache 确保 nil cache 不会 panic
|
||
func TestDecrementAccountWaitCount_NilCache(t *testing.T) {
|
||
svc := &ConcurrencyService{cache: nil}
|
||
svc.DecrementAccountWaitCount(context.Background(), 1)
|
||
}
|
||
|
||
// TestDecrementAccountWaitCount_CacheError 确保 cache 错误不会传播
|
||
func TestDecrementAccountWaitCount_CacheError(t *testing.T) {
|
||
cache := &stubConcurrencyCacheForTest{}
|
||
svc := NewConcurrencyService(cache)
|
||
svc.DecrementAccountWaitCount(context.Background(), 1)
|
||
}
|
||
|
||
// TestWaitingQueueFlow_IncrementThenDecrement 测试完整的等待队列增减流程
|
||
func TestWaitingQueueFlow_IncrementThenDecrement(t *testing.T) {
|
||
cache := &stubConcurrencyCacheForTest{waitAllowed: true}
|
||
svc := NewConcurrencyService(cache)
|
||
|
||
// 进入等待队列
|
||
allowed, err := svc.IncrementWaitCount(context.Background(), 1, 25)
|
||
require.NoError(t, err)
|
||
require.True(t, allowed)
|
||
|
||
// 离开等待队列(不应 panic)
|
||
svc.DecrementWaitCount(context.Background(), 1)
|
||
}
|
||
|
||
// TestWaitingQueueFlow_AccountLevel 测试账号级等待队列流程
|
||
func TestWaitingQueueFlow_AccountLevel(t *testing.T) {
|
||
cache := &stubConcurrencyCacheForTest{waitAllowed: true}
|
||
svc := NewConcurrencyService(cache)
|
||
|
||
// 进入账号等待队列
|
||
allowed, err := svc.IncrementAccountWaitCount(context.Background(), 42, 10)
|
||
require.NoError(t, err)
|
||
require.True(t, allowed)
|
||
|
||
// 离开账号等待队列
|
||
svc.DecrementAccountWaitCount(context.Background(), 42)
|
||
}
|
||
|
||
// TestWaitingQueueFull_Returns429Signal 测试等待队列满时返回 false
|
||
func TestWaitingQueueFull_Returns429Signal(t *testing.T) {
|
||
// waitAllowed=false 模拟队列已满
|
||
cache := &stubConcurrencyCacheForTest{waitAllowed: false}
|
||
svc := NewConcurrencyService(cache)
|
||
|
||
// 用户级等待队列满
|
||
allowed, err := svc.IncrementWaitCount(context.Background(), 1, 25)
|
||
require.NoError(t, err)
|
||
require.False(t, allowed, "等待队列满时应返回 false(调用方根据此返回 429)")
|
||
|
||
// 账号级等待队列满
|
||
allowed, err = svc.IncrementAccountWaitCount(context.Background(), 1, 10)
|
||
require.NoError(t, err)
|
||
require.False(t, allowed, "账号等待队列满时应返回 false")
|
||
}
|
||
|
||
// TestWaitingQueue_FailOpen_OnCacheError 测试 Redis 故障时 fail-open
|
||
func TestWaitingQueue_FailOpen_OnCacheError(t *testing.T) {
|
||
cache := &stubConcurrencyCacheForTest{waitErr: errors.New("redis connection refused")}
|
||
svc := NewConcurrencyService(cache)
|
||
|
||
// 用户级:Redis 错误时允许通过
|
||
allowed, err := svc.IncrementWaitCount(context.Background(), 1, 25)
|
||
require.NoError(t, err, "Redis 错误不应向调用方传播")
|
||
require.True(t, allowed, "Redis 故障时应 fail-open 放行")
|
||
|
||
// 账号级:同样 fail-open
|
||
allowed, err = svc.IncrementAccountWaitCount(context.Background(), 1, 10)
|
||
require.NoError(t, err, "Redis 错误不应向调用方传播")
|
||
require.True(t, allowed, "Redis 故障时应 fail-open 放行")
|
||
}
|
||
|
||
// TestCalculateMaxWait_Scenarios 测试最大等待队列大小计算
|
||
func TestCalculateMaxWait_Scenarios(t *testing.T) {
|
||
tests := []struct {
|
||
concurrency int
|
||
expected int
|
||
}{
|
||
{5, 25}, // 5 + 20
|
||
{10, 30}, // 10 + 20
|
||
{1, 21}, // 1 + 20
|
||
{0, 21}, // min(1) + 20
|
||
{-1, 21}, // min(1) + 20
|
||
{-10, 21}, // min(1) + 20
|
||
{100, 120}, // 100 + 20
|
||
}
|
||
for _, tt := range tests {
|
||
result := CalculateMaxWait(tt.concurrency)
|
||
require.Equal(t, tt.expected, result, "CalculateMaxWait(%d)", tt.concurrency)
|
||
}
|
||
}
|