- Home: new /api/v1/metrics/home endpoint buckets last 24h of assets,
jobs done/failed into hourly counts; sparklines now render real
time-series instead of decorative sine waves
- Home stat cards are now clickable (route to relevant page) and the
delta lines show real activity ("+N added in last 24h", "N completed")
- Home live-feed tiles use HlsPreview for recorders with a live_asset_id
- Users: row 3-dot menu is now a real popover with Rename / Reset
password / Delete actions wired to PATCH /users/:id and DELETE
- Users: role is now an inline <select> that PATCHes immediately
- Users: Created column replaces fake 'last active' (no last_login
tracking yet); group count is real
- Groups tab is now functional — list groups, create, expand to
show + manage members (add/remove), delete; backed by existing
/api/v1/groups CRUD
- Policies tab is now an honest 'coming soon' stub
- New icons: key, lock, edit; new .row-menu popover styles
206 lines
10 KiB
JavaScript
206 lines
10 KiB
JavaScript
// screens-home.jsx
|
|
|
|
function Home({ navigate }) {
|
|
const { RECORDERS, JOBS, ASSETS, NODES } = window.ZAMPP_DATA;
|
|
|
|
// Live (current-state) data from the boot-time data load
|
|
const liveRecorders = RECORDERS.filter(r => r.status === 'recording').slice(0, 4);
|
|
const runningJobs = JOBS.filter(j => j.status === 'running' || j.status === 'queued');
|
|
const recentAssets = [...ASSETS].sort((a, b) => new Date(b.created_at) - new Date(a.created_at)).slice(0, 6);
|
|
|
|
// Real historic sparklines from /metrics/home — buckets the last 24h.
|
|
const [metrics, setMetrics] = React.useState(null);
|
|
React.useEffect(() => {
|
|
let cancelled = false;
|
|
const load = () => {
|
|
window.ZAMPP_API.fetch('/metrics/home?hours=24')
|
|
.then(d => { if (!cancelled) setMetrics(d); })
|
|
.catch(() => {});
|
|
};
|
|
load();
|
|
const t = setInterval(load, 30_000);
|
|
return () => { cancelled = true; clearInterval(t); };
|
|
}, []);
|
|
|
|
const cards = metrics?.cards || {};
|
|
const vals = (s) => Array.isArray(s) ? s.map(p => p.v) : [];
|
|
|
|
// Card values come from /metrics so that "Library" reflects what's in the
|
|
// DB right now, not whatever ZAMPP_DATA happens to be cached as on first load.
|
|
const assetsTotal = cards.assets?.total ?? ASSETS.length;
|
|
const liveCount = cards.recorders?.live ?? liveRecorders.length;
|
|
const totalRecs = cards.recorders?.total ?? RECORDERS.length;
|
|
const runningCount = cards.jobs?.running ?? runningJobs.length;
|
|
const doneCount = cards.jobs?.done_total ?? JOBS.filter(j => j.status === 'done').length;
|
|
const failedCount = cards.jobs?.failed_total ?? JOBS.filter(j => j.status === 'failed').length;
|
|
const nodesOnline = cards.cluster?.online ?? NODES.filter(n => n.status === 'online' || n.online === true).length;
|
|
const nodesTotal = cards.cluster?.total ?? NODES.length;
|
|
|
|
// Sum the most recent hour of each bucketed series for the delta line so
|
|
// the "+N this hour" hint always reflects the latest bucket.
|
|
const lastBucket = (series) => (Array.isArray(series) && series.length ? series[series.length - 1].v : 0);
|
|
const sumWindow = (series) => (Array.isArray(series) ? series.reduce((a, p) => a + p.v, 0) : 0);
|
|
|
|
return (
|
|
<div className="page">
|
|
<div className="home-greeting">
|
|
<h1>Dragonflight</h1>
|
|
<p>
|
|
{liveCount > 0 ? liveCount + ' live · ' : ''}
|
|
{runningCount > 0 ? runningCount + ' job' + (runningCount > 1 ? 's running' : ' running') + ' · ' : ''}
|
|
{assetsTotal.toLocaleString()} assets
|
|
</p>
|
|
</div>
|
|
|
|
<div className="stat-row">
|
|
<div className="stat-card" onClick={() => navigate('library')} style={{ cursor: 'pointer' }}>
|
|
<div className="label"><Icon name="library" size={12} /> Library</div>
|
|
<div className="value">{assetsTotal.toLocaleString()}</div>
|
|
<div className="delta">
|
|
{sumWindow(cards.assets?.series) > 0
|
|
? '+' + sumWindow(cards.assets?.series) + ' added in last 24h'
|
|
: 'Total assets'}
|
|
</div>
|
|
<Sparkline data={vals(cards.assets?.series)} color="#5B7CFA" />
|
|
</div>
|
|
<div className="stat-card" onClick={() => navigate('recorders')} style={{ cursor: 'pointer' }}>
|
|
<div className="label"><Icon name="record" size={12} /> Live feeds</div>
|
|
<div className="value">{liveCount}</div>
|
|
<div className="delta">{totalRecs} recorder{totalRecs === 1 ? '' : 's'} configured</div>
|
|
<Sparkline data={vals(cards.recorders?.series)} color="#2DD4A8" />
|
|
</div>
|
|
<div className="stat-card" onClick={() => navigate('jobs')} style={{ cursor: 'pointer' }}>
|
|
<div className="label"><Icon name="jobs" size={12} /> Jobs</div>
|
|
<div className="value">{runningCount}<span style={{ color: 'var(--text-3)', fontWeight: 400, fontSize: 16 }}> / {doneCount} done</span></div>
|
|
<div className="delta" style={{ color: failedCount > 0 ? 'var(--warning)' : '' }}>
|
|
{failedCount > 0 ? failedCount + ' failed' : 'All clear'}
|
|
{sumWindow(cards.jobs?.series_done) > 0 && (
|
|
<> · {sumWindow(cards.jobs?.series_done)} completed in last 24h</>
|
|
)}
|
|
</div>
|
|
<Sparkline data={vals(cards.jobs?.series_done)} color="#B57CFA" />
|
|
</div>
|
|
<div className="stat-card" onClick={() => navigate('cluster')} style={{ cursor: 'pointer' }}>
|
|
<div className="label"><Icon name="hdd" size={12} /> Cluster nodes</div>
|
|
<div className="value">{nodesOnline}<span style={{ color: 'var(--text-3)', fontWeight: 400, fontSize: 16 }}> / {nodesTotal} online</span></div>
|
|
<div className="delta">Heartbeat within 2 min</div>
|
|
<Sparkline data={vals(cards.cluster?.series)} color="#F5A623" />
|
|
</div>
|
|
</div>
|
|
|
|
<div className="home-grid">
|
|
<div>
|
|
{liveRecorders.length > 0 && (
|
|
<>
|
|
<SectionHead title="Live now" onMore={() => navigate('recorders')} moreLabel="All recorders" />
|
|
<div className="live-feed-grid">
|
|
{liveRecorders.map(r => (
|
|
<div key={r.id} className="live-feed-tile" onClick={() => navigate('recorders')}>
|
|
<div className="live-feed-tile-badge"><span className="badge live">REC</span></div>
|
|
{r.live_asset_id
|
|
? <HlsPreview assetId={r.live_asset_id} />
|
|
: <FauxFrame />}
|
|
<div className="live-feed-tile-label">
|
|
<span className="name">{r.name}</span>
|
|
<span className="time">{r.elapsed}</span>
|
|
</div>
|
|
</div>
|
|
))}
|
|
</div>
|
|
<div style={{ height: 16 }} />
|
|
</>
|
|
)}
|
|
|
|
<SectionHead title="Recently added" onMore={() => navigate('library')} moreLabel="All assets" />
|
|
<div className="activity-feed">
|
|
{recentAssets.length === 0 ? (
|
|
<div style={{ padding: '16px 0', color: 'var(--text-3)', fontSize: 12.5 }}>No assets yet.</div>
|
|
) : recentAssets.map(a => (
|
|
<div key={a.id} className="activity-row">
|
|
<div className="activity-icon record">
|
|
<Icon name={a.type === 'audio' ? 'audio' : 'video'} size={12} />
|
|
</div>
|
|
<div className="activity-text">
|
|
<span className="target">{a.name}</span>
|
|
{a.project && <> in <strong>{a.project}</strong></>}
|
|
</div>
|
|
<div className="activity-time">{a.updated}</div>
|
|
</div>
|
|
))}
|
|
</div>
|
|
</div>
|
|
|
|
<div>
|
|
<SectionHead title="Job queue" onMore={() => navigate('jobs')} moreLabel="View all" />
|
|
<div className="panel" style={{ padding: 4 }}>
|
|
{JOBS.length === 0
|
|
? <div style={{ padding: '20px 12px', textAlign: 'center', color: 'var(--text-3)', fontSize: 12.5 }}>No jobs.</div>
|
|
: JOBS.slice(0, 5).map(j => <MiniJobRow key={j.id} job={j} />)}
|
|
</div>
|
|
|
|
<div style={{ height: 16 }} />
|
|
<SectionHead title="Cluster" onMore={() => navigate('cluster')} moreLabel="View all" />
|
|
<div className="panel" style={{ padding: 4 }}>
|
|
{NODES.length === 0
|
|
? <div style={{ padding: '16px 12px', color: 'var(--text-3)', fontSize: 12.5 }}>No nodes found.</div>
|
|
: NODES.slice(0, 4).map(n => {
|
|
const nodeId = n.id || n.hostname || n.name || 'node';
|
|
const isOnline = n.status === 'online' || n.online === true;
|
|
const cpuPct = n.cpu_percent ?? n.cpu ?? n.cpu_usage ?? null;
|
|
const memRaw = n.memory_used_gb ?? n.mem ?? (n.mem_used_mb != null ? n.mem_used_mb / 1024 : null);
|
|
const memGb = memRaw != null ? Number(memRaw) : null;
|
|
return (
|
|
<div key={nodeId} style={{ padding: '10px 12px', display: 'flex', alignItems: 'center', gap: 8, borderBottom: '1px solid var(--border)' }}>
|
|
<StatusDot status={isOnline ? 'online' : 'offline'} />
|
|
<span style={{ fontSize: 12.5, fontWeight: 500 }}>{nodeId}</span>
|
|
<span style={{ flex: 1 }} />
|
|
{cpuPct != null && <span className="mono" style={{ fontSize: 11, color: 'var(--text-3)' }}>CPU {Math.round(cpuPct)}%</span>}
|
|
{memGb != null && <span className="mono" style={{ fontSize: 11, color: 'var(--text-3)' }}>{memGb < 1 ? Math.round(memGb * 1024) + 'MB' : memGb.toFixed(1) + 'GB'} RAM</span>}
|
|
</div>
|
|
);
|
|
})}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function SectionHead({ title, onMore, moreLabel = 'View all' }) {
|
|
return (
|
|
<div style={{ display: 'flex', alignItems: 'center', padding: '16px 0 10px', gap: 12 }}>
|
|
<div style={{ fontSize: 13, fontWeight: 600, letterSpacing: '-0.01em', whiteSpace: 'nowrap' }}>{title}</div>
|
|
{onMore && (
|
|
<button className="btn ghost sm" style={{ marginLeft: 'auto', whiteSpace: 'nowrap' }} onClick={onMore}>
|
|
{moreLabel}<Icon name="arrowRight" size={11} />
|
|
</button>
|
|
)}
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function MiniJobRow({ job }) {
|
|
return (
|
|
<div style={{ padding: '10px 12px', display: 'flex', alignItems: 'center', gap: 10, borderBottom: '1px solid var(--border)' }}>
|
|
<StatusDot status={job.status} />
|
|
<div style={{ flex: 1, minWidth: 0 }}>
|
|
<div style={{ fontSize: 12, display: 'flex', gap: 6, alignItems: 'center' }}>
|
|
<span style={{ color: 'var(--text-2)' }}>{job.kind}</span>
|
|
<span className="muted" style={{ fontFamily: 'var(--font-mono)', fontSize: 10.5 }}>·</span>
|
|
<span style={{ overflow: 'hidden', textOverflow: 'ellipsis', whiteSpace: 'nowrap' }}>{job.asset}</span>
|
|
</div>
|
|
{job.status === 'running' && (
|
|
<div style={{ marginTop: 5, height: 3, background: 'var(--bg-3)', borderRadius: 99, overflow: 'hidden' }}>
|
|
<div style={{ width: job.progress + '%', height: '100%', background: 'var(--accent)', transition: 'width 300ms' }} />
|
|
</div>
|
|
)}
|
|
{job.status === 'failed' && <div style={{ marginTop: 3, fontSize: 11, color: 'var(--danger)' }}>{job.error}</div>}
|
|
</div>
|
|
<div style={{ fontFamily: 'var(--font-mono)', fontSize: 10.5, color: 'var(--text-3)', minWidth: 40, textAlign: 'right' }}>
|
|
{job.status === 'running' ? job.eta : job.status}
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
window.Home = Home;
|