mirror of
https://github.com/Crosstalk-Solutions/project-nomad.git
synced 2026-03-28 19:49:25 +01:00
Failed download jobs persist in BullMQ forever with no way to clear them, leaving stale error notifications in Content Explorer and Easy Setup. Adds a dismiss button (X) on failed download cards that removes the job from the queue via a new DELETE endpoint. - Backend: DELETE /api/downloads/jobs/:jobId endpoint - Frontend: X button on failed download cards with immediate refresh Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
78 lines
2.8 KiB
TypeScript
78 lines
2.8 KiB
TypeScript
import useDownloads, { useDownloadsProps } from '~/hooks/useDownloads'
|
|
import HorizontalBarChart from './HorizontalBarChart'
|
|
import { extractFileName } from '~/lib/util'
|
|
import StyledSectionHeader from './StyledSectionHeader'
|
|
import { IconAlertTriangle, IconX } from '@tabler/icons-react'
|
|
import api from '~/lib/api'
|
|
|
|
interface ActiveDownloadProps {
|
|
filetype?: useDownloadsProps['filetype']
|
|
withHeader?: boolean
|
|
}
|
|
|
|
const ActiveDownloads = ({ filetype, withHeader = false }: ActiveDownloadProps) => {
|
|
const { data: downloads, invalidate } = useDownloads({ filetype })
|
|
|
|
const handleDismiss = async (jobId: string) => {
|
|
await api.removeDownloadJob(jobId)
|
|
invalidate()
|
|
}
|
|
|
|
return (
|
|
<>
|
|
{withHeader && <StyledSectionHeader title="Active Downloads" className="mt-12 mb-4" />}
|
|
<div className="space-y-4">
|
|
{downloads && downloads.length > 0 ? (
|
|
downloads.map((download) => (
|
|
<div
|
|
key={download.jobId}
|
|
className={`bg-desert-white rounded-lg p-4 border shadow-sm hover:shadow-lg transition-shadow ${
|
|
download.status === 'failed'
|
|
? 'border-red-300'
|
|
: 'border-desert-stone-light'
|
|
}`}
|
|
>
|
|
{download.status === 'failed' ? (
|
|
<div className="flex items-center gap-2">
|
|
<IconAlertTriangle className="w-5 h-5 text-red-500 flex-shrink-0" />
|
|
<div className="flex-1 min-w-0">
|
|
<p className="text-sm font-medium text-gray-900 truncate">
|
|
{extractFileName(download.filepath) || download.url}
|
|
</p>
|
|
<p className="text-xs text-red-600 mt-0.5">
|
|
Download failed{download.failedReason ? `: ${download.failedReason}` : ''}
|
|
</p>
|
|
</div>
|
|
<button
|
|
onClick={() => handleDismiss(download.jobId)}
|
|
className="flex-shrink-0 p-1 rounded hover:bg-red-100 transition-colors"
|
|
title="Dismiss failed download"
|
|
>
|
|
<IconX className="w-4 h-4 text-red-400 hover:text-red-600" />
|
|
</button>
|
|
</div>
|
|
) : (
|
|
<HorizontalBarChart
|
|
items={[
|
|
{
|
|
label: extractFileName(download.filepath) || download.url,
|
|
value: download.progress,
|
|
total: '100%',
|
|
used: `${download.progress}%`,
|
|
type: download.filetype,
|
|
},
|
|
]}
|
|
/>
|
|
)}
|
|
</div>
|
|
))
|
|
) : (
|
|
<p className="text-text-muted">No active downloads</p>
|
|
)}
|
|
</div>
|
|
</>
|
|
)
|
|
}
|
|
|
|
export default ActiveDownloads
|