Files
Shadowbroker/frontend/src/components/TopRightControls.tsx
T
anoracleofra-code fc9eff865e v0.9.0: in-app auto-updater, ship toggle split, stable entity IDs, performance fixes
New features:
- In-app auto-updater with confirmation dialog, manual download fallback,
  restart polling, and protected file safety net
- Ship layers split into 4 independent toggles (Military/Carriers, Cargo/Tankers,
  Civilian, Cruise/Passenger) with per-category counts
- Stable entity IDs using MMSI/callsign instead of volatile array indices
- Dismissible threat alert bubbles (session-scoped, survives data refresh)

Performance:
- GDELT title fetching is now non-blocking (background enrichment)
- Removed duplicate startup fetch jobs
- Docker healthcheck start_period 15s → 90s

Bug fixes:
- Removed fake intelligence assessment generator (OSINT-only policy)
- Fixed carrier tracker GDELT 429/TypeError crash
- Fixed ETag collision (full payload hash)
- Added concurrent /api/refresh guard

Contributors: @imqdcr (ship split + stable IDs), @csysp (dismissible alerts, PR #48)

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>

Former-commit-id: a2c4c67da54345393f70a9b33b52e7e4fd6c049f
2026-03-13 11:32:16 -06:00

266 lines
13 KiB
TypeScript

"use client";
import { useState, useRef, useEffect } from "react";
import { Github, MessageSquare, Download, AlertCircle, CheckCircle2, RefreshCw, ExternalLink, X } from "lucide-react";
import { API_BASE } from "@/lib/api";
import packageJson from "../../package.json";
type UpdateStatus =
| "idle"
| "checking"
| "available"
| "uptodate"
| "error"
| "confirming"
| "updating"
| "restarting"
| "update_error";
export default function TopRightControls() {
const [updateStatus, setUpdateStatus] = useState<UpdateStatus>("idle");
const [latestVersion, setLatestVersion] = useState<string>("");
const [errorMessage, setErrorMessage] = useState("");
const pollRef = useRef<ReturnType<typeof setInterval> | null>(null);
const timeoutRef = useRef<ReturnType<typeof setTimeout> | null>(null);
const currentVersion = packageJson.version;
// Cleanup polling on unmount
useEffect(() => {
return () => {
if (pollRef.current) clearInterval(pollRef.current);
if (timeoutRef.current) clearTimeout(timeoutRef.current);
};
}, []);
const checkForUpdates = async () => {
setUpdateStatus("checking");
try {
const res = await fetch("https://api.github.com/repos/BigBodyCobain/Shadowbroker/releases/latest");
if (!res.ok) throw new Error("Failed to fetch");
const data = await res.json();
const latest = data.tag_name?.replace("v", "") || data.name?.replace("v", "");
const current = currentVersion.replace("v", "");
if (latest && latest !== current) {
setLatestVersion(latest);
setUpdateStatus("available");
} else {
setUpdateStatus("uptodate");
setTimeout(() => setUpdateStatus("idle"), 3000);
}
} catch (err) {
console.error("Update check failed:", err);
setUpdateStatus("error");
setTimeout(() => setUpdateStatus("idle"), 3000);
}
};
const triggerUpdate = async () => {
setUpdateStatus("updating");
setErrorMessage("");
try {
const res = await fetch(`${API_BASE}/api/system/update`, { method: "POST" });
const data = await res.json();
if (!res.ok) throw new Error(data.message || "Update failed");
setUpdateStatus("restarting");
// Poll /api/health until backend comes back
pollRef.current = setInterval(async () => {
try {
const h = await fetch(`${API_BASE}/api/health`);
if (h.ok) {
if (pollRef.current) clearInterval(pollRef.current);
if (timeoutRef.current) clearTimeout(timeoutRef.current);
window.location.reload();
}
} catch {
// Backend still down — keep polling
}
}, 3000);
// Give up after 90 seconds
timeoutRef.current = setTimeout(() => {
if (pollRef.current) clearInterval(pollRef.current);
setErrorMessage("Restart timed out — the app may need to be started manually.");
setUpdateStatus("update_error");
}, 90000);
} catch (err: any) {
setErrorMessage(err.message || "Unknown error");
setUpdateStatus("update_error");
}
};
// ── Confirmation Dialog ──
const renderConfirmDialog = () => (
<div className="absolute top-full right-0 mt-2 w-72 z-[9999]">
<div className="bg-[var(--bg-primary)]/95 backdrop-blur-md border border-cyan-800/60 rounded-lg shadow-[0_4px_30px_rgba(0,255,255,0.15)] overflow-hidden">
{/* Header */}
<div className="flex items-center justify-between px-3 py-2 border-b border-[var(--border-primary)]">
<span className="text-[10px] font-mono tracking-widest text-cyan-400">
UPDATE v{currentVersion} v{latestVersion}
</span>
<button
onClick={() => setUpdateStatus("available")}
className="text-[var(--text-muted)] hover:text-[var(--text-primary)] transition-colors"
>
<X size={12} />
</button>
</div>
{/* Actions */}
<div className="p-3 flex flex-col gap-2">
<button
onClick={triggerUpdate}
className="w-full flex items-center justify-center gap-2 px-3 py-2 bg-cyan-500/10 border border-cyan-500/40 rounded-md hover:bg-cyan-500/20 transition-all text-[10px] text-cyan-400 font-mono tracking-widest"
>
<Download size={12} />
AUTO UPDATE
</button>
<a
href="https://github.com/BigBodyCobain/Shadowbroker/releases/latest"
target="_blank"
rel="noreferrer"
className="w-full flex items-center justify-center gap-2 px-3 py-2 bg-[var(--bg-secondary)]/50 border border-[var(--border-primary)] rounded-md hover:border-[var(--text-muted)] transition-all text-[10px] text-[var(--text-muted)] font-mono tracking-widest"
>
<ExternalLink size={12} />
MANUAL DOWNLOAD
</a>
<button
onClick={() => setUpdateStatus("available")}
className="w-full flex items-center justify-center px-3 py-1.5 text-[9px] text-[var(--text-muted)] font-mono tracking-widest hover:text-[var(--text-secondary)] transition-colors"
>
CANCEL
</button>
</div>
</div>
</div>
);
// ── Error Dialog ──
const renderErrorDialog = () => (
<div className="absolute top-full right-0 mt-2 w-72 z-[9999]">
<div className="bg-[var(--bg-primary)]/95 backdrop-blur-md border border-red-800/60 rounded-lg shadow-[0_4px_30px_rgba(255,0,0,0.1)] overflow-hidden">
<div className="px-3 py-2 border-b border-red-900/40">
<span className="text-[10px] font-mono tracking-widest text-red-400">
UPDATE FAILED
</span>
</div>
<div className="p-3 flex flex-col gap-2">
<p className="text-[9px] font-mono text-[var(--text-muted)] leading-relaxed break-words">
{errorMessage}
</p>
<button
onClick={() => setUpdateStatus("confirming")}
className="w-full flex items-center justify-center gap-2 px-3 py-2 bg-cyan-500/10 border border-cyan-500/40 rounded-md hover:bg-cyan-500/20 transition-all text-[10px] text-cyan-400 font-mono tracking-widest"
>
<RefreshCw size={12} />
TRY AGAIN
</button>
<a
href="https://github.com/BigBodyCobain/Shadowbroker/releases/latest"
target="_blank"
rel="noreferrer"
className="w-full flex items-center justify-center gap-2 px-3 py-2 bg-[var(--bg-secondary)]/50 border border-[var(--border-primary)] rounded-md hover:border-[var(--text-muted)] transition-all text-[10px] text-[var(--text-muted)] font-mono tracking-widest"
>
<ExternalLink size={12} />
MANUAL DOWNLOAD
</a>
</div>
</div>
</div>
);
return (
<div className="relative flex items-center gap-2 mb-1 justify-end">
{/* Discussions link */}
<a
href="https://github.com/BigBodyCobain/Shadowbroker/discussions"
target="_blank"
rel="noreferrer"
className="flex items-center gap-1.5 px-2.5 py-1.5 bg-[var(--bg-primary)]/50 backdrop-blur-md border border-[var(--border-primary)] rounded-lg hover:border-cyan-500/50 hover:bg-[var(--hover-accent)] transition-all text-[10px] text-[var(--text-secondary)] font-mono cursor-pointer"
>
<MessageSquare size={12} className="text-cyan-400 w-3 h-3" />
<span className="tracking-widest">DISCUSSIONS</span>
</a>
{/* ── Update Available → opens confirmation ── */}
{updateStatus === "available" && (
<button
onClick={() => setUpdateStatus("confirming")}
className="flex items-center gap-1.5 px-2.5 py-1.5 bg-green-500/10 backdrop-blur-md border border-green-500/50 rounded-lg hover:bg-green-500/20 transition-all text-[10px] text-green-400 font-mono cursor-pointer shadow-[0_0_15px_rgba(34,197,94,0.3)]"
>
<Download size={12} className="w-3 h-3" />
<span className="tracking-widest animate-pulse">v{latestVersion} UPDATE!</span>
</button>
)}
{/* ── Confirming → show dialog ── */}
{updateStatus === "confirming" && (
<>
<button className="flex items-center gap-1.5 px-2.5 py-1.5 bg-green-500/10 backdrop-blur-md border border-green-500/50 rounded-lg text-[10px] text-green-400 font-mono shadow-[0_0_15px_rgba(34,197,94,0.3)]">
<Download size={12} className="w-3 h-3" />
<span className="tracking-widest">v{latestVersion} UPDATE!</span>
</button>
{renderConfirmDialog()}
</>
)}
{/* ── Updating → spinner ── */}
{updateStatus === "updating" && (
<div className="flex items-center gap-1.5 px-2.5 py-1.5 bg-cyan-500/10 backdrop-blur-md border border-cyan-500/50 rounded-lg text-[10px] text-cyan-400 font-mono">
<RefreshCw size={12} className="w-3 h-3 animate-spin" />
<span className="tracking-widest">DOWNLOADING UPDATE...</span>
</div>
)}
{/* ── Restarting → spinner + waiting ── */}
{updateStatus === "restarting" && (
<div className="flex items-center gap-1.5 px-2.5 py-1.5 bg-cyan-500/10 backdrop-blur-md border border-cyan-500/50 rounded-lg text-[10px] text-cyan-400 font-mono shadow-[0_0_15px_rgba(0,255,255,0.2)]">
<RefreshCw size={12} className="w-3 h-3 animate-spin" />
<span className="tracking-widest animate-pulse">RESTARTING...</span>
</div>
)}
{/* ── Error → show error dialog ── */}
{updateStatus === "update_error" && (
<>
<button
onClick={() => setUpdateStatus("confirming")}
className="flex items-center gap-1.5 px-2.5 py-1.5 bg-red-500/10 backdrop-blur-md border border-red-500/50 rounded-lg hover:bg-red-500/20 transition-all text-[10px] text-red-400 font-mono"
>
<AlertCircle size={12} className="w-3 h-3" />
<span className="tracking-widest">UPDATE FAILED</span>
</button>
{renderErrorDialog()}
</>
)}
{/* ── Default states: idle / checking / uptodate / check-error ── */}
{!["available", "confirming", "updating", "restarting", "update_error"].includes(updateStatus) && (
<button
onClick={checkForUpdates}
disabled={updateStatus === "checking"}
className="flex items-center gap-1.5 px-2.5 py-1.5 bg-[var(--bg-primary)]/50 backdrop-blur-md border border-[var(--border-primary)] rounded-lg hover:border-cyan-500/50 hover:bg-[var(--hover-accent)] transition-all text-[10px] text-[var(--text-secondary)] font-mono cursor-pointer disabled:opacity-50 disabled:cursor-not-allowed"
>
{updateStatus === "checking" && <Github size={12} className="w-3 h-3 animate-spin text-cyan-400" />}
{updateStatus === "idle" && <Github size={12} className="w-3 h-3 text-cyan-400" />}
{updateStatus === "uptodate" && <CheckCircle2 size={12} className="w-3 h-3 text-green-400" />}
{updateStatus === "error" && <AlertCircle size={12} className="w-3 h-3 text-red-400" />}
<span className="tracking-widest">
{updateStatus === "checking" ? "CHECKING..." :
updateStatus === "uptodate" ? "UP TO DATE" :
updateStatus === "error" ? "CHECK FAILED" :
"CHECK UPDATES"}
</span>
</button>
)}
</div>
);
}