2026-03-01 23:03:18 -05:00
'use strict' ;
2026-03-17 20:43:08 -04:00
// ── Auto-redirect on auth timeout ─────────────────────────────────────
2026-04-18 23:46:44 -04:00
// Wraps fetch so a 401 (Authelia session expired) forces a full reload.
// lt.api uses fetch internally, so this covers all API calls too.
2026-03-17 20:43:08 -04:00
( function ( ) {
const _fetch = window . fetch ;
window . fetch = async function ( ... args ) {
const resp = await _fetch ( ... args ) ;
2026-04-19 23:35:02 -04:00
if ( resp . status === 401 ) {
window . location . reload ( ) ;
throw new Error ( 'Session expired — reloading' ) ;
}
2026-03-17 20:43:08 -04:00
return resp ;
} ;
} ) ( ) ;
2026-04-18 23:46:44 -04:00
// ── Toast notifications — thin wrapper over lt.toast ──────────────────
2026-03-01 23:03:18 -05:00
function showToast ( msg , type = 'success' ) {
2026-03-14 21:40:20 -04:00
if ( type === 'error' ) return lt . toast . error ( msg ) ;
if ( type === 'warning' ) return lt . toast . warning ( msg ) ;
if ( type === 'info' ) return lt . toast . info ( msg ) ;
return lt . toast . success ( msg ) ;
2025-02-07 21:03:31 -05:00
}
2026-04-18 23:46:44 -04:00
// ── Normalise UTC timestamp string for Date() parsing ─────────────────
// Server returns "2026-03-14 14:14:21 UTC"; Date() needs ISO 8601.
function _toIso ( s ) {
if ( ! s ) return s ;
return s . replace ( ' UTC' , 'Z' ) . replace ( ' ' , 'T' ) ;
}
2026-03-01 23:03:18 -05:00
// ── Dashboard auto-refresh ────────────────────────────────────────────
async function refreshAll ( ) {
2026-04-19 23:35:02 -04:00
const refreshBtn = document . querySelector ( '[data-action="refresh"]' ) ;
if ( refreshBtn ) refreshBtn . classList . add ( 'is-loading' ) ;
2026-03-01 23:03:18 -05:00
try {
2026-04-19 23:35:02 -04:00
const [ netResult , statusResult ] = await Promise . allSettled ( [
2026-04-18 23:46:44 -04:00
lt . api . get ( '/api/network' ) ,
lt . api . get ( '/api/status' ) ,
2026-03-01 23:03:18 -05:00
] ) ;
2026-04-19 23:35:02 -04:00
if ( netResult . status === 'fulfilled' ) {
const net = netResult . value ;
updateHostGrid ( net . hosts || { } ) ;
updateUnifiTable ( net . unifi || [ ] ) ;
updateTopology ( net . hosts || { } ) ;
} else {
2026-04-29 17:50:00 -04:00
showToast ( 'Network data unavailable' , 'warning' ) ;
2026-04-19 23:35:02 -04:00
}
if ( statusResult . status === 'fulfilled' ) {
const status = statusResult . value ;
updateEventsTable ( status . events || [ ] , status . total _active ) ;
updateStatusBar ( status . summary || { } , status . last _check || '' , status . daemon _ok ) ;
} else {
2026-04-29 17:50:00 -04:00
showToast ( 'Status data unavailable' , 'warning' ) ;
2026-04-19 23:35:02 -04:00
}
} finally {
if ( refreshBtn ) refreshBtn . classList . remove ( 'is-loading' ) ;
2026-03-01 23:03:18 -05:00
}
2025-02-08 00:32:25 -05:00
}
2026-04-19 23:35:02 -04:00
function updateStatusBar ( summary , lastCheck , daemonOk ) {
2026-03-01 23:03:18 -05:00
const bar = document . querySelector ( '.status-chips' ) ;
if ( ! bar ) return ;
const chips = [ ] ;
2026-04-19 23:35:02 -04:00
if ( daemonOk === false ) chips . push ( '<span class="chip chip-critical">⚠ MONITOR OFFLINE</span>' ) ;
2026-03-02 12:43:11 -05:00
if ( summary . critical ) chips . push ( ` <span class="chip chip-critical">● ${ summary . critical } CRITICAL</span> ` ) ;
if ( summary . warning ) chips . push ( ` <span class="chip chip-warning">● ${ summary . warning } WARNING</span> ` ) ;
2026-04-19 23:35:02 -04:00
if ( ! summary . critical && ! summary . warning && daemonOk !== false ) chips . push ( '<span class="chip chip-ok">✔ ALL SYSTEMS NOMINAL</span>' ) ;
2026-03-01 23:03:18 -05:00
bar . innerHTML = chips . join ( '' ) ;
const lc = document . getElementById ( 'last-check' ) ;
2026-03-02 12:43:11 -05:00
if ( lc && lastCheck ) lc . textContent = lastCheck ;
2026-03-14 21:35:32 -04:00
const critCount = summary . critical || 0 ;
const warnCount = summary . warning || 0 ;
2026-04-18 23:46:44 -04:00
if ( critCount ) document . title = ` ( ${ critCount } CRIT) GANDALF ` ;
else if ( warnCount ) document . title = ` ( ${ warnCount } WARN) GANDALF ` ;
else document . title = 'GANDALF' ;
2026-03-14 21:35:32 -04:00
// Stale data banner: warn if last_check is older than 15 minutes
let staleBanner = document . getElementById ( 'stale-banner' ) ;
if ( lastCheck ) {
2026-04-18 23:46:44 -04:00
const checkAge = ( Date . now ( ) - new Date ( _toIso ( lastCheck ) ) ) / 1000 ;
if ( checkAge > 900 ) {
2026-03-14 21:35:32 -04:00
if ( ! staleBanner ) {
staleBanner = document . createElement ( 'div' ) ;
staleBanner . id = 'stale-banner' ;
2026-04-29 23:37:47 -04:00
staleBanner . className = 'lt-alert lt-alert--warning' ;
staleBanner . innerHTML = '<span class="lt-alert-icon">⚠</span><div class="lt-alert-body"><div class="lt-alert-msg"></div></div>' ;
2026-04-18 21:01:20 -04:00
document . querySelector ( '.lt-main' ) . prepend ( staleBanner ) ;
2026-03-14 21:35:32 -04:00
}
const mins = Math . floor ( checkAge / 60 ) ;
2026-04-29 23:37:47 -04:00
staleBanner . querySelector ( '.lt-alert-msg' ) . textContent =
` Monitoring data is stale — last check was ${ mins } minute ${ mins !== 1 ? 's' : '' } ago. The monitor daemon may be down. ` ;
2026-03-14 21:35:32 -04:00
staleBanner . style . display = '' ;
} else if ( staleBanner ) {
staleBanner . style . display = 'none' ;
}
}
2025-02-07 23:54:28 -05:00
}
2026-03-01 23:03:18 -05:00
function updateHostGrid ( hosts ) {
for ( const [ name , host ] of Object . entries ( hosts ) ) {
const card = document . querySelector ( ` .host-card[data-host=" ${ CSS . escape ( name ) } "] ` ) ;
if ( ! card ) continue ;
card . className = card . className . replace ( /host-card-(up|down|degraded|unknown)/g , '' ) ;
card . classList . add ( ` host-card- ${ host . status } ` ) ;
const dot = card . querySelector ( '.host-status-dot' ) ;
if ( dot ) dot . className = ` host-status-dot dot- ${ host . status } ` ;
2025-02-07 23:54:28 -05:00
2026-03-01 23:03:18 -05:00
const ifaceList = card . querySelector ( '.iface-list' ) ;
if ( ifaceList && host . interfaces && Object . keys ( host . interfaces ) . length > 0 ) {
ifaceList . innerHTML = Object . entries ( host . interfaces )
. sort ( ( [ a ] , [ b ] ) => a . localeCompare ( b ) )
. map ( ( [ iface , state ] ) => `
<div class="iface-row">
<span class="iface-dot dot- ${ state } "></span>
2026-04-18 23:46:44 -04:00
<span class="iface-name"> ${ lt . escHtml ( iface ) } </span>
2026-03-01 23:03:18 -05:00
<span class="iface-state state- ${ state } "> ${ state } </span>
</div>
` ) . join ( '' ) ;
2025-02-07 23:54:28 -05:00
}
2026-03-01 23:03:18 -05:00
}
2025-02-07 23:54:28 -05:00
}
2026-03-01 23:03:18 -05:00
function updateTopology ( hosts ) {
document . querySelectorAll ( '.topo-host' ) . forEach ( node => {
const name = node . dataset . host ;
const host = hosts [ name ] ;
if ( ! host ) return ;
2026-03-14 22:22:19 -04:00
node . className = node . className . replace ( /topo-v2-status-(up|down|degraded|unknown)/g , '' ) ;
2026-03-01 23:03:18 -05:00
node . className = node . className . replace ( /topo-status-(up|down|degraded|unknown)/g , '' ) ;
2026-03-14 22:22:19 -04:00
node . classList . add ( ` topo-v2-status- ${ host . status } ` ) ;
2026-03-01 23:03:18 -05:00
node . classList . add ( ` topo-status- ${ host . status } ` ) ;
const badge = node . querySelector ( '.topo-badge' ) ;
if ( badge ) {
badge . className = ` topo-badge topo-badge- ${ host . status } ` ;
badge . textContent = host . status ;
}
2026-04-29 23:37:47 -04:00
// Animate the 10G drop-wire red+dashed when host is down
document . querySelectorAll ( ` .topo-v2-wire-10g[data-host=" ${ CSS . escape ( name ) } "] ` ) . forEach ( wire => {
wire . classList . toggle ( 'wire-down' , host . status === 'down' ) ;
} ) ;
2026-03-01 23:03:18 -05:00
} ) ;
2025-02-07 21:28:54 -05:00
}
2026-03-01 23:03:18 -05:00
function updateUnifiTable ( devices ) {
const tbody = document . querySelector ( '#unifi-table tbody' ) ;
if ( ! tbody || ! devices . length ) return ;
tbody . innerHTML = devices . map ( d => {
const statusClass = d . connected ? '' : 'row-critical' ;
const dotClass = d . connected ? 'dot-up' : 'dot-down' ;
const statusText = d . connected ? 'Online' : 'Offline' ;
const suppressBtn = ! d . connected
2026-04-18 21:01:20 -04:00
? ` <button class="lt-btn lt-btn-ghost lt-btn-sm btn-suppress"
2026-03-03 15:39:48 -05:00
data-sup-type="unifi_device"
2026-04-18 23:46:44 -04:00
data-sup-name=" ${ lt . escHtml ( d . name ) } "
2026-04-29 17:50:00 -04:00
data-sup-detail=""
aria-label="Suppress alerts for ${ lt . escHtml ( d . name ) } ">🔕 Suppress</button> `
2026-03-01 23:03:18 -05:00
: '' ;
return `
<tr class=" ${ statusClass } ">
<td><span class=" ${ dotClass } "></span> ${ statusText } </td>
2026-04-18 23:46:44 -04:00
<td><strong> ${ lt . escHtml ( d . name ) } </strong></td>
<td> ${ lt . escHtml ( d . type ) } </td>
<td> ${ lt . escHtml ( d . model ) } </td>
<td> ${ lt . escHtml ( d . ip ) } </td>
2026-03-01 23:03:18 -05:00
<td> ${ suppressBtn } </td>
</tr> ` ;
} ) . join ( '' ) ;
2025-01-04 01:42:16 -05:00
}
2026-03-17 20:32:32 -04:00
function updateEventsTable ( events , totalActive ) {
2026-03-01 23:03:18 -05:00
const wrap = document . getElementById ( 'events-table-wrap' ) ;
if ( ! wrap ) return ;
2026-04-29 17:50:00 -04:00
const active = ( events || [ ] ) . filter ( e => e . severity !== 'info' ) ;
2026-03-01 23:03:18 -05:00
if ( ! active . length ) {
wrap . innerHTML = '<p class="empty-state">No active alerts ✔</p>' ;
return ;
}
2026-03-17 20:32:32 -04:00
const truncated = totalActive != null && totalActive > active . length ;
const countNotice = truncated
? ` <div class="pagination-notice">Showing ${ active . length } of ${ totalActive } active alerts — <a href="/api/events?limit=1000">view all via API</a></div> `
: '' ;
2026-03-01 23:03:18 -05:00
const rows = active . map ( e => {
const supType = e . event _type === 'unifi_device_offline' ? 'unifi_device'
: e . event _type === 'interface_down' ? 'interface'
: 'host' ;
2026-03-14 14:31:57 -04:00
const ticketBase = ( typeof GANDALF _CONFIG !== 'undefined' && GANDALF _CONFIG . ticket _web _url )
? GANDALF _CONFIG . ticket _web _url : 'http://t.lotusguild.org/ticket/' ;
2026-03-01 23:03:18 -05:00
const ticket = e . ticket _id
2026-04-18 23:46:44 -04:00
? ` <a href=" ${ lt . escHtml ( ticketBase ) } ${ lt . escHtml ( String ( e . ticket _id ) ) } " target="_blank"
2026-03-01 23:03:18 -05:00
class="ticket-link"># ${ e . ticket _id } </a> `
: '– ' ;
return `
<tr class="row- ${ e . severity } ">
2026-04-18 21:01:20 -04:00
<td><span class="lt-badge badge- ${ e . severity } "> ${ e . severity } </span></td>
2026-04-18 23:46:44 -04:00
<td> ${ lt . escHtml ( e . event _type . replace ( /_/g , ' ' ) ) } </td>
<td><strong> ${ lt . escHtml ( e . target _name ) } </strong></td>
<td> ${ lt . escHtml ( e . target _detail || '– ' ) } </td>
<td class="desc-cell" title=" ${ lt . escHtml ( e . description || '' ) } "> ${ lt . escHtml ( ( e . description || '' ) . substring ( 0 , 60 ) ) } ${ ( e . description || '' ) . length > 60 ? '…' : '' } </td>
<td class="ts-cell" title=" ${ lt . escHtml ( e . first _seen || '' ) } "> ${ lt . time . ago ( _toIso ( e . first _seen ) ) } </td>
<td class="ts-cell" title=" ${ lt . escHtml ( e . last _seen || '' ) } "> ${ lt . time . ago ( _toIso ( e . last _seen ) ) } </td>
2026-03-01 23:03:18 -05:00
<td> ${ e . consecutive _failures } </td>
<td> ${ ticket } </td>
<td>
2026-04-18 21:01:20 -04:00
<button class="lt-btn lt-btn-ghost lt-btn-sm btn-suppress"
2026-04-18 23:46:44 -04:00
data-sup-type=" ${ lt . escHtml ( supType ) } "
data-sup-name=" ${ lt . escHtml ( e . target _name ) } "
2026-04-29 17:50:00 -04:00
data-sup-detail=" ${ lt . escHtml ( e . target _detail || '' ) } "
aria-label="Suppress alert for ${ lt . escHtml ( e . target _name ) } ">🔕</button>
2026-03-01 23:03:18 -05:00
</td>
</tr> ` ;
} ) . join ( '' ) ;
wrap . innerHTML = `
2026-03-17 20:32:32 -04:00
${ countNotice }
2026-04-18 21:01:20 -04:00
<div class="lt-table-wrap">
<table class="lt-table" id="events-table">
<caption class="lt-sr-only">Active network alerts</caption>
2026-03-02 12:43:11 -05:00
<thead>
<tr>
<th>Sev</th><th>Type</th><th>Target</th><th>Detail</th>
2026-03-14 21:46:11 -04:00
<th>Description</th><th>First Seen</th><th>Last Seen</th><th>Failures</th><th>Ticket</th><th>Actions</th>
2026-03-02 12:43:11 -05:00
</tr>
</thead>
<tbody> ${ rows } </tbody>
</table>
</div> ` ;
2025-02-07 20:55:38 -05:00
}
2026-04-18 23:46:44 -04:00
// ── Suppression modal ─────────────────────────────────────────────────
2026-03-01 23:03:18 -05:00
function openSuppressModal ( type , name , detail ) {
const modal = document . getElementById ( 'suppress-modal' ) ;
if ( ! modal ) return ;
2026-04-18 23:46:44 -04:00
document . getElementById ( 'sup-type' ) . value = type ;
document . getElementById ( 'sup-name' ) . value = name ;
document . getElementById ( 'sup-detail' ) . value = detail ;
document . getElementById ( 'sup-reason' ) . value = '' ;
2026-03-01 23:03:18 -05:00
document . getElementById ( 'sup-expires' ) . value = '' ;
updateSuppressForm ( ) ;
2026-04-18 23:46:44 -04:00
lt . modal . open ( 'suppress-modal' ) ;
2026-03-01 23:03:18 -05:00
document . querySelectorAll ( '#suppress-modal .pill' ) . forEach ( p => p . classList . remove ( 'active' ) ) ;
const manualPill = document . querySelector ( '#suppress-modal .pill-manual' ) ;
if ( manualPill ) manualPill . classList . add ( 'active' ) ;
const hint = document . getElementById ( 'duration-hint' ) ;
if ( hint ) hint . textContent = 'Suppression will persist until manually removed.' ;
}
function closeSuppressModal ( ) {
2026-04-18 23:46:44 -04:00
lt . modal . close ( 'suppress-modal' ) ;
2026-03-01 23:03:18 -05:00
}
function updateSuppressForm ( ) {
const type = document . getElementById ( 'sup-type' ) . value ;
const nameGrp = document . getElementById ( 'sup-name-group' ) ;
const detailGrp = document . getElementById ( 'sup-detail-group' ) ;
if ( nameGrp ) nameGrp . style . display = ( type === 'all' ) ? 'none' : '' ;
if ( detailGrp ) detailGrp . style . display = ( type === 'interface' ) ? '' : 'none' ;
}
2026-03-03 15:39:48 -05:00
function setDuration ( mins , el ) {
2026-03-01 23:03:18 -05:00
document . getElementById ( 'sup-expires' ) . value = mins || '' ;
document . querySelectorAll ( '#suppress-modal .pill' ) . forEach ( p => p . classList . remove ( 'active' ) ) ;
2026-03-03 15:39:48 -05:00
if ( el ) el . classList . add ( 'active' ) ;
2026-03-01 23:03:18 -05:00
const hint = document . getElementById ( 'duration-hint' ) ;
if ( hint ) {
if ( mins ) {
const h = Math . floor ( mins / 60 ) , m = mins % 60 ;
hint . textContent = ` Expires in ${ h ? h + 'h ' : '' } ${ m ? m + 'm' : '' } . ` ;
} else {
hint . textContent = 'Suppression will persist until manually removed.' ;
}
}
2025-02-07 20:55:38 -05:00
}
2026-03-01 23:03:18 -05:00
async function submitSuppress ( e ) {
e . preventDefault ( ) ;
const type = document . getElementById ( 'sup-type' ) . value ;
const name = document . getElementById ( 'sup-name' ) . value ;
const detail = document . getElementById ( 'sup-detail' ) . value ;
const reason = document . getElementById ( 'sup-reason' ) . value ;
const expires = document . getElementById ( 'sup-expires' ) . value ;
if ( ! reason . trim ( ) ) { showToast ( 'Reason is required' , 'error' ) ; return ; }
if ( type !== 'all' && ! name . trim ( ) ) { showToast ( 'Target name is required' , 'error' ) ; return ; }
2025-02-07 20:55:38 -05:00
2026-03-01 23:03:18 -05:00
try {
2026-04-18 23:46:44 -04:00
await lt . api . post ( '/api/suppressions' , {
target _type : type ,
target _name : name ,
target _detail : detail ,
reason ,
expires _minutes : expires ? parseInt ( expires ) : null ,
2026-03-01 23:03:18 -05:00
} ) ;
2026-04-18 23:46:44 -04:00
closeSuppressModal ( ) ;
showToast ( 'Suppression applied ✔' , 'success' ) ;
setTimeout ( refreshAll , 500 ) ;
2026-03-01 23:03:18 -05:00
} catch ( err ) {
2026-04-18 23:46:44 -04:00
showToast ( err . message || 'Failed to apply suppression' , 'error' ) ;
2026-03-01 23:03:18 -05:00
}
2025-02-07 21:03:31 -05:00
}
2025-02-07 20:55:38 -05:00
2026-04-18 23:46:44 -04:00
// ── Global click delegation ───────────────────────────────────────────
2026-03-01 23:03:18 -05:00
document . addEventListener ( 'click' , e => {
2026-04-18 23:46:44 -04:00
// Refresh button
if ( e . target . closest ( '[data-action="refresh"]' ) ) {
lt . autoRefresh . now ( ) ;
return ;
}
// Duration pills (data-duration="" = manual/forever)
const pill = e . target . closest ( '.pill[data-duration]' ) ;
if ( pill ) {
const val = pill . dataset . duration ;
setDuration ( val ? parseInt ( val ) : null , pill ) ;
return ;
}
2026-03-03 15:39:48 -05:00
2026-04-18 23:46:44 -04:00
// Suppress buttons
2026-03-03 15:39:48 -05:00
const btn = e . target . closest ( '.btn-suppress[data-sup-type]' ) ;
if ( btn ) {
openSuppressModal (
btn . dataset . supType || '' ,
btn . dataset . supName || '' ,
btn . dataset . supDetail || '' ,
) ;
}
2026-03-01 23:03:18 -05:00
} ) ;