CodeReview/src/pages/AuditTasks.tsx

320 lines
12 KiB
TypeScript
Raw Normal View History

import { useState, useEffect } from "react";
import { Card, CardContent } from "@/components/ui/card";
import { Button } from "@/components/ui/button";
import { Badge } from "@/components/ui/badge";
import { Input } from "@/components/ui/input";
import {
Activity,
AlertTriangle,
CheckCircle,
Clock,
Search,
Play,
FileText,
Calendar
} from "lucide-react";
import { api } from "@/shared/config/database";
import type { AuditTask } from "@/shared/types";
import { Link } from "react-router-dom";
import { toast } from "sonner";
export default function AuditTasks() {
const [tasks, setTasks] = useState<AuditTask[]>([]);
const [loading, setLoading] = useState(true);
const [searchTerm, setSearchTerm] = useState("");
const [statusFilter, setStatusFilter] = useState<string>("all");
useEffect(() => {
loadTasks();
}, []);
const loadTasks = async () => {
try {
setLoading(true);
const data = await api.getAuditTasks();
setTasks(data);
} catch (error) {
console.error('Failed to load tasks:', error);
toast.error("加载任务失败");
} finally {
setLoading(false);
}
};
const getStatusColor = (status: string) => {
switch (status) {
case 'completed': return 'bg-green-100 text-green-800';
case 'running': return 'bg-blue-100 text-blue-800';
case 'failed': return 'bg-red-100 text-red-800';
default: return 'bg-gray-100 text-gray-800';
}
};
const getStatusIcon = (status: string) => {
switch (status) {
case 'completed': return <CheckCircle className="w-4 h-4" />;
case 'running': return <Activity className="w-4 h-4" />;
case 'failed': return <AlertTriangle className="w-4 h-4" />;
default: return <Clock className="w-4 h-4" />;
}
};
const formatDate = (dateString: string) => {
return new Date(dateString).toLocaleDateString('zh-CN', {
year: 'numeric',
month: 'short',
day: 'numeric',
hour: '2-digit',
minute: '2-digit'
});
};
const filteredTasks = tasks.filter(task => {
const matchesSearch = task.project?.name.toLowerCase().includes(searchTerm.toLowerCase()) ||
task.task_type.toLowerCase().includes(searchTerm.toLowerCase());
const matchesStatus = statusFilter === "all" || task.status === statusFilter;
return matchesSearch && matchesStatus;
});
if (loading) {
return (
<div className="flex items-center justify-center min-h-screen">
<div className="animate-spin rounded-full h-32 w-32 border-b-2 border-blue-600"></div>
</div>
);
}
return (
<div className="space-y-6 animate-fade-in">
{/* 页面标题 */}
<div className="flex flex-col sm:flex-row sm:items-center sm:justify-between gap-4">
<div>
<h1 className="page-title"></h1>
<p className="page-subtitle"></p>
</div>
<Button className="btn-primary">
<Play className="w-4 h-4 mr-2" />
</Button>
</div>
{/* 统计卡片 */}
<div className="grid grid-cols-2 md:grid-cols-4 gap-4">
<Card className="stat-card">
<CardContent className="p-5">
<div className="flex items-center justify-between">
<div>
<p className="stat-label"></p>
<p className="stat-value text-xl">{tasks.length}</p>
</div>
<div className="stat-icon from-blue-500 to-blue-600">
<Activity className="w-5 h-5 text-white" />
</div>
</div>
</CardContent>
</Card>
<Card className="stat-card">
<CardContent className="p-5">
<div className="flex items-center justify-between">
<div>
<p className="stat-label"></p>
<p className="stat-value text-xl">{tasks.filter(t => t.status === 'completed').length}</p>
</div>
<div className="stat-icon from-emerald-500 to-emerald-600">
<CheckCircle className="w-5 h-5 text-white" />
</div>
</div>
</CardContent>
</Card>
<Card className="stat-card">
<CardContent className="p-5">
<div className="flex items-center justify-between">
<div>
<p className="stat-label"></p>
<p className="stat-value text-xl">{tasks.filter(t => t.status === 'running').length}</p>
</div>
<div className="stat-icon from-orange-500 to-orange-600">
<Clock className="w-5 h-5 text-white" />
</div>
</div>
</CardContent>
</Card>
<Card className="stat-card">
<CardContent className="p-5">
<div className="flex items-center justify-between">
<div>
<p className="stat-label"></p>
<p className="stat-value text-xl">{tasks.filter(t => t.status === 'failed').length}</p>
</div>
<div className="stat-icon from-red-500 to-red-600">
<AlertTriangle className="w-5 h-5 text-white" />
</div>
</div>
</CardContent>
</Card>
</div>
{/* 搜索和筛选 */}
<Card>
<CardContent className="p-4">
<div className="flex items-center space-x-4">
<div className="flex-1 relative">
<Search className="absolute left-3 top-1/2 transform -translate-y-1/2 text-gray-400 w-4 h-4" />
<Input
placeholder="搜索项目名称或任务类型..."
value={searchTerm}
onChange={(e) => setSearchTerm(e.target.value)}
className="pl-10"
/>
</div>
<div className="flex space-x-2">
<Button
variant={statusFilter === "all" ? "default" : "outline"}
size="sm"
onClick={() => setStatusFilter("all")}
>
</Button>
<Button
variant={statusFilter === "running" ? "default" : "outline"}
size="sm"
onClick={() => setStatusFilter("running")}
>
</Button>
<Button
variant={statusFilter === "completed" ? "default" : "outline"}
size="sm"
onClick={() => setStatusFilter("completed")}
>
</Button>
<Button
variant={statusFilter === "failed" ? "default" : "outline"}
size="sm"
onClick={() => setStatusFilter("failed")}
>
</Button>
</div>
</div>
</CardContent>
</Card>
{/* 任务列表 */}
{filteredTasks.length > 0 ? (
<div className="space-y-4">
{filteredTasks.map((task) => (
<Card key={task.id} className="card-modern group">
<CardContent className="p-6">
<div className="flex items-center justify-between mb-6">
<div className="flex items-center space-x-4">
<div className={`w-10 h-10 rounded-lg flex items-center justify-center ${
task.status === 'completed' ? 'bg-emerald-100 text-emerald-600' :
task.status === 'running' ? 'bg-blue-100 text-blue-600' :
task.status === 'failed' ? 'bg-red-100 text-red-600' : 'bg-gray-100 text-gray-600'
}`}>
{getStatusIcon(task.status)}
</div>
<div>
<h3 className="font-semibold text-lg text-gray-900 group-hover:text-blue-600 transition-colors">
{task.project?.name || '未知项目'}
</h3>
<p className="text-sm text-gray-500">
{task.task_type === 'repository' ? '仓库审计任务' : '即时分析任务'}
</p>
</div>
</div>
<Badge className={getStatusColor(task.status)}>
{task.status === 'completed' ? '已完成' :
task.status === 'running' ? '运行中' :
task.status === 'failed' ? '失败' : '等待中'}
</Badge>
</div>
<div className="grid grid-cols-3 md:grid-cols-5 gap-6 mb-6">
<div className="text-center">
<p className="text-2xl font-bold text-gray-900">{task.total_files}</p>
<p className="text-xs text-gray-500 mt-1"></p>
</div>
<div className="text-center">
<p className="text-2xl font-bold text-gray-900">{task.total_lines}</p>
<p className="text-xs text-gray-500 mt-1"></p>
</div>
<div className="text-center">
<p className="text-2xl font-bold text-orange-600">{task.issues_count}</p>
<p className="text-xs text-gray-500 mt-1"></p>
</div>
<div className="text-center">
<p className="text-2xl font-bold text-blue-600">{task.quality_score.toFixed(1)}</p>
<p className="text-xs text-gray-500 mt-1"></p>
</div>
<div className="text-center">
<p className="text-2xl font-bold text-emerald-600">
{Math.round((task.scanned_files / task.total_files) * 100)}%
</p>
<p className="text-xs text-gray-500 mt-1"></p>
</div>
</div>
<div className="flex items-center justify-between pt-4 border-t border-gray-100">
<div className="flex items-center space-x-6 text-sm text-gray-500">
<div className="flex items-center">
<Calendar className="w-4 h-4 mr-2" />
{formatDate(task.created_at)}
</div>
{task.completed_at && (
<div className="flex items-center">
<CheckCircle className="w-4 h-4 mr-2" />
{formatDate(task.completed_at)}
</div>
)}
</div>
<div className="flex gap-3">
<Link to={`/tasks/${task.id}`}>
<Button variant="outline" size="sm" className="btn-secondary">
<FileText className="w-4 h-4 mr-2" />
</Button>
</Link>
{task.project && (
<Link to={`/projects/${task.project.id}`}>
<Button size="sm" className="btn-primary">
</Button>
</Link>
)}
</div>
</div>
</CardContent>
</Card>
))}
</div>
) : (
<Card className="card-modern">
<CardContent className="empty-state py-16">
<div className="empty-icon">
<Activity className="w-8 h-8 text-blue-600" />
</div>
<h3 className="text-lg font-medium text-gray-900 mb-2">
{searchTerm || statusFilter !== "all" ? '未找到匹配的任务' : '暂无审计任务'}
</h3>
<p className="text-gray-500 mb-6 max-w-md">
{searchTerm || statusFilter !== "all" ? '尝试调整搜索条件或筛选器' : '创建第一个审计任务开始代码质量分析'}
</p>
{!searchTerm && statusFilter === "all" && (
<Button className="btn-primary">
<Play className="w-4 h-4 mr-2" />
</Button>
)}
</CardContent>
</Card>
)}
</div>
);
}