当账号触发特定错误码和关键词匹配时,自动临时禁用调度: 后端: - 新增 TempUnschedCache Redis 缓存层 - RateLimitService 支持规则匹配和状态管理 - 添加 GET/DELETE /accounts/:id/temp-unschedulable API - 数据库迁移添加 temp_unschedulable_until/reason 字段 前端: - 账号状态指示器显示临时不可调度状态 - 新增 TempUnschedStatusModal 详情弹窗 - 创建/编辑账号时支持配置规则和预设模板 - 完整的中英文国际化支持
92 lines
2.3 KiB
Go
92 lines
2.3 KiB
Go
package repository
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"fmt"
|
|
"time"
|
|
|
|
"github.com/Wei-Shaw/sub2api/internal/service"
|
|
"github.com/redis/go-redis/v9"
|
|
)
|
|
|
|
const tempUnschedPrefix = "temp_unsched:account:"
|
|
|
|
var tempUnschedSetScript = redis.NewScript(`
|
|
local key = KEYS[1]
|
|
local new_until = tonumber(ARGV[1])
|
|
local new_value = ARGV[2]
|
|
local new_ttl = tonumber(ARGV[3])
|
|
|
|
local existing = redis.call('GET', key)
|
|
if existing then
|
|
local ok, existing_data = pcall(cjson.decode, existing)
|
|
if ok and existing_data and existing_data.until_unix then
|
|
local existing_until = tonumber(existing_data.until_unix)
|
|
if existing_until and new_until <= existing_until then
|
|
return 0
|
|
end
|
|
end
|
|
end
|
|
|
|
redis.call('SET', key, new_value, 'EX', new_ttl)
|
|
return 1
|
|
`)
|
|
|
|
type tempUnschedCache struct {
|
|
rdb *redis.Client
|
|
}
|
|
|
|
func NewTempUnschedCache(rdb *redis.Client) service.TempUnschedCache {
|
|
return &tempUnschedCache{rdb: rdb}
|
|
}
|
|
|
|
// SetTempUnsched 设置临时不可调度状态(只延长不缩短)
|
|
func (c *tempUnschedCache) SetTempUnsched(ctx context.Context, accountID int64, state *service.TempUnschedState) error {
|
|
key := fmt.Sprintf("%s%d", tempUnschedPrefix, accountID)
|
|
|
|
stateJSON, err := json.Marshal(state)
|
|
if err != nil {
|
|
return fmt.Errorf("marshal state: %w", err)
|
|
}
|
|
|
|
ttl := time.Until(time.Unix(state.UntilUnix, 0))
|
|
if ttl <= 0 {
|
|
return nil // 已过期,不设置
|
|
}
|
|
|
|
ttlSeconds := int(ttl.Seconds())
|
|
if ttlSeconds < 1 {
|
|
ttlSeconds = 1
|
|
}
|
|
|
|
_, err = tempUnschedSetScript.Run(ctx, c.rdb, []string{key}, state.UntilUnix, string(stateJSON), ttlSeconds).Result()
|
|
return err
|
|
}
|
|
|
|
// GetTempUnsched 获取临时不可调度状态
|
|
func (c *tempUnschedCache) GetTempUnsched(ctx context.Context, accountID int64) (*service.TempUnschedState, error) {
|
|
key := fmt.Sprintf("%s%d", tempUnschedPrefix, accountID)
|
|
|
|
val, err := c.rdb.Get(ctx, key).Result()
|
|
if err == redis.Nil {
|
|
return nil, nil
|
|
}
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
var state service.TempUnschedState
|
|
if err := json.Unmarshal([]byte(val), &state); err != nil {
|
|
return nil, fmt.Errorf("unmarshal state: %w", err)
|
|
}
|
|
|
|
return &state, nil
|
|
}
|
|
|
|
// DeleteTempUnsched 删除临时不可调度状态
|
|
func (c *tempUnschedCache) DeleteTempUnsched(ctx context.Context, accountID int64) error {
|
|
key := fmt.Sprintf("%s%d", tempUnschedPrefix, accountID)
|
|
return c.rdb.Del(ctx, key).Err()
|
|
}
|