"""
CTFd-MCP (server.py)
--------------------
FastMCP server running in API mode that exposes helper tools for
the CTFd platform.

Exposed endpoints:
1. /challenges - List of all challenges
2. /unsolved - List of unsolved challenges 
3. /challenge/{id} - Detailed information for a specific challenge
4. /submit - Submit a flag

Requirements
    pip install httpx fastapi mcp uvicorn

Environment variables
    CTFD_API_TOKEN   – the API token (omit leading "Token ")
    CTFD_BASE_URL    – e.g. https://ctf.example.com/api/v1
"""

from __future__ import annotations

import os
import random
from typing import Any, Dict, List, Optional, Union
import logging
from logging.handlers import RotatingFileHandler
from datetime import datetime
import asyncio
import signal
import sys
import json
from fastapi import FastAPI, HTTPException, Request, Body, Path, Query, Depends, status
from fastapi.middleware.cors import CORSMiddleware
from fastapi.responses import JSONResponse, StreamingResponse, HTMLResponse
from fastapi.exceptions import RequestValidationError
from pydantic import BaseModel, ValidationError
import uvicorn
from contextlib import asynccontextmanager

import httpx
from dotenv import load_dotenv

# Logging configuration
def setup_logging():
    # Create logs directory if it doesn't exist
    if not os.path.exists('logs'):
        os.makedirs('logs')
    
    # Configure logger
    logger = logging.getLogger('CTFd-MCP-Server')
    logger.setLevel(logging.INFO)
    
    # Get current timestamp for filename
    timestamp = datetime.now().strftime('%Y%m%d_%H%M%S')
    log_filename = f'logs/ctfd-mcp-server_{timestamp}.log'
    
    # Configure file handler
    file_handler = RotatingFileHandler(
        log_filename,
        maxBytes=5*1024*1024,
        backupCount=5,
        encoding='utf-8'
    )
    file_handler.setLevel(logging.INFO)
    
    # Configure console handler
    console_handler = logging.StreamHandler(sys.stdout)
    console_handler.setLevel(logging.INFO)
    
    # Configure formatter
    formatter = logging.Formatter(
        '%(asctime)s - %(levelname)s - %(message)s',
        datefmt='%Y-%m-%d %H:%M:%S'
    )
    file_handler.setFormatter(formatter)
    console_handler.setFormatter(formatter)
    
    # Add handlers
    logger.addHandler(file_handler)
    logger.addHandler(console_handler)
    
    # Disable handler buffering
    for handler in logger.handlers:
        handler.flush()
    
    return logger

logger = setup_logging()

load_dotenv()  # load environment variables from .env

# Global variables
http_client: Optional[httpx.AsyncClient] = None
shutdown_event = asyncio.Event()

# ──────────────────────────────
# Configuration
# ──────────────────────────────
CTFD_API_BASE: str = os.environ.get("CTFD_BASE_URL", "https://your-ctfd/api/v1")
if not CTFD_API_BASE.endswith('/api/v1'):
    CTFD_API_BASE += '/api/v1'
if not CTFD_API_BASE.endswith('/'):
    CTFD_API_BASE += '/'

CTFD_API_TOKEN: str | None = os.environ.get("CTFD_API_TOKEN")
if not CTFD_API_TOKEN:
    logger.error("CTFD_API_TOKEN environment variable not set")
    raise RuntimeError(
        "CTFD_API_TOKEN environment variable not set ‒ please export your token."
    )

# Remove any quotes from the token if present
CTFD_API_TOKEN = CTFD_API_TOKEN.strip("'").strip('"')

# Flag format configuration
FLAG_FORMAT: str = os.environ.get("FLAG_FORMAT", "flag{example_flag}")

USER_AGENT = "ctfd-mcp/1.0"

COMMON_HEADERS = {
    "User-Agent": USER_AGENT,
    "Authorization": f"Token {CTFD_API_TOKEN}",
    "Accept": "application/json",
    "Content-Type": "application/json"
}

# ──────────────────────────────
# API Models
# ──────────────────────────────
class FlagSubmission(BaseModel):
    challenge_id: int
    flag: str

class SuccessResponse(BaseModel):
    success: bool
    message: str

class ErrorResponse(BaseModel):
    error: str
    details: Optional[Dict[str, Any]] = None  # Field to store detailed error information

