6044b00685
- 上游 Key 命名改为 {prefix}-{upstream.id}-{safe_group_name}-{group_id}
- 唯一约束 (upstream_id, group_id, managed_prefix) 加 managed_prefix 列
- 上游检测成功时同步 Key 状态,远端已删/分组已删自动清理
- 重复分组导入跳过,目标网站已存在同名分组返回 exists
- 账号导入平台自动识别(auto/manual 模式)
- 全选可导入 Key 按钮 + 目标分组自动匹配
- 导入幂等:已导入过的 Key 校验远端账号,不存在则重建
- 新增同步接口 POST /sync-imported-upstream-keys
- account_exists() 通过拉取账号列表判断,避免 404 误判
- credentials.base_url 注入来源上游地址,避免 401
- 前端导入弹窗自动同步+刷新按钮+并发/优先级设置
- 新增 12 个测试覆盖同步、幂等、远端删除、校验失败路径
180 lines
4.4 KiB
Python
180 lines
4.4 KiB
Python
from datetime import datetime
|
|
from typing import Any, Optional
|
|
|
|
from pydantic import BaseModel, Field
|
|
|
|
|
|
class TestResult(BaseModel):
|
|
success: bool
|
|
message: str
|
|
detail: Optional[str] = None
|
|
|
|
|
|
class WebsiteCreate(BaseModel):
|
|
name: str
|
|
site_type: str = "sub2api"
|
|
base_url: str
|
|
api_prefix: str = "/api/v1/admin"
|
|
auth_type: str = "api_key"
|
|
auth_config: dict[str, Any] = {}
|
|
groups_endpoint: str = "/groups"
|
|
group_update_endpoint: str = "/groups/{id}"
|
|
enabled: bool = True
|
|
auto_sync_enabled: bool = True
|
|
timeout_seconds: int = 30
|
|
|
|
|
|
class WebsiteUpdate(BaseModel):
|
|
name: Optional[str] = None
|
|
site_type: Optional[str] = None
|
|
base_url: Optional[str] = None
|
|
api_prefix: Optional[str] = None
|
|
auth_type: Optional[str] = None
|
|
auth_config: Optional[dict[str, Any]] = None
|
|
groups_endpoint: Optional[str] = None
|
|
group_update_endpoint: Optional[str] = None
|
|
enabled: Optional[bool] = None
|
|
auto_sync_enabled: Optional[bool] = None
|
|
timeout_seconds: Optional[int] = None
|
|
|
|
|
|
class WebsiteResponse(BaseModel):
|
|
id: int
|
|
name: str
|
|
site_type: str
|
|
base_url: str
|
|
api_prefix: str
|
|
auth_type: str
|
|
auth_config_masked: dict[str, Any]
|
|
groups_endpoint: str
|
|
group_update_endpoint: str
|
|
enabled: bool
|
|
auto_sync_enabled: bool
|
|
timeout_seconds: int
|
|
last_status: str
|
|
last_checked_at: Optional[datetime]
|
|
last_error: Optional[str]
|
|
created_at: datetime
|
|
updated_at: datetime
|
|
|
|
model_config = {"from_attributes": True}
|
|
|
|
|
|
class WebsiteGroupResponse(BaseModel):
|
|
id: str
|
|
name: str
|
|
rate_multiplier: Optional[str] = None
|
|
raw: dict[str, Any] = {}
|
|
|
|
|
|
class BindingSourceGroup(BaseModel):
|
|
upstream_id: int
|
|
group_id: str
|
|
upstream_name: str = ""
|
|
group_name: str = ""
|
|
|
|
|
|
class BindingCreate(BaseModel):
|
|
website_id: int
|
|
target_group_id: str
|
|
target_group_name: str = ""
|
|
source_groups: list[BindingSourceGroup] = Field(default_factory=list)
|
|
percent: float = Field(default=0, ge=0)
|
|
algorithm: str = "max_plus_percent"
|
|
enabled: bool = True
|
|
|
|
|
|
class BindingUpdate(BaseModel):
|
|
website_id: Optional[int] = None
|
|
target_group_id: Optional[str] = None
|
|
target_group_name: Optional[str] = None
|
|
source_groups: Optional[list[BindingSourceGroup]] = None
|
|
percent: Optional[float] = Field(default=None, ge=0)
|
|
algorithm: Optional[str] = None
|
|
enabled: Optional[bool] = None
|
|
|
|
|
|
class BindingResponse(BaseModel):
|
|
id: int
|
|
website_id: int
|
|
website_name: str
|
|
target_group_id: str
|
|
target_group_name: str
|
|
source_groups: list[BindingSourceGroup]
|
|
percent: float
|
|
algorithm: str
|
|
enabled: bool
|
|
created_at: datetime
|
|
updated_at: datetime
|
|
|
|
|
|
class WebsiteSyncLogResponse(BaseModel):
|
|
id: int
|
|
website_id: int
|
|
binding_id: Optional[int]
|
|
target_group_id: str
|
|
target_group_name: str
|
|
algorithm: str
|
|
percent: float
|
|
source_rates: list[dict[str, Any]]
|
|
old_rate: Optional[str]
|
|
new_rate: Optional[str]
|
|
status: str
|
|
message: str
|
|
created_at: datetime
|
|
|
|
|
|
class ImportGroupsRequest(BaseModel):
|
|
group_ids: list[str] = Field(default_factory=list)
|
|
name_prefix: str = ""
|
|
|
|
|
|
class ImportGroupItem(BaseModel):
|
|
source_group_id: str
|
|
source_group_name: str
|
|
target_group_id: Optional[str] = None
|
|
target_group_name: str = ""
|
|
status: str
|
|
message: str = ""
|
|
raw: dict[str, Any] = {}
|
|
|
|
|
|
class ImportGroupsResponse(BaseModel):
|
|
success: bool
|
|
message: str
|
|
items: list[ImportGroupItem]
|
|
|
|
|
|
class SyncImportStatusRequest(BaseModel):
|
|
upstream_id: int = Field(default=0)
|
|
|
|
|
|
class ImportAccountsRequest(BaseModel):
|
|
upstream_key_ids: list[int] = Field(default_factory=list)
|
|
target_group_map: dict[str, str] = Field(default_factory=dict)
|
|
account_name_prefix: str = "SmartUp"
|
|
default_platform: str = "openai"
|
|
platform_mode: str = "auto" # "auto" | "manual"
|
|
concurrency: int = Field(default=10, ge=1)
|
|
priority: int = Field(default=1, ge=0)
|
|
|
|
|
|
class ImportAccountItem(BaseModel):
|
|
upstream_key_id: int
|
|
source_group_id: str
|
|
source_group_name: str
|
|
target_group_id: Optional[str] = None
|
|
account_id: Optional[str] = None
|
|
account_name: str = ""
|
|
platform: str = ""
|
|
upstream_base_url: str = ""
|
|
status: str
|
|
message: str = ""
|
|
raw: dict[str, Any] = {}
|
|
|
|
|
|
class ImportAccountsResponse(BaseModel):
|
|
success: bool
|
|
message: str
|
|
items: list[ImportAccountItem]
|