admin-projects/src/pages/Events.tsx
WangDL af76de4ff7
Some checks failed
Deploy Admin Frontend / build-and-deploy (push) Failing after 7s
feat: events queue page + restructure under 系统运维
2026-05-22 22:31:46 +08:00

72 lines
3.5 KiB
TypeScript

import { useState } from 'react'
import { useQuery, useQueryClient } from '@tanstack/react-query'
import { Table, Tag, Button, Typography, App, Space, Badge } from 'antd'
import { ReloadOutlined, RetweetOutlined, CloudServerOutlined } from '@ant-design/icons'
import { getQueueOverview, getFailedJobs, retryJob } from '@/services/events-api'
const { Title, Text } = Typography
function EventsPage() {
const { message } = App.useApp()
const qc = useQueryClient()
const [selectedQueue, setSelectedQueue] = useState<string | null>(null)
const { data: overview } = useQuery({ queryKey: ['events', 'overview'], queryFn: getQueueOverview, staleTime: 10_000 })
const { data: failed } = useQuery({
queryKey: ['events', 'failed', selectedQueue],
queryFn: () => selectedQueue ? getFailedJobs(selectedQueue) : null,
enabled: !!selectedQueue,
})
const handleRetry = async (queue: string, jobId: string) => {
await retryJob(queue, jobId)
message.success('已重试')
qc.invalidateQueries({ queryKey: ['events'] })
}
const overviewColumns = [
{ title: '队列', dataIndex: 'name', width: 160 },
{ title: '总计', dataIndex: 'total', width: 60, align: 'center' as const },
{ title: '等待', dataIndex: 'waiting', width: 60, align: 'center' as const, render: (v: number) => <Badge count={v} showZero color="blue" /> },
{ title: '进行中', dataIndex: 'active', width: 70, align: 'center' as const, render: (v: number) => <Badge count={v} showZero color="processing" /> },
{ title: '完成', dataIndex: 'completed', width: 60, align: 'center' as const, render: (v: number) => <Text type="success">{v}</Text> },
{ title: '失败', dataIndex: 'failed', width: 60, align: 'center' as const, render: (v: number, r: any) => (
v > 0 ? <Button type="link" size="small" danger onClick={() => setSelectedQueue(r.name)}>{v}</Button> : <Text type="secondary">0</Text>
)},
{ title: '延迟', dataIndex: 'delayed', width: 60, align: 'center' as const },
]
const failedColumns = [
{ title: 'Job ID', dataIndex: 'id', width: 160, ellipsis: true },
{ title: '名称', dataIndex: 'name', width: 150 },
{ title: '重试', dataIndex: 'attemptsMade', width: 60, align: 'center' as const },
{ title: '失败原因', dataIndex: 'failedReason', ellipsis: true },
{ title: '操作', width: 80, render: (_: any, r: any) => (
<Button type="link" size="small" icon={<RetweetOutlined />} onClick={() => handleRetry(selectedQueue!, r.id)}></Button>
)},
]
return (
<div>
<div style={{ display: 'flex', justifyContent: 'space-between', marginBottom: 16 }}>
<Title level={5} style={{ margin: 0 }}><CloudServerOutlined /> </Title>
<Button icon={<ReloadOutlined />} onClick={() => qc.invalidateQueries({ queryKey: ['events'] })}></Button>
</div>
<Table dataSource={overview?.queues || []} columns={overviewColumns} rowKey="name" pagination={false} style={{ marginBottom: 24 }} />
{selectedQueue && (
<>
<div style={{ display: 'flex', justifyContent: 'space-between', marginBottom: 12 }}>
<Title level={5} style={{ margin: 0, fontSize: 14 }}> · {selectedQueue}</Title>
<Button size="small" onClick={() => setSelectedQueue(null)}></Button>
</div>
<Table dataSource={failed?.jobs || []} columns={failedColumns} rowKey="id" pagination={false} size="small" />
</>
)}
</div>
)
}
export default EventsPage