# ──────────────────────────────
# FastAPI Lifespan & App
# ──────────────────────────────
@asynccontextmanager
async def lifespan(app: FastAPI):
    # Startup
    global http_client
    try:
        logger.info("Starting CTFd MCP API Server")
        http_client = httpx.AsyncClient(
            timeout=30.0,
            follow_redirects=True,
            limits=httpx.Limits(max_keepalive_connections=10, max_connections=20)
        )
        logger.info("API Server started successfully")
        yield
    except Exception as e:
        logger.error(f"Startup error: {str(e)}")
        raise
    finally:
        # Shutdown
        logger.info("Shutting down CTFd MCP API Server")
        if http_client:
            await http_client.aclose()
            http_client = None
        logger.info("Cleanup completed")

app = FastAPI(
    lifespan=lifespan,
    title="CTFd-MCP API",
    description="API for CTFd integration and LLM Agent MCP Server",
    version="1.0.0"
)

# CORS Configuration
app.add_middleware(
    CORSMiddleware,
    allow_origins=["*"],
    allow_credentials=True,
    allow_methods=["*"],
    allow_headers=["*"],
)

# ──────────────────────────────
# HTTP Client
# ──────────────────────────────
async def get_http_client() -> httpx.AsyncClient:
    global http_client
    if http_client is None:
        http_client = httpx.AsyncClient(
            timeout=30.0,
            follow_redirects=True,
            limits=httpx.Limits(max_keepalive_connections=10, max_connections=20)
        )
    return http_client

async def cleanup_http_client():
    global http_client
    if http_client is not None:
        await http_client.aclose()
        http_client = None

# ──────────────────────────────
# Low-level HTTP helper
# ──────────────────────────────
async def ctfd_request(
    method: str,
    endpoint: str,
    *,
    json: Optional[dict] = None,
    params: Optional[dict] = None,
) -> Optional[dict[str, Any]]:
    """
    Perform an HTTP request against the CTFd API and return parsed JSON.

    Returns None if any exception is raised or the response code is not 2xx.
    """
    url = f"{CTFD_API_BASE}{endpoint.lstrip('/')}"
    
    # Log request details
    logger.info("======= CTFd API Request =======")
    logger.info(f"URL: {url}")
    logger.info(f"Method: {method}")
    
    # Log headers (mask sensitive information)
    masked_headers = COMMON_HEADERS.copy()
    if 'Authorization' in masked_headers:
        token = masked_headers['Authorization'].split(" ")[1]
        masked_token = token[:4] + "*" * (len(token) - 8) + token[-4:]
        masked_headers['Authorization'] = f"Token {masked_token}"
    
    logger.info(f"Headers: {masked_headers}")
    logger.info(f"Params: {params}")
    logger.info(f"JSON payload: {json}")
    
    try:
        client = await get_http_client()
        start_time = datetime.now()
        
        resp = await client.request(
            method,
            url,
            headers=COMMON_HEADERS,
            json=json,
            params=params
        )
        
        # Calculate response time
        elapsed_time = (datetime.now() - start_time).total_seconds()
        
        # Log response details
        logger.info("======= CTFd API Response =======")
        logger.info(f"Status Code: {resp.status_code}")
        logger.info(f"Response Time: {elapsed_time:.3f} seconds")
        
        # Log response headers
        logger.info("Response Headers:")
        for key, value in resp.headers.items():
            logger.info(f"  {key}: {value}")
        
        # Log response body (max 1000 chars)
        if len(resp.text) > 1000:
            logger.info(f"Response Body (truncated): {resp.text[:1000]}...")
        else:
            logger.info(f"Response Body: {resp.text}")
        
        resp.raise_for_status()
        return resp.json()
    except Exception as e:
        logger.error("======= CTFd API Error =======")
        logger.error(f"Error Type: {type(e).__name__}")
        logger.error(f"Error Message: {str(e)}")
        if hasattr(e, 'response') and e.response is not None:
            logger.error(f"Response Status Code: {e.response.status_code}")
            if hasattr(e.response, 'text'):
                logger.error(f"Response Body: {e.response.text}")
        return None

