2026-03-16 00:45:17 +08:00
|
|
|
|
//go:build unit
|
|
|
|
|
|
|
|
|
|
|
|
package service
|
|
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
|
"context"
|
|
|
|
|
|
"testing"
|
|
|
|
|
|
"time"
|
|
|
|
|
|
|
|
|
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
type updateAccountOveragesRepoStub struct {
|
|
|
|
|
|
mockAccountRepoForGemini
|
|
|
|
|
|
account *Account
|
|
|
|
|
|
updateCalls int
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func (r *updateAccountOveragesRepoStub) GetByID(ctx context.Context, id int64) (*Account, error) {
|
|
|
|
|
|
return r.account, nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func (r *updateAccountOveragesRepoStub) Update(ctx context.Context, account *Account) error {
|
|
|
|
|
|
r.updateCalls++
|
|
|
|
|
|
r.account = account
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-03-16 04:31:22 +08:00
|
|
|
|
func TestUpdateAccount_DisableOveragesClearsAICreditsKey(t *testing.T) {
|
2026-03-16 00:45:17 +08:00
|
|
|
|
accountID := int64(101)
|
|
|
|
|
|
repo := &updateAccountOveragesRepoStub{
|
|
|
|
|
|
account: &Account{
|
|
|
|
|
|
ID: accountID,
|
|
|
|
|
|
Platform: PlatformAntigravity,
|
|
|
|
|
|
Type: AccountTypeOAuth,
|
|
|
|
|
|
Status: StatusActive,
|
|
|
|
|
|
Extra: map[string]any{
|
|
|
|
|
|
"allow_overages": true,
|
|
|
|
|
|
"mixed_scheduling": true,
|
2026-03-16 04:31:22 +08:00
|
|
|
|
modelRateLimitsKey: map[string]any{
|
2026-03-16 00:45:17 +08:00
|
|
|
|
"claude-sonnet-4-5": map[string]any{
|
2026-03-16 04:31:22 +08:00
|
|
|
|
"rate_limited_at": "2026-03-15T00:00:00Z",
|
|
|
|
|
|
"rate_limit_reset_at": "2099-03-15T00:00:00Z",
|
|
|
|
|
|
},
|
|
|
|
|
|
creditsExhaustedKey: map[string]any{
|
|
|
|
|
|
"rate_limited_at": "2026-03-15T00:00:00Z",
|
|
|
|
|
|
"rate_limit_reset_at": time.Now().Add(5 * time.Hour).UTC().Format(time.RFC3339),
|
2026-03-16 00:45:17 +08:00
|
|
|
|
},
|
|
|
|
|
|
},
|
|
|
|
|
|
},
|
|
|
|
|
|
},
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
svc := &adminServiceImpl{accountRepo: repo}
|
|
|
|
|
|
updated, err := svc.UpdateAccount(context.Background(), accountID, &UpdateAccountInput{
|
|
|
|
|
|
Extra: map[string]any{
|
|
|
|
|
|
"mixed_scheduling": true,
|
2026-03-16 04:31:22 +08:00
|
|
|
|
modelRateLimitsKey: map[string]any{
|
|
|
|
|
|
"claude-sonnet-4-5": map[string]any{
|
|
|
|
|
|
"rate_limited_at": "2026-03-15T00:00:00Z",
|
|
|
|
|
|
"rate_limit_reset_at": "2099-03-15T00:00:00Z",
|
|
|
|
|
|
},
|
|
|
|
|
|
creditsExhaustedKey: map[string]any{
|
|
|
|
|
|
"rate_limited_at": "2026-03-15T00:00:00Z",
|
|
|
|
|
|
"rate_limit_reset_at": time.Now().Add(5 * time.Hour).UTC().Format(time.RFC3339),
|
|
|
|
|
|
},
|
|
|
|
|
|
},
|
2026-03-16 00:45:17 +08:00
|
|
|
|
},
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
require.NotNil(t, updated)
|
|
|
|
|
|
require.Equal(t, 1, repo.updateCalls)
|
|
|
|
|
|
require.False(t, updated.IsOveragesEnabled())
|
|
|
|
|
|
|
2026-03-16 04:31:22 +08:00
|
|
|
|
// 关闭 overages 后,AICredits key 应被清除
|
|
|
|
|
|
rawLimits, ok := repo.account.Extra[modelRateLimitsKey].(map[string]any)
|
|
|
|
|
|
if ok {
|
|
|
|
|
|
_, exists := rawLimits[creditsExhaustedKey]
|
|
|
|
|
|
require.False(t, exists, "关闭 overages 时应清除 AICredits 限流 key")
|
|
|
|
|
|
}
|
|
|
|
|
|
// 普通模型限流应保留
|
|
|
|
|
|
require.True(t, ok)
|
|
|
|
|
|
_, exists := rawLimits["claude-sonnet-4-5"]
|
|
|
|
|
|
require.True(t, exists, "普通模型限流应保留")
|
2026-03-16 00:45:17 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
func TestUpdateAccount_EnableOveragesClearsModelRateLimitsBeforePersist(t *testing.T) {
|
|
|
|
|
|
accountID := int64(102)
|
|
|
|
|
|
repo := &updateAccountOveragesRepoStub{
|
|
|
|
|
|
account: &Account{
|
|
|
|
|
|
ID: accountID,
|
|
|
|
|
|
Platform: PlatformAntigravity,
|
|
|
|
|
|
Type: AccountTypeOAuth,
|
|
|
|
|
|
Status: StatusActive,
|
|
|
|
|
|
Extra: map[string]any{
|
|
|
|
|
|
"mixed_scheduling": true,
|
|
|
|
|
|
modelRateLimitsKey: map[string]any{
|
|
|
|
|
|
"claude-sonnet-4-5": map[string]any{
|
|
|
|
|
|
"rate_limited_at": "2026-03-15T00:00:00Z",
|
|
|
|
|
|
"rate_limit_reset_at": "2099-03-15T00:00:00Z",
|
|
|
|
|
|
},
|
|
|
|
|
|
},
|
|
|
|
|
|
},
|
|
|
|
|
|
},
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
svc := &adminServiceImpl{accountRepo: repo}
|
|
|
|
|
|
updated, err := svc.UpdateAccount(context.Background(), accountID, &UpdateAccountInput{
|
|
|
|
|
|
Extra: map[string]any{
|
|
|
|
|
|
"mixed_scheduling": true,
|
|
|
|
|
|
"allow_overages": true,
|
|
|
|
|
|
},
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
require.NotNil(t, updated)
|
|
|
|
|
|
require.Equal(t, 1, repo.updateCalls)
|
|
|
|
|
|
require.True(t, updated.IsOveragesEnabled())
|
|
|
|
|
|
|
|
|
|
|
|
_, exists := repo.account.Extra[modelRateLimitsKey]
|
|
|
|
|
|
require.False(t, exists, "开启 overages 时应在持久化前清掉旧模型限流")
|
|
|
|
|
|
}
|
2026-03-16 16:22:31 +08:00
|
|
|
|
|
|
|
|
|
|
func TestUpdateAccount_EmptyExtraPayloadCanClearQuotaLimits(t *testing.T) {
|
|
|
|
|
|
accountID := int64(103)
|
|
|
|
|
|
repo := &updateAccountOveragesRepoStub{
|
|
|
|
|
|
account: &Account{
|
|
|
|
|
|
ID: accountID,
|
|
|
|
|
|
Platform: PlatformAnthropic,
|
|
|
|
|
|
Type: AccountTypeAPIKey,
|
|
|
|
|
|
Status: StatusActive,
|
|
|
|
|
|
Extra: map[string]any{
|
|
|
|
|
|
"quota_limit": 100.0,
|
|
|
|
|
|
"quota_daily_limit": 10.0,
|
|
|
|
|
|
"quota_weekly_limit": 40.0,
|
|
|
|
|
|
},
|
|
|
|
|
|
},
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
svc := &adminServiceImpl{accountRepo: repo}
|
|
|
|
|
|
updated, err := svc.UpdateAccount(context.Background(), accountID, &UpdateAccountInput{
|
|
|
|
|
|
// 显式空对象:语义是“清空 extra 中的可配置键”(例如关闭配额限制)
|
|
|
|
|
|
Extra: map[string]any{},
|
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
require.NotNil(t, updated)
|
|
|
|
|
|
require.Equal(t, 1, repo.updateCalls)
|
|
|
|
|
|
require.NotNil(t, repo.account.Extra)
|
|
|
|
|
|
require.NotContains(t, repo.account.Extra, "quota_limit")
|
|
|
|
|
|
require.NotContains(t, repo.account.Extra, "quota_daily_limit")
|
|
|
|
|
|
require.NotContains(t, repo.account.Extra, "quota_weekly_limit")
|
|
|
|
|
|
require.Len(t, repo.account.Extra, 0)
|
|
|
|
|
|
}
|