Matches the 1.0.0-38 product display rename. Touches every
infrastructure identifier:
- container_name: truenas-burnin → nas-burnin
- forge URL in /api/v1/updates/check
- security-scan: REPO_URL, REPO, DEPLOY_DIR, systemd unit description
- run-tests.sh default container name
- doc paths in README/SPEC/CLAUDE
- in-app instruction strings (login.html, settings.html, auth_cli.py)
Maple migration done in lockstep:
docker compose down (truenas-burnin)
mv ~/docker/stacks/{truenas-burnin,nas-burnin}
systemd unit ExecStart updated + daemon-reload
docker compose up -d --build → container nas-burnin
Old image truenas-burnin-app removed (~12 GB reclaimed)
Stale top-level orphans cleaned (config.py, poller.py, routes.py,
truenas.py, tests/) — all dead since pre-split refactors
Forge repo rename (git.hellocomputer.xyz/brandon/truenas-burnin →
nas-burnin) is a separate UI-only step. Forgejo redirects the old
URL after rename, so this commit can be pushed to the existing
remote first; remote URL gets updated locally once you rename.
136 lines
5 KiB
Python
136 lines
5 KiB
Python
"""System-level endpoints with no business-logic dependencies.
|
|
|
|
GET /health — readiness probe (DB write + poller + SSH)
|
|
GET /api/v1/updates/check — check Forgejo for newer release
|
|
WS /ws/terminal — xterm.js bridge to TrueNAS SSH PTY
|
|
"""
|
|
|
|
from __future__ import annotations
|
|
|
|
from datetime import datetime, timezone
|
|
|
|
import aiosqlite
|
|
from fastapi import APIRouter, Depends, WebSocket
|
|
from fastapi.responses import JSONResponse
|
|
|
|
from app import poller
|
|
from app.config import settings
|
|
from app.database import get_db
|
|
|
|
router = APIRouter()
|
|
|
|
|
|
@router.get("/health")
|
|
async def health(db: aiosqlite.Connection = Depends(get_db)):
|
|
"""Real readiness check, not just process-is-running.
|
|
|
|
Verifies (a) DB writable, (b) poller has succeeded recently relative
|
|
to the configured stale_threshold_seconds, (c) SSH reachable when
|
|
configured. Returns 503 when any check fails so a proxy/orchestrator
|
|
health probe can take the container out of rotation.
|
|
"""
|
|
from app import ssh_client as _ssh
|
|
|
|
checks: dict[str, dict] = {}
|
|
|
|
# DB probe — actually exercise the write path (read-only mounts,
|
|
# full disks, broken WAL all silently pass a journal_mode read).
|
|
# Uses a temp table that lives only inside the connection so the
|
|
# round-trip touches the writer without polluting real data.
|
|
try:
|
|
await db.execute(
|
|
"CREATE TEMP TABLE IF NOT EXISTS _hc (k INTEGER PRIMARY KEY, v TEXT)"
|
|
)
|
|
await db.execute("INSERT OR REPLACE INTO _hc (k, v) VALUES (1, ?)",
|
|
(datetime.now(timezone.utc).isoformat(),))
|
|
cur = await db.execute("SELECT v FROM _hc WHERE k=1")
|
|
row = await cur.fetchone()
|
|
await db.commit()
|
|
checks["db"] = {"ok": bool(row)}
|
|
except Exception as exc:
|
|
checks["db"] = {"ok": False, "error": str(exc)}
|
|
|
|
ps = poller.get_state()
|
|
last = ps.get("last_poll_at")
|
|
poll_age = None
|
|
if last:
|
|
try:
|
|
t = datetime.fromisoformat(last)
|
|
if t.tzinfo is None:
|
|
t = t.replace(tzinfo=timezone.utc)
|
|
poll_age = (datetime.now(timezone.utc) - t).total_seconds()
|
|
except Exception:
|
|
poll_age = None
|
|
poll_ok = ps.get("healthy") and (
|
|
poll_age is None or poll_age <= settings.stale_threshold_seconds * 3
|
|
)
|
|
checks["poller"] = {
|
|
"ok": bool(poll_ok),
|
|
"last_error": ps.get("last_error"),
|
|
"last_poll_at": last,
|
|
"age_seconds": int(poll_age) if poll_age is not None else None,
|
|
}
|
|
|
|
# SSH probe — only when configured. Cheap (single sensors -j).
|
|
if _ssh.is_configured():
|
|
try:
|
|
r = await _ssh.test_connection()
|
|
checks["ssh"] = {"ok": bool(r.get("ok")),
|
|
"error": r.get("error")}
|
|
except Exception as exc:
|
|
checks["ssh"] = {"ok": False, "error": str(exc)}
|
|
else:
|
|
checks["ssh"] = {"ok": True, "skipped": True}
|
|
|
|
cur = await db.execute("SELECT COUNT(*) FROM drives")
|
|
row = await cur.fetchone()
|
|
drives_tracked = row[0] if row else 0
|
|
|
|
status_ok = all(c["ok"] for c in checks.values())
|
|
body = {
|
|
"status": "ok" if status_ok else "degraded",
|
|
"checks": checks,
|
|
"drives_tracked": drives_tracked,
|
|
"poll_interval_s": settings.poll_interval_seconds,
|
|
"version": settings.app_version,
|
|
}
|
|
return JSONResponse(body, status_code=200 if status_ok else 503)
|
|
|
|
|
|
@router.websocket("/ws/terminal")
|
|
async def terminal_ws(websocket: WebSocket):
|
|
"""WebSocket endpoint bridging the browser xterm.js terminal to an SSH PTY."""
|
|
from app import terminal as _term
|
|
await _term.handle(websocket)
|
|
|
|
|
|
@router.get("/api/v1/updates/check")
|
|
async def check_updates():
|
|
"""Check for a newer release on Forgejo."""
|
|
import httpx
|
|
current = settings.app_version
|
|
try:
|
|
async with httpx.AsyncClient(timeout=8.0) as client:
|
|
r = await client.get(
|
|
"https://git.hellocomputer.xyz/api/v1/repos/brandon/nas-burnin/releases/latest",
|
|
headers={"Accept": "application/json"},
|
|
)
|
|
if r.status_code == 200:
|
|
data = r.json()
|
|
latest = data.get("tag_name", "").lstrip("v")
|
|
up_to_date = not latest or latest == current
|
|
return {
|
|
"current": current,
|
|
"latest": latest or None,
|
|
"update_available": not up_to_date,
|
|
"message": None,
|
|
}
|
|
elif r.status_code == 404:
|
|
return {"current": current, "latest": None, "update_available": False,
|
|
"message": "No releases published yet"}
|
|
else:
|
|
return {"current": current, "latest": None, "update_available": False,
|
|
"message": f"Forgejo API returned {r.status_code}"}
|
|
except Exception as exc:
|
|
return {"current": current, "latest": None, "update_available": False,
|
|
"message": f"Could not reach update server: {exc}"}
|