Files
homelab-infra/apps/dashboard/dashboard.html
T
Micha bbdf2ffb60 updates
Repo sauber machen
2026-04-15 13:40:03 +02:00

954 lines
43 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)); }
}
/* Density / reference lock overrides */
.wrapper { max-width: 1640px; padding: 8px 12px 20px; }
.header { padding: 10px 0 10px; margin-bottom: 8px; gap: 16px; }
.logo-text { font-size: 18px; letter-spacing: 2px; }
.logo-sub, .overall-status, .date-str, #last-updated { font-size: 10px; }
.clock { font-size: 46px; }
.section-header { margin-bottom: 5px; font-size: 9px; letter-spacing: 3px; }
.widget-row { gap: 6px; margin-bottom: 6px; }
.row-5 { grid-template-columns: repeat(5, minmax(0, 1fr)); }
.card {
max-height: 120px;
min-height: 118px;
height: 118px;
padding: 8px 10px 8px;
border-radius: 10px;
background: linear-gradient(180deg, rgba(7, 17, 14, 0.84), rgba(5, 11, 9, 0.72));
box-shadow: inset 0 1px 0 rgba(255,255,255,0.03), 0 10px 28px rgba(0,0,0,0.22), 0 0 18px rgba(0,220,140,0.05);
overflow: hidden;
}
.card-title { margin-bottom: 6px; min-height: 26px; }
.card-title-left { gap: 7px; }
.card-title .dot { width: 6px; height: 6px; }
.stats-grid {
display: grid;
grid-auto-flow: column;
grid-auto-columns: minmax(0, 1fr);
align-items: end;
justify-content: stretch;
gap: 10px;
flex-wrap: nowrap;
}
.stat-block {
min-width: 0;
text-align: center;
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
gap: 2px;
}
.stat-num {
font-size: 22px;
line-height: 0.95;
color: var(--teal-bright);
text-shadow: 0 0 12px var(--teal-glow);
}
.stat-num.dim { font-size: 18px; color: var(--text-bright); }
.stat-label { font-size: 8px; margin-top: 2px; }
#cpu-percent, #ram-percent, #net-rx, #uptime-days, #docker-running { font-size: 30px; }
.service-icon {
width: 28px;
height: 28px;
border-radius: 8px;
position: relative;
display: inline-flex;
align-items: center;
justify-content: center;
color: var(--icon-color, var(--teal-bright));
background:
radial-gradient(circle at 30% 30%, rgba(255,255,255,0.16), transparent 45%),
linear-gradient(180deg, rgba(16, 36, 29, 0.96), rgba(7, 14, 11, 0.92));
border: 1px solid rgba(255,255,255,0.06);
box-shadow:
inset 0 0 0 1px rgba(255,255,255,0.02),
0 0 16px rgba(0,220,140,0.08);
}
.service-icon::after {
content: "";
position: absolute;
inset: 0;
border-radius: inherit;
box-shadow: inset 0 0 14px rgba(0,0,0,0.18);
pointer-events: none;
}
.icon-glyph {
width: 15px;
height: 15px;
display: block;
position: relative;
color: inherit;
opacity: 0.96;
}
.icon-cpu { --icon-color: #82ffbf; }
.icon-memory { --icon-color: #98ffbf; }
.icon-network { --icon-color: #7dc8ff; }
.icon-host { --icon-color: #95ffbf; }
.icon-docker { --icon-color: #81ffc8; }
.icon-storage { --icon-color: #7effb6; }
.icon-matrix { --icon-color: #7fc9ff; }
.icon-scrutiny { --icon-color: #7effb0; }
.icon-ha { --icon-color: #8ea6ff; }
.icon-kuma { --icon-color: #89ffaf; }
.icon-immich { --icon-color: #ffd84f; }
.icon-backrest { --icon-color: #74d7ff; }
.icon-adguard { --icon-color: #66f0ba; }
.icon-services { --icon-color: #8affbe; }
.glyph-cpu {
border: 1.6px solid currentColor;
border-radius: 3px;
}
.glyph-cpu::before {
content: "";
position: absolute;
inset: 3px;
border: 1.4px solid currentColor;
border-radius: 2px;
opacity: 0.92;
}
.glyph-cpu::after {
content: "";
position: absolute;
inset: -3px;
background:
linear-gradient(currentColor,currentColor) 2px 1px / 1px 3px no-repeat,
linear-gradient(currentColor,currentColor) 7px 1px / 1px 3px no-repeat,
linear-gradient(currentColor,currentColor) 12px 1px / 1px 3px no-repeat,
linear-gradient(currentColor,currentColor) 2px calc(100% - 1px) / 1px 3px no-repeat,
linear-gradient(currentColor,currentColor) 7px calc(100% - 1px) / 1px 3px no-repeat,
linear-gradient(currentColor,currentColor) 12px calc(100% - 1px) / 1px 3px no-repeat,
linear-gradient(currentColor,currentColor) 1px 2px / 3px 1px no-repeat,
linear-gradient(currentColor,currentColor) 1px 7px / 3px 1px no-repeat,
linear-gradient(currentColor,currentColor) calc(100% - 1px) 2px / 3px 1px no-repeat,
linear-gradient(currentColor,currentColor) calc(100% - 1px) 7px / 3px 1px no-repeat;
opacity: 0.78;
}
.glyph-memory {
background:
linear-gradient(currentColor,currentColor) 2px 9px / 2px 4px no-repeat,
linear-gradient(currentColor,currentColor) 6px 6px / 2px 7px no-repeat,
linear-gradient(currentColor,currentColor) 10px 3px / 2px 10px no-repeat;
}
.glyph-memory::before {
content: "";
position: absolute;
inset: 1px;
border: 1.4px solid rgba(152,255,191,0.34);
border-radius: 3px;
}
.glyph-network {
background:
radial-gradient(circle at 2px 11px, currentColor 0 2px, transparent 2.4px),
radial-gradient(circle at 13px 11px, currentColor 0 2px, transparent 2.4px),
radial-gradient(circle at 7.5px 3px, currentColor 0 2px, transparent 2.4px),
linear-gradient(currentColor,currentColor) 7px 4px / 1.4px 7px no-repeat,
linear-gradient(32deg, transparent 44%, currentColor 45% 55%, transparent 56%) 2px 5px / 10px 7px no-repeat,
linear-gradient(-32deg, transparent 44%, currentColor 45% 55%, transparent 56%) 4px 5px / 10px 7px no-repeat;
}
.glyph-host {
border: 1.6px solid currentColor;
border-radius: 3px;
}
.glyph-host::before {
content: "";
position: absolute;
left: 3px;
right: 3px;
bottom: -2px;
height: 1.6px;
background: currentColor;
}
.glyph-host::after {
content: "";
position: absolute;
left: 5px;
right: 5px;
bottom: -5px;
height: 1.6px;
background: currentColor;
border-radius: 999px;
}
.glyph-docker {
background:
linear-gradient(currentColor,currentColor) 1px 3px / 4px 4px no-repeat,
linear-gradient(currentColor,currentColor) 6px 3px / 4px 4px no-repeat,
linear-gradient(currentColor,currentColor) 11px 3px / 4px 4px no-repeat,
linear-gradient(currentColor,currentColor) 6px 8px / 4px 4px no-repeat,
linear-gradient(currentColor,currentColor) 1px 12px / 14px 1.5px no-repeat;
border-radius: 3px;
}
.glyph-storage {
background:
linear-gradient(currentColor,currentColor) 1px 5px / 13px 1.6px no-repeat,
linear-gradient(currentColor,currentColor) 1px 8px / 13px 1.6px no-repeat,
linear-gradient(currentColor,currentColor) 1px 11px / 13px 1.6px no-repeat;
}
.glyph-storage::before {
content: "";
position: absolute;
inset: 1px 1px 2px;
border: 1.4px solid rgba(126,255,182,0.36);
border-radius: 4px;
}
.glyph-matrix {
background:
linear-gradient(currentColor,currentColor) 1px 1px / 5px 5px no-repeat,
linear-gradient(currentColor,currentColor) 9px 1px / 5px 5px no-repeat,
linear-gradient(currentColor,currentColor) 1px 9px / 5px 5px no-repeat,
linear-gradient(currentColor,currentColor) 9px 9px / 5px 5px no-repeat;
opacity: 0.95;
}
.glyph-scrutiny {
border: 1.6px solid currentColor;
border-radius: 50%;
}
.glyph-scrutiny::before {
content: "";
position: absolute;
inset: 3px;
border: 1.4px solid currentColor;
border-radius: 50%;
}
.glyph-scrutiny::after {
content: "";
position: absolute;
left: 7px;
top: 1px;
width: 1.4px;
height: 13px;
background: currentColor;
box-shadow: -6px 6px 0 -5px currentColor, 6px 6px 0 -5px currentColor;
opacity: 0.9;
}
.glyph-home {
background:
linear-gradient(-35deg, transparent 45%, currentColor 46% 56%, transparent 57%) 0 1px / 8px 7px no-repeat,
linear-gradient(35deg, transparent 45%, currentColor 46% 56%, transparent 57%) 7px 1px / 8px 7px no-repeat,
linear-gradient(currentColor,currentColor) 3px 7px / 9px 6px no-repeat;
}
.glyph-kuma {
background:
linear-gradient(currentColor,currentColor) 1px 8px / 3px 1.5px no-repeat,
linear-gradient(55deg, transparent 43%, currentColor 44% 56%, transparent 57%) 3px 6px / 4px 5px no-repeat,
linear-gradient(-55deg, transparent 43%, currentColor 44% 56%, transparent 57%) 6px 4px / 4px 7px no-repeat,
linear-gradient(55deg, transparent 43%, currentColor 44% 56%, transparent 57%) 9px 6px / 4px 5px no-repeat,
linear-gradient(currentColor,currentColor) 12px 8px / 3px 1.5px no-repeat;
}
.glyph-image {
border: 1.5px solid currentColor;
border-radius: 3px;
}
.glyph-image::before {
content: "";
position: absolute;
left: 2px;
right: 2px;
bottom: 2px;
height: 5px;
background:
linear-gradient(140deg, transparent 35%, currentColor 36% 48%, transparent 49%) 0 0 / 8px 5px no-repeat,
linear-gradient(45deg, transparent 32%, currentColor 33% 45%, transparent 46%) 5px 0 / 8px 5px no-repeat;
}
.glyph-image::after {
content: "";
position: absolute;
right: 2px;
top: 2px;
width: 3px;
height: 3px;
border-radius: 50%;
background: currentColor;
}
.glyph-backrest {
border: 1.5px solid currentColor;
border-radius: 3px;
}
.glyph-backrest::before {
content: "";
position: absolute;
left: 2px;
right: 2px;
top: 3px;
height: 2px;
background: currentColor;
box-shadow: 0 4px 0 currentColor;
}
.glyph-backrest::after {
content: "";
position: absolute;
left: 5px;
right: 5px;
bottom: -2px;
height: 1.5px;
background: currentColor;
}
.glyph-shield {
background:
linear-gradient(currentColor,currentColor) 7px 2px / 1.8px 9px no-repeat;
clip-path: polygon(50% 0%, 88% 16%, 88% 50%, 50% 100%, 12% 50%, 12% 16%);
background-color: transparent;
border: 1.5px solid currentColor;
}
.glyph-services {
background:
radial-gradient(circle at 2px 7px, currentColor 0 2px, transparent 2.4px),
radial-gradient(circle at 13px 2px, currentColor 0 2px, transparent 2.4px),
radial-gradient(circle at 13px 12px, currentColor 0 2px, transparent 2.4px),
linear-gradient(currentColor,currentColor) 4px 6px / 7px 1.4px no-repeat,
linear-gradient(currentColor,currentColor) 10px 4px / 1.4px 6px no-repeat;
}
.service-name { font-size: 10px; letter-spacing: 1.4px; }
.status-pill {
width: 10px;
height: 10px;
min-width: 10px;
border-radius: 999px;
padding: 0;
font-size: 0;
border: none;
display: inline-flex;
align-items: center;
justify-content: center;
box-shadow: 0 0 12px rgba(0,0,0,0.18);
}
.pill-online { box-shadow: 0 0 10px rgba(0,220,140,0.32); }
.pill-degraded { box-shadow: 0 0 10px rgba(255,204,68,0.28); }
.pill-offline { box-shadow: 0 0 10px rgba(255,68,102,0.28); }
.progress-wrap {
display: flex;
flex-direction: column;
gap: 4px;
margin-top: 6px;
}
.progress-meta {
display: flex;
justify-content: space-between;
gap: 8px;
font-family: var(--font-mono);
font-size: 8px;
color: var(--text-dim);
text-transform: uppercase;
letter-spacing: 0.8px;
}
.progress-bar { height: 4px; border-radius: 999px; background: rgba(0,220,140,0.08); }
.mini-graph {
display: flex;
align-items: end;
gap: 3px;
height: 16px;
}
.mini-bar {
flex: 1;
min-width: 0;
border-radius: 2px 2px 0 0;
background: linear-gradient(180deg, rgba(127,255,199,0.9), rgba(0,220,140,0.22));
box-shadow: 0 0 8px rgba(0,220,140,0.12);
}
.mini-bar.warn { background: linear-gradient(180deg, rgba(255,204,68,0.95), rgba(255,204,68,0.24)); }
.mini-bar.danger { background: linear-gradient(180deg, rgba(255,68,102,0.95), rgba(255,68,102,0.24)); }
.mini-bar.blue { background: linear-gradient(180deg, rgba(68,170,255,0.95), rgba(68,170,255,0.24)); }
.services-grid {
display: grid;
grid-template-columns: repeat(3, minmax(0, 1fr));
gap: 6px;
margin-bottom: 6px;
}
.service-card { display: flex; flex-direction: column; justify-content: space-between; }
.service-footer {
display: flex;
align-items: center;
justify-content: space-between;
gap: 8px;
margin-top: 4px;
font-family: var(--font-mono);
font-size: 8px;
color: var(--text-dim);
text-transform: uppercase;
}
.micro-strip { display: flex; gap: 3px; min-height: 8px; align-items: center; }
.micro-seg { width: 7px; height: 7px; border-radius: 999px; background: rgba(255,255,255,0.1); }
.micro-seg.up { background: var(--teal); box-shadow: 0 0 8px rgba(0,220,140,0.22); }
.micro-seg.down { background: var(--red); box-shadow: 0 0 8px rgba(255,68,102,0.22); }
.micro-seg.warn { background: var(--yellow); box-shadow: 0 0 8px rgba(255,204,68,0.22); }
#quick-access-grid {
display: grid;
grid-template-columns: repeat(6, minmax(0, 1fr));
gap: 6px;
}
.quick-tile {
min-height: 74px;
padding: 8px 10px;
border-radius: 10px;
display: flex;
align-items: center;
justify-content: flex-start;
gap: 9px;
text-align: left;
}
.quick-tile-icon {
width: 28px;
height: 28px;
display: inline-flex;
align-items: center;
justify-content: center;
border-radius: 8px;
font-size: 11px;
font-family: var(--font-display);
background: linear-gradient(135deg, var(--teal-bright), var(--teal));
color: #04110d;
}
.quick-tile-copy {
display: flex;
flex-direction: column;
gap: 2px;
min-width: 0;
}
.quick-tile-label { font-size: 9px; color: var(--text-bright); }
.quick-tile-meta {
font-family: var(--font-mono);
font-size: 7px;
color: var(--text-dim);
text-transform: uppercase;
letter-spacing: 1px;
}
.quick-tile-icon-ha { background: linear-gradient(135deg, #6cb8ff, #9e8dff); }
.quick-tile-icon-komodo { background: linear-gradient(135deg, #00e2b3, #68c7ff); }
.quick-tile-icon-kuma { background: linear-gradient(135deg, #00d98a, #7fffc7); }
.quick-tile-icon-paperless { background: linear-gradient(135deg, #89ffdc, #46cfa0); }
.quick-tile-icon-mealie { background: linear-gradient(135deg, #7ec2ff, #f6ff8c); }
.quick-tile-icon-immich { background: linear-gradient(135deg, #ffd15c, #ff9d4d); }
.quick-tile-icon-gitea { background: linear-gradient(135deg, #9cff87, #3cd675); }
.quick-tile-icon-code { background: linear-gradient(135deg, #5cc4ff, #5f92ff); }
.quick-tile-icon-files { background: linear-gradient(135deg, #6ec8ff, #9be1ff); }
.quick-tile-icon-backrest { background: linear-gradient(135deg, #8bb4ff, #6fd8ff); }
.quick-tile-icon-vault { background: linear-gradient(135deg, #ffe173, #ffaa5c); }
.quick-tile-icon-adguard { background: linear-gradient(135deg, #57ffaa, #57d8ff); }
.quick-tile-icon-traefik { background: linear-gradient(135deg, #8f9fff, #6cc4ff); }
.quick-tile-icon-scrutiny { background: linear-gradient(135deg, #9cffcf, #61ffaa); }
.storage-layout {
display: grid;
grid-template-columns: repeat(3, minmax(0, 1fr));
gap: 6px;
margin-bottom: 6px;
align-items: stretch;
}
.storage-layout > div:first-child { display: contents; }
#storage-grid { display: contents; }
.scrutiny-row { padding: 1px 0; border-bottom: none; gap: 6px; font-size: 8px; }
.scrutiny-offline { font-size: 8px; padding: 2px 0; }
.scrutiny-strip,
.storage-strip {
display: flex;
gap: 4px;
flex-wrap: wrap;
overflow: hidden;
margin-top: 4px;
}
.scrutiny-chip,
.storage-chip {
font-family: var(--font-mono);
font-size: 7px;
color: var(--text-dim);
padding: 2px 5px;
border-radius: 999px;
border: 1px solid rgba(0,220,140,0.12);
background: rgba(255,255,255,0.02);
white-space: nowrap;
}
.scrutiny-chip.ok { color: var(--teal-bright); border-color: rgba(0,220,140,0.22); }
.scrutiny-chip.fail { color: var(--red); border-color: rgba(255,68,102,0.24); }
.scrutiny-chip.unk { color: var(--text-dim); border-color: rgba(255,255,255,0.08); }
.scrutiny-chip strong {
color: currentColor;
font-family: var(--font-display);
font-size: 7px;
letter-spacing: 0.6px;
margin-right: 4px;
}
.storage-matrix-card .stats-grid { margin-bottom: 6px; }
.storage-matrix-card .service-footer { margin-top: auto; }
.system-card .card-title,
.service-card .card-title,
.storage-card .card-title { align-items: center; }
@media (max-width: 1360px) {
.services-grid { grid-template-columns: repeat(3, minmax(0, 1fr)); }
#quick-access-grid { grid-template-columns: repeat(4, minmax(0, 1fr)); }
.row-5 { grid-template-columns: repeat(3, minmax(0, 1fr)); }
}
@media (max-width: 960px) {
.services-grid, #quick-access-grid, #storage-grid, .storage-layout, .row-5 { grid-template-columns: 1fr; }
.card { max-height: none; }
.stats-grid { grid-auto-flow: row; grid-template-columns: repeat(2, minmax(0, 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>&#x2B21;</span> SYSTEM</div>
<div class="widget-row row-5" id="stats-row" style="margin-bottom:8px;">
<div class="card service-card system-card">
<div class="card-title">
<div class="card-title-left"><span class="service-icon icon-cpu"><span class="icon-glyph glyph-cpu"></span></span><span class="service-name">CPU</span></div>
<span class="status-pill pill-online">OK</span>
</div>
<div class="stats-grid">
<div class="stat-block"><div class="stat-num" id="cpu-percent">&#x2014;</div><div class="stat-label">Usage %</div></div>
<div class="stat-block"><div class="stat-num dim" id="cpu-cores">&#x2014;</div><div class="stat-label">Cores</div></div>
<div class="stat-block"><div class="stat-num dim" id="cpu-load">&#x2014;</div><div class="stat-label">Load 5m</div></div>
</div>
<div class="progress-wrap">
<div class="progress-meta"><span>Compute Load</span><span id="cpu-progress-label">0%</span></div>
<div class="progress-bar"><div class="progress-fill" id="cpu-progress"></div></div>
<div class="mini-graph" id="cpu-graph"></div>
</div>
</div>
<div class="card service-card system-card">
<div class="card-title">
<div class="card-title-left"><span class="service-icon icon-memory"><span class="icon-glyph glyph-memory"></span></span><span class="service-name">MEMORY</span></div>
<span class="status-pill pill-online">OK</span>
</div>
<div class="stats-grid">
<div class="stat-block"><div class="stat-num" id="ram-percent">&#x2014;</div><div class="stat-label">Usage %</div></div>
<div class="stat-block"><div class="stat-num dim" id="ram-used">&#x2014;</div><div class="stat-label">Used GB</div></div>
<div class="stat-block"><div class="stat-num dim" id="ram-total">&#x2014;</div><div class="stat-label">Total GB</div></div>
</div>
<div class="progress-wrap">
<div class="progress-meta"><span>Memory Pool</span><span id="ram-progress-label">0%</span></div>
<div class="progress-bar"><div class="progress-fill" id="ram-progress"></div></div>
<div class="mini-graph" id="ram-graph"></div>
</div>
</div>
<div class="card service-card system-card">
<div class="card-title">
<div class="card-title-left"><span class="service-icon icon-network"><span class="icon-glyph glyph-network"></span></span><span class="service-name">NETWORK</span></div>
<span class="status-pill pill-online">OK</span>
</div>
<div class="stats-grid">
<div class="stat-block"><div class="stat-num" id="net-rx">&#x2014;</div><div class="stat-label">&#x2193; Mbps</div></div>
<div class="stat-block"><div class="stat-num" id="net-tx">&#x2014;</div><div class="stat-label">&#x2191; Mbps</div></div>
</div>
<div class="progress-wrap">
<div class="progress-meta"><span>Traffic Flow</span><span id="net-progress-label">0 Mbps</span></div>
<div class="progress-bar"><div class="progress-fill" id="net-progress"></div></div>
<div class="mini-graph" id="net-graph"></div>
</div>
</div>
<div class="card service-card system-card">
<div class="card-title">
<div class="card-title-left"><span class="service-icon icon-host"><span class="icon-glyph glyph-host"></span></span><span class="service-name">HOST</span></div>
<span class="status-pill pill-online">OK</span>
</div>
<div class="stats-grid">
<div class="stat-block"><div class="stat-num" id="uptime-days">&#x2014;</div><div class="stat-label">Uptime d</div></div>
<div class="stat-block"><div class="stat-num dim" id="host-platform">&#x2014;</div><div class="stat-label">OS</div></div>
</div>
<div class="progress-wrap">
<div class="progress-meta"><span>Host Runtime</span><span id="host-progress-label">&#x2014;</span></div>
<div class="progress-bar"><div class="progress-fill" id="host-progress"></div></div>
<div class="mini-graph" id="host-graph"></div>
</div>
</div>
<div class="card service-card system-card">
<div class="card-title">
<div class="card-title-left"><span class="service-icon icon-docker"><span class="icon-glyph glyph-docker"></span></span><span class="service-name">DOCKER</span></div>
<span class="status-pill pill-online">OK</span>
</div>
<div class="stats-grid">
<div class="stat-block"><div class="stat-num" id="docker-running">&#x2014;</div><div class="stat-label">Running</div></div>
<div class="stat-block"><div class="stat-num dim" id="docker-stopped">&#x2014;</div><div class="stat-label">Stopped</div></div>
<div class="stat-block"><div class="stat-num dim" id="docker-total">&#x2014;</div><div class="stat-label">Total</div></div>
</div>
<div class="progress-wrap">
<div class="progress-meta"><span>Runtime Surface</span><span id="docker-progress-label">0%</span></div>
<div class="progress-bar"><div class="progress-fill" id="docker-progress"></div></div>
<div class="mini-graph" id="docker-graph"></div>
</div>
</div>
</div>
<!-- STORAGE + SCRUTINY ROW -->
<div class="section-header"><span>&#x2B21;</span> STORAGE &amp; HEALTH</div>
<div class="storage-layout">
<div>
<div class="widget-row row-3" id="storage-grid">
<!-- Disk cards injected by renderer -->
</div>
</div>
<div class="card service-card">
<div class="card-title">
<div class="card-title-left"><span class="service-icon icon-scrutiny"><span class="icon-glyph glyph-scrutiny"></span></span><span class="service-name">SCRUTINY</span></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">&#x2014;</div><div class="stat-label">Disks</div></div>
<div class="stat-block"><div class="stat-num" id="scrutiny-passed">&#x2014;</div><div class="stat-label">Passed</div></div>
<div class="stat-block"><div class="stat-num danger" id="scrutiny-failed">&#x2014;</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>&#x2B21;</span> SERVICES</div>
<div class="services-grid">
<div class="card service-card">
<div class="card-title">
<div class="card-title-left">
<span class="service-icon icon-ha"><span class="icon-glyph glyph-home"></span></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">&#x2014;</div><div class="stat-label">Lights</div></div>
<div class="stat-block"><div class="stat-num" id="ha-climate">&#x2014;</div><div class="stat-label">Climate</div></div>
<div class="stat-block"><div class="stat-num" id="ha-doors">&#x2014;</div><div class="stat-label">Doors</div></div>
<div class="stat-block"><div class="stat-num danger" id="ha-alerts">&#x2014;</div><div class="stat-label">Alerts</div></div>
</div>
<div class="service-footer"><span id="ha-version">Core automation hub</span><span></span></div>
</div>
<div class="card service-card">
<div class="card-title">
<div class="card-title-left">
<span class="service-icon icon-kuma"><span class="icon-glyph glyph-kuma"></span></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">&#x2014;</div><div class="stat-label">Up</div></div>
<div class="stat-block"><div class="stat-num danger" id="uk-down">&#x2014;</div><div class="stat-label">Down</div></div>
<div class="stat-block"><div class="stat-num warn" id="uk-paused">&#x2014;</div><div class="stat-label">Paused</div></div>
<div class="stat-block"><div class="stat-num" id="uk-uptime">&#x2014;</div><div class="stat-label">24h %</div></div>
</div>
<div class="service-footer"><span id="uk-footer">No monitor data</span><div class="micro-strip" id="uk-bars"></div></div>
</div>
<div class="card service-card">
<div class="card-title">
<div class="card-title-left">
<span class="service-icon icon-immich"><span class="icon-glyph glyph-image"></span></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">&#x2014;</div><div class="stat-label">Photos</div></div>
<div class="stat-block"><div class="stat-num" id="immich-videos">&#x2014;</div><div class="stat-label">Videos</div></div>
<div class="stat-block"><div class="stat-num dim" id="immich-storage">&#x2014;</div><div class="stat-label">Storage</div></div>
</div>
</div>
<div class="card service-card">
<div class="card-title">
<div class="card-title-left">
<span class="service-icon icon-backrest"><span class="icon-glyph glyph-backrest"></span></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">&#x2014;</div><div class="stat-label">Last Backup</div></div>
<div class="stat-block"><div class="stat-num" id="backrest-repos">&#x2014;</div><div class="stat-label">Repos</div></div>
<div class="stat-block"><div class="stat-num danger" id="backrest-errors">&#x2014;</div><div class="stat-label">Errors</div></div>
</div>
</div>
<div class="card service-card">
<div class="card-title">
<div class="card-title-left">
<span class="service-icon icon-adguard"><span class="icon-glyph glyph-shield"></span></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">&#x2014;</div><div class="stat-label">Queries</div></div>
<div class="stat-block"><div class="stat-num" id="adguard-blocked">&#x2014;</div><div class="stat-label">Blocked</div></div>
<div class="stat-block"><div class="stat-num dim" id="adguard-blocked-pct">&#x2014;</div><div class="stat-label">Block %</div></div>
<div class="stat-block"><div class="stat-num dim" id="adguard-latency">&#x2014;</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 service-card">
<div class="card-title">
<div class="card-title-left">
<span class="service-icon icon-services"><span class="icon-glyph glyph-services"></span></span>
<span class="service-name">SERVICES OVERVIEW</span>
</div>
<span class="status-pill pill-offline" id="services-pill">&#x2014;</span>
</div>
<div class="stats-grid">
<div class="stat-block"><div class="stat-num" id="svc-online">&#x2014;</div><div class="stat-label">Online</div></div>
<div class="stat-block"><div class="stat-num warn" id="svc-degraded">&#x2014;</div><div class="stat-label">Degraded</div></div>
<div class="stat-block"><div class="stat-num danger" id="svc-offline">&#x2014;</div><div class="stat-label">Offline</div></div>
<div class="stat-block"><div class="stat-num dim" id="svc-total">&#x2014;</div><div class="stat-label">Total</div></div>
</div>
</div>
</div>
<!-- QUICK ACCESS -->
<div class="section-header"><span>&#x2B21;</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>