# ──────────────────────────────
# API Routes/Functions
# ──────────────────────────────
@app.get("/", response_class=HTMLResponse)
async def root():
    """MCP Server documentation page for LLM agents"""
    html_content = f"""
    <!DOCTYPE html>
    <html>
    <head>
        <title>CTFd-MCP API Documentation</title>
        <meta charset="UTF-8">
        <meta name="viewport" content="width=device-width, initial-scale=1.0">
        <style>
            body {{
                font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Oxygen, Ubuntu, Cantarell, 'Open Sans', 'Helvetica Neue', sans-serif;
                line-height: 1.6;
                color: #333;
                max-width: 1200px;
                margin: 0 auto;
                padding: 20px;
            }}
            h1, h2, h3 {{
                color: #1a73e8;
            }}
            .endpoint {{
                background-color: #f5f7f9;
                border-left: 4px solid #1a73e8;
                padding: 15px;
                margin: 20px 0;
                border-radius: 4px;
            }}
            .method {{
                display: inline-block;
                padding: 5px 10px;
                color: white;
                font-weight: bold;
                border-radius: 4px;
                margin-right: 10px;
            }}
            .get {{
                background-color: #61affe;
            }}
            .post {{
                background-color: #49cc90;
            }}
            pre {{
                background-color: #272822;
                color: #f8f8f2;
                padding: 15px;
                border-radius: 4px;
                overflow-x: auto;
            }}
            code {{
                font-family: 'Courier New', Courier, monospace;
            }}
            table {{
                width: 100%;
                border-collapse: collapse;
            }}
            th, td {{
                padding: 8px;
                text-align: left;
                border-bottom: 1px solid #ddd;
            }}
            th {{
                background-color: #f5f7f9;
            }}
        </style>
    </head>
    <body>
        <h1>CTFd-MCP Server for LLM Agents</h1>
        <p>This API server provides endpoints for LLM agents to interact with CTFd platform.</p>
        
        <h2>Base URL</h2>
        <p>All endpoints are relative to: <code>http://your-server-address:8000</code></p>
        
        <h2>Available Endpoints</h2>
        
        <div class="endpoint">
            <span class="method get">GET</span>
            <code>/challenges</code>
            <p>Returns a list of all challenges with essential information.</p>
            <h3>Response Format</h3>
            <pre><code>[
  {{
    "id": 123,
    "name": "Challenge Name",
    "value": 100,
    "solved_by_me": true,
    "category": "Web"
  }},
  ...
]</code></pre>
        </div>
        
        <div class="endpoint">
            <span class="method get">GET</span>
            <code>/unsolved</code>
            <p>Returns a list of unsolved challenges with essential information.</p>
            <h3>Response Format</h3>
            <pre><code>[
  {{
    "id": 456,
    "name": "Unsolved Challenge",
    "value": 200,
    "solved_by_me": false,
    "category": "Crypto"
  }},
  ...
]</code></pre>
        </div>
        
        <div class="endpoint">
            <span class="method get">GET</span>
            <code>/challenge/{{challenge_id}}</code>
            <p>Returns detailed information about a specific challenge.</p>
            <h3>Path Parameters</h3>
            <table>
                <tr>
                    <th>Parameter</th>
                    <th>Type</th>
                    <th>Description</th>
                </tr>
                <tr>
                    <td>challenge_id</td>
                    <td>integer</td>
                    <td>The ID of the challenge</td>
                </tr>
            </table>
            <h3>Response Format</h3>
            <pre><code>{{
  "id": 123,
  "name": "Challenge Name",
  "category": "Web",
  "description": "Challenge description goes here...",
  "files": [
    {{
      "id": 1,
      "type": "challenge",
      "location": "filename.zip"
    }}
  ],
  "connection_info": "nc challenge.com 1337",
  "hints": []
}}</code></pre>
        </div>
        
        <div class="endpoint">
            <span class="method post">POST</span>
            <code>/submit</code>
            <p>Submit a flag for a challenge.</p>
            <h3>Request Body</h3>
            <pre><code>{{
  "challenge_id": 123,
  "flag": "{FLAG_FORMAT}"
}}</code></pre>
            <h3>Response Format</h3>
            <pre><code>{{
  "success": true,
  "message": "[CORRECT] Flag is correct!"
}}</code></pre>
        </div>
        
        <div class="endpoint">
            <span class="method get">GET</span>
            <code>/files/{{file_path}}</code>
            <p>Download a file associated with a challenge.</p>
            <h3>Path Parameters</h3>
            <table>
                <tr>
                    <th>Parameter</th>
                    <th>Type</th>
                    <th>Description</th>
                </tr>
                <tr>
                    <td>file_path</td>
                    <td>string</td>
                    <td>The path to the file</td>
                </tr>
            </table>
            <p>Returns the file content as a downloadable response.</p>
        </div>
        
        <h2>Error Handling</h2>
        <p>When an error occurs, the API returns a JSON response with the following structure:</p>
        <pre><code>{{
  "error": "Error message",
  "details": {{
    // Additional error information
  }}
}}</code></pre>
        
        <h2>Example Usage (Python)</h2>
        <pre><code>import requests

# Server base URL
base_url = "http://localhost:8000"

# Get all challenges
challenges = requests.get(f"{{base_url}}/challenges").json()

# Get challenge details
challenge_id = 123
challenge = requests.get(f"{{base_url}}/challenge/{{challenge_id}}").json()

# Submit a flag
submission = {{
    "challenge_id": 123,
    "flag": "{FLAG_FORMAT}"
}}
result = requests.post(f"{{base_url}}/submit", json=submission).json()</code></pre>
        
        <hr>
        <p><small>CTFd-MCP API Server v1.0.0</small></p>
    </body>
    </html>
    """
    return html_content

