Initial web ui and control for robot

This commit is contained in:
2026-04-30 21:35:18 +00:00
parent d64e1c24c8
commit 91f5f4d3ab
23 changed files with 2688 additions and 4 deletions
+105
View File
@@ -0,0 +1,105 @@
import asyncio
import logging
from contextlib import asynccontextmanager
from pathlib import Path
from fastapi import FastAPI, WebSocket, WebSocketDisconnect
from fastapi.responses import FileResponse, JSONResponse
from ros_node import get_node, start_ros, stop_ros
logging.basicConfig(level=logging.INFO)
logger = logging.getLogger(__name__)
STATIC_DIR = Path(__file__).parent / 'static'
class ConnectionManager:
def __init__(self) -> None:
self.connections: list[WebSocket] = []
self._loop: asyncio.AbstractEventLoop | None = None
def set_loop(self, loop: asyncio.AbstractEventLoop) -> None:
self._loop = loop
async def connect(self, ws: WebSocket) -> None:
await ws.accept()
self.connections.append(ws)
def disconnect(self, ws: WebSocket) -> None:
if ws in self.connections:
self.connections.remove(ws)
def broadcast_from_thread(self, data: dict) -> None:
if self._loop and self._loop.is_running():
asyncio.run_coroutine_threadsafe(self._broadcast(data), self._loop)
async def _broadcast(self, data: dict) -> None:
dead = []
for ws in self.connections:
try:
await ws.send_json(data)
except Exception:
dead.append(ws)
for ws in dead:
self.disconnect(ws)
manager = ConnectionManager()
@asynccontextmanager
async def lifespan(app: FastAPI):
manager.set_loop(asyncio.get_event_loop())
node = start_ros()
node.on_joint_states_callbacks.append(manager.broadcast_from_thread)
node.on_ultrasonic_callbacks.append(manager.broadcast_from_thread)
yield
stop_ros()
app = FastAPI(lifespan=lifespan)
@app.websocket('/ws')
async def websocket_endpoint(ws: WebSocket):
await manager.connect(ws)
node = get_node()
try:
while True:
data = await ws.receive_json()
msg_type = data.get('type')
if msg_type == 'cmd_vel':
node.publish_cmd_vel(data.get('linear_x', 0.0), data.get('angular_z', 0.0))
elif msg_type == 'joint_command':
node.publish_joint_command(data.get('pan', 0.0), data.get('tilt', 0.0))
except WebSocketDisconnect:
pass
except Exception as e:
logger.error('WebSocket error: %s', e)
finally:
manager.disconnect(ws)
try:
node.publish_cmd_vel(0.0, 0.0)
except Exception:
pass
@app.get('/{full_path:path}')
async def serve_spa(full_path: str):
index = STATIC_DIR / 'index.html'
if not index.exists():
return JSONResponse(
{'detail': 'Frontend not built. In dev mode open http://localhost:5173'},
status_code=404,
)
candidate = (STATIC_DIR / full_path).resolve()
static_root = STATIC_DIR.resolve()
if str(candidate).startswith(str(static_root)) and full_path and candidate.is_file():
return FileResponse(candidate)
return FileResponse(index)
if __name__ == '__main__':
import uvicorn
uvicorn.run(app, host='0.0.0.0', port=8080)