feat: add RetryFailedButton component and API endpoint for retrying failed queue jobs
Build Images and Deploy / Update-PROD-Stack (push) Successful in 1m19s
Build Images and Deploy / Update-PROD-Stack (push) Successful in 1m19s
This commit is contained in:
@@ -1,5 +1,6 @@
|
|||||||
import { priceUpdateQueue, maintenanceQueue, schedulerQueue } from '@/lib/queue'
|
import { priceUpdateQueue, maintenanceQueue, schedulerQueue } from '@/lib/queue'
|
||||||
import { formatDistanceToNow } from 'date-fns'
|
import { formatDistanceToNow } from 'date-fns'
|
||||||
|
import RetryFailedButton from '@/components/admin/RetryFailedButton'
|
||||||
|
|
||||||
export const dynamic = 'force-dynamic'
|
export const dynamic = 'force-dynamic'
|
||||||
|
|
||||||
@@ -89,6 +90,7 @@ export default async function AdminQueuePage() {
|
|||||||
<div className="flex items-center justify-between px-4 py-3 border-b border-surface-border">
|
<div className="flex items-center justify-between px-4 py-3 border-b border-surface-border">
|
||||||
<h3 className="font-medium text-sm">{q.name}</h3>
|
<h3 className="font-medium text-sm">{q.name}</h3>
|
||||||
<div className="flex items-center gap-3 text-xs">
|
<div className="flex items-center gap-3 text-xs">
|
||||||
|
<RetryFailedButton queueName={q.name} count={q.failed} />
|
||||||
<Badge label="waiting" count={q.waiting} color="slate" />
|
<Badge label="waiting" count={q.waiting} color="slate" />
|
||||||
<Badge label="active" count={q.active} color="indigo" />
|
<Badge label="active" count={q.active} color="indigo" />
|
||||||
<Badge label="delayed" count={q.delayed} color="amber" />
|
<Badge label="delayed" count={q.delayed} color="amber" />
|
||||||
|
|||||||
@@ -0,0 +1,40 @@
|
|||||||
|
import { NextRequest, NextResponse } from 'next/server'
|
||||||
|
import { getServerSession } from 'next-auth'
|
||||||
|
import { authOptions } from '@/lib/auth'
|
||||||
|
import { priceUpdateQueue, maintenanceQueue, schedulerQueue } from '@/lib/queue'
|
||||||
|
import { Queue } from 'bullmq'
|
||||||
|
|
||||||
|
const QUEUES: Record<string, Queue> = {
|
||||||
|
'hashex-price-updates': priceUpdateQueue,
|
||||||
|
'hashex-maintenance': maintenanceQueue,
|
||||||
|
'hashex-scheduler': schedulerQueue,
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function POST(
|
||||||
|
req: NextRequest,
|
||||||
|
{ params }: { params: { name: string } }
|
||||||
|
) {
|
||||||
|
const session = await getServerSession(authOptions)
|
||||||
|
if (!session?.user?.isAdmin) {
|
||||||
|
return NextResponse.json({ error: 'Forbidden' }, { status: 403 })
|
||||||
|
}
|
||||||
|
|
||||||
|
const queue = QUEUES[params.name]
|
||||||
|
if (!queue) {
|
||||||
|
return NextResponse.json({ error: 'Queue not found' }, { status: 404 })
|
||||||
|
}
|
||||||
|
|
||||||
|
const { action } = await req.json() as { action: string }
|
||||||
|
|
||||||
|
if (action === 'retry-failed') {
|
||||||
|
await queue.retryJobs({ count: 100, state: 'failed' })
|
||||||
|
return NextResponse.json({ ok: true })
|
||||||
|
}
|
||||||
|
|
||||||
|
if (action === 'clean-failed') {
|
||||||
|
await queue.clean(0, 100, 'failed')
|
||||||
|
return NextResponse.json({ ok: true })
|
||||||
|
}
|
||||||
|
|
||||||
|
return NextResponse.json({ error: 'Unknown action' }, { status: 400 })
|
||||||
|
}
|
||||||
@@ -0,0 +1,41 @@
|
|||||||
|
'use client'
|
||||||
|
|
||||||
|
import { useState } from 'react'
|
||||||
|
import { useRouter } from 'next/navigation'
|
||||||
|
|
||||||
|
export default function RetryFailedButton({
|
||||||
|
queueName,
|
||||||
|
count,
|
||||||
|
}: {
|
||||||
|
queueName: string
|
||||||
|
count: number
|
||||||
|
}) {
|
||||||
|
const [loading, setLoading] = useState(false)
|
||||||
|
const router = useRouter()
|
||||||
|
|
||||||
|
if (count === 0) return null
|
||||||
|
|
||||||
|
async function handleRetry() {
|
||||||
|
setLoading(true)
|
||||||
|
try {
|
||||||
|
await fetch(`/api/admin/queues/${encodeURIComponent(queueName)}`, {
|
||||||
|
method: 'POST',
|
||||||
|
headers: { 'Content-Type': 'application/json' },
|
||||||
|
body: JSON.stringify({ action: 'retry-failed' }),
|
||||||
|
})
|
||||||
|
} finally {
|
||||||
|
router.refresh()
|
||||||
|
setLoading(false)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<button
|
||||||
|
onClick={handleRetry}
|
||||||
|
disabled={loading}
|
||||||
|
className="text-xs px-2 py-1 rounded bg-red-500/20 text-red-400 hover:bg-red-500/30 disabled:opacity-50 transition-colors"
|
||||||
|
>
|
||||||
|
{loading ? 'Retrying…' : `Retry ${count} failed`}
|
||||||
|
</button>
|
||||||
|
)
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user