Feature: show suppression status on active alert rows
Lint / Python (flake8) (push) Successful in 45s
Lint / JS (eslint) (push) Successful in 7s
Security / Python Security (bandit) (push) Successful in 47s
Test / Python Tests (pytest) (push) Successful in 1m11s
Lint / Notify on failure (push) Has been skipped
Lint / Deploy (push) Successful in 3s
Lint / Python (flake8) (push) Successful in 45s
Lint / JS (eslint) (push) Successful in 7s
Security / Python Security (bandit) (push) Successful in 47s
Test / Python Tests (pytest) (push) Successful in 1m11s
Lint / Notify on failure (push) Has been skipped
Lint / Deploy (push) Successful in 3s
Active events now carry an is_suppressed boolean (added in api_status() and the index() route via check_suppressed() against the pre-loaded suppression list). The events table renders a muted '🔕 sup' badge next to the severity and dims the entire row (.row-suppressed) so operators can immediately see which firing alerts are silenced. Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
@@ -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,
|
||||||
|
|||||||
@@ -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>
|
||||||
|
|||||||
Reference in New Issue
Block a user