from fastapi import FastAPI, HTTPException
import asyncio
from typing import Any, Dict, Optional
from collections import deque
from dataclasses import dataclass, field

app = FastAPI()

@dataclass
class SignalEntry:
    """FIFO queue entry for an individual signal key."""

    event: asyncio.Event = field(default_factory=asyncio.Event)
    queue: deque[Any] = field(default_factory=deque)

# Global dictionary to store queues: key -> SignalEntry
signals: Dict[str, SignalEntry] = {}
signals_lock = asyncio.Lock()

@app.get("/wait/{key}")
async def wait_for_signal(key: str, timeout: Optional[float] = 30.0):
    """
    Wait for a signal with the given key. If the signal is not yet submitted,
    wait until it is available or timeout, then return the data.
    """
    print(f"Waiting for signal with key: '{key}' with timeout: {timeout}")
    while True:
        entry = await _ensure_entry(key)
        async with signals_lock:
            if entry.queue:
                data = entry.queue.popleft()
                if not entry.queue:
                    entry.event.clear()
                    # 清理空的 entry
                    await _cleanup_empty_entry(key)
                return {"data": data}
            wait_event = entry.event

        try:
            await asyncio.wait_for(wait_event.wait(), timeout=timeout)
        except asyncio.TimeoutError as timeout_err:
            async with signals_lock:
                entry = signals.get(key)
                if entry:
                    entry.event.clear()
                    # 超时后清理空队列的 entry
                    await _cleanup_empty_entry(key)
            raise HTTPException(status_code=408, detail="Request timeout") from timeout_err

@app.post("/submit/{key}")
async def submit_signal(key: str, data: Dict[str, Any]):
    """
    Submit data for a signal with the given key. This will trigger any waiting requests.
    """
    entry = await _ensure_entry(key)
    async with signals_lock:
        entry.queue.append(data)
        entry.event.set()
        queue_length = len(entry.queue)
    return {"status": "submitted", "queue_length": queue_length}

async def _ensure_entry(key: str) -> SignalEntry:
    async with signals_lock:
        entry = signals.get(key)
        if entry is None:
            entry = SignalEntry()
            signals[key] = entry
        return entry

async def _cleanup_empty_entry(key: str):
    """
    清理空的 SignalEntry，释放内存。
    注意：此函数应在持有 signals_lock 的情况下调用。
    """
    entry = signals.get(key)
    if entry and not entry.queue:
        del signals[key]
        print(f"Cleaned up empty entry for key: '{key}'")

@app.get("/health")
async def health_check():
    """
    Health check endpoint.
    """
    return {"status": "healthy"}