mirror of
https://github.com/dograh-hq/dograh.git
synced 2026-06-22 08:38:13 +02:00
Initial Commit 🚀 🚀
This commit is contained in:
commit
4f2a629340
444 changed files with 76863 additions and 0 deletions
122
api/app.py
Normal file
122
api/app.py
Normal file
|
|
@ -0,0 +1,122 @@
|
|||
"""Set up logging before importing anything else"""
|
||||
|
||||
import sentry_sdk
|
||||
|
||||
from api.constants import ENABLE_SENTRY, REDIS_URL, SENTRY_DSN
|
||||
from api.logging_config import ENVIRONMENT, setup_logging
|
||||
|
||||
# Set up logging and get the listener for cleanup
|
||||
logging_queue_listener = setup_logging()
|
||||
|
||||
|
||||
if ENABLE_SENTRY:
|
||||
if not SENTRY_DSN:
|
||||
print(
|
||||
"Warning: ENABLE_SENTRY is true but SENTRY_DSN is not configured. Sentry disabled."
|
||||
)
|
||||
else:
|
||||
sentry_sdk.init(
|
||||
dsn=SENTRY_DSN,
|
||||
# Add data like request headers and IP for users,
|
||||
# see https://docs.sentry.io/platforms/python/data-management/data-collected/ for more info
|
||||
send_default_pii=True,
|
||||
environment=ENVIRONMENT,
|
||||
)
|
||||
print(f"Sentry initialized in environment: {ENVIRONMENT}")
|
||||
else:
|
||||
print(f"Sentry disabled (ENABLE_SENTRY=false)")
|
||||
|
||||
|
||||
import asyncio
|
||||
from contextlib import asynccontextmanager
|
||||
from typing import Optional
|
||||
|
||||
import redis.asyncio as aioredis
|
||||
from fastapi import APIRouter, FastAPI
|
||||
from fastapi.middleware.cors import CORSMiddleware
|
||||
from loguru import logger
|
||||
|
||||
from api.routes.main import router as main_router
|
||||
from api.routes.rtc_offer import pcs_map
|
||||
from api.services.telephony.worker_event_subscriber import (
|
||||
WorkerEventSubscriber,
|
||||
setup_worker_subscriber,
|
||||
)
|
||||
from api.tasks.arq import get_arq_redis
|
||||
|
||||
API_PREFIX = "/api/v1"
|
||||
|
||||
# Global reference to worker subscriber for graceful shutdown
|
||||
worker_subscriber_instance: Optional[WorkerEventSubscriber] = None
|
||||
|
||||
|
||||
@asynccontextmanager
|
||||
async def lifespan(app: FastAPI):
|
||||
global worker_subscriber_instance
|
||||
|
||||
# warmup arq pool
|
||||
await get_arq_redis()
|
||||
|
||||
# Setup Redis connection for distributed mode
|
||||
redis = await aioredis.from_url(REDIS_URL, decode_responses=True)
|
||||
|
||||
# Setup worker subscriber (ARI Manager runs separately)
|
||||
worker_subscriber = await setup_worker_subscriber(redis)
|
||||
worker_subscriber_instance = worker_subscriber
|
||||
|
||||
# Store worker ID in app state for health check
|
||||
app.state.worker_id = worker_subscriber.worker_id
|
||||
app.state.worker_subscriber = worker_subscriber
|
||||
|
||||
yield # Run app
|
||||
|
||||
# Shutdown sequence - this runs when FastAPI is shutting down
|
||||
logger.info("Starting graceful shutdown...")
|
||||
|
||||
# First, try graceful shutdown with timeout
|
||||
if worker_subscriber:
|
||||
try:
|
||||
# Check if we should do graceful shutdown (e.g., if SIGTERM was received)
|
||||
# For now, we'll attempt graceful shutdown for all shutdowns
|
||||
await worker_subscriber.graceful_shutdown(max_wait_seconds=300)
|
||||
except Exception as e:
|
||||
logger.error(f"Error during graceful shutdown: {e}")
|
||||
# Fall back to immediate stop
|
||||
await worker_subscriber.stop()
|
||||
|
||||
# close all dangling pipecat connections
|
||||
coros = [pc.close() for pc in pcs_map.values()]
|
||||
await asyncio.gather(*coros)
|
||||
pcs_map.clear()
|
||||
|
||||
await redis.aclose()
|
||||
|
||||
if logging_queue_listener is not None:
|
||||
logging_queue_listener.stop()
|
||||
|
||||
|
||||
app = FastAPI(
|
||||
title="Dograh API",
|
||||
description="API for the Dograh app",
|
||||
version="1.0.0",
|
||||
openapi_url=f"{API_PREFIX}/openapi.json",
|
||||
lifespan=lifespan,
|
||||
)
|
||||
|
||||
|
||||
# Configure CORS
|
||||
app.add_middleware(
|
||||
CORSMiddleware,
|
||||
allow_origins=["*"], # Allows all origins
|
||||
allow_credentials=True,
|
||||
allow_methods=["*"], # Allows all methods
|
||||
allow_headers=["*"], # Allows all headers
|
||||
)
|
||||
|
||||
api_router = APIRouter()
|
||||
|
||||
# include subrouters here
|
||||
api_router.include_router(main_router)
|
||||
|
||||
# main router with api prefix
|
||||
app.include_router(api_router, prefix=API_PREFIX)
|
||||
Loading…
Add table
Add a link
Reference in a new issue