2 Commits

Author SHA1 Message Date
jared b393d94e81 Upgrade page headers to lt-page-header/lt-page-title across all pages
Lint / Python (flake8) (push) Successful in 1m7s
Lint / JS (eslint) (push) Successful in 10s
Security / Python Security (bandit) (push) Failing after 1m17s
Test / Python Tests (pytest) (push) Successful in 1m23s
Lint / Notify on failure (push) Has been skipped
Lint / Deploy (push) Successful in 4s
Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-05-01 01:09:30 -04:00
jared 4cb36a47a9 Add stat cards, lt-frame alert queue, and timeline for resolved alerts
Lint / Python (flake8) (push) Successful in 54s
Lint / JS (eslint) (push) Successful in 7s
Security / Python Security (bandit) (push) Failing after 40s
Test / Python Tests (pytest) (push) Successful in 50s
Lint / Notify on failure (push) Has been skipped
Lint / Deploy (push) Successful in 3s
- Four lt-stat-card widgets (Critical, Warning, Hosts, Resolved 24h)
  below the status bar; Critical card pulses red when count > 0
- Clicking Critical or Warning card filters the events table by severity
- Events table wrapped in lt-frame with ASCII corner ornaments and
  lt-section-header; filter bar moved to lt-toolbar with lt-search icon
