Compare commits
3 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 906869f425 | |||
| c027b5422a | |||
| d3e8191f26 |
@@ -160,6 +160,16 @@ def index():
|
|||||||
last_check = db.get_state('last_check', 'Never')
|
last_check = db.get_state('last_check', 'Never')
|
||||||
snapshot = json.loads(snapshot_raw) if snapshot_raw else {}
|
snapshot = json.loads(snapshot_raw) if snapshot_raw else {}
|
||||||
suppressions = db.get_active_suppressions()
|
suppressions = db.get_active_suppressions()
|
||||||
|
for ev in events:
|
||||||
|
sup_type = (
|
||||||
|
'unifi_device' if ev.get('event_type') == 'unifi_device_offline'
|
||||||
|
else 'interface' if ev.get('event_type') == 'interface_down'
|
||||||
|
else 'host'
|
||||||
|
)
|
||||||
|
ev['is_suppressed'] = db.check_suppressed(
|
||||||
|
suppressions, sup_type,
|
||||||
|
ev.get('target_name', ''), ev.get('target_detail', '') or '',
|
||||||
|
)
|
||||||
recent_resolved = db.get_recent_resolved(hours=24, limit=10)
|
recent_resolved = db.get_recent_resolved(hours=24, limit=10)
|
||||||
return render_template(
|
return render_template(
|
||||||
'index.html',
|
'index.html',
|
||||||
@@ -214,6 +224,17 @@ def suppressions_page():
|
|||||||
@require_auth
|
@require_auth
|
||||||
def api_status():
|
def api_status():
|
||||||
active = db.get_active_events(limit=_PAGE_LIMIT)
|
active = db.get_active_events(limit=_PAGE_LIMIT)
|
||||||
|
suppressions = db.get_active_suppressions()
|
||||||
|
for ev in active:
|
||||||
|
sup_type = (
|
||||||
|
'unifi_device' if ev.get('event_type') == 'unifi_device_offline'
|
||||||
|
else 'interface' if ev.get('event_type') == 'interface_down'
|
||||||
|
else 'host'
|
||||||
|
)
|
||||||
|
ev['is_suppressed'] = db.check_suppressed(
|
||||||
|
suppressions, sup_type,
|
||||||
|
ev.get('target_name', ''), ev.get('target_detail', '') or '',
|
||||||
|
)
|
||||||
last_check = db.get_state('last_check', 'Never')
|
last_check = db.get_state('last_check', 'Never')
|
||||||
return jsonify({
|
return jsonify({
|
||||||
'summary': db.get_status_summary(),
|
'summary': db.get_status_summary(),
|
||||||
|
|||||||
+5
-2
@@ -222,9 +222,12 @@ function updateEventsTable(events, totalActive) {
|
|||||||
? `<a href="${lt.escHtml(ticketBase)}${lt.escHtml(String(e.ticket_id))}" target="_blank"
|
? `<a href="${lt.escHtml(ticketBase)}${lt.escHtml(String(e.ticket_id))}" target="_blank"
|
||||||
class="ticket-link">#${e.ticket_id}</a>`
|
class="ticket-link">#${e.ticket_id}</a>`
|
||||||
: '–';
|
: '–';
|
||||||
|
const supBadge = e.is_suppressed
|
||||||
|
? `<span class="lt-badge badge-suppressed" title="Alert suppressed">🔕 sup</span>`
|
||||||
|
: '';
|
||||||
return `
|
return `
|
||||||
<tr class="row-${e.severity}">
|
<tr class="row-${e.severity}${e.is_suppressed ? ' row-suppressed' : ''}">
|
||||||
<td><span class="lt-badge badge-${e.severity}">${e.severity}</span></td>
|
<td><span class="lt-badge badge-${e.severity}">${e.severity}</span>${supBadge}</td>
|
||||||
<td>${lt.escHtml(e.event_type.replace(/_/g,' '))}</td>
|
<td>${lt.escHtml(e.event_type.replace(/_/g,' '))}</td>
|
||||||
<td><strong>${lt.escHtml(e.target_name)}</strong></td>
|
<td><strong>${lt.escHtml(e.target_name)}</strong></td>
|
||||||
<td>${lt.escHtml(e.target_detail || '–')}</td>
|
<td>${lt.escHtml(e.target_detail || '–')}</td>
|
||||||
|
|||||||
@@ -203,6 +203,8 @@
|
|||||||
.lt-table tr.row-warning td { background: rgba(255,107,0,.04); }
|
.lt-table tr.row-warning td { background: rgba(255,107,0,.04); }
|
||||||
.lt-table tr.row-warning td:first-child { border-left: 2px solid var(--orange); }
|
.lt-table tr.row-warning td:first-child { border-left: 2px solid var(--orange); }
|
||||||
.lt-table tr.row-resolved td { opacity: .65; }
|
.lt-table tr.row-resolved td { opacity: .65; }
|
||||||
|
.lt-table tr.row-suppressed td { opacity: .6; }
|
||||||
|
.lt-table tr.row-suppressed td:first-child{ border-left-color: var(--text-muted) !important; }
|
||||||
|
|
||||||
/* ── Table size modifier ─────────────────────────────────────────── */
|
/* ── Table size modifier ─────────────────────────────────────────── */
|
||||||
.lt-table-sm th,
|
.lt-table-sm th,
|
||||||
@@ -408,6 +410,9 @@
|
|||||||
background: linear-gradient(to bottom, var(--cyan), var(--green));
|
background: linear-gradient(to bottom, var(--cyan), var(--green));
|
||||||
opacity: .7;
|
opacity: .7;
|
||||||
}
|
}
|
||||||
|
.topo-vc-wire--wan { background: linear-gradient(to bottom, var(--cyan), rgba(0,212,255,.3)); opacity: .7; }
|
||||||
|
.topo-vc-wire--10g { background: var(--amber); opacity: .6; }
|
||||||
|
.topo-vc-wire--mgmt { background: var(--border-color); opacity: .5; }
|
||||||
/* Blurred copy of the wire for a soft glow halo */
|
/* Blurred copy of the wire for a soft glow halo */
|
||||||
.topo-vc-wire::before {
|
.topo-vc-wire::before {
|
||||||
content: '';
|
content: '';
|
||||||
@@ -465,6 +470,7 @@
|
|||||||
.topo-v2-sub { font-size: .58em; color: var(--text-muted); letter-spacing: .02em; }
|
.topo-v2-sub { font-size: .58em; color: var(--text-muted); letter-spacing: .02em; }
|
||||||
.topo-v2-vlan { font-size: .54em; color: var(--cyan); opacity: .75; }
|
.topo-v2-vlan { font-size: .54em; color: var(--cyan); opacity: .75; }
|
||||||
|
|
||||||
|
.topo-v2-host--bus { min-width: 80px; max-width: 96px; }
|
||||||
.topo-v2-internet { border-color: var(--cyan); color: var(--cyan); text-shadow: var(--glow-cyan); box-shadow: 0 0 12px rgba(0,212,255,.12); }
|
.topo-v2-internet { border-color: var(--cyan); color: var(--cyan); text-shadow: var(--glow-cyan); box-shadow: 0 0 12px rgba(0,212,255,.12); }
|
||||||
.topo-v2-router { border-color: var(--cyan); color: var(--cyan); text-shadow: var(--glow-cyan); box-shadow: 0 0 12px rgba(0,212,255,.14); }
|
.topo-v2-router { border-color: var(--cyan); color: var(--cyan); text-shadow: var(--glow-cyan); box-shadow: 0 0 12px rgba(0,212,255,.14); }
|
||||||
.topo-v2-switch { border-color: var(--amber); color: var(--amber); text-shadow: var(--glow-amber); box-shadow: 0 0 12px rgba(255,179,0,.12); }
|
.topo-v2-switch { border-color: var(--amber); color: var(--amber); text-shadow: var(--glow-amber); box-shadow: 0 0 12px rgba(255,179,0,.12); }
|
||||||
@@ -594,6 +600,10 @@
|
|||||||
color: var(--text-muted);
|
color: var(--text-muted);
|
||||||
font-family: var(--font);
|
font-family: var(--font);
|
||||||
}
|
}
|
||||||
|
.topo-legend-item--offrack {
|
||||||
|
border: 1px dashed var(--border-color);
|
||||||
|
padding: 1px 5px;
|
||||||
|
}
|
||||||
.topo-legend-line-10g { width: 24px; height: 2px; background: var(--green); display: inline-block; box-shadow: 0 0 4px rgba(0,255,136,.5); }
|
.topo-legend-line-10g { width: 24px; height: 2px; background: var(--green); display: inline-block; box-shadow: 0 0 4px rgba(0,255,136,.5); }
|
||||||
.topo-legend-line-1g { width: 24px; height: 0; border-top: 2px dashed var(--amber); display: inline-block; }
|
.topo-legend-line-1g { width: 24px; height: 0; border-top: 2px dashed var(--amber); display: inline-block; }
|
||||||
.topo-legend-line-wan { width: 24px; height: 2px; background: linear-gradient(to right, var(--cyan), var(--green)); display: inline-block; }
|
.topo-legend-line-wan { width: 24px; height: 2px; background: linear-gradient(to right, var(--cyan), var(--green)); display: inline-block; }
|
||||||
|
|||||||
+13
-10
@@ -111,8 +111,11 @@
|
|||||||
<tbody>
|
<tbody>
|
||||||
{% for e in events %}
|
{% for e in events %}
|
||||||
{% if e.severity != 'info' %}
|
{% if e.severity != 'info' %}
|
||||||
<tr class="row-{{ e.severity }}">
|
<tr class="row-{{ e.severity }}{% if e.is_suppressed %} row-suppressed{% endif %}">
|
||||||
<td><span class="lt-badge badge-{{ e.severity }}">{{ e.severity }}</span></td>
|
<td>
|
||||||
|
<span class="lt-badge badge-{{ e.severity }}">{{ e.severity }}</span>
|
||||||
|
{% if e.is_suppressed %}<span class="lt-badge badge-suppressed" title="Alert suppressed">🔕 sup</span>{% endif %}
|
||||||
|
</td>
|
||||||
<td>{{ e.event_type | replace('_', ' ') }}</td>
|
<td>{{ e.event_type | replace('_', ' ') }}</td>
|
||||||
<td><strong>{{ e.target_name }}</strong></td>
|
<td><strong>{{ e.target_name }}</strong></td>
|
||||||
<td>{{ e.target_detail or '–' }}</td>
|
<td>{{ e.target_detail or '–' }}</td>
|
||||||
@@ -185,9 +188,9 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<!-- WAN wire: cyan → green gradient, labeled -->
|
<!-- WAN wire: cyan → WAN gradient -->
|
||||||
<div class="topo-vc">
|
<div class="topo-vc">
|
||||||
<div class="topo-vc-wire" style="background:linear-gradient(to bottom,var(--cyan),rgba(0,212,255,.3)); opacity:.7;"></div>
|
<div class="topo-vc-wire topo-vc-wire--wan"></div>
|
||||||
<span class="topo-vc-label">WAN · 10G SFP+</span>
|
<span class="topo-vc-label">WAN · 10G SFP+</span>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@@ -205,7 +208,7 @@
|
|||||||
|
|
||||||
<!-- UDM-Pro → USW-Agg (10G SFP+) -->
|
<!-- UDM-Pro → USW-Agg (10G SFP+) -->
|
||||||
<div class="topo-vc">
|
<div class="topo-vc">
|
||||||
<div class="topo-vc-wire" style="background:var(--amber);opacity:.6;"></div>
|
<div class="topo-vc-wire topo-vc-wire--10g"></div>
|
||||||
<span class="topo-vc-label">10G SFP+</span>
|
<span class="topo-vc-label">10G SFP+</span>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@@ -224,7 +227,7 @@
|
|||||||
|
|
||||||
<!-- USW-Agg → Pro 24 PoE (10G trunk) -->
|
<!-- USW-Agg → Pro 24 PoE (10G trunk) -->
|
||||||
<div class="topo-vc">
|
<div class="topo-vc">
|
||||||
<div class="topo-vc-wire" style="background:var(--amber);opacity:.6;"></div>
|
<div class="topo-vc-wire topo-vc-wire--10g"></div>
|
||||||
<span class="topo-vc-label">10G trunk</span>
|
<span class="topo-vc-label">10G trunk</span>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@@ -243,7 +246,7 @@
|
|||||||
|
|
||||||
<!-- Pro 24 PoE → host bus section -->
|
<!-- Pro 24 PoE → host bus section -->
|
||||||
<div class="topo-vc">
|
<div class="topo-vc">
|
||||||
<div class="topo-vc-wire" style="background:var(--border-color);opacity:.5;"></div>
|
<div class="topo-vc-wire topo-vc-wire--mgmt"></div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<!-- ══════════════════════════════════════════════════════════════
|
<!-- ══════════════════════════════════════════════════════════════
|
||||||
@@ -283,8 +286,8 @@
|
|||||||
<div class="topo-v2-wire-1g" data-host="{{ hname }}" title="1G → Pro 24 PoE"></div>
|
<div class="topo-v2-wire-1g" data-host="{{ hname }}" title="1G → Pro 24 PoE"></div>
|
||||||
</div>
|
</div>
|
||||||
<!-- host box -->
|
<!-- host box -->
|
||||||
<div class="topo-v2-node topo-v2-host topo-host topo-v2-status-{{ st }}{{ ' topo-v2-offrack' if off_rack else '' }}"
|
<div class="topo-v2-node topo-v2-host topo-host topo-v2-status-{{ st }}{{ ' topo-v2-offrack' if off_rack else '' }} topo-v2-host--bus"
|
||||||
data-host="{{ hname }}" style="min-width:80px; max-width:96px;">
|
data-host="{{ hname }}">
|
||||||
<span class="topo-v2-icon">▣</span>
|
<span class="topo-v2-icon">▣</span>
|
||||||
<span class="topo-v2-label">{{ hlabel }}</span>
|
<span class="topo-v2-label">{{ hlabel }}</span>
|
||||||
<span class="topo-v2-sub">{{ hsub }}</span>
|
<span class="topo-v2-sub">{{ hsub }}</span>
|
||||||
@@ -302,7 +305,7 @@
|
|||||||
<div class="topo-legend-item"><span class="topo-legend-line-wan"></span> WAN / uplink</div>
|
<div class="topo-legend-item"><span class="topo-legend-line-wan"></span> WAN / uplink</div>
|
||||||
<div class="topo-legend-item"><span class="topo-legend-line-10g"></span> 10G SFP+ (Ceph / VLAN90)</div>
|
<div class="topo-legend-item"><span class="topo-legend-line-10g"></span> 10G SFP+ (Ceph / VLAN90)</div>
|
||||||
<div class="topo-legend-item"><span class="topo-legend-line-1g"></span> 1G DHCP (mgmt)</div>
|
<div class="topo-legend-item"><span class="topo-legend-line-1g"></span> 1G DHCP (mgmt)</div>
|
||||||
<div class="topo-legend-item" style="border:1px dashed var(--border-color); padding:1px 5px; font-size:.56em; color:var(--text-muted);">dashed border = off-rack</div>
|
<div class="topo-legend-item topo-legend-item--offrack">dashed border = off-rack</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
</div><!-- /topo-v2 -->
|
</div><!-- /topo-v2 -->
|
||||||
|
|||||||
@@ -6,7 +6,7 @@
|
|||||||
<div class="lt-page-header">
|
<div class="lt-page-header">
|
||||||
<div>
|
<div>
|
||||||
<h1 class="lt-page-title">Network Inspector</h1>
|
<h1 class="lt-page-title">Network Inspector</h1>
|
||||||
<p class="g-page-sub" style="margin-top:4px">
|
<p class="g-page-sub">
|
||||||
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:8px"></span>
|
<span id="inspector-updated" style="margin-left:8px"></span>
|
||||||
</p>
|
</p>
|
||||||
|
|||||||
@@ -6,7 +6,7 @@
|
|||||||
<div class="lt-page-header">
|
<div class="lt-page-header">
|
||||||
<div>
|
<div>
|
||||||
<h1 class="lt-page-title">Link Debug</h1>
|
<h1 class="lt-page-title">Link Debug</h1>
|
||||||
<p class="g-page-sub" style="margin-top:4px">
|
<p class="g-page-sub">
|
||||||
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.
|
||||||
<span id="links-updated" style="margin-left:8px"></span>
|
<span id="links-updated" style="margin-left:8px"></span>
|
||||||
</p>
|
</p>
|
||||||
|
|||||||
@@ -6,7 +6,7 @@
|
|||||||
<div class="lt-page-header">
|
<div class="lt-page-header">
|
||||||
<div>
|
<div>
|
||||||
<h1 class="lt-page-title">Alert Suppressions</h1>
|
<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>
|
<p class="g-page-sub">Manage maintenance windows and per-target alert suppression rules.</p>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@@ -188,7 +188,7 @@
|
|||||||
<span class="lt-frame-bl">╚</span>
|
<span class="lt-frame-bl">╚</span>
|
||||||
<span class="lt-frame-br">╝</span>
|
<span class="lt-frame-br">╝</span>
|
||||||
<div class="lt-section-header">Host & Interface Reference</div>
|
<div class="lt-section-header">Host & Interface Reference</div>
|
||||||
<div style="padding:12px 14px">
|
<div class="lt-section-body">
|
||||||
<div class="targets-grid">
|
<div class="targets-grid">
|
||||||
{% for name, host in snapshot.hosts.items() %}
|
{% for name, host in snapshot.hosts.items() %}
|
||||||
<div class="target-card">
|
<div class="target-card">
|
||||||
|
|||||||
Reference in New Issue
Block a user