@app.get("/challenges", response_model=List[Dict[str, Any]])
async def list_all_challenges():
    """Return list of all challenges"""
    logger.info("======= Tool call: list_all_challenges =======")
    data = await ctfd_request("GET", "challenges")
    if data:
        challenges = data.get("data", [])
        logger.info(f"Successfully retrieved {len(challenges)} challenges")
        
        # Extract only needed fields
        simplified_challenges = []
        for challenge in challenges:
            simplified_challenges.append({
                "id": challenge.get("id"),
                "name": challenge.get("name"),
                "value": challenge.get("value"),
                "solved_by_me": challenge.get("solved_by_me"),
                "category": challenge.get("category"),
                "solves" : challenge.get("solves")
            })
        
        # Randomly shuffle the challenges
        random.shuffle(simplified_challenges)
        
        return simplified_challenges
    else:
        logger.error("Failed to retrieve challenges")
        raise HTTPException(status_code=500, detail="Failed to retrieve challenges")

@app.get("/unsolved", response_model=List[Dict[str, Any]])
async def list_unsolved_challenges():
    """Return list of unsolved challenges"""
    logger.info("======= Tool call: list_unsolved_challenges =======")
    
    # Get solved challenges
    logger.info("Fetching solved challenges")
    solves = await ctfd_request("GET", "users/me/solves")
    if not solves:
        logger.error("Failed to retrieve solved challenges")
        raise HTTPException(status_code=500, detail="Failed to retrieve solved challenges")
    
    solved_ids = {s["challenge_id"] for s in solves.get("data", [])}
    logger.info(f"Found {len(solved_ids)} solved challenges")

    # Get all challenges
    logger.info("Fetching all challenges")
    all_challenges = await ctfd_request("GET", "challenges")
    if not all_challenges:
        logger.error("Failed to retrieve challenges")
        raise HTTPException(status_code=500, detail="Failed to retrieve challenges")

    # Filter unsolved challenges
    unsolved = [c for c in all_challenges["data"] if c["id"] not in solved_ids]
    logger.info(f"Found {len(unsolved)} unsolved challenges")
    
    # Extract only needed fields
    simplified_challenges = []
    for challenge in unsolved:
        simplified_challenges.append({
            "id": challenge.get("id"),
            "name": challenge.get("name"),
            "value": challenge.get("value"),
            "solved_by_me": challenge.get("solved_by_me", False),
            "category": challenge.get("category"),
            "solves": challenge.get("solves")
        })
    
    # Randomly shuffle the challenges
    random.shuffle(simplified_challenges)
    logger.info(f"Simplified unsolved challenges data and randomized the order")
    
    return simplified_challenges

@app.get("/challenge/{challenge_id}", response_model=Dict[str, Any])
async def get_challenge_details(challenge_id: int = Path(..., description="Challenge ID")):
    """Return detailed information for a specific challenge"""
    logger.info(f"======= Tool call: get_challenge_details =======")
    logger.info(f"Challenge ID: {challenge_id}")
    
    if not isinstance(challenge_id, int) or challenge_id <= 0:
        raise HTTPException(
            status_code=400, 
            detail="Invalid challenge_id. It must be a positive integer."
        )
    
    detail = await ctfd_request("GET", f"challenges/{challenge_id}")
    if not detail:
        logger.error(f"Challenge not found: {challenge_id}")
        raise HTTPException(status_code=404, detail=f"Challenge with ID {challenge_id} not found.")

    d = detail["data"]
    result = {
        "id": d["id"],
        "name": d["name"],
        "category": d["category"],
        "description": d.get("description", ""),
        "files": d["files"],
        "connection_info": d["connection_info"],
        "hints": d["hints"],
        "solves": d["solves"],
        "value": d["value"],
    }
    
    logger.info(f"Successfully retrieved details for challenge: {d['name']}")
    return result

