feat(frontend): add theming terminal and brand assets

This commit is contained in:
2026-06-20 17:22:25 -05:00
parent d7f3e2054d
commit 6c50debcf4
5 changed files with 775 additions and 160 deletions

View File

@@ -11,107 +11,14 @@
<link href="https://cdn.jsdelivr.net/npm/bootstrap@5.3.8/dist/css/bootstrap.min.css" rel="stylesheet" integrity="sha384-sRIl4kxILFvY47J16cr9ZwB07vP4J8+LH7qKQnuqkuIAvNWLzeN8tE5YBujZqJLB" crossorigin="anonymous">
<link href="https://cdn.jsdelivr.net/npm/@fortawesome/[email protected]/css/all.min.css" rel="stylesheet">
<link href="https://cdn.jsdelivr.net/npm/@tabler/icons-webfont@3.44.0/dist/tabler-icons.min.css" rel="stylesheet">
<link href="https://cdn.jsdelivr.net/npm/@xterm/xterm@5.5.0/css/xterm.min.css" rel="stylesheet">
<link rel="stylesheet" href="/static/css/app.css">
</head>
<body class="theme-{{.ThemeClass}} bg-body-tertiary">
{{if eq .Shell "auth"}}
<main class="min-vh-100 d-flex align-items-center auth-shell py-5">
<div class="container">
<div class="row justify-content-center">
<div class="col-12 col-lg-10 col-xl-9">
<div class="card border-0 shadow-lg auth-card overflow-hidden">
<div class="row g-0">
<div class="col-lg-6 auth-brand-col">
<div class="auth-brand h-100 p-4 p-lg-5">
<span class="badge rounded-pill text-bg-primary-subtle text-primary-emphasis px-3 py-2 mb-4">Self-hosted fleet control</span>
<div class="d-flex align-items-center gap-3 mb-4">
<div class="brand-mark-lg">
<i class="ti ti-server-2"></i>
</div>
<div>
<div class="text-uppercase small text-primary fw-semibold">Maintainarr</div>
<div class="text-body-secondary">VM operations dashboard</div>
</div>
</div>
<h1 class="display-5 fw-bold mb-3">Operate Linux nodes without leaving the browser.</h1>
<p class="lead text-body-secondary mb-4">Registration, OTP-backed sign-in, fleet health, remote actions, SSH access, and automation hooks in one self-hosted build.</p>
<div class="row g-3">
<div class="col-sm-6">
<div class="auth-feature">
<i class="ti ti-shield-lock text-primary"></i>
<span>Role-based access with OTP 2FA</span>
</div>
</div>
<div class="col-sm-6">
<div class="auth-feature">
<i class="ti ti-brand-ubuntu text-primary"></i>
<span>Ubuntu, Debian, Arch and more</span>
</div>
</div>
<div class="col-sm-6">
<div class="auth-feature">
<i class="ti ti-terminal-2 text-primary"></i>
<span>Web SSH console</span>
</div>
</div>
<div class="col-sm-6">
<div class="auth-feature">
<i class="ti ti-clock-bolt text-primary"></i>
<span>Recurring automation jobs</span>
</div>
</div>
</div>
</div>
</div>
<div class="col-lg-6">
<div class="auth-form-wrap p-4 p-lg-5">
{{template "content" .}}
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</main>
{{else}}
<div class="d-lg-flex app-shell">
<aside class="sidebar border-end">
<div class="p-3 p-lg-4">
<div class="d-flex align-items-center gap-3 mb-4">
<div class="brand-mark">
<i class="ti ti-layout-dashboard-filled"></i>
</div>
<div>
<div class="fw-bold fs-5">Maintainarr</div>
<div class="small text-body-secondary">{{with .Organization}}{{.Name}}{{end}}</div>
</div>
</div>
<div class="list-group list-group-flush sidebar-nav">
<a href="/dashboard" class="list-group-item list-group-item-action {{if eq .CurrentPath "/dashboard"}}active{{end}}"><i class="ti ti-dashboard me-2"></i>Dashboard</a>
<a href="/groups" class="list-group-item list-group-item-action {{if eq .CurrentPath "/groups"}}active{{end}}"><i class="ti ti-stack-2 me-2"></i>VM Groups</a>
<a href="/automations" class="list-group-item list-group-item-action {{if eq .CurrentPath "/automations"}}active{{end}}"><i class="ti ti-clock-cog me-2"></i>Automations</a>
<a href="/settings" class="list-group-item list-group-item-action {{if eq .CurrentPath "/settings"}}active{{end}}"><i class="ti ti-palette me-2"></i>Theme System</a>
</div>
<div class="card mt-4 border-0 sidebar-status">
<div class="card-body">
<div class="small text-uppercase text-body-secondary mb-2">Signed in as</div>
<div class="fw-semibold">{{with .User}}{{.Name}}{{end}}</div>
<div class="text-body-secondary small">{{with .User}}{{.Role}}{{end}}</div>
<hr>
<a class="btn btn-outline-secondary w-100" href="/logout"><i class="fa-solid fa-right-from-bracket me-2"></i>Sign out</a>
</div>
</div>
</div>
</aside>
<main class="content flex-grow-1">
<div class="container-fluid px-3 px-lg-4 py-4">
{{template "content" .}}
</div>
</main>
</div>
{{end}}
<body class="theme-{{.ThemeClass}} bg-body-tertiary" data-bs-theme="{{.ThemeMode}}">
{{template "shell" .}}
<script src="https://cdn.jsdelivr.net/npm/bootstrap@5.3.8/dist/js/bootstrap.bundle.min.js" integrity="sha384-FKyoEForCGlyvwx9Hj09JcYn3nv7wiPVlz7YYwJrWVcXK/BmnVDxM+D2scQbITxI" crossorigin="anonymous"></script>
<script src="https://cdn.jsdelivr.net/npm/@xterm/xterm@5.5.0/lib/xterm.min.js"></script>
<script src="https://cdn.jsdelivr.net/npm/@xterm/addon-fit@0.10.0/lib/addon-fit.min.js"></script>
<script src="/static/js/app.js"></script>
</body>
</html>

