import { useState, useEffect } from "react"; import { useParams, Link } from "react-router-dom"; import { Button } from "@/components/ui/button"; import { Badge } from "@/components/ui/badge"; import { Input } from "@/components/ui/input"; import { Label } from "@/components/ui/label"; import { Textarea } from "@/components/ui/textarea"; import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from "@/components/ui/select"; import { Tabs, TabsContent, TabsList, TabsTrigger } from "@/components/ui/tabs"; import { Progress } from "@/components/ui/progress"; import { ArrowLeft, Edit, ExternalLink, Code, Shield, Activity, AlertTriangle, CheckCircle, Clock, Play, FileText } from "lucide-react"; import { api } from "@/shared/config/database"; import { runRepositoryAudit, scanZipFile } from "@/features/projects/services"; import type { Project, AuditTask, CreateProjectForm } from "@/shared/types"; import { loadZipFile } from "@/shared/utils/zipStorage"; import { toast } from "sonner"; import CreateTaskDialog from "@/components/audit/CreateTaskDialog"; import TerminalProgressDialog from "@/components/audit/TerminalProgressDialog"; import { SUPPORTED_LANGUAGES } from "@/shared/constants"; export default function ProjectDetail() { const { id } = useParams<{ id: string }>(); const [project, setProject] = useState(null); const [tasks, setTasks] = useState([]); const [loading, setLoading] = useState(true); const [scanning, setScanning] = useState(false); const [showCreateTaskDialog, setShowCreateTaskDialog] = useState(false); const [showTerminalDialog, setShowTerminalDialog] = useState(false); const [currentTaskId, setCurrentTaskId] = useState(null); const [editForm, setEditForm] = useState({ name: "", description: "", repository_url: "", repository_type: "github", default_branch: "main", programming_languages: [] }); const [activeTab, setActiveTab] = useState("overview"); const [latestIssues, setLatestIssues] = useState([]); const [loadingIssues, setLoadingIssues] = useState(false); useEffect(() => { if (activeTab === 'issues' && tasks.length > 0) { loadLatestIssues(); } }, [activeTab, tasks]); const loadLatestIssues = async () => { const completedTasks = tasks.filter(t => t.status === 'completed').sort((a, b) => new Date(b.created_at).getTime() - new Date(a.created_at).getTime()); if (completedTasks.length > 0) { setLoadingIssues(true); try { const issues = await api.getAuditIssues(completedTasks[0].id); setLatestIssues(issues); } catch (error) { console.error('Failed to load issues:', error); toast.error("加载问题列表失败"); } finally { setLoadingIssues(false); } } }; const handleOpenSettings = () => { if (!project) return; // 初始化编辑表单 setEditForm({ name: project.name, description: project.description || "", repository_url: project.repository_url || "", repository_type: project.repository_type || "github", default_branch: project.default_branch || "main", programming_languages: project.programming_languages ? JSON.parse(project.programming_languages) : [] }); setActiveTab("settings"); }; // 将小写语言名转换为显示格式 const formatLanguageName = (lang: string): string => { const nameMap: Record = { 'javascript': 'JavaScript', 'typescript': 'TypeScript', 'python': 'Python', 'java': 'Java', 'go': 'Go', 'rust': 'Rust', 'cpp': 'C++', 'csharp': 'C#', 'php': 'PHP', 'ruby': 'Ruby', 'swift': 'Swift', 'kotlin': 'Kotlin' }; return nameMap[lang] || lang.charAt(0).toUpperCase() + lang.slice(1); }; const supportedLanguages = SUPPORTED_LANGUAGES.map(formatLanguageName); useEffect(() => { if (id) { loadProjectData(); } }, [id]); const loadProjectData = async () => { if (!id) return; try { setLoading(true); const [projectData, tasksData] = await Promise.all([ api.getProjectById(id), api.getAuditTasks(id) ]); setProject(projectData); setTasks(tasksData); } catch (error) { console.error('Failed to load project data:', error); toast.error("加载项目数据失败"); } finally { setLoading(false); } }; const handleRunAudit = async () => { if (!project || !id) return; // 检查是否有仓库地址 if (project.repository_url) { // 有仓库地址,启动仓库审计 try { setScanning(true); console.log('开始启动仓库审计任务...'); const taskId = await runRepositoryAudit({ projectId: id, repoUrl: project.repository_url, branch: project.default_branch || 'main', createdBy: undefined }); console.log('审计任务创建成功,taskId:', taskId); // 显示终端进度窗口 setCurrentTaskId(taskId); setShowTerminalDialog(true); // 重新加载项目数据 loadProjectData(); } catch (e: any) { console.error('启动审计失败:', e); toast.error(e?.message || '启动审计失败'); } finally { setScanning(false); } } else { // 没有仓库地址,尝试从IndexedDB加载保存的ZIP文件 try { setScanning(true); const file = await loadZipFile(id); if (file) { console.log('找到保存的ZIP文件,开始启动审计...'); try { // 启动ZIP文件审计 const taskId = await scanZipFile({ projectId: id, zipFile: file, excludePatterns: ['node_modules/**', '.git/**', 'dist/**', 'build/**'], createdBy: 'local-user' }); console.log('审计任务创建成功,taskId:', taskId); // 显示终端进度窗口 setCurrentTaskId(taskId); setShowTerminalDialog(true); // 重新加载项目数据 loadProjectData(); } catch (e: any) { console.error('启动审计失败:', e); toast.error(e?.message || '启动审计失败'); } finally { setScanning(false); } } else { setScanning(false); toast.warning('此项目未配置仓库地址,也未上传ZIP文件。请先在项目设置中配置仓库地址,或通过"新建任务"上传ZIP文件。'); // 不自动打开对话框,让用户自己选择 } } catch (error) { console.error('启动审计失败:', error); setScanning(false); toast.error('读取ZIP文件失败,请检查项目配置'); } } }; const handleSaveSettings = async () => { if (!id) return; if (!editForm.name.trim()) { toast.error("项目名称不能为空"); return; } try { await api.updateProject(id, editForm); toast.success("项目信息已保存"); loadProjectData(); } catch (error) { console.error('Failed to update project:', error); toast.error("保存失败"); } }; const handleToggleLanguage = (lang: string) => { const currentLanguages = editForm.programming_languages || []; const newLanguages = currentLanguages.includes(lang) ? currentLanguages.filter(l => l !== lang) : [...currentLanguages, lang]; setEditForm({ ...editForm, programming_languages: newLanguages }); }; 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 ; case 'running': return ; case 'failed': return ; default: return ; } }; const formatDate = (dateString: string) => { return new Date(dateString).toLocaleDateString('zh-CN', { year: 'numeric', month: 'short', day: 'numeric', hour: '2-digit', minute: '2-digit' }); }; const handleCreateTask = () => { setShowCreateTaskDialog(true); }; const handleTaskCreated = () => { toast.success("审计任务已创建", { description: '因为网络和代码文件大小等因素,审计时长通常至少需要1分钟,请耐心等待...', duration: 5000 }); loadProjectData(); // 重新加载项目数据以显示新任务 }; if (loading) { return (

加载项目数据...

); } if (!project) { return (

项目未找到

请检查项目ID是否正确

); } return (
{/* 页面标题 */}

{project.name}

{project.is_active ? '活跃' : '暂停'}

{project.description || '暂无项目描述'}

{/* ... (stats cards remain same) ... */}
{/* ... (stats cards content) ... */}

审计任务

{tasks.length}

已完成

{tasks.filter(t => t.status === 'completed').length}

发现问题

{tasks.reduce((sum, task) => sum + task.issues_count, 0)}

平均质量分

{tasks.length > 0 ? (tasks.reduce((sum, task) => sum + task.quality_score, 0) / tasks.length).toFixed(1) : '0.0' }

{/* 主要内容 */} 项目概览 审计任务 问题管理 项目设置 {/* ... (overview content remains same) ... */}
{/* 项目信息 */}

项目信息

{project.repository_url && (
仓库地址 查看仓库
)}
仓库类型 {project.repository_type === 'github' ? 'GitHub' : project.repository_type === 'gitlab' ? 'GitLab' : '其他'}
默认分支 {project.default_branch}
创建时间 {formatDate(project.created_at)}
所有者 {project.owner?.full_name || project.owner?.phone || '未知'}
{/* 编程语言 */} {project.programming_languages && (

支持的编程语言

{JSON.parse(project.programming_languages).map((lang: string) => ( {lang} ))}
)}
{/* 最近活动 */}

最近活动

{tasks.length > 0 ? (
{tasks.slice(0, 5).map((task) => (
{getStatusIcon(task.status)}

{task.task_type === 'repository' ? '仓库审计' : '即时分析'}

{formatDate(task.created_at)}

{task.status === 'completed' ? '已完成' : task.status === 'running' ? '运行中' : task.status === 'failed' ? '失败' : '等待中'}
))}
) : (

暂无活动记录

)}
{/* ... (tasks content remains same) ... */}

审计任务列表

{tasks.length > 0 ? (
{tasks.map((task) => (
{getStatusIcon(task.status)}

{task.task_type === 'repository' ? '仓库审计任务' : '即时分析任务'}

创建于 {formatDate(task.created_at)}

{task.status === 'completed' ? '已完成' : task.status === 'running' ? '运行中' : task.status === 'failed' ? '失败' : '等待中'}

{task.total_files}

总文件数

{task.total_lines}

代码行数

{task.issues_count}

发现问题

{task.quality_score.toFixed(1)}

质量评分

{task.status === 'completed' && (
质量评分 {task.quality_score.toFixed(1)}/100
)}
))}
) : (

暂无审计任务

创建第一个审计任务开始代码质量分析

)}

最新发现的问题

{tasks.length > 0 && (

来自最近一次审计 ({formatDate(tasks[0].created_at)})

)}
{loadingIssues ? (

正在加载问题列表...

) : latestIssues.length > 0 ? (
{latestIssues.map((issue, index) => (

{issue.title}

{issue.file_path}:{issue.line_number} {issue.category}
{issue.severity === 'critical' ? '严重' : issue.severity === 'high' ? '高' : issue.severity === 'medium' ? '中等' : '低'}

{issue.description}

))}
) : (

未发现问题

最近一次审计未发现明显问题,或尚未进行审计。

)}

编辑项目配置

{/* 基本信息 */}
setEditForm({ ...editForm, name: e.target.value })} placeholder="输入项目名称" className="retro-input mt-1" />