|
4 | 4 | import json |
5 | 5 | from contextlib import suppress |
6 | 6 | from datetime import timedelta |
7 | | -from typing import Dict, List, Optional |
| 7 | +from typing import Dict, List, Optional, Tuple |
8 | 8 | from fastapi import WebSocket |
9 | 9 |
|
10 | 10 | from models import TankInfo |
|
14 | 14 | class ConnectionManager: |
15 | 15 | """Manages WebSocket connections to tanks.""" |
16 | 16 |
|
17 | | - def __init__(self) -> None: |
| 17 | + def __init__( |
| 18 | + self, |
| 19 | + *, |
| 20 | + stale_timeout_seconds: int = 600, |
| 21 | + prune_interval_seconds: int = 30, |
| 22 | + ) -> None: |
18 | 23 | self._tanks: Dict[str, TankInfo] = {} |
19 | 24 | self._lock = asyncio.Lock() |
20 | | - self._stale_timeout = timedelta(minutes=10) |
| 25 | + self._stale_timeout = timedelta(seconds=max(1, stale_timeout_seconds)) |
| 26 | + self._prune_interval = timedelta(seconds=max(5, prune_interval_seconds)) |
| 27 | + self._maintenance_task: Optional[asyncio.Task] = None |
21 | 28 |
|
22 | | - async def _prune_stale(self) -> None: |
| 29 | + async def start(self) -> None: |
| 30 | + """Start background maintenance tasks.""" |
| 31 | + if self._maintenance_task and not self._maintenance_task.done(): |
| 32 | + return |
| 33 | + self._maintenance_task = asyncio.create_task(self._run_auto_prune()) |
| 34 | + |
| 35 | + async def stop(self) -> None: |
| 36 | + """Stop maintenance tasks and wait for completion.""" |
| 37 | + task = self._maintenance_task |
| 38 | + if task: |
| 39 | + task.cancel() |
| 40 | + with suppress(asyncio.CancelledError): |
| 41 | + await task |
| 42 | + self._maintenance_task = None |
| 43 | + |
| 44 | + async def _run_auto_prune(self) -> None: |
| 45 | + """Periodically prune stale tank connections.""" |
| 46 | + try: |
| 47 | + while True: |
| 48 | + await asyncio.sleep(self._prune_interval.total_seconds()) |
| 49 | + await self._prune_stale(reason="auto-prune") |
| 50 | + except asyncio.CancelledError: |
| 51 | + pass |
| 52 | + |
| 53 | + async def _prune_stale(self, *, reason: str = "stale") -> None: |
23 | 54 | """Remove tanks that have been inactive for longer than the timeout.""" |
24 | 55 | now = utcnow() |
25 | | - to_close: List[WebSocket] = [] |
| 56 | + to_close: List[Tuple[str, WebSocket]] = [] |
26 | 57 | async with self._lock: |
27 | 58 | for tank_id, info in list(self._tanks.items()): |
28 | 59 | if (now - info.last_seen) > self._stale_timeout: |
29 | 60 | websocket = info.websocket |
30 | 61 | if websocket is not None: |
31 | | - to_close.append(websocket) |
| 62 | + to_close.append((tank_id, websocket)) |
32 | 63 | self._tanks.pop(tank_id, None) |
33 | | - for websocket in to_close: |
| 64 | + for tank_id, websocket in to_close: |
34 | 65 | with suppress(Exception): |
35 | 66 | await websocket.close(code=1011) |
| 67 | + print(f"[MANAGER] Pruned tank '{tank_id}' due to {reason}") |
36 | 68 |
|
37 | 69 | async def register_tank(self, tank_id: str, websocket: WebSocket) -> TankInfo: |
38 | 70 | """Register a new tank connection.""" |
@@ -110,3 +142,28 @@ async def update_last_seen(self, tank_id: str, payload: Optional[dict]) -> None: |
110 | 142 | info.last_seen = utcnow() |
111 | 143 | if payload is not None: |
112 | 144 | info.last_payload = payload |
| 145 | + |
| 146 | + async def force_reset(self, tank_id: str) -> bool: |
| 147 | + """Forcefully terminate and remove a tank connection.""" |
| 148 | + async with self._lock: |
| 149 | + info = self._tanks.pop(tank_id, None) |
| 150 | + if not info: |
| 151 | + return False |
| 152 | + websocket = info.websocket |
| 153 | + if websocket: |
| 154 | + with suppress(Exception): |
| 155 | + await websocket.close(code=1012) |
| 156 | + print(f"[MANAGER] Forced reset for tank '{tank_id}'") |
| 157 | + return True |
| 158 | + |
| 159 | + async def close_all(self) -> None: |
| 160 | + """Close all tracked tank connections.""" |
| 161 | + async with self._lock: |
| 162 | + entries = list(self._tanks.items()) |
| 163 | + self._tanks.clear() |
| 164 | + for tank_id, info in entries: |
| 165 | + websocket = info.websocket |
| 166 | + if websocket: |
| 167 | + with suppress(Exception): |
| 168 | + await websocket.close(code=1001) |
| 169 | + print(f"[MANAGER] Closed connection for tank '{tank_id}' during shutdown") |
0 commit comments