View File

@@ -15,13 +15,104 @@
--bs-link-color-rgb: var(--color-primary-600);
--bs-link-hover-color-rgb: var(--color-primary-700);
--ma-border: rgba(15, 23, 42, 0.08);
--ma-surface-sidebar: #16181d;
--ma-surface-base: #1d2026;
--ma-surface-1: #22262d;
--ma-surface-2: #272c34;
--ma-surface-3: #2d333c;
--ma-surface-overlay: #343b45;
}
body.theme-dark,
body.theme-light,
body.theme-blue {
--color-primary-50: 239 246 255;
--color-primary-100: 219 234 254;
--color-primary-200: 191 219 254;
--color-primary-300: 147 197 253;
--color-primary-400: 96 165 250;
--color-primary-500: 59 130 246;
--color-primary-600: 37 99 235;
--color-primary-700: 29 78 216;
--color-primary-800: 30 64 175;
--color-primary-900: 30 58 138;
--color-primary-950: 23 37 84;
}
body.theme-green {
--color-primary-50: 236 253 245;
--color-primary-100: 209 250 229;
--color-primary-200: 167 243 208;
--color-primary-300: 110 231 183;
--color-primary-400: 52 211 153;
--color-primary-500: 16 185 129;
--color-primary-600: 5 150 105;
--color-primary-700: 4 120 87;
--color-primary-800: 6 95 70;
--color-primary-900: 6 78 59;
--color-primary-950: 2 44 34;
}
body.theme-red {
--color-primary-50: 254 242 242;
--color-primary-100: 254 226 226;
--color-primary-200: 254 202 202;
--color-primary-300: 252 165 165;
--color-primary-400: 248 113 113;
--color-primary-500: 239 68 68;
--color-primary-600: 220 38 38;
--color-primary-700: 185 28 28;
--color-primary-800: 153 27 27;
--color-primary-900: 127 29 29;
--color-primary-950: 69 10 10;
}
body.theme-blue,
body.theme-green,
body.theme-red,
body.theme-dark,
body.theme-light {
--bs-primary: rgb(var(--color-primary-600));
--bs-primary-rgb: var(--color-primary-600);
--bs-link-color-rgb: var(--color-primary-600);
--bs-link-hover-color-rgb: var(--color-primary-700);
}
body {
font-family: "Inter", system-ui, sans-serif;
overflow: hidden;
}
html,
body {
height: 100%;
}
body[data-bs-theme="dark"] {
--ma-border: rgba(255, 255, 255, 0.06);
--ma-surface-sidebar: #191b20;
--ma-surface-base: #21252b;
--ma-surface-1: #242930;
--ma-surface-2: #282e36;
--ma-surface-3: #2c333c;
--ma-surface-overlay: #313944;
background:
radial-gradient(circle at top left, rgba(var(--color-primary-100), 0.55), transparent 28%),
linear-gradient(180deg, #f8fbff 0%, #f4f7fb 55%, #eef3f8 100%);
radial-gradient(circle at top left, rgba(var(--color-primary-700), 0.28), transparent 28%),
linear-gradient(180deg, #171a1f 0%, #1b1f26 55%, #20252c 100%);
color: #e5eefb;
}
body[data-bs-theme="light"] {
--ma-border: rgba(15, 23, 42, 0.08);
--ma-surface-sidebar: #e2e5ea;
--ma-surface-base: #eceff3;
--ma-surface-1: #e9ecf0;
--ma-surface-2: #e6e9ee;
--ma-surface-3: #e3e7ec;
--ma-surface-overlay: #dde2e8;
background:
radial-gradient(circle at top left, rgba(var(--color-primary-200), 0.55), transparent 28%),
linear-gradient(180deg, #f7f9fc 0%, #f0f3f7 55%, #eaedf2 100%);
color: #0f172a;
}
@@ -30,7 +121,6 @@ a {
}
.auth-card,
.sidebar,
.node-chip,
.sidebar-status,
.hero-stat,
@@ -41,11 +131,21 @@ a {
}
.auth-card {
background: rgba(255, 255, 255, 0.92);
background: color-mix(in srgb, var(--ma-surface-overlay) 92%, transparent);
backdrop-filter: blur(14px);
}
body[data-bs-theme="light"] .auth-card {
background: color-mix(in srgb, white 58%, var(--ma-surface-base));
}
.auth-brand {
background:
radial-gradient(circle at top, rgba(var(--color-primary-600), 0.32), transparent 35%),
linear-gradient(155deg, #111827 0%, #0f172a 70%, #172554 100%);
}
body[data-bs-theme="light"] .auth-brand {
background:
radial-gradient(circle at top, rgba(var(--color-primary-200), 0.55), transparent 35%),
linear-gradient(155deg, #ffffff 0%, #eef5ff 70%, #e4ecff 100%);
@@ -57,11 +157,16 @@ a {
align-items: center;
padding: 0.85rem 1rem;
border-radius: 1rem;
background: rgba(255, 255, 255, 0.72);
border: 1px solid rgba(var(--color-primary-200), 0.7);
background: color-mix(in srgb, var(--ma-surface-2) 88%, transparent);
border: 1px solid rgba(var(--color-primary-500), 0.35);
font-weight: 500;
}
body[data-bs-theme="light"] .auth-feature {
background: color-mix(in srgb, white 55%, var(--ma-surface-1));
border: 1px solid rgba(var(--color-primary-200), 0.7);
}
.brand-mark,
.brand-mark-lg,
.stat-icon,
@@ -94,52 +199,196 @@ a {
font-size: 1.4rem;
}
.sidebar {
width: 300px;
min-height: 100vh;
background: rgba(255, 255, 255, 0.82);
.content {
min-width: 0;
min-height: 0;
background: color-mix(in srgb, var(--ma-surface-base) 96%, transparent);
overflow: hidden;
}
.app-header,
.app-footer {
background: color-mix(in srgb, var(--ma-surface-2) 94%, transparent);
backdrop-filter: blur(14px);
}
.sidebar-nav .list-group-item {
border: 0;
border-radius: 0.95rem !important;
margin-bottom: 0.35rem;
color: #475569;
font-weight: 600;
background: transparent;
body[data-bs-theme="light"] .app-header,
body[data-bs-theme="light"] .app-footer {
background: color-mix(in srgb, white 42%, var(--ma-surface-2));
}
.sidebar-nav .list-group-item:hover {
background: rgba(var(--color-primary-100), 0.8);
color: #0f172a;
.app-sidebar {
width: 280px;
background: color-mix(in srgb, var(--ma-surface-sidebar) 97%, transparent);
backdrop-filter: blur(14px);
overflow: auto;
}
.sidebar-nav .list-group-item.active {
background: linear-gradient(135deg, rgba(var(--color-primary-500), 0.12), rgba(var(--color-primary-700), 0.18));
color: rgb(var(--color-primary-800));
body[data-bs-theme="light"] .app-sidebar {
background: color-mix(in srgb, white 22%, var(--ma-surface-sidebar));
}
.sidebar-status {
background: rgba(var(--color-primary-50), 0.9);
.app-sidebar-nav .btn {
justify-content: flex-start;
align-items: center;
gap: 0.4rem;
}
.content {
.sidebar-logo {
width: 3rem;
height: 3rem;
object-fit: cover;
border-radius: 0.95rem;
box-shadow: 0 0.75rem 2rem rgba(var(--color-primary-700), 0.18);
}
.min-w-0 {
min-width: 0;
}
.app-shell {
height: 100vh;
overflow: hidden;
}
.container-fluid {
min-height: 0;
}
.content > .container-fluid {
height: 100%;
overflow: auto;
}
.node-chip {
transition: transform 0.18s ease, box-shadow 0.18s ease;
}
.node-chip:hover {
transform: translateY(-3px);
box-shadow: 0 1rem 2rem rgba(15, 23, 42, 0.08);
box-shadow: 0 1rem 2rem rgba(2, 6, 23, 0.35);
}
.dashboard-loader {
min-height: 220px;
}
.dashboard-spinner {
display: grid;
place-items: center;
min-height: 160px;
}
.kpi-card {
overflow: hidden;
position: relative;
}
.kpi-graph {
position: absolute;
inset: 0;
opacity: 0.55;
pointer-events: none;
}
.kpi-graph svg {
width: 100%;
height: 100%;
}
.node-tile {
display: grid;
grid-template-columns: 64px minmax(0, 1fr);
min-height: 118px;
}
.node-tile-icon {
display: flex;
align-items: center;
justify-content: center;
padding: 0.65rem;
border-right: 1px solid var(--ma-border);
}
.node-tile-main {
padding: 0.7rem 0.8rem 0.7rem 0.75rem;
min-width: 0;
position: relative;
}
.node-tile-metrics {
display: grid;
gap: 0.45rem;
}
.node-metric {
display: grid;
grid-template-columns: 2rem minmax(0, 1fr) 2.4rem;
align-items: center;
gap: 0.45rem;
width: 100%;
}
.node-metric .progress {
--bs-progress-height: 0.35rem;
margin: 0;
}
.node-tile .chip-icon {
width: 2.35rem;
height: 2.35rem;
font-size: 1rem;
}
.node-tile-main .badge {
font-size: 0.65rem;
padding: 0.35rem 0.45rem;
}
.node-tile-main h3 {
font-size: 0.95rem;
}
.node-tile-main .small {
font-size: 0.74rem;
}
.node-status {
width: 0.5rem;
height: 0.5rem;
border-radius: 999px;
display: inline-block;
position: absolute;
top: 0.75rem;
right: 0.8rem;
}
.node-status.is-online {
background: #22c55e;
box-shadow: 0 0 0 0.2rem rgba(34, 197, 94, 0.15);
}
.node-status.is-offline {
background: #64748b;
box-shadow: 0 0 0 0.2rem rgba(100, 116, 139, 0.12);
}
.node-metric-label,
.node-metric-value {
font-size: 0.72rem;
font-weight: 600;
line-height: 1;
}
.node-metric-value {
text-align: right;
font-variant-numeric: tabular-nums;
justify-self: end;
}
.progress {
--bs-progress-height: 0.6rem;
background: #e7eef8;
background: color-mix(in srgb, black 18%, var(--ma-surface-1));
}
.card,
@@ -155,6 +404,121 @@ a {
border-color: var(--ma-border);
}
.card,
.list-group-item,
.table,
.modal-content,
.dropdown-menu,
.sidebar-status,
.theme-card,
.add-vm-panel,
.auth-toggle-option,
.option-check {
background: var(--ma-surface-1);
}
.stat-card,
.node-chip,
.dashboard-loader > .card,
.preview-card.dim {
background: var(--ma-surface-2);
}
.modal-content {
background: var(--ma-surface-overlay);
}
body[data-bs-theme="light"] .card,
body[data-bs-theme="light"] .list-group-item,
body[data-bs-theme="light"] .table,
body[data-bs-theme="light"] .modal-content,
body[data-bs-theme="light"] .dropdown-menu,
body[data-bs-theme="light"] .sidebar-status,
body[data-bs-theme="light"] .theme-card,
body[data-bs-theme="light"] .add-vm-panel,
body[data-bs-theme="light"] .auth-toggle-option,
body[data-bs-theme="light"] .option-check {
background: var(--ma-surface-1);
}
.add-vm-form {
display: grid;
gap: 1rem;
}
.add-vm-panel {
border: 1px solid var(--ma-border);
border-radius: 1.1rem;
padding: 1rem;
}
body[data-bs-theme="light"] .add-vm-panel {
background: var(--ma-surface-1);
}
.add-vm-panel-title {
font-size: 0.76rem;
font-weight: 700;
letter-spacing: 0.08em;
text-transform: uppercase;
color: var(--bs-secondary-color);
margin-bottom: 0.9rem;
}
.auth-toggle {
display: grid;
grid-template-columns: repeat(2, minmax(0, 1fr));
gap: 0.75rem;
}
.auth-toggle-option {
display: grid;
gap: 0.2rem;
padding: 0.85rem 0.95rem;
border: 1px solid var(--ma-border);
border-radius: 0.95rem;
cursor: pointer;
transition: border-color 0.18s ease, background-color 0.18s ease, transform 0.18s ease;
}
body[data-bs-theme="light"] .auth-toggle-option {
background: var(--ma-surface-1);
}
.auth-toggle-label {
font-weight: 600;
color: var(--bs-emphasis-color);
}
.auth-toggle-text {
font-size: 0.78rem;
color: var(--bs-secondary-color);
}
.btn-check:checked + .auth-toggle-option {
border-color: rgba(var(--color-primary-500), 0.7);
background: rgba(var(--color-primary-500), 0.12);
transform: translateY(-1px);
}
.auth-mode-panel {
min-height: 74px;
}
.option-check {
display: flex;
align-items: center;
gap: 0.7rem;
min-height: calc(1.5em + 0.75rem + 2px);
padding: 0.7rem 0.85rem;
border: 1px solid var(--ma-border);
border-radius: 0.95rem;
}
body[data-bs-theme="light"] .option-check {
background: var(--ma-surface-1);
}
.run-pre,
.code-block,
.console-output {
@@ -166,7 +530,7 @@ a {
.run-pre,
.code-block {
background: #0f172a;
background: #13161b;
color: #dbeafe;
padding: 1rem;
border-radius: 1rem;
@@ -176,15 +540,42 @@ a {
min-height: 420px;
max-height: 60vh;
overflow: auto;
background: #0b1220;
background: #151920;
color: #dbeafe;
padding: 1rem;
padding: 0;
}
.console-shell {
position: relative;
}
.console-terminal {
min-height: 420px;
height: 60vh;
overflow: hidden;
}
.console-panel {
min-height: calc(100vh - 12rem);
}
.console-panel .console-terminal {
height: calc(100vh - 12rem);
}
.console-terminal .xterm {
height: 100%;
padding: 0.9rem 1rem;
}
.console-terminal .xterm-viewport {
border-radius: 1.25rem;
}
.preview-card {
min-height: 120px;
border: 1px solid rgba(var(--color-primary-200), 0.9);
background: linear-gradient(135deg, rgba(var(--color-primary-100), 0.9), rgba(var(--color-primary-300), 0.9));
border: 1px solid rgba(var(--color-primary-700), 0.55);
background: linear-gradient(135deg, rgba(var(--color-primary-900), 0.95), rgba(var(--color-primary-700), 0.8));
}
.preview-card.accent {
@@ -192,6 +583,15 @@ a {
}
.preview-card.dim {
background: linear-gradient(135deg, #0f172a, #1e293b);
}
body[data-bs-theme="light"] .preview-card {
background: linear-gradient(135deg, rgba(var(--color-primary-100), 0.9), rgba(var(--color-primary-300), 0.9));
border: 1px solid rgba(var(--color-primary-200), 0.9);
}
body[data-bs-theme="light"] .preview-card.dim {
background: linear-gradient(135deg, #ffffff, #e2e8f0);
}
@@ -200,9 +600,69 @@ a {
height: auto;
}
.theme-card {
padding: 1rem;
border: 1px solid var(--ma-border);
border-radius: 1rem;
cursor: pointer;
}
body[data-bs-theme="light"] .theme-card {
background: var(--ma-surface-1);
}
.theme-card strong,
.theme-card small {
display: block;
}
.theme-card small {
color: var(--bs-secondary-color);
}
.theme-swatch {
display: block;
height: 88px;
border-radius: 0.85rem;
margin-bottom: 0.85rem;
border: 1px solid rgba(255, 255, 255, 0.08);
}
.swatch-dark {
background: linear-gradient(135deg, #0f172a, #1e293b);
}
.swatch-light {
background: linear-gradient(135deg, #ffffff, #e2e8f0);
}
.swatch-green {
background: linear-gradient(135deg, rgb(16 185 129), rgb(4 120 87));
}
.swatch-red {
background: linear-gradient(135deg, rgb(239 68 68), rgb(185 28 28));
}
.swatch-blue {
background: linear-gradient(135deg, rgb(59 130 246), rgb(29 78 216));
}
.btn-check:checked + .theme-card {
border-color: rgb(var(--color-primary-500));
box-shadow: 0 0 0 0.2rem rgba(var(--color-primary-500), 0.2);
}
@media (max-width: 991.98px) {
.sidebar {
.app-shell {
flex-direction: column;
}
.app-sidebar {
width: 100%;
min-height: auto;
}
.auth-toggle {
grid-template-columns: 1fr;
}
}

Binary file not shown.

After

Width:  |  Height:  |  Size: 711 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 806 KiB

View File

@@ -1,34 +1,282 @@
document.addEventListener("DOMContentLoaded", () => {
const form = document.getElementById("console-form");
const output = document.getElementById("console");
const input = document.getElementById("console-input");
const triggerType = document.getElementById("trigger_type");
const scheduleKind = document.getElementById("schedule_kind");
const addCommandButton = document.getElementById("add-command-step");
const commandSteps = document.getElementById("command-steps");
const nodeLive = document.querySelector(".node-live");
const dashboardNodes = document.getElementById("dashboard-nodes");
const authToggle = document.querySelector("[data-auth-toggle]");
if (!form || !output || !input) {
return;
}
const protocol = window.location.protocol === "https:" ? "wss:" : "ws:";
const socket = new WebSocket(`${protocol}//${window.location.host}${form.dataset.ws}`);
socket.addEventListener("message", (event) => {
output.textContent += event.data;
output.scrollTop = output.scrollHeight;
});
socket.addEventListener("open", () => {
output.textContent += "Connected to remote shell.\n";
});
socket.addEventListener("close", () => {
output.textContent += "\nSession closed.\n";
});
form.addEventListener("submit", (event) => {
event.preventDefault();
if (socket.readyState !== WebSocket.OPEN) {
const attachXtermConsole = (consoleOutput) => {
const wsPath = consoleOutput.dataset.ws;
if (!wsPath || typeof Terminal === "undefined") {
return;
}
socket.send(`${input.value}\n`);
input.value = "";
const term = new Terminal({
cursorBlink: true,
convertEol: true,
fontFamily: '"Cascadia Code", "SFMono-Regular", Consolas, "Liberation Mono", monospace',
fontSize: 14,
lineHeight: 1.25,
theme: {
background: "#0b1220",
foreground: "#dbeafe",
cursor: "#93c5fd",
cursorAccent: "#0b1220",
selectionBackground: "rgba(147, 197, 253, 0.28)",
},
});
const fitAddon = typeof FitAddon !== "undefined" ? new FitAddon.FitAddon() : null;
if (fitAddon) {
term.loadAddon(fitAddon);
}
term.open(consoleOutput);
const protocol = window.location.protocol === "https:" ? "wss:" : "ws:";
const socket = new WebSocket(`${protocol}//${window.location.host}${wsPath}`);
const fit = () => {
fitAddon?.fit();
term.focus();
};
window.setTimeout(fit, 0);
window.addEventListener("resize", fit);
if (typeof ResizeObserver !== "undefined") {
const observer = new ResizeObserver(fit);
observer.observe(consoleOutput);
}
socket.addEventListener("message", (event) => {
term.write(event.data);
});
socket.addEventListener("open", () => {
fit();
});
socket.addEventListener("close", () => {
term.write("\r\n[session closed]\r\n");
});
socket.addEventListener("error", () => {
term.write("\r\n[connection error]\r\n");
});
term.onData((value) => {
if (socket.readyState === WebSocket.OPEN) {
socket.send(value);
}
});
};
if (output?.dataset.xterm === "true") {
attachXtermConsole(output);
}
const syncScheduleUi = () => {
if (!triggerType || !scheduleKind) {
return;
}
const schedulePanel = document.querySelector(".schedule-panel");
const weeklyRows = document.querySelectorAll(".weekly-only");
const intervalRows = document.querySelectorAll(".interval-only");
const showSchedule = triggerType.value !== "triggered";
const showWeekly = showSchedule && scheduleKind.value === "weekly";
const showInterval = showSchedule && scheduleKind.value === "interval";
if (schedulePanel) {
schedulePanel.classList.toggle("opacity-50", !showSchedule);
}
weeklyRows.forEach((row) => row.classList.toggle("d-none", !showWeekly));
intervalRows.forEach((row) => row.classList.toggle("d-none", !showInterval));
};
triggerType?.addEventListener("change", syncScheduleUi);
scheduleKind?.addEventListener("change", syncScheduleUi);
syncScheduleUi();
addCommandButton?.addEventListener("click", () => {
if (!commandSteps) {
return;
}
const nextIndex = commandSteps.querySelectorAll(".command-step").length + 1;
const wrapper = document.createElement("div");
wrapper.className = "input-group command-step";
wrapper.innerHTML = `
<span class="input-group-text">${nextIndex}</span>
<input type="text" class="form-control" name="command_steps[]" placeholder="command ${nextIndex}" required>
<button class="btn btn-outline-danger remove-command-step" type="button">Remove</button>
`;
commandSteps.appendChild(wrapper);
});
commandSteps?.addEventListener("click", (event) => {
const target = event.target;
if (!(target instanceof HTMLElement) || !target.classList.contains("remove-command-step")) {
return;
}
const steps = commandSteps.querySelectorAll(".command-step");
if (steps.length <= 1) {
return;
}
target.closest(".command-step")?.remove();
commandSteps.querySelectorAll(".command-step").forEach((step, index) => {
const badge = step.querySelector(".input-group-text");
if (badge) {
badge.textContent = String(index + 1);
}
});
});
if (nodeLive instanceof HTMLElement) {
const nodeId = nodeLive.dataset.nodeId;
const history = {
cpu: [],
ram: [],
disk: [],
uptime: [],
};
const uptimeLabel = (seconds) => {
const days = Math.floor(seconds / 86400);
const hours = Math.floor((seconds % 86400) / 3600);
const minutes = Math.floor((seconds % 3600) / 60);
if (days > 0) return `${days}d ${hours}h ${minutes}m`;
return `${hours}h ${minutes}m`;
};
const renderSparkline = (values) => {
if (values.length === 0) {
return "";
}
const max = Math.max(...values, 1);
const points = values.map((value, index) => {
const x = (index / Math.max(values.length - 1, 1)) * 100;
const y = 100 - (value / max) * 80 - 10;
return `${x},${y}`;
}).join(" ");
return `
<svg viewBox="0 0 100 100" preserveAspectRatio="none">
<polyline points="${points}" fill="none" stroke="rgba(255,255,255,0.28)" stroke-width="3" stroke-linecap="round" stroke-linejoin="round"></polyline>
</svg>
`;
};
const pushValue = (key, value) => {
history[key].push(value);
if (history[key].length > 18) {
history[key].shift();
}
const card = nodeLive.querySelector(`.kpi-card[data-kpi="${key}"] .kpi-graph`);
if (card) {
card.innerHTML = renderSparkline(history[key]);
}
};
const updateNodeStats = async () => {
if (!nodeId) {
return;
}
try {
const response = await fetch(`/nodes/${nodeId}/stats`, { headers: { Accept: "application/json" } });
if (!response.ok) {
return;
}
const stats = await response.json();
const cpu = Number(stats.cpu_usage || 0);
const ram = Number(stats.ram_usage || 0);
const disk = Number(stats.disk_usage || 0);
const uptime = Number(stats.uptime_seconds || 0);
const cpuValue = nodeLive.querySelector('[data-kpi-value="cpu"]');
const ramValue = nodeLive.querySelector('[data-kpi-value="ram"]');
const diskValue = nodeLive.querySelector('[data-kpi-value="disk"]');
const uptimeValue = nodeLive.querySelector('[data-kpi-value="uptime"]');
if (cpuValue) cpuValue.textContent = `${cpu.toFixed(1)}%`;
if (ramValue) ramValue.textContent = `${ram.toFixed(1)}%`;
if (diskValue) diskValue.textContent = `${disk.toFixed(1)}%`;
if (uptimeValue) uptimeValue.textContent = uptimeLabel(uptime);
pushValue("cpu", cpu);
pushValue("ram", ram);
pushValue("disk", disk);
pushValue("uptime", Math.min(uptime / 3600, 100));
} catch (_) {
// Ignore transient poll failures.
}
};
updateNodeStats();
window.setInterval(updateNodeStats, 5000);
}
if (dashboardNodes instanceof HTMLElement) {
const url = dashboardNodes.dataset.dashboardNodesUrl;
if (url) {
fetch(url, { headers: { Accept: "text/html" } })
.then((response) => {
if (!response.ok) {
throw new Error("failed");
}
return response.text();
})
.then((html) => {
dashboardNodes.innerHTML = html;
dashboardNodes.classList.add("is-loaded");
})
.catch(() => {
dashboardNodes.innerHTML = `
<div class="card border-0 shadow-sm">
<div class="card-body py-5 text-center text-body-secondary">
Failed to load nodes.
</div>
</div>
`;
});
}
}
if (authToggle instanceof HTMLElement) {
const passwordRadio = document.getElementById("authModePassword");
const keyRadio = document.getElementById("authModeKey");
const passwordPanel = document.querySelector('[data-auth-panel="password"]');
const keyPanel = document.querySelector('[data-auth-panel="key"]');
const passwordInput = document.querySelector('[data-auth-input="password"]');
const keyInput = document.querySelector('[data-auth-input="key"]');
const syncAuthMode = () => {
const passwordMode = passwordRadio instanceof HTMLInputElement && passwordRadio.checked;
passwordPanel?.classList.toggle("d-none", !passwordMode);
keyPanel?.classList.toggle("d-none", passwordMode);
if (passwordInput instanceof HTMLInputElement) {
passwordInput.disabled = !passwordMode;
passwordInput.required = passwordMode;
if (!passwordMode) {
passwordInput.value = "";
}
}
if (keyInput instanceof HTMLInputElement) {
keyInput.disabled = passwordMode;
keyInput.required = !passwordMode;
if (passwordMode) {
keyInput.value = "";
}
}
};
passwordRadio?.addEventListener("change", syncAuthMode);
keyRadio?.addEventListener("change", syncAuthMode);
syncAuthMode();
}
});