""" Proxy Configuration API Router Endpoints for managing browser proxy settings and testing connectivity. """ import socket import time from pathlib import Path from typing import Optional from fastapi import APIRouter from fastapi.responses import JSONResponse from pydantic import BaseModel, Field, field_validator router = APIRouter(prefix="/api/proxy", tags=["proxy"]) # Config file path _CONFIG_DIR = Path(__file__).parent.parent.parent _PROXY_CONFIG_FILE = _CONFIG_DIR / "proxy_config.json" class ProxyConfig(BaseModel): """Proxy configuration model.""" enabled: bool = False address: str = "http://127.0.0.1:7890" @field_validator("address") @classmethod def validate_address(cls, v: str) -> str: """Validate proxy address format.""" v = v.strip() if v and not (v.startswith("http://") or v.startswith("https://")): raise ValueError("Proxy address must start with http:// or https://") return v class ProxyTestRequest(BaseModel): """Request model for proxy test.""" address: str = Field(..., description="Proxy address") test_url: str = Field( default="http://httpbin.org/get", description="Test target URL" ) class ProxyTestResult(BaseModel): """Result of proxy connectivity test.""" success: bool message: str latency_ms: Optional[float] = None def _load_config() -> ProxyConfig: """Load proxy config from file.""" import json if _PROXY_CONFIG_FILE.exists(): try: data = json.loads(_PROXY_CONFIG_FILE.read_text(encoding="utf-8")) return ProxyConfig(**data) except Exception: pass return ProxyConfig() def _save_config(config: ProxyConfig) -> None: """Save proxy config to file.""" import json _PROXY_CONFIG_FILE.write_text( json.dumps(config.model_dump(), ensure_ascii=False, indent=2), encoding="utf-8", ) @router.get("/config") async def get_proxy_config() -> JSONResponse: """Get current proxy configuration.""" config = _load_config() return JSONResponse(content=config.model_dump()) @router.post("/config") async def update_proxy_config(config: ProxyConfig) -> JSONResponse: """Update proxy configuration.""" _save_config(config) return JSONResponse(content={"success": True, "config": config.model_dump()}) @router.post("/test") async def test_proxy_connectivity(request: ProxyTestRequest) -> JSONResponse: """Test proxy connectivity.""" import httpx proxy_addr = request.address.strip() test_url = request.test_url if not proxy_addr: return JSONResponse( content=ProxyTestResult( success=False, message="Proxy address cannot be empty" ).model_dump(), status_code=400, ) try: start_time = time.monotonic() async with httpx.AsyncClient( proxy=proxy_addr, timeout=15.0, follow_redirects=True, ) as client: response = await client.get(test_url) latency = (time.monotonic() - start_time) * 1000 if 200 <= response.status_code < 300: return JSONResponse( content=ProxyTestResult( success=True, message=f"Connection successful (HTTP {response.status_code})", latency_ms=round(latency, 2), ).model_dump() ) else: return JSONResponse( content=ProxyTestResult( success=False, message=f"HTTP Error: {response.status_code}", latency_ms=round(latency, 2), ).model_dump() ) except httpx.ProxyError as e: return JSONResponse( content=ProxyTestResult( success=False, message=f"Proxy error: {e}" ).model_dump() ) except httpx.ConnectTimeout: return JSONResponse( content=ProxyTestResult( success=False, message="Connection timeout" ).model_dump() ) except httpx.ReadTimeout: return JSONResponse( content=ProxyTestResult(success=False, message="Read timeout").model_dump() ) except socket.gaierror as e: return JSONResponse( content=ProxyTestResult( success=False, message=f"DNS resolution failed: {e}" ).model_dump() ) except Exception as e: return JSONResponse( content=ProxyTestResult( success=False, message=f"Unknown error: {e}" ).model_dump() )