Files
multiverse/jumpa-iam/app/supreme-admin/telemetry/page.tsx
T

289 lines
12 KiB
TypeScript

"use client";
import { useEffect, useState, useRef } from "react";
import { useDictionary } from "@/lib/dictionary";
import { useRouter } from "next/navigation";
export default function PanopticonTelemetry() {
const { t } = useDictionary();
const [logs, setLogs] = useState<{ id: string; time: string; action: string; target: string; status: string }[]>([]);
const [activeConnections, setActiveConnections] = useState(14029);
const [bandwidth, setBandwidth] = useState(24.5);
const [targetKillId, setTargetKillId] = useState("");
const canvasRef = useRef<HTMLCanvasElement>(null);
const router = useRouter();
const [isCheckingAuth, setIsCheckingAuth] = useState(true);
useEffect(() => {
const checkAuth = async () => {
try {
const res = await fetch("/api/superadmin/supreme-dashboard");
const data = await res.json();
if (data.error) {
router.push("/");
} else {
setIsCheckingAuth(false);
}
} catch (err) {
router.push("/");
}
};
checkAuth();
}, [router]);
// Move conditional return AFTER all hooks to comply with React Rules of Hooks
// (early return is placed after line 124 instead)
// WebGL-Style Waveform Simulation using Canvas for Zero-Error absolute performance
useEffect(() => {
const canvas = canvasRef.current;
if (!canvas) return;
const ctx = canvas.getContext("2d");
if (!ctx) return;
let animationFrameId: number;
let time = 0;
const resize = () => {
canvas.width = canvas.clientWidth;
canvas.height = canvas.clientHeight;
};
window.addEventListener("resize", resize);
resize();
const draw = () => {
time += 0.05;
ctx.clearRect(0, 0, canvas.width, canvas.height);
const width = canvas.width;
const height = canvas.height;
const centerY = height / 2;
// Draw multiple waves
for (let i = 0; i < 3; i++) {
ctx.beginPath();
ctx.moveTo(0, centerY);
for (let x = 0; x < width; x++) {
const frequency = 0.01 + (i * 0.005);
const amplitude = 30 + (i * 15) + Math.sin(time + x * 0.02) * 10;
const y = Math.sin(x * frequency + time + i * 2) * amplitude;
ctx.lineTo(x, centerY + y);
}
// WhatsApp Emerald to Zoom Blue Gradient
const gradient = ctx.createLinearGradient(0, 0, width, 0);
gradient.addColorStop(0, "rgba(37, 211, 102, 0.4)"); // WhatsApp Green
gradient.addColorStop(0.5, "rgba(11, 92, 255, 0.6)"); // Zoom Blue
gradient.addColorStop(1, "rgba(255, 0, 128, 0.3)"); // XCU Magenta
ctx.strokeStyle = gradient;
ctx.lineWidth = 2 + i;
ctx.stroke();
}
animationFrameId = requestAnimationFrame(draw);
};
draw();
return () => {
window.removeEventListener("resize", resize);
cancelAnimationFrame(animationFrameId);
};
}, []);
// Fetch real telemetry data from PANOPTICON backend
useEffect(() => {
const fetchTelemetry = async () => {
try {
const res = await fetch("/api/superadmin/telemetry");
if (res.ok) {
const data = await res.json();
// Transform quantum logs into UI log format
const formattedLogs = data.logs.map((log: any) => ({
id: log.id,
time: log.nanoTimestamp ? new Date(log.nanoTimestamp).toISOString().split("T")[1].substring(0, 8) : "00:00:00",
action: log.action,
target: log.targetId,
status: log.action.includes("KILL") ? "TERMINATED" : "OK",
}));
setLogs(formattedLogs);
setActiveConnections(data.stats.totalTelemetryRecords);
setBandwidth(parseFloat(data.stats.totalBandwidthBytes) / 1024 / 1024 / 1024);
}
} catch (err) {
console.error("PANOPTICON Sync Error:", err);
}
};
fetchTelemetry();
const interval = setInterval(fetchTelemetry, 2000);
return () => clearInterval(interval);
}, []);
// Auth guard AFTER all hooks (React Rules of Hooks compliance)
if (isCheckingAuth) {
return <div className="min-h-screen bg-black flex items-center justify-center text-purple-500 font-mono text-sm">INITIALIZING SECURE CONNECTION...</div>;
}
const executeLiveKill = async () => {
if (!targetKillId) return;
// Spectactular Kill Trigger Effect
document.body.style.animation = "shake 0.5s cubic-bezier(.36,.07,.19,.97) both";
setTimeout(() => document.body.style.animation = "", 500);
const res = await fetch("/api/telemetry/live-kill", {
method: "POST",
headers: { "Content-Type": "application/json" },
body: JSON.stringify({
targetType: "USER",
targetId: targetKillId,
reason: "SUPREME_ADMIN_OVERRIDE"
})
});
if (res.ok) {
setLogs(prev => [{
id: "KILL_" + Date.now(),
time: new Date().toISOString().split("T")[1].substring(0, 8),
action: "LIVE_KILL_EXECUTE",
target: targetKillId,
status: "TERMINATED"
}, ...prev]);
setTargetKillId("");
alert("XCU MoQ Ejection Berhasil. Sesi Dihanguskan.");
} else {
alert("Akses Ditolak: Quantum Jurisdiction");
}
};
return (
<div className="min-h-screen bg-[#050914] text-gray-200 font-sans selection:bg-blue-500/30 overflow-hidden relative">
<style dangerouslySetInnerHTML={{__html: `
@keyframes shake {
10%, 90% { transform: translate3d(-1px, 0, 0); }
20%, 80% { transform: translate3d(2px, 0, 0); }
30%, 50%, 70% { transform: translate3d(-4px, 0, 0); }
40%, 60% { transform: translate3d(4px, 0, 0); }
}
.matrix-bg {
background-image:
linear-gradient(rgba(11, 92, 255, 0.05) 1px, transparent 1px),
linear-gradient(90deg, rgba(37, 211, 102, 0.05) 1px, transparent 1px);
background-size: 30px 30px;
}
`}} />
{/* Decorative Grid */}
<div className="absolute inset-0 matrix-bg pointer-events-none z-0"></div>
<div className="relative z-10 p-6 md:p-10 max-w-7xl mx-auto flex flex-col gap-8 h-screen">
{/* HEADER */}
<header className="flex justify-between items-center border-b border-white/10 pb-6">
<div>
<h1 className="text-3xl font-black tracking-tighter text-transparent bg-clip-text bg-gradient-to-r from-[#25D366] via-[#0b5cff] to-[#ff0080]">
PANOPTICON OMNISCIENT
</h1>
<p className="text-sm font-mono text-blue-400/80 uppercase tracking-widest mt-1">
Supreme Admin XCU Level 1 Akses Absolut
</p>
</div>
<div className="flex gap-4 items-center">
<div className="px-4 py-2 rounded-lg bg-green-500/10 border border-green-500/30 flex items-center gap-2 shadow-[0_0_15px_rgba(37,211,102,0.2)]">
<span className="w-2 h-2 rounded-full bg-green-500 animate-pulse"></span>
<span className="font-mono text-green-400 text-sm font-bold">CLUSTER STABLE</span>
</div>
<a href="/supreme-admin" className="px-4 py-2 rounded-lg bg-white/5 hover:bg-white/10 border border-white/10 text-sm font-bold transition-all">
Tutup Dasbor
</a>
</div>
</header>
{/* TOP METRICS */}
<div className="grid grid-cols-1 md:grid-cols-3 gap-6">
<div className="bg-black/40 backdrop-blur-xl border border-white/5 rounded-2xl p-6 relative overflow-hidden group">
<div className="absolute inset-0 bg-gradient-to-br from-blue-500/10 to-transparent opacity-0 group-hover:opacity-100 transition-all duration-500"></div>
<p className="text-sm font-mono text-gray-500 uppercase">XCU Active MoC Streams</p>
<p className="text-5xl font-black mt-2 text-white font-mono tracking-tighter">
{activeConnections.toLocaleString()}
</p>
</div>
<div className="bg-black/40 backdrop-blur-xl border border-white/5 rounded-2xl p-6 relative overflow-hidden group">
<div className="absolute inset-0 bg-gradient-to-br from-green-500/10 to-transparent opacity-0 group-hover:opacity-100 transition-all duration-500"></div>
<p className="text-sm font-mono text-gray-500 uppercase">Global Bandwidth (Gbps)</p>
<p className="text-5xl font-black mt-2 text-transparent bg-clip-text bg-gradient-to-r from-white to-gray-500 font-mono tracking-tighter">
{bandwidth.toFixed(2)}
</p>
</div>
<div className="bg-black/40 backdrop-blur-xl border border-red-500/20 rounded-2xl p-6 relative shadow-[0_0_30px_rgba(255,0,0,0.1)]">
<p className="text-sm font-bold text-red-500 uppercase mb-4 flex items-center gap-2">
<svg className="w-4 h-4" fill="currentColor" viewBox="0 0 24 24"><path d="M12 2L1 21h22M12 6l7.5 13h-15M11 10h2v5h-2M11 16h2v2h-2"/></svg>
GUILLOTINE PANEL
</p>
<div className="flex gap-2">
<input
type="text"
value={targetKillId}
onChange={(e) => setTargetKillId(e.target.value)}
placeholder="Target ID (e.g. tsm@pc24.id)"
className="w-full bg-black/50 border border-red-500/30 rounded-lg px-3 py-2 text-sm font-mono focus:outline-none focus:border-red-500 text-white placeholder-gray-600"
/>
<button
onClick={executeLiveKill}
className="bg-red-600 hover:bg-red-500 text-white px-4 py-2 rounded-lg font-bold text-sm tracking-widest shadow-[0_0_15px_rgba(255,0,0,0.4)] transition-all hover:scale-105 active:scale-95"
>
KILL
</button>
</div>
</div>
</div>
{/* VISUALIZATION & LOGS */}
<div className="grid grid-cols-1 lg:grid-cols-3 gap-6 flex-1 min-h-0">
{/* Waveform Canvas */}
<div className="lg:col-span-2 bg-black/40 backdrop-blur-xl border border-white/5 rounded-2xl relative overflow-hidden flex flex-col">
<div className="p-4 border-b border-white/5 flex justify-between items-center bg-black/40">
<p className="text-xs font-mono text-blue-400">LIVE XCU eBPF TELEMETRY (ZOOM x WA MULTIPLEXING)</p>
<div className="flex gap-2">
<span className="w-2 h-2 rounded-full bg-blue-500"></span>
<span className="w-2 h-2 rounded-full bg-green-500"></span>
</div>
</div>
<div className="flex-1 relative">
<canvas ref={canvasRef} className="absolute inset-0 w-full h-full opacity-80 mix-blend-screen"></canvas>
</div>
</div>
{/* Hacker Log Stream */}
<div className="bg-black/60 backdrop-blur-xl border border-white/5 rounded-2xl flex flex-col overflow-hidden relative shadow-inner">
<div className="absolute inset-x-0 top-0 h-8 bg-gradient-to-b from-blue-500/10 to-transparent z-10 pointer-events-none"></div>
<div className="p-4 border-b border-white/5 bg-[#0a0a0a]">
<p className="text-xs font-mono text-gray-400 flex items-center gap-2">
<span className="w-2 h-2 rounded-full bg-red-500 animate-ping"></span>
QUANTUM TRACE LOGS
</p>
</div>
<div className="flex-1 overflow-y-auto p-4 font-mono text-xs flex flex-col gap-1">
{logs.map((log, i) => (
<div key={log.id + i} className="flex gap-3 hover:bg-white/5 px-2 py-1 rounded transition-colors group">
<span className="text-gray-600">[{log.time}]</span>
<span className={log.status === 'TERMINATED' ? 'text-red-500 font-bold' : log.status === 'WARN' ? 'text-yellow-500' : 'text-green-400'}>
{log.action}
</span>
<span className="text-blue-300 ml-auto opacity-70 group-hover:opacity-100">{log.target}</span>
</div>
))}
</div>
</div>
</div>
</div>
</div>
);
}