431 lines
23 KiB
HTML
431 lines
23 KiB
HTML
<!DOCTYPE html>
|
|
<html lang="de">
|
|
<head>
|
|
<meta charset="UTF-8">
|
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
|
<title>KalliLab Control Panel</title>
|
|
<link href="https://fonts.googleapis.com/css2?family=Orbitron:wght@400;600;700&family=Share+Tech+Mono&family=Exo+2:wght@300;400;600&display=swap" rel="stylesheet">
|
|
<style>
|
|
:root {
|
|
--bg: #060b09;
|
|
--bg2: #0a1210;
|
|
--card: rgba(8, 18, 15, 0.88);
|
|
--card-border: rgba(0, 220, 140, 0.18);
|
|
--card-hover: rgba(0, 220, 140, 0.28);
|
|
--teal: #00dc8c;
|
|
--teal-dim: #009e65;
|
|
--teal-bright: #00ffaa;
|
|
--teal-glow: rgba(0, 220, 140, 0.4);
|
|
--text: #b8d4cc;
|
|
--text-dim: #5a8a7a;
|
|
--text-bright: #d8f0e8;
|
|
--clr-warn: #ff4466;
|
|
--red: #ff4466;
|
|
--yellow: #ffcc44;
|
|
--blue: #44aaff;
|
|
--graph: #00cc88;
|
|
--font-display: 'Orbitron', monospace;
|
|
--font-mono: 'Share Tech Mono', monospace;
|
|
--font-body: 'Exo 2', sans-serif;
|
|
}
|
|
* { margin: 0; padding: 0; box-sizing: border-box; }
|
|
body {
|
|
background: var(--bg);
|
|
color: var(--text);
|
|
font-family: var(--font-body);
|
|
min-height: 100vh;
|
|
overflow-x: hidden;
|
|
position: relative;
|
|
}
|
|
body::before {
|
|
content: '';
|
|
position: fixed;
|
|
inset: 0;
|
|
background-image:
|
|
linear-gradient(rgba(0, 220, 140, 0.025) 1px, transparent 1px),
|
|
linear-gradient(90deg, rgba(0, 220, 140, 0.025) 1px, transparent 1px);
|
|
background-size: 40px 40px;
|
|
z-index: 0;
|
|
pointer-events: none;
|
|
}
|
|
@keyframes scanline {
|
|
0% { top: -5%; }
|
|
100% { top: 105%; }
|
|
}
|
|
.scanline {
|
|
position: fixed;
|
|
left: 0; right: 0;
|
|
height: 2px;
|
|
background: linear-gradient(90deg, transparent, rgba(0,220,140,0.06), rgba(0,220,140,0.10), rgba(0,220,140,0.06), transparent);
|
|
z-index: 1;
|
|
animation: scanline 8s linear infinite;
|
|
pointer-events: none;
|
|
}
|
|
.wrapper {
|
|
position: relative;
|
|
z-index: 2;
|
|
max-width: 1340px;
|
|
margin: 0 auto;
|
|
padding: 0 10px 32px;
|
|
}
|
|
.header {
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: space-between;
|
|
padding: 10px 0 8px;
|
|
border-bottom: 1px solid var(--card-border);
|
|
margin-bottom: 10px;
|
|
gap: 12px;
|
|
}
|
|
.header-logo { display: flex; align-items: center; gap: 10px; }
|
|
.logo-text {
|
|
font-family: var(--font-display);
|
|
font-size: 16px;
|
|
font-weight: 700;
|
|
color: var(--teal-bright);
|
|
text-shadow: 0 0 20px var(--teal-glow);
|
|
letter-spacing: 2px;
|
|
}
|
|
.logo-sub {
|
|
font-family: var(--font-mono);
|
|
font-size: 9px;
|
|
color: var(--text-dim);
|
|
letter-spacing: 2px;
|
|
text-transform: uppercase;
|
|
}
|
|
.header-center { display: flex; flex-direction: column; align-items: center; gap: 2px; }
|
|
.overall-status { font-family: var(--font-mono); font-size: 10px; letter-spacing: 2px; color: var(--teal-dim); }
|
|
.status-indicator {
|
|
display: flex; align-items: center; gap: 6px;
|
|
font-family: var(--font-display); font-size: 11px;
|
|
color: var(--teal-bright); text-shadow: 0 0 10px var(--teal-glow);
|
|
}
|
|
.status-dot-main { width: 8px; height: 8px; border-radius: 50%; background: var(--teal); box-shadow: 0 0 8px var(--teal-glow); }
|
|
.header-right { display: flex; flex-direction: column; align-items: flex-end; gap: 2px; }
|
|
.clock {
|
|
font-family: var(--font-display); font-size: 24px; font-weight: 700;
|
|
color: var(--teal-bright); text-shadow: 0 0 20px var(--teal-glow), 0 0 40px rgba(0,220,140,0.2);
|
|
letter-spacing: 2px;
|
|
}
|
|
.date-str { font-family: var(--font-mono); font-size: 10px; color: var(--text-dim); text-align: right; }
|
|
#last-updated { font-family: var(--font-mono); font-size: 9px; color: var(--text-dim); text-align: right; }
|
|
.section-header {
|
|
display: flex; align-items: center; gap: 8px; margin-bottom: 6px;
|
|
font-family: var(--font-display); font-size: 8px; font-weight: 600;
|
|
letter-spacing: 3px; text-transform: uppercase; color: var(--teal-dim);
|
|
}
|
|
.section-header::after { content: ''; flex: 1; height: 1px; background: linear-gradient(90deg, var(--card-border), transparent); }
|
|
.widget-row { display: grid; gap: 8px; margin-bottom: 8px; }
|
|
.row-5 { grid-template-columns: repeat(5, 1fr); }
|
|
.row-4 { grid-template-columns: repeat(4, 1fr); }
|
|
.row-3 { grid-template-columns: repeat(3, 1fr); }
|
|
.row-2 { grid-template-columns: repeat(2, 1fr); }
|
|
.row-2-1 { grid-template-columns: 2fr 1fr; }
|
|
.row-1-2 { grid-template-columns: 1fr 2fr; }
|
|
.row-3-2 { grid-template-columns: 3fr 2fr; }
|
|
.card {
|
|
background: rgba(6, 14, 11, 0.78);
|
|
border: 1px solid var(--card-border);
|
|
border-radius: 8px;
|
|
padding: 8px 10px;
|
|
transition: border-color 0.2s, box-shadow 0.2s;
|
|
backdrop-filter: blur(14px) saturate(1.4);
|
|
-webkit-backdrop-filter: blur(14px) saturate(1.4);
|
|
}
|
|
.card:hover { border-color: rgba(0,220,140,0.32); box-shadow: 0 0 16px rgba(0,220,140,0.07), inset 0 0 20px rgba(0,220,140,0.02); }
|
|
.card-title {
|
|
font-family: var(--font-display); font-size: 8px; font-weight: 600;
|
|
letter-spacing: 2px; text-transform: uppercase; color: var(--teal-dim);
|
|
margin-bottom: 6px; display: flex; align-items: center; justify-content: space-between; gap: 6px;
|
|
}
|
|
.card-title-left { display: flex; align-items: center; gap: 6px; }
|
|
.card-title .dot { width: 5px; height: 5px; border-radius: 50%; background: var(--teal); box-shadow: 0 0 5px var(--teal-glow); flex-shrink: 0; }
|
|
.stats-grid { display: flex; justify-content: space-around; gap: 6px; flex-wrap: wrap; }
|
|
.stat-block { text-align: center; min-width: 40px; }
|
|
.stat-num { font-family: var(--font-display); font-size: 17px; font-weight: 700; color: var(--teal-bright); text-shadow: 0 0 10px var(--teal-glow); line-height: 1.1; }
|
|
.stat-num.dim { color: var(--teal-dim); text-shadow: none; font-size: 14px; }
|
|
.stat-num.warn { color: var(--yellow); text-shadow: 0 0 10px rgba(255,204,68,0.4); }
|
|
.stat-num.danger { color: var(--red); text-shadow: 0 0 10px rgba(255,68,102,0.4); }
|
|
.stat-num.blue { color: var(--blue); text-shadow: 0 0 10px rgba(68,170,255,0.4); }
|
|
.stat-label { font-family: var(--font-mono); font-size: 8px; color: var(--text-dim); letter-spacing: 1px; text-transform: uppercase; margin-top: 1px; }
|
|
.status-pill { font-family: var(--font-mono); font-size: 7px; letter-spacing: 1px; padding: 1px 5px; border-radius: 3px; font-weight: 700; }
|
|
.pill-online { background: rgba(0,220,140,0.12); color: var(--teal); border: 1px solid rgba(0,220,140,0.3); }
|
|
.pill-offline { background: rgba(255,68,102,0.1); color: var(--red); border: 1px solid rgba(255,68,102,0.25); }
|
|
.pill-degraded { background: rgba(255,204,68,0.1); color: var(--yellow); border: 1px solid rgba(255,204,68,0.25); }
|
|
.progress-wrap { margin-top: 5px; }
|
|
.progress-label { display: flex; justify-content: space-between; font-family: var(--font-mono); font-size: 9px; color: var(--text-dim); margin-bottom: 2px; }
|
|
.progress-bar { height: 3px; background: rgba(0,220,140,0.1); border-radius: 2px; overflow: hidden; }
|
|
.progress-fill { height: 100%; background: linear-gradient(90deg, var(--teal-dim), var(--teal-bright)); border-radius: 2px; box-shadow: 0 0 5px var(--teal-glow); transition: width 1s ease; }
|
|
.progress-fill.warn { background: linear-gradient(90deg, #cc8800, var(--yellow)); }
|
|
.progress-fill.danger { background: linear-gradient(90deg, #cc2244, var(--red)); }
|
|
.service-header { display: flex; align-items: center; gap: 7px; margin-bottom: 7px; }
|
|
.service-icon { width: 24px; height: 24px; border-radius: 5px; display: flex; align-items: center; justify-content: center; font-size: 14px; flex-shrink: 0; }
|
|
.service-name { font-family: var(--font-display); font-size: 9px; font-weight: 600; color: var(--text-bright); letter-spacing: 1px; flex: 1; }
|
|
.service-version { font-family: var(--font-mono); font-size: 8px; color: var(--text-dim); }
|
|
.sys-grid { display: grid; grid-template-columns: 1fr 1fr; gap: 1px 10px; font-family: var(--font-mono); font-size: 10px; }
|
|
.sys-row { display: flex; justify-content: space-between; padding: 1px 0; }
|
|
.sys-key { color: var(--text-dim); }
|
|
.sys-val { color: var(--teal); }
|
|
.disk-header { display: flex; justify-content: space-between; align-items: baseline; margin-bottom: 3px; }
|
|
.disk-name { font-family: var(--font-display); font-size: 9px; color: var(--text-bright); letter-spacing: 1px; }
|
|
.disk-usage { font-family: var(--font-mono); font-size: 9px; color: var(--teal); }
|
|
.disk-sub { font-family: var(--font-mono); font-size: 8px; color: var(--text-dim); margin-bottom: 4px; }
|
|
.scrutiny-row { display: flex; align-items: center; gap: 6px; padding: 2px 0; font-family: var(--font-mono); font-size: 9px; border-bottom: 1px solid rgba(0,220,140,0.05); }
|
|
.scrutiny-row:last-child { border-bottom: none; }
|
|
.disk-icon { font-size: 10px; font-weight: bold; width: 12px; text-align: center; }
|
|
.disk-ok { color: var(--teal); }
|
|
.disk-fail { color: var(--red); }
|
|
.disk-unk { color: var(--text-dim); }
|
|
.disk-name-col { flex: 1; color: var(--text-bright); min-width: 0; overflow: hidden; text-overflow: ellipsis; white-space: nowrap; }
|
|
.disk-model { color: var(--text-dim); font-size: 8px; max-width: 90px; overflow: hidden; text-overflow: ellipsis; white-space: nowrap; }
|
|
.disk-temp { color: var(--teal-dim); font-size: 8px; white-space: nowrap; }
|
|
.scrutiny-offline { font-family: var(--font-mono); font-size: 9px; color: var(--text-dim); padding: 4px 0; }
|
|
.uk-monitor-row { display: flex; align-items: center; gap: 6px; margin-bottom: 3px; }
|
|
.uk-monitor-name { font-family: var(--font-mono); font-size: 8px; color: var(--text-dim); min-width: 70px; max-width: 90px; overflow: hidden; text-overflow: ellipsis; white-space: nowrap; }
|
|
.uk-bar { display: flex; gap: 1px; flex: 1; }
|
|
.hb-seg { height: 7px; flex: 1; border-radius: 1px; }
|
|
.hb-up { background: var(--teal); box-shadow: 0 0 3px var(--teal-glow); opacity: 0.85; }
|
|
.hb-down { background: var(--red); box-shadow: 0 0 3px rgba(255,68,102,0.4); opacity: 0.85; }
|
|
.uk-down-name { display: block; font-family: var(--font-mono); font-size: 8px; color: var(--red); padding: 1px 0; }
|
|
.status-dot { display: inline-block; width: 7px; height: 7px; border-radius: 50%; flex-shrink: 0; }
|
|
.dot-ok { background: var(--teal); box-shadow: 0 0 5px var(--teal-glow); }
|
|
.dot-err { background: var(--red); box-shadow: 0 0 5px rgba(255,68,102,0.4); }
|
|
.dot-unk { background: var(--text-dim); }
|
|
.adguard-bar-wrap { margin-top: 5px; }
|
|
.adguard-bar { height: 3px; background: rgba(0,220,140,0.1); border-radius: 2px; overflow: hidden; position: relative; }
|
|
.adguard-bar-fill { height: 100%; background: linear-gradient(90deg, var(--teal-dim), var(--teal-bright)); border-radius: 2px; box-shadow: 0 0 5px var(--teal-glow); width: 0%; transition: width 1s ease; }
|
|
.net-health-grid { display: grid; grid-template-columns: 1fr 1fr; gap: 8px; }
|
|
#quick-access-grid { display: grid; grid-template-columns: repeat(auto-fill, minmax(100px, 1fr)); gap: 6px; }
|
|
.quick-tile { display: flex; flex-direction: column; align-items: center; gap: 4px; padding: 8px 6px 7px; background: rgba(6, 14, 11, 0.72); border: 1px solid var(--card-border); border-radius: 7px; cursor: pointer; transition: all 0.18s; text-decoration: none; color: var(--text); backdrop-filter: blur(14px); -webkit-backdrop-filter: blur(14px); }
|
|
.quick-tile:hover { border-color: rgba(0,220,140,0.4); background: rgba(0,220,140,0.05); transform: translateY(-2px); box-shadow: 0 4px 18px rgba(0,220,140,0.10); }
|
|
.quick-tile-icon { font-size: 18px; line-height: 1; }
|
|
.quick-tile-label { font-family: var(--font-mono); font-size: 9px; color: var(--text-dim); text-align: center; line-height: 1.2; }
|
|
.quick-tile:hover .quick-tile-label { color: var(--teal); }
|
|
.docker-row { display: flex; gap: 6px; font-family: var(--font-mono); font-size: 9px; margin-top: 4px; flex-wrap: wrap; }
|
|
.docker-chip { padding: 2px 6px; border-radius: 3px; background: rgba(0,220,140,0.07); border: 1px solid rgba(0,220,140,0.15); color: var(--teal-dim); }
|
|
.docker-chip.running { color: var(--teal); border-color: rgba(0,220,140,0.3); }
|
|
.docker-chip.stopped { color: var(--yellow); border-color: rgba(255,204,68,0.3); background: rgba(255,204,68,0.06); }
|
|
.docker-chip.unhealthy { color: var(--red); border-color: rgba(255,68,102,0.3); background: rgba(255,68,102,0.06); }
|
|
@media (max-width: 1100px) {
|
|
.row-5 { grid-template-columns: repeat(3, 1fr); }
|
|
.row-4 { grid-template-columns: repeat(2, 1fr); }
|
|
}
|
|
@media (max-width: 780px) {
|
|
.row-5, .row-4, .row-3 { grid-template-columns: repeat(2, 1fr); }
|
|
.row-2, .row-2-1, .row-1-2, .row-3-2 { grid-template-columns: 1fr; }
|
|
.net-health-grid { grid-template-columns: 1fr; }
|
|
#quick-access-grid { grid-template-columns: repeat(auto-fill, minmax(80px, 1fr)); }
|
|
}
|
|
</style>
|
|
</head>
|
|
<body>
|
|
<div class="scanline"></div>
|
|
<div class="wrapper">
|
|
|
|
<!-- HEADER -->
|
|
<header class="header" id="header">
|
|
<div class="header-logo">
|
|
<div>
|
|
<div class="logo-text">KALLILAB</div>
|
|
<div class="logo-sub">Control Panel</div>
|
|
</div>
|
|
</div>
|
|
<div class="header-center">
|
|
<div class="overall-status">SYSTEM STATUS</div>
|
|
<div class="status-indicator">
|
|
<span class="status-dot-main" id="overall-dot"></span>
|
|
<span id="overall-status-text">LOADING</span>
|
|
</div>
|
|
</div>
|
|
<div class="header-right">
|
|
<div class="clock" id="clock">--:--:--</div>
|
|
<div class="date-str" id="date-str">---</div>
|
|
<div id="last-updated">never updated</div>
|
|
</div>
|
|
</header>
|
|
|
|
<!-- SYSTEM STATS ROW -->
|
|
<div class="section-header"><span>⬡</span> SYSTEM</div>
|
|
<div class="widget-row row-5" id="stats-row" style="margin-bottom:8px;">
|
|
<div class="card">
|
|
<div class="card-title"><span class="dot"></span>CPU</div>
|
|
<div class="stats-grid">
|
|
<div class="stat-block"><div class="stat-num" id="cpu-percent">—</div><div class="stat-label">Usage %</div></div>
|
|
<div class="stat-block"><div class="stat-num dim" id="cpu-cores">—</div><div class="stat-label">Cores</div></div>
|
|
<div class="stat-block"><div class="stat-num dim" id="cpu-load">—</div><div class="stat-label">Load 5m</div></div>
|
|
</div>
|
|
</div>
|
|
<div class="card">
|
|
<div class="card-title"><span class="dot"></span>MEMORY</div>
|
|
<div class="stats-grid">
|
|
<div class="stat-block"><div class="stat-num" id="ram-percent">—</div><div class="stat-label">Usage %</div></div>
|
|
<div class="stat-block"><div class="stat-num dim" id="ram-used">—</div><div class="stat-label">Used GB</div></div>
|
|
<div class="stat-block"><div class="stat-num dim" id="ram-total">—</div><div class="stat-label">Total GB</div></div>
|
|
</div>
|
|
</div>
|
|
<div class="card">
|
|
<div class="card-title"><span class="dot"></span>NETWORK</div>
|
|
<div class="stats-grid">
|
|
<div class="stat-block"><div class="stat-num" id="net-rx">—</div><div class="stat-label">↓ Mbps</div></div>
|
|
<div class="stat-block"><div class="stat-num" id="net-tx">—</div><div class="stat-label">↑ Mbps</div></div>
|
|
</div>
|
|
</div>
|
|
<div class="card">
|
|
<div class="card-title"><span class="dot"></span>HOST</div>
|
|
<div class="stats-grid">
|
|
<div class="stat-block"><div class="stat-num" id="uptime-days">—</div><div class="stat-label">Uptime d</div></div>
|
|
<div class="stat-block"><div class="stat-num dim" id="host-platform">—</div><div class="stat-label">OS</div></div>
|
|
</div>
|
|
</div>
|
|
<div class="card">
|
|
<div class="card-title"><span class="dot"></span>DOCKER</div>
|
|
<div class="stats-grid">
|
|
<div class="stat-block"><div class="stat-num" id="docker-running">—</div><div class="stat-label">Running</div></div>
|
|
<div class="stat-block"><div class="stat-num dim" id="docker-stopped">—</div><div class="stat-label">Stopped</div></div>
|
|
<div class="stat-block"><div class="stat-num dim" id="docker-total">—</div><div class="stat-label">Total</div></div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- STORAGE + SCRUTINY ROW -->
|
|
<div class="section-header"><span>⬡</span> STORAGE & HEALTH</div>
|
|
<div style="display:grid; grid-template-columns: 1fr 280px; gap:8px; margin-bottom:8px;">
|
|
<div>
|
|
<div class="widget-row row-3" id="storage-grid">
|
|
<!-- Disk cards injected by renderer -->
|
|
</div>
|
|
</div>
|
|
<div class="card">
|
|
<div class="card-title">
|
|
<div class="card-title-left"><span class="dot"></span>SCRUTINY</div>
|
|
<span class="status-pill pill-offline" id="scrutiny-pill">OFFLINE</span>
|
|
</div>
|
|
<div class="stats-grid" style="margin-bottom:5px;">
|
|
<div class="stat-block"><div class="stat-num" id="scrutiny-total">—</div><div class="stat-label">Disks</div></div>
|
|
<div class="stat-block"><div class="stat-num" id="scrutiny-passed">—</div><div class="stat-label">Passed</div></div>
|
|
<div class="stat-block"><div class="stat-num danger" id="scrutiny-failed">—</div><div class="stat-label">Failed</div></div>
|
|
</div>
|
|
<div id="scrutiny-list"></div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- SERVICE WIDGETS ROW 1 -->
|
|
<div class="section-header"><span>⬡</span> SERVICES</div>
|
|
<div class="widget-row row-3" style="margin-bottom:8px;">
|
|
<div class="card">
|
|
<div class="card-title">
|
|
<div class="card-title-left">
|
|
<span class="service-icon" style="background:rgba(65,105,225,0.15);">🏠</span>
|
|
<span class="service-name">HOME ASSISTANT</span>
|
|
</div>
|
|
<span class="status-pill pill-offline" id="ha-pill">OFFLINE</span>
|
|
</div>
|
|
<div class="stats-grid">
|
|
<div class="stat-block"><div class="stat-num" id="ha-lights">—</div><div class="stat-label">Lights</div></div>
|
|
<div class="stat-block"><div class="stat-num" id="ha-climate">—</div><div class="stat-label">Climate</div></div>
|
|
<div class="stat-block"><div class="stat-num" id="ha-doors">—</div><div class="stat-label">Doors</div></div>
|
|
<div class="stat-block"><div class="stat-num danger" id="ha-alerts">—</div><div class="stat-label">Alerts</div></div>
|
|
</div>
|
|
<div style="margin-top:4px; font-family:var(--font-mono); font-size:8px; color:var(--text-dim);" id="ha-version"></div>
|
|
</div>
|
|
<div class="card">
|
|
<div class="card-title">
|
|
<div class="card-title-left">
|
|
<span class="service-icon" style="background:rgba(0,220,140,0.12);">🐻</span>
|
|
<span class="service-name">UPTIME KUMA</span>
|
|
</div>
|
|
<span class="status-pill pill-offline" id="uk-pill">OFFLINE</span>
|
|
</div>
|
|
<div class="stats-grid" style="margin-bottom:5px;">
|
|
<div class="stat-block"><div class="stat-num" id="uk-up">—</div><div class="stat-label">Up</div></div>
|
|
<div class="stat-block"><div class="stat-num danger" id="uk-down">—</div><div class="stat-label">Down</div></div>
|
|
<div class="stat-block"><div class="stat-num" id="uk-uptime">—</div><div class="stat-label">24h %</div></div>
|
|
</div>
|
|
<div id="uk-down-list"></div>
|
|
<div id="uk-bars"></div>
|
|
</div>
|
|
<div class="card">
|
|
<div class="card-title">
|
|
<div class="card-title-left">
|
|
<span class="service-icon" style="background:rgba(255,204,68,0.12);">📷</span>
|
|
<span class="service-name">IMMICH</span>
|
|
</div>
|
|
<span class="status-pill pill-offline" id="immich-pill">OFFLINE</span>
|
|
</div>
|
|
<div class="stats-grid">
|
|
<div class="stat-block"><div class="stat-num" id="immich-photos">—</div><div class="stat-label">Photos</div></div>
|
|
<div class="stat-block"><div class="stat-num" id="immich-videos">—</div><div class="stat-label">Videos</div></div>
|
|
<div class="stat-block"><div class="stat-num dim" id="immich-storage">—</div><div class="stat-label">Storage</div></div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- SERVICE WIDGETS ROW 2 -->
|
|
<div class="widget-row row-3" style="margin-bottom:8px;">
|
|
<div class="card">
|
|
<div class="card-title">
|
|
<div class="card-title-left">
|
|
<span class="service-icon" style="background:rgba(68,170,255,0.12);">💾</span>
|
|
<span class="service-name">BACKREST</span>
|
|
<span class="status-dot dot-unk" id="backrest-status-dot" title="unknown"></span>
|
|
</div>
|
|
<span class="status-pill pill-offline" id="backrest-pill">OFFLINE</span>
|
|
</div>
|
|
<div class="stats-grid">
|
|
<div class="stat-block"><div class="stat-num dim" id="backrest-last">—</div><div class="stat-label">Last Backup</div></div>
|
|
<div class="stat-block"><div class="stat-num" id="backrest-repos">—</div><div class="stat-label">Repos</div></div>
|
|
<div class="stat-block"><div class="stat-num danger" id="backrest-errors">—</div><div class="stat-label">Errors</div></div>
|
|
</div>
|
|
</div>
|
|
<div class="card">
|
|
<div class="card-title">
|
|
<div class="card-title-left">
|
|
<span class="service-icon" style="background:rgba(0,220,140,0.12);">🛡️</span>
|
|
<span class="service-name">ADGUARD DNS</span>
|
|
</div>
|
|
<span class="status-pill pill-offline" id="adguard-pill">OFFLINE</span>
|
|
</div>
|
|
<div class="stats-grid" style="margin-bottom:5px;">
|
|
<div class="stat-block"><div class="stat-num" id="adguard-total">—</div><div class="stat-label">Queries</div></div>
|
|
<div class="stat-block"><div class="stat-num" id="adguard-blocked">—</div><div class="stat-label">Blocked</div></div>
|
|
<div class="stat-block"><div class="stat-num dim" id="adguard-blocked-pct">—</div><div class="stat-label">Block %</div></div>
|
|
<div class="stat-block"><div class="stat-num dim" id="adguard-latency">—</div><div class="stat-label">Latency</div></div>
|
|
</div>
|
|
<div class="adguard-bar-wrap"><div class="adguard-bar"><div class="adguard-bar-fill" id="adguard-bar-fill"></div></div></div>
|
|
</div>
|
|
<div class="card">
|
|
<div class="card-title">
|
|
<div class="card-title-left"><span class="dot"></span>SERVICES OVERVIEW</div>
|
|
<span class="status-pill pill-offline" id="services-pill">—</span>
|
|
</div>
|
|
<div class="stats-grid">
|
|
<div class="stat-block"><div class="stat-num" id="svc-online">—</div><div class="stat-label">Online</div></div>
|
|
<div class="stat-block"><div class="stat-num warn" id="svc-degraded">—</div><div class="stat-label">Degraded</div></div>
|
|
<div class="stat-block"><div class="stat-num danger" id="svc-offline">—</div><div class="stat-label">Offline</div></div>
|
|
<div class="stat-block"><div class="stat-num dim" id="svc-total">—</div><div class="stat-label">Total</div></div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- QUICK ACCESS -->
|
|
<div class="section-header"><span>⬡</span> QUICK ACCESS</div>
|
|
<div id="quick-access-grid"></div>
|
|
|
|
</div><!-- /wrapper -->
|
|
|
|
<script type="module" src="/assets/js/app.js"></script>
|
|
<script>
|
|
// Clock
|
|
function updateClock() {
|
|
const now = new Date();
|
|
const pad = n => String(n).padStart(2, '0');
|
|
document.getElementById('clock').textContent =
|
|
`${pad(now.getHours())}:${pad(now.getMinutes())}:${pad(now.getSeconds())}`;
|
|
document.getElementById('date-str').textContent =
|
|
now.toLocaleDateString('de-DE', { weekday: 'short', day: '2-digit', month: '2-digit', year: 'numeric' });
|
|
}
|
|
updateClock();
|
|
setInterval(updateClock, 1000);
|
|
</script>
|
|
</body>
|
|
</html> |