mirror of
https://gitee.com/wanwujie/sub2api
synced 2026-04-03 06:52:13 +08:00
UpdateAccount previously required len(input.Extra) > 0, causing explicit
empty payloads (extra:{}) to be silently skipped. Change condition to
input.Extra != nil so clearing quota keys actually persists.
156 lines
4.4 KiB
Go
156 lines
4.4 KiB
Go
//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
|
||
}
|
||
|
||
func TestUpdateAccount_DisableOveragesClearsAICreditsKey(t *testing.T) {
|
||
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,
|
||
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),
|
||
},
|
||
},
|
||
},
|
||
},
|
||
}
|
||
|
||
svc := &adminServiceImpl{accountRepo: repo}
|
||
updated, err := svc.UpdateAccount(context.Background(), accountID, &UpdateAccountInput{
|
||
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",
|
||
},
|
||
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),
|
||
},
|
||
},
|
||
},
|
||
})
|
||
|
||
require.NoError(t, err)
|
||
require.NotNil(t, updated)
|
||
require.Equal(t, 1, repo.updateCalls)
|
||
require.False(t, updated.IsOveragesEnabled())
|
||
|
||
// 关闭 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, "普通模型限流应保留")
|
||
}
|
||
|
||
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 时应在持久化前清掉旧模型限流")
|
||
}
|
||
|
||
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)
|
||
}
|