- Recently Resolved table replaced with lt-timeline component
- updateStatusBar() and updateHostGrid() keep stat card values live

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-04-30 22:19:50 -04:00
6 changed files with 128 additions and 54 deletions
+11
View File
@@ -89,6 +89,14 @@ function updateStatusBar(summary, lastCheck, daemonOk) {
alertBadge.style.display = total ? '' : 'none'; alertBadge.style.display = total ? '' : 'none';
} }
// Update stat cards
const scCrit = document.getElementById('stat-critical-val');
const scWarn = document.getElementById('stat-warning-val');
if (scCrit) scCrit.textContent = critCount;
if (scWarn) scWarn.textContent = warnCount;
const statCritCard = document.getElementById('stat-critical');
if (statCritCard) statCritCard.classList.toggle('lt-stat-card--alert', critCount > 0);
// Stale data banner: warn if last_check is older than 15 minutes // Stale data banner: warn if last_check is older than 15 minutes
let staleBanner = document.getElementById('stale-banner'); let staleBanner = document.getElementById('stale-banner');
if (lastCheck) { if (lastCheck) {
@@ -112,6 +120,9 @@ function updateStatusBar(summary, lastCheck, daemonOk) {
} }
function updateHostGrid(hosts) { function updateHostGrid(hosts) {
const scHosts = document.getElementById('stat-hosts-val');
if (scHosts) scHosts.textContent = Object.keys(hosts).length;
for (const [name, host] of Object.entries(hosts)) { for (const [name, host] of Object.entries(hosts)) {
const card = document.querySelector(`.host-card[data-host="${CSS.escape(name)}"]`); const card = document.querySelector(`.host-card[data-host="${CSS.escape(name)}"]`);
if (!card) continue; if (!card) continue;
+11
View File
@@ -995,6 +995,17 @@
.diag-pulse-link a { color: var(--cyan); } .diag-pulse-link a { color: var(--cyan); }
.diag-pulse-link a:hover { text-shadow: var(--glow-cyan); } .diag-pulse-link a:hover { text-shadow: var(--glow-cyan); }
/* ── Stat card alert variant (pulsing border when critical > 0) ─── */
.lt-stat-card--alert {
border-color: var(--red) !important;
box-shadow: 0 0 8px rgba(255,45,85,.25) !important;
animation: topo-pulse-down 2s ease-in-out infinite;
}
.lt-stat-card--alert::before { background: var(--red); box-shadow: var(--glow-red); }
/* ── lt-frame inside g-section: no extra bottom margin ────────────── */
.g-section > .lt-frame { margin-bottom: 0; }
/* ── Responsive ───────────────────────────────────────────────────── */ /* ── Responsive ───────────────────────────────────────────────────── */
@media (max-width: 768px) { @media (max-width: 768px) {
.host-grid { grid-template-columns: 1fr; } .host-grid { grid-template-columns: 1fr; }
+78 -31
View File
@@ -26,6 +26,42 @@
</div> </div>
</div> </div>
<!-- ── Stats summary cards ──────────────────────────────────────────── -->
<div class="lt-stats-grid">
<div class="lt-stat-card{% if summary.critical %} lt-stat-card--alert{% endif %}"
id="stat-critical" role="button" tabindex="0"
data-stat-filter="critical" aria-label="{{ summary.critical or 0 }} critical alerts">
<span class="lt-stat-icon" aria-hidden="true" style="color:var(--red);text-shadow:var(--glow-red)"></span>
<div class="lt-stat-info">
<span class="lt-stat-value" id="stat-critical-val" style="color:var(--red)">{{ summary.critical or 0 }}</span>
<span class="lt-stat-label">Critical</span>
</div>
</div>
<div class="lt-stat-card"
id="stat-warning" role="button" tabindex="0"
data-stat-filter="warning" aria-label="{{ summary.warning or 0 }} warning alerts">
<span class="lt-stat-icon" aria-hidden="true" style="color:var(--amber)"></span>
<div class="lt-stat-info">
<span class="lt-stat-value" id="stat-warning-val" style="color:var(--amber)">{{ summary.warning or 0 }}</span>
<span class="lt-stat-label">Warning</span>
</div>
</div>
<div class="lt-stat-card" id="stat-hosts" aria-label="Monitored hosts">
<span class="lt-stat-icon" aria-hidden="true" style="color:var(--cyan)"></span>
<div class="lt-stat-info">
<span class="lt-stat-value" id="stat-hosts-val" style="color:var(--cyan)">{{ snapshot.hosts | length }}</span>
<span class="lt-stat-label">Hosts</span>
</div>
</div>
<div class="lt-stat-card" id="stat-resolved" aria-label="{{ recent_resolved | length }} alerts resolved in last 24 hours">
<span class="lt-stat-icon" aria-hidden="true" style="color:var(--green);text-shadow:var(--glow)"></span>
<div class="lt-stat-info">
<span class="lt-stat-value" id="stat-resolved-val" style="color:var(--green)">{{ recent_resolved | length }}</span>
<span class="lt-stat-label">Resolved 24h</span>
</div>
</div>
</div>
<!-- ── Network topology + host grid ───────────────────────────────── --> <!-- ── Network topology + host grid ───────────────────────────────── -->
<section class="g-section"> <section class="g-section">
<div class="g-section-header"> <div class="g-section-header">
@@ -282,10 +318,13 @@
<h2 class="g-section-title">Active Alerts</h2> <h2 class="g-section-title">Active Alerts</h2>
<span class="g-section-badge" id="alert-count-badge" <span class="g-section-badge" id="alert-count-badge"
{% if not summary.critical and not summary.warning %}style="display:none"{% endif %}>{{ (summary.critical or 0) + (summary.warning or 0) }}</span> {% if not summary.critical and not summary.warning %}style="display:none"{% endif %}>{{ (summary.critical or 0) + (summary.warning or 0) }}</span>
<div class="g-section-actions"> </div>
<div class="events-filter-bar"> <div class="lt-toolbar">
<input type="search" class="lt-input lt-input-sm" id="events-search" <div class="lt-toolbar-left">
<div class="lt-search">
<input type="search" class="lt-input lt-search-input" id="events-search"
placeholder="Filter by target, type, description…" autocomplete="off"> placeholder="Filter by target, type, description…" autocomplete="off">
</div>
<div class="sev-pills"> <div class="sev-pills">
<button type="button" class="pill active" data-sev="">All</button> <button type="button" class="pill active" data-sev="">All</button>
<button type="button" class="pill" data-sev="critical">Critical</button> <button type="button" class="pill" data-sev="critical">Critical</button>
@@ -293,7 +332,10 @@
</div> </div>
</div> </div>
</div> </div>
</div> <div class="lt-frame">
<span class="lt-frame-bl">&#x255A;</span>
<span class="lt-frame-br">&#x255D;</span>
<div class="lt-section-header">Alert Queue</div>
<div id="events-table-wrap"> <div id="events-table-wrap">
{% if events %} {% if events %}
{% if total_active is defined and total_active > events|length %} {% if total_active is defined and total_active > events|length %}
@@ -364,7 +406,8 @@
<div class="lt-empty-state-title">No active alerts</div> <div class="lt-empty-state-title">No active alerts</div>
</div> </div>
{% endif %} {% endif %}
</div> </div><!-- /#events-table-wrap -->
</div><!-- /.lt-frame -->
</section> </section>
<!-- ── Recently Resolved (last 24h) ───────────────────────────────── --> <!-- ── Recently Resolved (last 24h) ───────────────────────────────── -->
@@ -374,34 +417,24 @@
<h2 class="g-section-title">Recently Resolved</h2> <h2 class="g-section-title">Recently Resolved</h2>
<span class="g-section-badge g-section-badge-resolved">{{ recent_resolved | length }} in last 24h</span> <span class="g-section-badge g-section-badge-resolved">{{ recent_resolved | length }} in last 24h</span>
</div> </div>
<div class="lt-table-wrap"> <div class="lt-timeline">
<table class="lt-table">
<caption class="lt-sr-only">Recently resolved alerts</caption>
<thead>
<tr>
<th>Sev</th>
<th>Type</th>
<th>Target</th>
<th>Detail</th>
<th>Resolved</th>
<th>Duration</th>
</tr>
</thead>
<tbody>
{% for e in recent_resolved %} {% for e in recent_resolved %}
<tr class="row-resolved"> {%- set dot_cls = 'lt-timeline-item--green' if e.severity == 'info' else 'lt-timeline-item--dim' -%}
<td><span class="lt-badge badge-resolved">{{ e.severity }}</span></td> <div class="lt-timeline-item {{ dot_cls }}">
<td>{{ e.event_type | replace('_', ' ') }}</td> <div class="lt-timeline-meta">
<td><strong>{{ e.target_name }}</strong></td> <strong class="lt-timeline-actor">{{ e.target_name }}</strong>
<td>{{ e.target_detail or '' }}</td> {% if e.target_detail %}<span>· {{ e.target_detail }}</span>{% endif %}
<td class="ts-cell"> <span class="lt-timeline-time event-age" data-ts="{{ e.resolved_at }}">{{ e.resolved_at }}</span>
<span class="event-age" data-ts="{{ e.resolved_at }}">{{ e.resolved_at }}</span> </div>
</td> <div class="lt-timeline-body">
<td class="ts-cell event-duration" data-first="{{ e.first_seen }}" data-resolved="{{ e.resolved_at }}"></td> {{ e.event_type | replace('_', ' ') }}
</tr> &nbsp;·&nbsp;
<span class="lt-badge badge-resolved">{{ e.severity }}</span>
&nbsp;·&nbsp; duration
<span class="event-duration" data-first="{{ e.first_seen }}" data-resolved="{{ e.resolved_at }}"></span>
</div>
</div>
{% endfor %} {% endfor %}
</tbody>
</table>
</div> </div>
</section> </section>
{% endif %} {% endif %}
@@ -529,5 +562,19 @@
// Re-apply filter after dynamic table updates // Re-apply filter after dynamic table updates
new MutationObserver(applyEventsFilter) new MutationObserver(applyEventsFilter)
.observe(document.getElementById('events-table-wrap'), { childList: true, subtree: true }); .observe(document.getElementById('events-table-wrap'), { childList: true, subtree: true });
// Stat card clicks — filter events table by severity
document.querySelectorAll('.lt-stat-card[data-stat-filter]').forEach(card => {
card.addEventListener('click', () => {
const sev = card.dataset.statFilter;
document.querySelectorAll('.sev-pills .pill').forEach(p => p.classList.remove('active'));
const matchPill = document.querySelector(`.sev-pills .pill[data-sev="${sev}"]`);
if (matchPill) matchPill.classList.add('active');
_filterSev = sev;
applyEventsFilter();
document.getElementById('events-table-wrap')?.scrollIntoView({ behavior: 'smooth', block: 'start' });
});
card.addEventListener('keydown', e => { if (e.key === 'Enter' || e.key === ' ') { e.preventDefault(); card.click(); } });
});
</script> </script>
{% endblock %} {% endblock %}
+6 -4
View File
@@ -3,12 +3,14 @@
{% block content %} {% block content %}
<div class="g-page-header"> <div class="lt-page-header">
<h1 class="g-page-title">Network Inspector</h1> <div>
<p class="g-page-sub"> <h1 class="lt-page-title">Network Inspector</h1>
<p class="g-page-sub" style="margin-top:4px">
Visual switch chassis diagrams. Click a port to see detailed stats and LLDP path debug. Visual switch chassis diagrams. Click a port to see detailed stats and LLDP path debug.
<span id="inspector-updated" style="margin-left:10px; color:var(--text-muted); font-size:.85em;"></span> <span id="inspector-updated" style="margin-left:8px"></span>
</p> </p>
</div>
</div> </div>
<div class="inspector-layout"> <div class="inspector-layout">
+6 -5
View File
@@ -3,13 +3,14 @@
{% block content %} {% block content %}
<div class="g-page-header"> <div class="lt-page-header">
<h1 class="g-page-title">Link Debug</h1> <div>
<p class="g-page-sub"> <h1 class="lt-page-title">Link Debug</h1>
<p class="g-page-sub" style="margin-top:4px">
Per-interface stats: speed, duplex, SFP optical levels, TX/RX rates, errors, and carrier changes. Per-interface stats: speed, duplex, SFP optical levels, TX/RX rates, errors, and carrier changes.
Data collected via Prometheus node_exporter + SSH ethtool (servers) and UniFi API (switches) every poll cycle. <span id="links-updated" style="margin-left:8px"></span>
<span id="links-updated" style="margin-left:10px; color:var(--text-muted); font-size:.85em;"></span>
</p> </p>
</div>
</div> </div>
<div id="links-container"> <div id="links-container">
+5 -3
View File
@@ -3,9 +3,11 @@
{% block content %} {% block content %}
<div class="g-page-header"> <div class="lt-page-header">
<h1 class="g-page-title">Alert Suppressions</h1> <div>
<p class="g-page-sub">Manage maintenance windows and per-target alert suppression rules.</p> <h1 class="lt-page-title">Alert Suppressions</h1>
<p class="g-page-sub" style="margin-top:4px">Manage maintenance windows and per-target alert suppression rules.</p>
</div>
</div> </div>
<!-- ── Create suppression ─────────────────────────────────────────── --> <!-- ── Create suppression ─────────────────────────────────────────── -->