mirror of
https://github.com/farion1231/cc-switch.git
synced 2026-05-25 15:31:10 +08:00
0dd823ae3a
* fix(proxy): fix Codex 404 errors with custom base_url prefixes - handlers.rs:268: Remove hardcoded /v1 prefix in Codex forwarding - codex.rs:140: Only add /v1 for origin-only base_urls, dedupe /v1/v1 - stream_check.rs:364: Try /responses first, fallback to /v1/responses - provider.rs:427: Don't force /v1 for custom prefix base_urls Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * fix(codex): always add /v1 for custom prefix base_urls Changed logic to always add /v1 prefix unless base_url already ends with /v1. This fixes 504 timeout errors with relay services that expect /v1 in the path. - Most relay services follow OpenAI standard format: /v1/responses - Users can opt-out by adding /v1 to their base_url configuration - Updated test case to reflect new behavior Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * fix(proxy): allow system proxy on localhost with different ports - Only bypass system proxy if it points to CC Switch's own port (15721) - Allow other localhost proxies (e.g., Clash on 7890) to be used - Add INFO level logging for request URLs to aid debugging This fixes connection timeout issues when using local proxy tools. Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * fix(codex): don't add /v1 for custom prefix base_urls Reverted logic to not add /v1 for base_urls with custom prefixes. Many relay services use custom paths without /v1. - Pure origin (e.g., https://api.openai.com) → adds /v1 - With /v1 (e.g., https://api.openai.com/v1) → no change - Custom prefix (e.g., https://example.com/openai) → no /v1 This fixes 404 errors with relay services that don't use /v1 in their paths. Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * fix(proxy): use dynamic port for system proxy detection Instead of hardcoding port 15721, now uses the actual configured listen_port from proxy settings. - Added set_proxy_port() to update the port when proxy server starts - Added get_proxy_port() to retrieve current port for detection - Updated server.rs to call set_proxy_port() on startup - Updated tests to reflect new behavior This allows users to change the proxy port in settings without breaking the system proxy detection logic. Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * fix(proxy): change default proxy port from 15721 to 5000 Update the default fallback port in get_proxy_port() from 15721 to 5000 to match the project's standard default port configuration. Also updated test cases to use port 5000 consistently. Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com> * fix(proxy): revert default port back to 15721 Revert the default fallback port in get_proxy_port() from 5000 back to 15721 to align with the project's updated default port configuration. Also updated test cases to use port 15721 consistently. Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com> --------- Co-authored-by: ozbombor <ozbombor@users.noreply.github.com> Co-authored-by: Claude Opus 4.5 <noreply@anthropic.com>
272 lines
9.9 KiB
Rust
272 lines
9.9 KiB
Rust
//! HTTP代理服务器
|
||
//!
|
||
//! 基于Axum的HTTP服务器,处理代理请求
|
||
|
||
use super::{
|
||
failover_switch::FailoverSwitchManager, handlers, log_codes::srv as log_srv,
|
||
provider_router::ProviderRouter, types::*, ProxyError,
|
||
};
|
||
use crate::database::Database;
|
||
use axum::{
|
||
extract::DefaultBodyLimit,
|
||
routing::{get, post},
|
||
Router,
|
||
};
|
||
use std::net::SocketAddr;
|
||
use std::sync::Arc;
|
||
use tokio::sync::{oneshot, RwLock};
|
||
use tokio::task::JoinHandle;
|
||
use tower_http::cors::{Any, CorsLayer};
|
||
|
||
/// 代理服务器状态(共享)
|
||
#[derive(Clone)]
|
||
pub struct ProxyState {
|
||
pub db: Arc<Database>,
|
||
pub config: Arc<RwLock<ProxyConfig>>,
|
||
pub status: Arc<RwLock<ProxyStatus>>,
|
||
pub start_time: Arc<RwLock<Option<std::time::Instant>>>,
|
||
/// 每个应用类型当前使用的 provider (app_type -> (provider_id, provider_name))
|
||
pub current_providers: Arc<RwLock<std::collections::HashMap<String, (String, String)>>>,
|
||
/// 共享的 ProviderRouter(持有熔断器状态,跨请求保持)
|
||
pub provider_router: Arc<ProviderRouter>,
|
||
/// AppHandle,用于发射事件和更新托盘菜单
|
||
pub app_handle: Option<tauri::AppHandle>,
|
||
/// 故障转移切换管理器
|
||
pub failover_manager: Arc<FailoverSwitchManager>,
|
||
}
|
||
|
||
/// 代理HTTP服务器
|
||
pub struct ProxyServer {
|
||
config: ProxyConfig,
|
||
state: ProxyState,
|
||
shutdown_tx: Arc<RwLock<Option<oneshot::Sender<()>>>>,
|
||
/// 服务器任务句柄,用于等待服务器实际关闭
|
||
server_handle: Arc<RwLock<Option<JoinHandle<()>>>>,
|
||
}
|
||
|
||
impl ProxyServer {
|
||
pub fn new(
|
||
config: ProxyConfig,
|
||
db: Arc<Database>,
|
||
app_handle: Option<tauri::AppHandle>,
|
||
) -> Self {
|
||
// 创建共享的 ProviderRouter(熔断器状态将跨所有请求保持)
|
||
let provider_router = Arc::new(ProviderRouter::new(db.clone()));
|
||
// 创建故障转移切换管理器
|
||
let failover_manager = Arc::new(FailoverSwitchManager::new(db.clone()));
|
||
|
||
let state = ProxyState {
|
||
db,
|
||
config: Arc::new(RwLock::new(config.clone())),
|
||
status: Arc::new(RwLock::new(ProxyStatus::default())),
|
||
start_time: Arc::new(RwLock::new(None)),
|
||
current_providers: Arc::new(RwLock::new(std::collections::HashMap::new())),
|
||
provider_router,
|
||
app_handle,
|
||
failover_manager,
|
||
};
|
||
|
||
Self {
|
||
config,
|
||
state,
|
||
shutdown_tx: Arc::new(RwLock::new(None)),
|
||
server_handle: Arc::new(RwLock::new(None)),
|
||
}
|
||
}
|
||
|
||
pub async fn start(&self) -> Result<ProxyServerInfo, ProxyError> {
|
||
// 检查是否已在运行
|
||
if self.shutdown_tx.read().await.is_some() {
|
||
return Err(ProxyError::AlreadyRunning);
|
||
}
|
||
|
||
let addr: SocketAddr =
|
||
format!("{}:{}", self.config.listen_address, self.config.listen_port)
|
||
.parse()
|
||
.map_err(|e| ProxyError::BindFailed(format!("无效的地址: {e}")))?;
|
||
|
||
// 创建关闭通道
|
||
let (shutdown_tx, shutdown_rx) = oneshot::channel();
|
||
|
||
// 构建路由
|
||
let app = self.build_router();
|
||
|
||
// 绑定监听器
|
||
let listener = tokio::net::TcpListener::bind(&addr)
|
||
.await
|
||
.map_err(|e| ProxyError::BindFailed(e.to_string()))?;
|
||
|
||
log::info!("[{}] 代理服务器启动于 {addr}", log_srv::STARTED);
|
||
|
||
// 更新全局代理端口,用于系统代理检测
|
||
crate::proxy::http_client::set_proxy_port(self.config.listen_port);
|
||
|
||
// 保存关闭句柄
|
||
*self.shutdown_tx.write().await = Some(shutdown_tx);
|
||
|
||
// 更新状态
|
||
let mut status = self.state.status.write().await;
|
||
status.running = true;
|
||
status.address = self.config.listen_address.clone();
|
||
status.port = self.config.listen_port;
|
||
drop(status);
|
||
|
||
// 记录启动时间
|
||
*self.state.start_time.write().await = Some(std::time::Instant::now());
|
||
|
||
// 启动服务器
|
||
let state = self.state.clone();
|
||
let handle = tokio::spawn(async move {
|
||
axum::serve(listener, app)
|
||
.with_graceful_shutdown(async {
|
||
shutdown_rx.await.ok();
|
||
})
|
||
.await
|
||
.ok();
|
||
|
||
// 服务器停止后更新状态
|
||
state.status.write().await.running = false;
|
||
*state.start_time.write().await = None;
|
||
});
|
||
|
||
// 保存服务器任务句柄
|
||
*self.server_handle.write().await = Some(handle);
|
||
|
||
Ok(ProxyServerInfo {
|
||
address: self.config.listen_address.clone(),
|
||
port: self.config.listen_port,
|
||
started_at: chrono::Utc::now().to_rfc3339(),
|
||
})
|
||
}
|
||
|
||
pub async fn stop(&self) -> Result<(), ProxyError> {
|
||
// 1. 发送关闭信号
|
||
if let Some(tx) = self.shutdown_tx.write().await.take() {
|
||
let _ = tx.send(());
|
||
} else {
|
||
return Err(ProxyError::NotRunning);
|
||
}
|
||
|
||
// 2. 等待服务器任务结束(带 5 秒超时保护)
|
||
if let Some(handle) = self.server_handle.write().await.take() {
|
||
match tokio::time::timeout(std::time::Duration::from_secs(5), handle).await {
|
||
Ok(Ok(())) => {
|
||
log::info!("[{}] 代理服务器已完全停止", log_srv::STOPPED);
|
||
Ok(())
|
||
}
|
||
Ok(Err(e)) => {
|
||
log::warn!("[{}] 代理服务器任务异常终止: {e}", log_srv::TASK_ERROR);
|
||
Err(ProxyError::StopFailed(e.to_string()))
|
||
}
|
||
Err(_) => {
|
||
log::warn!(
|
||
"[{}] 代理服务器停止超时(5秒),强制继续",
|
||
log_srv::STOP_TIMEOUT
|
||
);
|
||
Err(ProxyError::StopTimeout)
|
||
}
|
||
}
|
||
} else {
|
||
Ok(())
|
||
}
|
||
}
|
||
|
||
pub async fn get_status(&self) -> ProxyStatus {
|
||
let mut status = self.state.status.read().await.clone();
|
||
|
||
// 计算运行时间
|
||
if let Some(start) = *self.state.start_time.read().await {
|
||
status.uptime_seconds = start.elapsed().as_secs();
|
||
}
|
||
|
||
// 从 current_providers HashMap 获取每个应用类型当前正在使用的 provider
|
||
let current_providers = self.state.current_providers.read().await;
|
||
status.active_targets = current_providers
|
||
.iter()
|
||
.map(|(app_type, (provider_id, provider_name))| ActiveTarget {
|
||
app_type: app_type.clone(),
|
||
provider_id: provider_id.clone(),
|
||
provider_name: provider_name.clone(),
|
||
})
|
||
.collect();
|
||
|
||
status
|
||
}
|
||
|
||
/// 更新某个应用类型当前“目标供应商”(用于 UI 展示 active_targets)
|
||
///
|
||
/// 注意:这不代表该供应商一定已经处理过请求,而是用于“热切换/启用故障转移立即切 P1”
|
||
/// 等场景下,让 UI 能立刻反映最新目标。
|
||
pub async fn set_active_target(&self, app_type: &str, provider_id: &str, provider_name: &str) {
|
||
let mut current_providers = self.state.current_providers.write().await;
|
||
current_providers.insert(
|
||
app_type.to_string(),
|
||
(provider_id.to_string(), provider_name.to_string()),
|
||
);
|
||
}
|
||
|
||
fn build_router(&self) -> Router {
|
||
let cors = CorsLayer::new()
|
||
.allow_origin(Any)
|
||
.allow_methods(Any)
|
||
.allow_headers(Any);
|
||
|
||
Router::new()
|
||
// 健康检查
|
||
.route("/health", get(handlers::health_check))
|
||
.route("/status", get(handlers::get_status))
|
||
// Claude API (支持带前缀和不带前缀两种格式)
|
||
.route("/v1/messages", post(handlers::handle_messages))
|
||
.route("/claude/v1/messages", post(handlers::handle_messages))
|
||
// OpenAI Chat Completions API (Codex CLI,支持带前缀和不带前缀)
|
||
.route("/chat/completions", post(handlers::handle_chat_completions))
|
||
.route(
|
||
"/v1/chat/completions",
|
||
post(handlers::handle_chat_completions),
|
||
)
|
||
.route(
|
||
"/v1/v1/chat/completions",
|
||
post(handlers::handle_chat_completions),
|
||
)
|
||
.route(
|
||
"/codex/v1/chat/completions",
|
||
post(handlers::handle_chat_completions),
|
||
)
|
||
// OpenAI Responses API (Codex CLI,支持带前缀和不带前缀)
|
||
.route("/responses", post(handlers::handle_responses))
|
||
.route("/v1/responses", post(handlers::handle_responses))
|
||
.route("/v1/v1/responses", post(handlers::handle_responses))
|
||
.route("/codex/v1/responses", post(handlers::handle_responses))
|
||
// Gemini API (支持带前缀和不带前缀)
|
||
.route("/v1beta/*path", post(handlers::handle_gemini))
|
||
.route("/gemini/v1beta/*path", post(handlers::handle_gemini))
|
||
// 提高默认请求体大小限制(避免 413 Payload Too Large)
|
||
.layer(DefaultBodyLimit::max(200 * 1024 * 1024))
|
||
.layer(cors)
|
||
.with_state(self.state.clone())
|
||
}
|
||
|
||
/// 在不重启服务的情况下更新运行时配置
|
||
pub async fn apply_runtime_config(&self, config: &ProxyConfig) {
|
||
*self.state.config.write().await = config.clone();
|
||
}
|
||
|
||
/// 热更新熔断器配置
|
||
///
|
||
/// 将新配置应用到所有已创建的熔断器实例
|
||
pub async fn update_circuit_breaker_configs(
|
||
&self,
|
||
config: super::circuit_breaker::CircuitBreakerConfig,
|
||
) {
|
||
self.state.provider_router.update_all_configs(config).await;
|
||
}
|
||
|
||
/// 重置指定 Provider 的熔断器
|
||
pub async fn reset_provider_circuit_breaker(&self, provider_id: &str, app_type: &str) {
|
||
self.state
|
||
.provider_router
|
||
.reset_provider_breaker(provider_id, app_type)
|
||
.await;
|
||
}
|
||
}
|