mirror of
https://github.com/CJackHwang/ds2api.git
synced 2026-05-05 00:45:29 +08:00
147 lines
5.8 KiB
Python
147 lines
5.8 KiB
Python
# -*- coding: utf-8 -*-
|
||
"""账号认证与管理模块"""
|
||
import random
|
||
from fastapi import HTTPException, Request
|
||
|
||
from .config import CONFIG, logger
|
||
from .deepseek import login_deepseek_via_account, BASE_HEADERS
|
||
|
||
# -------------------------- 全局账号队列 --------------------------
|
||
account_queue = [] # 维护所有可用账号
|
||
claude_api_key_queue = [] # 维护所有可用的Claude API keys
|
||
|
||
|
||
def init_account_queue():
|
||
"""初始化时从配置加载账号"""
|
||
global account_queue
|
||
account_queue = CONFIG.get("accounts", [])[:] # 深拷贝
|
||
random.shuffle(account_queue) # 初始随机排序
|
||
|
||
|
||
def init_claude_api_key_queue():
|
||
"""Claude API keys由用户自己的token提供,这里初始化为空"""
|
||
global claude_api_key_queue
|
||
claude_api_key_queue = []
|
||
|
||
|
||
# 初始化
|
||
init_account_queue()
|
||
init_claude_api_key_queue()
|
||
|
||
|
||
# ----------------------------------------------------------------------
|
||
# 辅助函数:获取账号唯一标识(优先 email,否则 mobile)
|
||
# ----------------------------------------------------------------------
|
||
def get_account_identifier(account: dict) -> str:
|
||
"""返回账号的唯一标识,优先使用 email,否则使用 mobile"""
|
||
return account.get("email", "").strip() or account.get("mobile", "").strip()
|
||
|
||
|
||
# ----------------------------------------------------------------------
|
||
# 账号选择与释放
|
||
# ----------------------------------------------------------------------
|
||
def choose_new_account(exclude_ids=None):
|
||
"""选择策略:
|
||
1. 优先选择已有 token 的账号(避免登录)
|
||
2. 遍历队列,找到第一个未被 exclude_ids 包含的账号
|
||
3. 从队列中移除该账号
|
||
4. 返回该账号(由后续逻辑保证最终会重新入队)
|
||
"""
|
||
if exclude_ids is None:
|
||
exclude_ids = []
|
||
|
||
# 第一轮:优先选择已有 token 的账号
|
||
for i in range(len(account_queue)):
|
||
acc = account_queue[i]
|
||
acc_id = get_account_identifier(acc)
|
||
if acc_id and acc_id not in exclude_ids:
|
||
if acc.get("token", "").strip(): # 已有 token
|
||
logger.info(f"[choose_new_account] 选择已有token的账号: {acc_id}")
|
||
return account_queue.pop(i)
|
||
|
||
# 第二轮:选择任意账号(需要登录)
|
||
for i in range(len(account_queue)):
|
||
acc = account_queue[i]
|
||
acc_id = get_account_identifier(acc)
|
||
if acc_id and acc_id not in exclude_ids:
|
||
logger.info(f"[choose_new_account] 选择需登录的账号: {acc_id}")
|
||
return account_queue.pop(i)
|
||
|
||
logger.warning("[choose_new_account] 没有可用的账号或所有账号都在使用中")
|
||
return None
|
||
|
||
|
||
def release_account(account: dict):
|
||
"""将账号重新加入队列末尾"""
|
||
account_queue.append(account)
|
||
|
||
|
||
# ----------------------------------------------------------------------
|
||
# Claude API key 管理函数(简化版本)
|
||
# ----------------------------------------------------------------------
|
||
def choose_claude_api_key():
|
||
"""选择一个可用的Claude API key - 现在直接由用户提供"""
|
||
return None
|
||
|
||
|
||
def release_claude_api_key(api_key):
|
||
"""释放Claude API key - 现在无需操作"""
|
||
pass
|
||
|
||
|
||
# ----------------------------------------------------------------------
|
||
# 判断调用模式:配置模式 vs 用户自带 token
|
||
# ----------------------------------------------------------------------
|
||
def determine_mode_and_token(request: Request):
|
||
"""
|
||
根据请求头 Authorization 判断使用哪种模式:
|
||
- 如果 Bearer token 出现在 CONFIG["keys"] 中,则为配置模式,从 CONFIG["accounts"] 中随机选择一个账号(排除已尝试账号),
|
||
检查该账号是否已有 token,否则调用登录接口获取;
|
||
- 否则,直接使用请求中的 Bearer 值作为 DeepSeek token。
|
||
结果存入 request.state.deepseek_token;配置模式下同时存入 request.state.account 与 request.state.tried_accounts。
|
||
"""
|
||
auth_header = request.headers.get("Authorization", "")
|
||
if not auth_header.startswith("Bearer "):
|
||
raise HTTPException(
|
||
status_code=401, detail="Unauthorized: missing Bearer token."
|
||
)
|
||
caller_key = auth_header.replace("Bearer ", "", 1).strip()
|
||
config_keys = CONFIG.get("keys", [])
|
||
if caller_key in config_keys:
|
||
request.state.use_config_token = True
|
||
request.state.tried_accounts = [] # 初始化已尝试账号
|
||
selected_account = choose_new_account()
|
||
if not selected_account:
|
||
raise HTTPException(
|
||
status_code=429,
|
||
detail="No accounts configured or all accounts are busy.",
|
||
)
|
||
if not selected_account.get("token", "").strip():
|
||
try:
|
||
login_deepseek_via_account(selected_account)
|
||
except Exception as e:
|
||
logger.error(
|
||
f"[determine_mode_and_token] 账号 {get_account_identifier(selected_account)} 登录失败:{e}"
|
||
)
|
||
raise HTTPException(status_code=500, detail="Account login failed.")
|
||
|
||
request.state.deepseek_token = selected_account.get("token")
|
||
request.state.account = selected_account
|
||
|
||
else:
|
||
request.state.use_config_token = False
|
||
request.state.deepseek_token = caller_key
|
||
|
||
|
||
def get_auth_headers(request: Request) -> dict:
|
||
"""返回 DeepSeek 请求所需的公共请求头"""
|
||
return {**BASE_HEADERS, "authorization": f"Bearer {request.state.deepseek_token}"}
|
||
|
||
|
||
# ----------------------------------------------------------------------
|
||
# Claude 认证相关函数
|
||
# ----------------------------------------------------------------------
|
||
def determine_claude_mode_and_token(request: Request):
|
||
"""Claude认证:沿用现有的OpenAI接口认证逻辑"""
|
||
determine_mode_and_token(request)
|