80 lines
2.6 KiB
Python
80 lines
2.6 KiB
Python
"""
|
|
Wartungs-Router: Container/Toolbox aktualisieren.
|
|
|
|
Der konkrete Befehl steckt in MC_UPDATE_CMD (z.B. kyuz0 refresh-Skript) und
|
|
laeuft als Hintergrund-Job mit Live-Log. Spaeter wandert hier ggf. mehr
|
|
Server-Wartung hinein (siehe Roadmap: Server-Management).
|
|
"""
|
|
|
|
import shlex
|
|
import subprocess
|
|
import asyncio
|
|
|
|
from fastapi import APIRouter, Depends, HTTPException, WebSocket, WebSocketDisconnect
|
|
from pydantic import BaseModel
|
|
|
|
from auth import auth
|
|
from config import UPDATE_CMD
|
|
from jobengine import start_job
|
|
|
|
router = APIRouter(prefix="/api", dependencies=[Depends(auth)])
|
|
|
|
class PwdReq(BaseModel):
|
|
password: str
|
|
|
|
@router.post("/update")
|
|
def update():
|
|
if not UPDATE_CMD:
|
|
raise HTTPException(400, "Kein Update-Befehl gesetzt (MC_UPDATE_CMD).")
|
|
job_id = start_job(shlex.split(UPDATE_CMD), "update containers")
|
|
return {"job_id": job_id}
|
|
|
|
@router.post("/os-update")
|
|
def os_update(req: PwdReq):
|
|
cmd = f"echo {shlex.quote(req.password)} | sudo -S bash -c 'apt-get update && DEBIAN_FRONTEND=noninteractive apt-get upgrade -y'"
|
|
job_id = start_job(["bash", "-c", cmd], "os update")
|
|
return {"job_id": job_id}
|
|
|
|
@router.post("/service/{name}/restart")
|
|
def service_restart(name: str):
|
|
if name not in ("llama-swap", "mission-control"):
|
|
raise HTTPException(400, "Dienst nicht erlaubt.")
|
|
subprocess.Popen(["sudo", "systemctl", "restart", name])
|
|
return {"ok": True, "note": f"Restart {name} getriggert."}
|
|
|
|
@router.post("/reboot")
|
|
def reboot(req: PwdReq):
|
|
cmd = f"echo {shlex.quote(req.password)} | sudo -S reboot"
|
|
subprocess.Popen(["bash", "-c", cmd])
|
|
return {"ok": True, "note": "Reboot getriggert."}
|
|
|
|
@router.websocket("/logs/{service}")
|
|
async def stream_logs(websocket: WebSocket, service: str):
|
|
await websocket.accept()
|
|
if service not in ("llama-swap", "mission-control"):
|
|
await websocket.close(code=1008)
|
|
return
|
|
|
|
process = await asyncio.create_subprocess_exec(
|
|
"sudo", "journalctl", "-u", service, "-n", "100", "-f",
|
|
stdout=asyncio.subprocess.PIPE,
|
|
stderr=asyncio.subprocess.PIPE
|
|
)
|
|
try:
|
|
while True:
|
|
line = await process.stdout.readline()
|
|
if not line:
|
|
break
|
|
line_str = line.decode("utf-8", errors="replace")
|
|
# Filter spammy polls
|
|
if '"GET /running HTTP' in line_str or '"GET /api/' in line_str:
|
|
continue
|
|
await websocket.send_text(line_str)
|
|
except WebSocketDisconnect:
|
|
pass
|
|
finally:
|
|
try:
|
|
process.terminate()
|
|
except OSError:
|
|
pass
|