2026-03-01 23:03:18 -05:00
'use strict' ;
2026-03-17 20:43:08 -04:00
// ── Auto-redirect on auth timeout ─────────────────────────────────────
// Intercept all fetch() calls: if the server returns 401 (auth expired),
// reload the page so Authelia redirects to the login screen.
( function ( ) {
const _fetch = window . fetch ;
window . fetch = async function ( ... args ) {
const resp = await _fetch ( ... args ) ;
if ( resp . status === 401 ) {
window . location . reload ( ) ;
}
return resp ;
} ;
} ) ( ) ;
2026-03-14 21:40:20 -04:00
// ── Toast notifications — delegates to lt.toast from base.js ─────────
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-03-01 23:03:18 -05:00
// ── Dashboard auto-refresh ────────────────────────────────────────────
async function refreshAll ( ) {
try {
const [ netResp , statusResp ] = await Promise . all ( [
fetch ( '/api/network' ) ,
fetch ( '/api/status' ) ,
] ) ;
if ( ! netResp . ok || ! statusResp . ok ) return ;
const net = await netResp . json ( ) ;
const status = await statusResp . json ( ) ;
updateHostGrid ( net . hosts || { } ) ;
updateUnifiTable ( net . unifi || [ ] ) ;
2026-03-17 20:32:32 -04:00
updateEventsTable ( status . events || [ ] , status . total _active ) ;
2026-03-01 23:03:18 -05:00
updateStatusBar ( status . summary || { } , status . last _check || '' ) ;
updateTopology ( net . hosts || { } ) ;
} catch ( e ) {
console . warn ( 'Refresh failed:' , e ) ;
}
2025-02-08 00:32:25 -05:00
}
2026-03-01 23:03:18 -05:00
function updateStatusBar ( summary , lastCheck ) {
const bar = document . querySelector ( '.status-chips' ) ;
if ( ! bar ) return ;
const chips = [ ] ;
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> ` ) ;
if ( ! summary . critical && ! summary . warning ) 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
// Update browser tab title with alert count
const critCount = summary . critical || 0 ;
const warnCount = summary . warning || 0 ;
if ( critCount ) {
document . title = ` ( ${ critCount } CRIT) GANDALF ` ;
} else if ( warnCount ) {
document . title = ` ( ${ warnCount } WARN) GANDALF ` ;
} else {
document . title = 'GANDALF' ;
}
// Stale data banner: warn if last_check is older than 15 minutes
let staleBanner = document . getElementById ( 'stale-banner' ) ;
if ( lastCheck ) {
// last_check format: "2026-03-14 14:14:21 UTC"
const checkAge = ( Date . now ( ) - new Date ( lastCheck . replace ( ' UTC' , 'Z' ) . replace ( ' ' , 'T' ) ) ) / 1000 ;
if ( checkAge > 900 ) { // 15 minutes
if ( ! staleBanner ) {
staleBanner = document . createElement ( 'div' ) ;
staleBanner . id = 'stale-banner' ;
staleBanner . className = 'stale-banner' ;
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 ) ;
staleBanner . textContent = ` ⚠ Monitoring data is stale — last check was ${ mins } minute ${ mins !== 1 ? 's' : '' } ago. The monitor daemon may be down. ` ;
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 ;
// Update card border class
card . className = card . className . replace ( /host-card-(up|down|degraded|unknown)/g , '' ) ;
card . classList . add ( ` host-card- ${ host . status } ` ) ;
// Update status dot in header
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
// Update interface rows
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>
<span class="iface-name"> ${ escHtml ( iface ) } </span>
<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 ;
}
} ) ;
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"
data-sup-name=" ${ escHtml ( d . name ) } "
data-sup-detail="">🔕 Suppress</button> `
2026-03-01 23:03:18 -05:00
: '' ;
return `
<tr class=" ${ statusClass } ">
<td><span class=" ${ dotClass } "></span> ${ statusText } </td>
<td><strong> ${ escHtml ( d . name ) } </strong></td>
<td> ${ escHtml ( d . type ) } </td>
<td> ${ escHtml ( d . model ) } </td>
<td> ${ escHtml ( d . ip ) } </td>
<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 ;
const active = events . filter ( e => e . severity !== 'info' ) ;
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-03-14 14:31:57 -04:00
? ` <a href=" ${ ticketBase } ${ 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-03-01 23:03:18 -05:00
<td> ${ escHtml ( e . event _type . replace ( /_/g , ' ' ) ) } </td>
<td><strong> ${ escHtml ( e . target _name ) } </strong></td>
<td> ${ escHtml ( e . target _detail || '– ' ) } </td>
<td class="desc-cell" title=" ${ escHtml ( e . description || '' ) } "> ${ escHtml ( ( e . description || '' ) . substring ( 0 , 60 ) ) } ${ ( e . description || '' ) . length > 60 ? '…' : '' } </td>
2026-03-14 21:46:11 -04:00
<td class="ts-cell" title=" ${ escHtml ( e . first _seen || '' ) } "> ${ fmtRelTime ( e . first _seen ) } </td>
<td class="ts-cell" title=" ${ escHtml ( e . last _seen || '' ) } "> ${ fmtRelTime ( 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-03-03 15:39:48 -05:00
data-sup-type=" ${ escHtml ( supType ) } "
data-sup-name=" ${ escHtml ( e . target _name ) } "
data-sup-detail=" ${ escHtml ( e . target _detail || '' ) } ">🔕</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-03-01 23:03:18 -05:00
// ── Suppression modal (dashboard) ────────────────────────────────────
function openSuppressModal ( type , name , detail ) {
const modal = document . getElementById ( 'suppress-modal' ) ;
if ( ! modal ) return ;
document . getElementById ( 'sup-type' ) . value = type ;
document . getElementById ( 'sup-name' ) . value = name ;
document . getElementById ( 'sup-detail' ) . value = detail ;
document . getElementById ( 'sup-reason' ) . value = '' ;
document . getElementById ( 'sup-expires' ) . value = '' ;
updateSuppressForm ( ) ;
2026-04-18 21:42:04 -04:00
modal . classList . add ( 'is-open' ) ;
modal . removeAttribute ( 'aria-hidden' ) ;
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 ( ) {
const modal = document . getElementById ( 'suppress-modal' ) ;
2026-04-18 21:42:04 -04:00
if ( ! modal ) return ;
modal . classList . remove ( 'is-open' ) ;
modal . setAttribute ( 'aria-hidden' , 'true' ) ;
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 {
const resp = await fetch ( '/api/suppressions' , {
method : 'POST' ,
headers : { 'Content-Type' : 'application/json' } ,
body : JSON . stringify ( {
target _type : type ,
target _name : name ,
target _detail : detail ,
reason : reason ,
expires _minutes : expires ? parseInt ( expires ) : null ,
} ) ,
} ) ;
const data = await resp . json ( ) ;
if ( data . success ) {
closeSuppressModal ( ) ;
showToast ( 'Suppression applied ✔' , 'success' ) ;
setTimeout ( refreshAll , 500 ) ;
} else {
showToast ( data . error || 'Failed to apply suppression' , 'error' ) ;
}
} catch ( err ) {
showToast ( 'Network error' , 'error' ) ;
}
2025-02-07 21:03:31 -05:00
}
2025-02-07 20:55:38 -05:00
2026-03-03 15:39:48 -05:00
// ── Global click handler: modal backdrop + suppress button delegation ─
2026-03-01 23:03:18 -05:00
document . addEventListener ( 'click' , e => {
2026-03-03 15:39:48 -05:00
// Close modal when clicking backdrop
2026-03-01 23:03:18 -05:00
const modal = document . getElementById ( 'suppress-modal' ) ;
2026-03-03 15:39:48 -05:00
if ( modal && e . target === modal ) { closeSuppressModal ( ) ; return ; }
// Suppress button via data attributes (avoids inline onclick XSS)
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
} ) ;
2026-03-14 21:46:11 -04:00
// ── Relative time ─────────────────────────────────────────────────────
function fmtRelTime ( tsStr ) {
if ( ! tsStr ) return '– ' ;
const d = new Date ( tsStr . replace ( ' UTC' , 'Z' ) . replace ( ' ' , 'T' ) ) ;
if ( isNaN ( d ) ) return tsStr ;
const secs = Math . floor ( ( Date . now ( ) - d ) / 1000 ) ;
if ( secs < 60 ) return ` ${ secs } s ago ` ;
if ( secs < 3600 ) return ` ${ Math . floor ( secs / 60 ) } m ago ` ;
if ( secs < 86400 ) return ` ${ Math . floor ( secs / 3600 ) } h ago ` ;
return ` ${ Math . floor ( secs / 86400 ) } d ago ` ;
}
2026-03-01 23:03:18 -05:00
// ── Utility ───────────────────────────────────────────────────────────
function escHtml ( str ) {
if ( str === null || str === undefined ) return '' ;
return String ( str )
. replace ( /&/g , '&' )
. replace ( /</g , '<' )
. replace ( />/g , '>' )
. replace ( /"/g , '"' ) ;
}