@app.post("/submit", response_model=SuccessResponse)
async def submit_flag(submission: FlagSubmission):
    """Submit a flag"""
    logger.info(f"======= Tool call: submit_flag =======")
    try:
        challenge_id = submission.challenge_id
        flag = submission.flag
        
        if not flag or not flag.strip():
            raise HTTPException(
                status_code=400, 
                detail="Flag cannot be an empty string."
            )
        
        logger.info(f"Challenge ID: {challenge_id}")
        logger.info(f"Flag length: {len(flag)}")
        
        payload = {
            "challenge_id": challenge_id,
            "submission": flag.strip()  # Remove any whitespace
        }
        
        # Direct request to specific URL
        submit_url = "http://127.0.0.1:5000/api/v1/challenges/attempt"
        
        logger.info("======= Submit Flag Request =======")
        logger.info(f"URL: {submit_url}")
        logger.info(f"Method: POST")
        
        # Log headers (mask sensitive information)
        masked_headers = COMMON_HEADERS.copy()
        if 'Authorization' in masked_headers:
            token = masked_headers['Authorization'].split(" ")[1]
            masked_token = token[:4] + "*" * (len(token) - 8) + token[-4:]
            masked_headers['Authorization'] = f"Token {masked_token}"
        
        logger.info(f"Headers: {masked_headers}")
        logger.info(f"JSON payload: {payload}")
        
        try:
            client = await get_http_client()
            start_time = datetime.now()
            
            resp = await client.request(
                "POST",
                submit_url,
                headers=COMMON_HEADERS,
                json=payload
            )
            
            # Calculate response time
            elapsed_time = (datetime.now() - start_time).total_seconds()
            
            # Log response details
            logger.info("======= Submit Flag Response =======")
            logger.info(f"Status Code: {resp.status_code}")
            logger.info(f"Response Time: {elapsed_time:.3f} seconds")
            
            # Log response headers
            logger.info("Response Headers:")
            for key, value in resp.headers.items():
                logger.info(f"  {key}: {value}")
            
            # Log response body (max 1000 chars)
            if len(resp.text) > 1000:
                logger.info(f"Response Body (truncated): {resp.text[:1000]}...")
            else:
                logger.info(f"Response Body: {resp.text}")
            
            resp.raise_for_status()
            result = resp.json()
            
        except Exception as e:
            logger.error("======= Submit Flag Error =======")
            logger.error(f"Error Type: {type(e).__name__}")
            logger.error(f"Error Message: {str(e)}")
            if hasattr(e, 'response') and e.response is not None:
                logger.error(f"Response Status Code: {e.response.status_code}")
                if hasattr(e.response, 'text'):
                    logger.error(f"Response Body: {e.response.text}")
            result = None
        
        if not result:
            error_msg = "Submission failed: server error or malformed request."
            logger.error(error_msg)
            raise HTTPException(
                status_code=500, 
                detail=error_msg,
                headers={"X-Error-Source": "CTFd API"}
            )
        
        if "data" not in result:
            logger.error(f"Unexpected response format: {result}")
            raise HTTPException(
                status_code=500, 
                detail="CTFd API returned an unexpected response format.",
                headers={"X-Error-Source": "Response Format"}
            )

        data = result["data"]
        status = data.get("status", "unknown")
        message = data.get("message", "")
        
        response_message = f"[{status.upper()}] {message}"
        logger.info(f"Flag submission result - {response_message}")
        
        return SuccessResponse(
            success=(status == "correct"), 
            message=response_message
        )
        
    except ValueError as e:
        error_msg = f"Invalid challenge ID: {submission.challenge_id}"
        logger.error(error_msg)
        raise HTTPException(
            status_code=400, 
            detail=error_msg,
            headers={"X-Error-Type": "ValueError"}
        )
    except ValidationError as e:
        error_msg = f"Validation error: {str(e)}"
        logger.error(error_msg)
        raise HTTPException(
            status_code=422,
            detail=error_msg,
            headers={"X-Error-Type": "ValidationError"}
        )
    except Exception as e:
        error_msg = f"Unexpected error during flag submission: {str(e)}"
        logger.error(error_msg)
        raise HTTPException(
            status_code=500, 
            detail=error_msg,
            headers={"X-Error-Type": type(e).__name__}
        )

