|
from fastapi import FastAPI, Request |
|
from fastapi.middleware.cors import CORSMiddleware |
|
from fastapi.responses import JSONResponse |
|
from contextlib import asynccontextmanager |
|
from agentpress.thread_manager import ThreadManager |
|
from services.supabase import DBConnection |
|
from datetime import datetime, timezone |
|
from dotenv import load_dotenv |
|
from utils.config import config, EnvMode |
|
import asyncio |
|
from utils.logger import logger |
|
import uuid |
|
import time |
|
from collections import OrderedDict |
|
|
|
|
|
from agent import api as agent_api |
|
from sandbox import api as sandbox_api |
|
from services import billing as billing_api |
|
|
|
|
|
load_dotenv() |
|
|
|
|
|
db = DBConnection() |
|
thread_manager = None |
|
instance_id = "single" |
|
|
|
|
|
ip_tracker = OrderedDict() |
|
MAX_CONCURRENT_IPS = 25 |
|
|
|
@asynccontextmanager |
|
async def lifespan(app: FastAPI): |
|
|
|
global thread_manager |
|
logger.info(f"Starting up FastAPI application with instance ID: {instance_id} in {config.ENV_MODE.value} mode") |
|
|
|
try: |
|
|
|
await db.initialize() |
|
thread_manager = ThreadManager() |
|
|
|
|
|
agent_api.initialize( |
|
thread_manager, |
|
db, |
|
instance_id |
|
) |
|
|
|
|
|
sandbox_api.initialize(db) |
|
|
|
|
|
from services import redis |
|
try: |
|
await redis.initialize_async() |
|
logger.info("Redis connection initialized successfully") |
|
except Exception as e: |
|
logger.error(f"Failed to initialize Redis connection: {e}") |
|
|
|
|
|
|
|
asyncio.create_task(agent_api.restore_running_agent_runs()) |
|
|
|
yield |
|
|
|
|
|
logger.info("Cleaning up agent resources") |
|
await agent_api.cleanup() |
|
|
|
|
|
try: |
|
logger.info("Closing Redis connection") |
|
await redis.close() |
|
logger.info("Redis connection closed successfully") |
|
except Exception as e: |
|
logger.error(f"Error closing Redis connection: {e}") |
|
|
|
|
|
logger.info("Disconnecting from database") |
|
await db.disconnect() |
|
except Exception as e: |
|
logger.error(f"Error during application startup: {e}") |
|
raise |
|
|
|
app = FastAPI(lifespan=lifespan) |
|
|
|
@app.middleware("http") |
|
async def log_requests_middleware(request: Request, call_next): |
|
start_time = time.time() |
|
client_ip = request.client.host |
|
method = request.method |
|
url = str(request.url) |
|
path = request.url.path |
|
query_params = str(request.query_params) |
|
|
|
|
|
logger.info(f"Request started: {method} {path} from {client_ip} | Query: {query_params}") |
|
|
|
try: |
|
response = await call_next(request) |
|
process_time = time.time() - start_time |
|
logger.debug(f"Request completed: {method} {path} | Status: {response.status_code} | Time: {process_time:.2f}s") |
|
return response |
|
except Exception as e: |
|
process_time = time.time() - start_time |
|
logger.error(f"Request failed: {method} {path} | Error: {str(e)} | Time: {process_time:.2f}s") |
|
raise |
|
|
|
|
|
allowed_origins = ["https://www.suna.so","https://suna-gold.vercel.app", "https://suna.so", "https://staging.suna.so", "http://localhost:3000"] |
|
|
|
|
|
if config.ENV_MODE == EnvMode.STAGING: |
|
allowed_origins.append("http://localhost:3006") |
|
|
|
|
|
if config.ENV_MODE == EnvMode.LOCAL: |
|
allowed_origins.append("http://localhost:3006") |
|
|
|
app.add_middleware( |
|
CORSMiddleware, |
|
allow_origins=allowed_origins, |
|
allow_credentials=True, |
|
allow_methods=["GET", "POST", "PUT", "DELETE", "OPTIONS"], |
|
allow_headers=["Content-Type", "Authorization"], |
|
) |
|
|
|
|
|
app.include_router(agent_api.router, prefix="/api") |
|
|
|
|
|
app.include_router(sandbox_api.router, prefix="/api") |
|
|
|
|
|
app.include_router(billing_api.router, prefix="/api") |
|
|
|
@app.get("/api/health") |
|
async def health_check(): |
|
"""Health check endpoint to verify API is working.""" |
|
logger.info("Health check endpoint called") |
|
return { |
|
"status": "ok", |
|
"timestamp": datetime.now(timezone.utc).isoformat(), |
|
"instance_id": instance_id |
|
} |
|
|
|
if __name__ == "__main__": |
|
import uvicorn |
|
|
|
workers = 2 |
|
|
|
logger.info(f"Starting server on 0.0.0.0:8000 with {workers} workers") |
|
uvicorn.run( |
|
"api:app", |
|
host="0.0.0.0", |
|
port=7860, |
|
workers=workers, |
|
|
|
) |