当账号触发特定错误码和关键词匹配时,自动临时禁用调度: 后端: - 新增 TempUnschedCache Redis 缓存层 - RateLimitService 支持规则匹配和状态管理 - 添加 GET/DELETE /accounts/:id/temp-unschedulable API - 数据库迁移添加 temp_unschedulable_until/reason 字段 前端: - 账号状态指示器显示临时不可调度状态 - 新增 TempUnschedStatusModal 详情弹窗 - 创建/编辑账号时支持配置规则和预设模板 - 完整的中英文国际化支持
312 lines
8.4 KiB
Go
312 lines
8.4 KiB
Go
package dto
|
|
|
|
import "github.com/Wei-Shaw/sub2api/internal/service"
|
|
|
|
func UserFromServiceShallow(u *service.User) *User {
|
|
if u == nil {
|
|
return nil
|
|
}
|
|
return &User{
|
|
ID: u.ID,
|
|
Email: u.Email,
|
|
Username: u.Username,
|
|
Notes: u.Notes,
|
|
Role: u.Role,
|
|
Balance: u.Balance,
|
|
Concurrency: u.Concurrency,
|
|
Status: u.Status,
|
|
AllowedGroups: u.AllowedGroups,
|
|
CreatedAt: u.CreatedAt,
|
|
UpdatedAt: u.UpdatedAt,
|
|
}
|
|
}
|
|
|
|
func UserFromService(u *service.User) *User {
|
|
if u == nil {
|
|
return nil
|
|
}
|
|
out := UserFromServiceShallow(u)
|
|
if len(u.ApiKeys) > 0 {
|
|
out.ApiKeys = make([]ApiKey, 0, len(u.ApiKeys))
|
|
for i := range u.ApiKeys {
|
|
k := u.ApiKeys[i]
|
|
out.ApiKeys = append(out.ApiKeys, *ApiKeyFromService(&k))
|
|
}
|
|
}
|
|
if len(u.Subscriptions) > 0 {
|
|
out.Subscriptions = make([]UserSubscription, 0, len(u.Subscriptions))
|
|
for i := range u.Subscriptions {
|
|
s := u.Subscriptions[i]
|
|
out.Subscriptions = append(out.Subscriptions, *UserSubscriptionFromService(&s))
|
|
}
|
|
}
|
|
return out
|
|
}
|
|
|
|
func ApiKeyFromService(k *service.ApiKey) *ApiKey {
|
|
if k == nil {
|
|
return nil
|
|
}
|
|
return &ApiKey{
|
|
ID: k.ID,
|
|
UserID: k.UserID,
|
|
Key: k.Key,
|
|
Name: k.Name,
|
|
GroupID: k.GroupID,
|
|
Status: k.Status,
|
|
CreatedAt: k.CreatedAt,
|
|
UpdatedAt: k.UpdatedAt,
|
|
User: UserFromServiceShallow(k.User),
|
|
Group: GroupFromServiceShallow(k.Group),
|
|
}
|
|
}
|
|
|
|
func GroupFromServiceShallow(g *service.Group) *Group {
|
|
if g == nil {
|
|
return nil
|
|
}
|
|
return &Group{
|
|
ID: g.ID,
|
|
Name: g.Name,
|
|
Description: g.Description,
|
|
Platform: g.Platform,
|
|
RateMultiplier: g.RateMultiplier,
|
|
IsExclusive: g.IsExclusive,
|
|
Status: g.Status,
|
|
SubscriptionType: g.SubscriptionType,
|
|
DailyLimitUSD: g.DailyLimitUSD,
|
|
WeeklyLimitUSD: g.WeeklyLimitUSD,
|
|
MonthlyLimitUSD: g.MonthlyLimitUSD,
|
|
CreatedAt: g.CreatedAt,
|
|
UpdatedAt: g.UpdatedAt,
|
|
AccountCount: g.AccountCount,
|
|
}
|
|
}
|
|
|
|
func GroupFromService(g *service.Group) *Group {
|
|
if g == nil {
|
|
return nil
|
|
}
|
|
out := GroupFromServiceShallow(g)
|
|
if len(g.AccountGroups) > 0 {
|
|
out.AccountGroups = make([]AccountGroup, 0, len(g.AccountGroups))
|
|
for i := range g.AccountGroups {
|
|
ag := g.AccountGroups[i]
|
|
out.AccountGroups = append(out.AccountGroups, *AccountGroupFromService(&ag))
|
|
}
|
|
}
|
|
return out
|
|
}
|
|
|
|
func AccountFromServiceShallow(a *service.Account) *Account {
|
|
if a == nil {
|
|
return nil
|
|
}
|
|
return &Account{
|
|
ID: a.ID,
|
|
Name: a.Name,
|
|
Platform: a.Platform,
|
|
Type: a.Type,
|
|
Credentials: a.Credentials,
|
|
Extra: a.Extra,
|
|
ProxyID: a.ProxyID,
|
|
Concurrency: a.Concurrency,
|
|
Priority: a.Priority,
|
|
Status: a.Status,
|
|
ErrorMessage: a.ErrorMessage,
|
|
LastUsedAt: a.LastUsedAt,
|
|
CreatedAt: a.CreatedAt,
|
|
UpdatedAt: a.UpdatedAt,
|
|
Schedulable: a.Schedulable,
|
|
RateLimitedAt: a.RateLimitedAt,
|
|
RateLimitResetAt: a.RateLimitResetAt,
|
|
OverloadUntil: a.OverloadUntil,
|
|
TempUnschedulableUntil: a.TempUnschedulableUntil,
|
|
TempUnschedulableReason: a.TempUnschedulableReason,
|
|
SessionWindowStart: a.SessionWindowStart,
|
|
SessionWindowEnd: a.SessionWindowEnd,
|
|
SessionWindowStatus: a.SessionWindowStatus,
|
|
GroupIDs: a.GroupIDs,
|
|
}
|
|
}
|
|
|
|
func AccountFromService(a *service.Account) *Account {
|
|
if a == nil {
|
|
return nil
|
|
}
|
|
out := AccountFromServiceShallow(a)
|
|
out.Proxy = ProxyFromService(a.Proxy)
|
|
if len(a.AccountGroups) > 0 {
|
|
out.AccountGroups = make([]AccountGroup, 0, len(a.AccountGroups))
|
|
for i := range a.AccountGroups {
|
|
ag := a.AccountGroups[i]
|
|
out.AccountGroups = append(out.AccountGroups, *AccountGroupFromService(&ag))
|
|
}
|
|
}
|
|
if len(a.Groups) > 0 {
|
|
out.Groups = make([]*Group, 0, len(a.Groups))
|
|
for _, g := range a.Groups {
|
|
out.Groups = append(out.Groups, GroupFromServiceShallow(g))
|
|
}
|
|
}
|
|
return out
|
|
}
|
|
|
|
func AccountGroupFromService(ag *service.AccountGroup) *AccountGroup {
|
|
if ag == nil {
|
|
return nil
|
|
}
|
|
return &AccountGroup{
|
|
AccountID: ag.AccountID,
|
|
GroupID: ag.GroupID,
|
|
Priority: ag.Priority,
|
|
CreatedAt: ag.CreatedAt,
|
|
Account: AccountFromServiceShallow(ag.Account),
|
|
Group: GroupFromServiceShallow(ag.Group),
|
|
}
|
|
}
|
|
|
|
func ProxyFromService(p *service.Proxy) *Proxy {
|
|
if p == nil {
|
|
return nil
|
|
}
|
|
return &Proxy{
|
|
ID: p.ID,
|
|
Name: p.Name,
|
|
Protocol: p.Protocol,
|
|
Host: p.Host,
|
|
Port: p.Port,
|
|
Username: p.Username,
|
|
Password: p.Password,
|
|
Status: p.Status,
|
|
CreatedAt: p.CreatedAt,
|
|
UpdatedAt: p.UpdatedAt,
|
|
}
|
|
}
|
|
|
|
func ProxyWithAccountCountFromService(p *service.ProxyWithAccountCount) *ProxyWithAccountCount {
|
|
if p == nil {
|
|
return nil
|
|
}
|
|
return &ProxyWithAccountCount{
|
|
Proxy: *ProxyFromService(&p.Proxy),
|
|
AccountCount: p.AccountCount,
|
|
}
|
|
}
|
|
|
|
func RedeemCodeFromService(rc *service.RedeemCode) *RedeemCode {
|
|
if rc == nil {
|
|
return nil
|
|
}
|
|
return &RedeemCode{
|
|
ID: rc.ID,
|
|
Code: rc.Code,
|
|
Type: rc.Type,
|
|
Value: rc.Value,
|
|
Status: rc.Status,
|
|
UsedBy: rc.UsedBy,
|
|
UsedAt: rc.UsedAt,
|
|
Notes: rc.Notes,
|
|
CreatedAt: rc.CreatedAt,
|
|
GroupID: rc.GroupID,
|
|
ValidityDays: rc.ValidityDays,
|
|
User: UserFromServiceShallow(rc.User),
|
|
Group: GroupFromServiceShallow(rc.Group),
|
|
}
|
|
}
|
|
|
|
func UsageLogFromService(l *service.UsageLog) *UsageLog {
|
|
if l == nil {
|
|
return nil
|
|
}
|
|
return &UsageLog{
|
|
ID: l.ID,
|
|
UserID: l.UserID,
|
|
ApiKeyID: l.ApiKeyID,
|
|
AccountID: l.AccountID,
|
|
RequestID: l.RequestID,
|
|
Model: l.Model,
|
|
GroupID: l.GroupID,
|
|
SubscriptionID: l.SubscriptionID,
|
|
InputTokens: l.InputTokens,
|
|
OutputTokens: l.OutputTokens,
|
|
CacheCreationTokens: l.CacheCreationTokens,
|
|
CacheReadTokens: l.CacheReadTokens,
|
|
CacheCreation5mTokens: l.CacheCreation5mTokens,
|
|
CacheCreation1hTokens: l.CacheCreation1hTokens,
|
|
InputCost: l.InputCost,
|
|
OutputCost: l.OutputCost,
|
|
CacheCreationCost: l.CacheCreationCost,
|
|
CacheReadCost: l.CacheReadCost,
|
|
TotalCost: l.TotalCost,
|
|
ActualCost: l.ActualCost,
|
|
RateMultiplier: l.RateMultiplier,
|
|
BillingType: l.BillingType,
|
|
Stream: l.Stream,
|
|
DurationMs: l.DurationMs,
|
|
FirstTokenMs: l.FirstTokenMs,
|
|
CreatedAt: l.CreatedAt,
|
|
User: UserFromServiceShallow(l.User),
|
|
ApiKey: ApiKeyFromService(l.ApiKey),
|
|
Account: AccountFromService(l.Account),
|
|
Group: GroupFromServiceShallow(l.Group),
|
|
Subscription: UserSubscriptionFromService(l.Subscription),
|
|
}
|
|
}
|
|
|
|
func SettingFromService(s *service.Setting) *Setting {
|
|
if s == nil {
|
|
return nil
|
|
}
|
|
return &Setting{
|
|
ID: s.ID,
|
|
Key: s.Key,
|
|
Value: s.Value,
|
|
UpdatedAt: s.UpdatedAt,
|
|
}
|
|
}
|
|
|
|
func UserSubscriptionFromService(sub *service.UserSubscription) *UserSubscription {
|
|
if sub == nil {
|
|
return nil
|
|
}
|
|
return &UserSubscription{
|
|
ID: sub.ID,
|
|
UserID: sub.UserID,
|
|
GroupID: sub.GroupID,
|
|
StartsAt: sub.StartsAt,
|
|
ExpiresAt: sub.ExpiresAt,
|
|
Status: sub.Status,
|
|
DailyWindowStart: sub.DailyWindowStart,
|
|
WeeklyWindowStart: sub.WeeklyWindowStart,
|
|
MonthlyWindowStart: sub.MonthlyWindowStart,
|
|
DailyUsageUSD: sub.DailyUsageUSD,
|
|
WeeklyUsageUSD: sub.WeeklyUsageUSD,
|
|
MonthlyUsageUSD: sub.MonthlyUsageUSD,
|
|
AssignedBy: sub.AssignedBy,
|
|
AssignedAt: sub.AssignedAt,
|
|
Notes: sub.Notes,
|
|
CreatedAt: sub.CreatedAt,
|
|
UpdatedAt: sub.UpdatedAt,
|
|
User: UserFromServiceShallow(sub.User),
|
|
Group: GroupFromServiceShallow(sub.Group),
|
|
AssignedByUser: UserFromServiceShallow(sub.AssignedByUser),
|
|
}
|
|
}
|
|
|
|
func BulkAssignResultFromService(r *service.BulkAssignResult) *BulkAssignResult {
|
|
if r == nil {
|
|
return nil
|
|
}
|
|
subs := make([]UserSubscription, 0, len(r.Subscriptions))
|
|
for i := range r.Subscriptions {
|
|
subs = append(subs, *UserSubscriptionFromService(&r.Subscriptions[i]))
|
|
}
|
|
return &BulkAssignResult{
|
|
SuccessCount: r.SuccessCount,
|
|
FailedCount: r.FailedCount,
|
|
Subscriptions: subs,
|
|
Errors: r.Errors,
|
|
}
|
|
}
|