dragonflight/services/web-ui/public/screens-jobs.jsx

214 lines
9.2 KiB
React
Raw Normal View History

// screens-jobs.jsx
// Pick the most-meaningful timestamp + label for a job's current state.
// Returns { label, iso } — caller renders "<label> <relative-time>" with
// the full ISO as a tooltip.
function _jobTimeFor(job) {
if (job.status === 'done' && job.completed_at) return { label: 'done', iso: job.completed_at };
if (job.status === 'failed' && job.failed_at) return { label: 'failed', iso: job.failed_at };
if (job.status === 'running' && job.started_at) return { label: 'started', iso: job.started_at };
if (job.created_at) return { label: 'queued', iso: job.created_at };
return null;
}
function _fmtAbsolute(iso) {
if (!iso) return '';
try {
return new Date(iso).toLocaleString(undefined, {
month: 'short', day: 'numeric',
hour: 'numeric', minute: '2-digit', second: '2-digit',
});
} catch { return iso; }
}
function Jobs({ navigate }) {
const [tab, setTab] = React.useState('all');
const [jobs, setJobs] = React.useState(window.ZAMPP_DATA.JOBS);
const [lastFetch, setLastFetch] = React.useState(Date.now());
const normalizeJob = (j) => {
const statusMap = { waiting: 'queued', active: 'running', completed: 'done', failed: 'failed' };
const kindMap = { proxy: 'Proxy', thumbnail: 'Thumbnail', conform: 'Conform', transcode: 'Transcode' };
const meta = j.metadata || {};
return {
...j,
status: statusMap[j.status] || j.status,
kind: kindMap[j.type] || j.type || 'Job',
asset: j.asset_name || meta.filename || '—',
eta: '—',
node: meta.node || '—',
priority: meta.priority || 'normal',
error: j.error || null,
progress: j.progress || 0,
};
};
const refresh = React.useCallback(() => {
window.ZAMPP_API.fetch('/jobs')
.then(raw => {
const norm = (raw || []).map(normalizeJob);
window.ZAMPP_DATA.JOBS = norm;
setJobs(norm);
setLastFetch(Date.now());
})
.catch(() => {});
}, []);
React.useEffect(() => {
const i = setInterval(refresh, 5000);
return () => clearInterval(i);
}, [refresh]);
const handleRetry = React.useCallback((job) => {
window.ZAMPP_API.fetch('/jobs/' + job.id + '/retry', { method: 'POST' })
.then(() => refresh())
.catch(e => alert('Retry failed: ' + e.message));
}, [refresh]);
const handleDelete = React.useCallback((job) => {
if (!window.confirm('Remove this job from the queue?')) return;
window.ZAMPP_API.fetch('/jobs/' + job.id, { method: 'DELETE' })
.then(() => setJobs(prev => prev.filter(j => j.id !== job.id)))
.catch(e => alert('Delete failed: ' + e.message));
}, []);
// Retry every failed job at once. Useful after a transient infra issue
// (S3 outage, hung worker) — one click per job is painful with 20+ failures.
const handleRetryAll = React.useCallback(() => {
const failedJobs = jobs.filter(j => j.status === 'failed');
if (failedJobs.length === 0) return;
if (!window.confirm(`Re-queue all ${failedJobs.length} failed jobs?`)) return;
Promise.allSettled(
failedJobs.map(j => window.ZAMPP_API.fetch('/jobs/' + j.id + '/retry', { method: 'POST' }))
).then(refresh);
}, [jobs, refresh]);
const counts = {
all: jobs.length,
running: jobs.filter(j => j.status === 'running').length,
queued: jobs.filter(j => j.status === 'queued').length,
done: jobs.filter(j => j.status === 'done').length,
failed: jobs.filter(j => j.status === 'failed').length,
};
const filtered = tab === 'all' ? jobs : jobs.filter(j => j.status === tab);
return (
<div className="page">
<div className="page-header">
<h1>Jobs</h1>
<span className="subtitle">Proxy generation, transcoding, and processing queue</span>
<div className="spacer" />
{counts.failed > 0 && (
<button className="btn ghost sm" onClick={handleRetryAll} title={`Retry all ${counts.failed} failed jobs`}>
<Icon name="refresh" />Retry all failed
</button>
)}
<button className="btn ghost sm" onClick={refresh}>
<Icon name="refresh" />Refresh
</button>
</div>
<div className="page-body">
<div className="jobs-stats">
<div className="stat-card">
<div className="label">Running</div>
<div className="value">{counts.running}</div>
<div className="delta">{counts.queued} queued</div>
</div>
<div className="stat-card">
<div className="label">Completed</div>
<div className="value">{counts.done}</div>
<div className="delta">Total done</div>
</div>
<div className="stat-card">
<div className="label">Failed</div>
<div className="value">{counts.failed}</div>
<div className="delta" style={{ color: counts.failed > 0 ? 'var(--warning)' : '' }}>
{counts.failed > 0 ? 'Needs attention' : 'All clear'}
</div>
</div>
<div className="stat-card">
<div className="label">Total jobs</div>
<div className="value">{counts.all}</div>
<div className="delta muted" style={{ fontSize: 10.5 }}>Updated {Math.round((Date.now()-lastFetch)/1000)}s ago</div>
</div>
</div>
<div className="tab-group" style={{ marginTop: 20, width: 'fit-content' }}>
{[
{ id: 'all', label: 'All · ' + counts.all },
{ id: 'running', label: 'Running · ' + counts.running },
{ id: 'queued', label: 'Queued · ' + counts.queued },
{ id: 'done', label: 'Done · ' + counts.done },
{ id: 'failed', label: 'Failed · ' + counts.failed },
].map(t => (
<button key={t.id} className={tab === t.id ? 'active' : ''} onClick={() => setTab(t.id)}>{t.label}</button>
))}
</div>
<div className="panel" style={{ marginTop: 12 }}>
<div className="job-row head">
<div></div><div>Job</div><div>Asset</div><div>Node</div><div>Progress</div><div>Time</div><div>Priority</div><div></div>
</div>
{filtered.length === 0
? <div style={{ padding: '24px', textAlign: 'center', color: 'var(--text-3)' }}>No jobs in this category.</div>
: filtered.map(j => <JobRow key={j.id} job={j} onRetry={handleRetry} onDelete={handleDelete} />)}
</div>
</div>
</div>
);
}
function JobRow({ job, onRetry, onDelete }) {
const iconMap = { Proxy: 'proxy', Transcode: 'film', Thumbnail: 'image', Conform: 'layers' };
return (
<div className="job-row">
<div><StatusDot status={job.status} /></div>
<div style={{ display: 'flex', alignItems: 'center', gap: 8 }}>
<Icon name={iconMap[job.kind] || 'jobs'} size={13} style={{ color: 'var(--text-3)' }} />
<span style={{ fontWeight: 500 }}>{job.kind}</span>
</div>
<div style={{ overflow: 'hidden', textOverflow: 'ellipsis', whiteSpace: 'nowrap', color: 'var(--text-2)' }}>{job.asset}</div>
<div className="mono" style={{ fontSize: 11.5, color: 'var(--text-3)' }}>{job.node}</div>
<div>
{job.status === 'running' && (
<div className="job-progress-wrap">
<div className="job-progress-bar"><div className="job-progress-fill" style={{ width: job.progress + '%' }} /></div>
<span className="mono" style={{ fontSize: 10.5, color: 'var(--text-3)', minWidth: 32, textAlign: 'right' }}>{Math.round(job.progress)}%</span>
</div>
)}
{job.status === 'done' && <span className="badge success" style={{ background: 'transparent', padding: 0 }}><Icon name="check" size={12} /> Complete</span>}
{job.status === 'queued' && <span style={{ fontSize: 12, color: 'var(--text-3)' }}>Waiting</span>}
{job.status === 'failed' && (
<span title={job.error || 'Failed'}
style={{ fontSize: 12, color: 'var(--danger)', display: 'flex', alignItems: 'center', gap: 4, overflow: 'hidden' }}>
<Icon name="alert" size={12} />
<span style={{ overflow: 'hidden', textOverflow: 'ellipsis', whiteSpace: 'nowrap', minWidth: 0 }}>
{(job.error || 'Failed').slice(0, 120)}
</span>
</span>
)}
</div>
<div className="mono" style={{ fontSize: 11.5, color: 'var(--text-3)', overflow: 'hidden', textOverflow: 'ellipsis', whiteSpace: 'nowrap' }}
title={(() => { const t = _jobTimeFor(job); return t ? t.label + ' at ' + _fmtAbsolute(t.iso) : ''; })()}>
{(() => {
const t = _jobTimeFor(job);
if (!t) return '—';
return t.label + ' ' + window.ZAMPP_API.fmtRelative(t.iso);
})()}
</div>
<div><span className={'badge ' + (job.priority === 'high' ? 'warning' : 'outline')}>{job.priority}</span></div>
<div style={{ display: 'flex', gap: 4 }}>
{job.status === 'failed' && (
<button className="btn ghost sm" onClick={() => onRetry(job)}><Icon name="refresh" />Retry</button>
)}
{(job.status === 'queued' || job.status === 'done') && (
<button className="icon-btn" title="Remove job" onClick={() => onDelete(job)}><Icon name="x" /></button>
)}
</div>
</div>
);
}
window.Jobs = Jobs;