@app.get("/files/{file_path:path}")
async def download_file(file_path: str, request: Request):
    """Download a file"""
    if shutdown_event.is_set():
        raise HTTPException(status_code=503, detail="Server is shutting down")
        
    base_url = CTFD_API_BASE.replace('/api/v1/', '')
    original_file_url = f"{base_url}/files/{file_path}"
    
    query_params = request.query_params
    if query_params:
        original_file_url += '?' + str(request.query_params)
        
    logger.info(f"Requesting file: {file_path}")

    try:
        cookies = dict(request.cookies)
        headers = {
            "User-Agent": USER_AGENT
        }
        
        client = await get_http_client()
        response = await client.get(original_file_url, headers=headers, cookies=cookies)
        response.raise_for_status()
        
        file_content = response.content
        logger.info(f"File downloaded successfully: {file_path}")
        
        response_headers = dict(response.headers)
        response_headers['Content-Disposition'] = f'attachment; filename="{os.path.basename(file_path)}"'
        
        return StreamingResponse(
            iter([file_content]), 
            headers=response_headers,
            media_type=response.headers.get("content-type", "application/octet-stream")
        )
    except httpx.HTTPStatusError as e:
        logger.error(f"File download error: {str(e)}")
        raise HTTPException(status_code=e.response.status_code, detail=str(e))
    except Exception as e:
        logger.error(f"File download error: {str(e)}")
        raise HTTPException(status_code=500, detail="Internal Server Error")

# Exception handlers
@app.exception_handler(RequestValidationError)
async def validation_exception_handler(request: Request, exc: RequestValidationError):
    """Provide detailed error information when request validation fails"""
    errors = []
    for error in exc.errors():
        error_info = {
            "loc": error.get("loc", []),
            "msg": error.get("msg", ""),
            "type": error.get("type", "")
        }
        errors.append(error_info)
    
    error_message = "Request format is invalid"
    logger.error(f"Validation Error: {error_message}")
    logger.error(f"Details: {json.dumps(errors, ensure_ascii=False)}")
    
    return JSONResponse(
        status_code=status.HTTP_422_UNPROCESSABLE_ENTITY,
        content={"error": error_message, "details": errors}
    )

@app.exception_handler(HTTPException)
async def http_exception_handler(request: Request, exc: HTTPException):
    """Provide detailed information when HTTP exceptions occur"""
    error_info = {
        "status_code": exc.status_code,
        "detail": exc.detail
    }
    if exc.headers:
        error_info["headers"] = exc.headers
        
    logger.error(f"HTTP Error: {exc.detail}")
    return JSONResponse(
        status_code=exc.status_code,
        content={"error": exc.detail, "details": error_info}
    )

@app.exception_handler(Exception)
async def general_exception_handler(request: Request, exc: Exception):
    """General exception handling"""
    error_message = f"Server error: {str(exc)}"
    error_type = type(exc).__name__
    
    logger.error(f"Unexpected error: {error_message}")
    logger.error(f"Error type: {error_type}")
    logger.exception("Detailed error:")
    
    return JSONResponse(
        status_code=status.HTTP_500_INTERNAL_SERVER_ERROR,
        content={
            "error": error_message,
            "details": {
                "type": error_type,
                "request_path": request.url.path,
                "request_method": request.method
            }
        }
    )

# ──────────────────────────────
# Main entry point
# ──────────────────────────────
if __name__ == "__main__":
    try:
        # Signal handler setup
        for sig in (signal.SIGTERM, signal.SIGINT):
            signal.signal(sig, lambda s, f: shutdown_event.set())
        
        # Server configuration
        config = uvicorn.Config(
            app,
            host="0.0.0.0",
            port=8000,
            log_level="info",
            timeout_keep_alive=60,
            limit_concurrency=100,
            limit_max_requests=10000,
            access_log=False
        )
        
        # Logging configuration
        logging.getLogger("uvicorn.error").handlers = logger.handlers
        logging.getLogger("uvicorn.access").handlers = logger.handlers
        logging.getLogger("uvicorn.error").propagate = False
        logging.getLogger("uvicorn.access").propagate = False
        
        # Run server
        server = uvicorn.Server(config)
        server.run()
    except KeyboardInterrupt:
        logger.info("Received keyboard interrupt")
    except Exception as e:
        logger.error(f"Error: {str(e)}")
    finally:
        # Cleanup
        asyncio.run(cleanup_http_client())
