feat: Phase 8 - Frontend main views (T-026 to T-031)

Implement all main frontend views for the MITRE ATT&CK coverage platform:

- T-026: Dashboard with coverage summary cards and tactic breakdown table

- T-027: Interactive ATT&CK matrix with filtering by status, tactic, platform

- T-028: Technique detail page with tests, intel items, and review actions

- T-029: Test creation form with technique selector and validation

- T-030: Test detail page with drag and drop evidence upload and download

- T-031: System admin panel with MITRE sync and intel scan controls

New components: CoverageSummaryCard, TacticCoverageChart, AttackMatrix, TechniqueCell, TestForm, EvidenceUpload, EvidenceList

New API modules: metrics.ts, techniques.ts, tests.ts, evidence.ts, system.ts

All views use TanStack Query for data fetching with proper loading and error states. Role-based UI controls for admin/lead actions.
This commit is contained in:
2026-02-06 16:21:14 +01:00
parent 591b5df250
commit cb447f3803
22 changed files with 3092 additions and 27 deletions

View File

@@ -1,15 +1,134 @@
import { Shield } from "lucide-react";
import { useQuery } from "@tanstack/react-query";
import {
Shield,
CheckCircle,
AlertTriangle,
XCircle,
Clock,
HelpCircle,
Percent,
Loader2,
AlertCircle,
} from "lucide-react";
import { getCoverageSummary, getCoverageByTactic } from "../api/metrics";
import CoverageSummaryCard from "../components/CoverageSummaryCard";
import TacticCoverageChart from "../components/TacticCoverageChart";
export default function DashboardPage() {
return (
<div className="space-y-4">
<h1 className="text-2xl font-bold text-white">Dashboard</h1>
<div className="flex items-center gap-3 rounded-xl border border-gray-800 bg-gray-900 p-6">
<Shield className="h-8 w-8 text-cyan-400" />
<p className="text-gray-300">
Coverage metrics will appear here.
const {
data: summary,
isLoading: summaryLoading,
error: summaryError,
} = useQuery({
queryKey: ["metrics", "summary"],
queryFn: getCoverageSummary,
});
const {
data: tactics,
isLoading: tacticsLoading,
error: tacticsError,
} = useQuery({
queryKey: ["metrics", "by-tactic"],
queryFn: getCoverageByTactic,
});
if (summaryLoading || tacticsLoading) {
return (
<div className="flex h-64 items-center justify-center">
<Loader2 className="h-8 w-8 animate-spin text-cyan-400" />
</div>
);
}
if (summaryError || tacticsError) {
return (
<div className="flex h-64 flex-col items-center justify-center gap-2">
<AlertCircle className="h-10 w-10 text-red-400" />
<p className="text-red-400">Failed to load metrics</p>
<p className="text-sm text-gray-500">
{summaryError?.message || tacticsError?.message || "Unknown error"}
</p>
</div>
);
}
return (
<div className="space-y-6">
{/* Header */}
<div className="flex items-center justify-between">
<div>
<h1 className="text-2xl font-bold text-white">Dashboard</h1>
<p className="mt-1 text-sm text-gray-400">
MITRE ATT&CK coverage overview
</p>
</div>
{summary && (
<div className="flex items-center gap-2 rounded-lg border border-cyan-500/30 bg-cyan-500/10 px-4 py-2">
<Percent className="h-5 w-5 text-cyan-400" />
<span className="text-lg font-bold text-cyan-400">
{summary.coverage_percentage.toFixed(1)}%
</span>
<span className="text-sm text-gray-400">Coverage</span>
</div>
)}
</div>
{/* Summary Cards */}
{summary && (
<div className="grid grid-cols-1 gap-4 sm:grid-cols-2 lg:grid-cols-3 xl:grid-cols-6">
<CoverageSummaryCard
title="Total Techniques"
value={summary.total_techniques}
icon={<Shield className="h-6 w-6 text-cyan-400" />}
colorClass="text-cyan-400"
bgClass="bg-gray-900"
/>
<CoverageSummaryCard
title="Validated"
value={summary.validated}
total={summary.total_techniques}
icon={<CheckCircle className="h-6 w-6 text-green-400" />}
colorClass="text-green-400"
bgClass="bg-green-950/20"
/>
<CoverageSummaryCard
title="Partial"
value={summary.partial}
total={summary.total_techniques}
icon={<AlertTriangle className="h-6 w-6 text-yellow-400" />}
colorClass="text-yellow-400"
bgClass="bg-yellow-950/20"
/>
<CoverageSummaryCard
title="In Progress"
value={summary.in_progress}
total={summary.total_techniques}
icon={<Clock className="h-6 w-6 text-blue-400" />}
colorClass="text-blue-400"
bgClass="bg-blue-950/20"
/>
<CoverageSummaryCard
title="Not Covered"
value={summary.not_covered}
total={summary.total_techniques}
icon={<XCircle className="h-6 w-6 text-red-400" />}
colorClass="text-red-400"
bgClass="bg-red-950/20"
/>
<CoverageSummaryCard
title="Not Evaluated"
value={summary.not_evaluated}
total={summary.total_techniques}
icon={<HelpCircle className="h-6 w-6 text-gray-400" />}
colorClass="text-gray-400"
bgClass="bg-gray-900"
/>
</div>
)}
{/* Tactic Coverage Table */}
{tactics && <TacticCoverageChart data={tactics} />}
</div>
);
}

View File

@@ -0,0 +1,197 @@
import { useState, useMemo } from "react";
import { useQuery } from "@tanstack/react-query";
import { Loader2, AlertCircle, Filter, X } from "lucide-react";
import { getTechniques, type TechniqueSummary } from "../api/techniques";
import AttackMatrix from "../components/AttackMatrix";
import type { TechniqueStatus } from "../types/models";
const STATUS_OPTIONS: { value: TechniqueStatus | "all"; label: string; color: string }[] = [
{ value: "all", label: "All Statuses", color: "text-gray-400" },
{ value: "validated", label: "Validated", color: "text-green-400" },
{ value: "partial", label: "Partial", color: "text-yellow-400" },
{ value: "in_progress", label: "In Progress", color: "text-blue-400" },
{ value: "not_covered", label: "Not Covered", color: "text-red-400" },
{ value: "not_evaluated", label: "Not Evaluated", color: "text-gray-400" },
];
const PLATFORM_OPTIONS = ["all", "windows", "linux", "macos", "cloud", "network"] as const;
export default function MatrixPage() {
const [statusFilter, setStatusFilter] = useState<TechniqueStatus | "all">("all");
const [platformFilter, setPlatformFilter] = useState<string>("all");
const [tacticFilter, setTacticFilter] = useState<string>("all");
const {
data: techniques,
isLoading,
error,
} = useQuery({
queryKey: ["techniques"],
queryFn: () => getTechniques(),
});
// Extract unique tactics from techniques
const availableTactics = useMemo(() => {
if (!techniques) return [];
const tactics = new Set<string>();
for (const tech of techniques) {
if (tech.tactic) {
tech.tactic.split(",").forEach((t) => tactics.add(t.trim().toLowerCase()));
}
}
return Array.from(tactics).sort();
}, [techniques]);
// Apply filters
const filteredTechniques = useMemo(() => {
if (!techniques) return [];
return techniques.filter((tech: TechniqueSummary) => {
// Status filter
if (statusFilter !== "all" && tech.status_global !== statusFilter) {
return false;
}
// Tactic filter
if (tacticFilter !== "all") {
const techTactics = tech.tactic?.split(",").map((t) => t.trim().toLowerCase()) || [];
if (!techTactics.includes(tacticFilter)) {
return false;
}
}
// Platform filter is handled client-side since we don't have platform in summary
// For now we show all - platform filtering would need the full technique data
return true;
});
}, [techniques, statusFilter, tacticFilter]);
const hasActiveFilters = statusFilter !== "all" || tacticFilter !== "all" || platformFilter !== "all";
const clearFilters = () => {
setStatusFilter("all");
setPlatformFilter("all");
setTacticFilter("all");
};
if (isLoading) {
return (
<div className="flex h-64 items-center justify-center">
<Loader2 className="h-8 w-8 animate-spin text-cyan-400" />
</div>
);
}
if (error) {
return (
<div className="flex h-64 flex-col items-center justify-center gap-2">
<AlertCircle className="h-10 w-10 text-red-400" />
<p className="text-red-400">Failed to load techniques</p>
</div>
);
}
return (
<div className="space-y-6">
{/* Header */}
<div>
<h1 className="text-2xl font-bold text-white">ATT&CK Matrix</h1>
<p className="mt-1 text-sm text-gray-400">
Interactive MITRE ATT&CK coverage matrix click any technique for details
</p>
</div>
{/* Filters */}
<div className="flex flex-wrap items-center gap-4 rounded-xl border border-gray-800 bg-gray-900 p-4">
<div className="flex items-center gap-2">
<Filter className="h-4 w-4 text-gray-400" />
<span className="text-sm font-medium text-gray-400">Filters:</span>
</div>
{/* Status filter */}
<select
value={statusFilter}
onChange={(e) => setStatusFilter(e.target.value as TechniqueStatus | "all")}
className="rounded-lg border border-gray-700 bg-gray-800 px-3 py-1.5 text-sm text-gray-200 focus:border-cyan-500 focus:outline-none"
>
{STATUS_OPTIONS.map((opt) => (
<option key={opt.value} value={opt.value}>
{opt.label}
</option>
))}
</select>
{/* Tactic filter */}
<select
value={tacticFilter}
onChange={(e) => setTacticFilter(e.target.value)}
className="rounded-lg border border-gray-700 bg-gray-800 px-3 py-1.5 text-sm text-gray-200 focus:border-cyan-500 focus:outline-none"
>
<option value="all">All Tactics</option>
{availableTactics.map((tactic) => (
<option key={tactic} value={tactic}>
{tactic
.split("-")
.map((w) => w.charAt(0).toUpperCase() + w.slice(1))
.join(" ")}
</option>
))}
</select>
{/* Platform filter */}
<select
value={platformFilter}
onChange={(e) => setPlatformFilter(e.target.value)}
className="rounded-lg border border-gray-700 bg-gray-800 px-3 py-1.5 text-sm text-gray-200 focus:border-cyan-500 focus:outline-none"
>
{PLATFORM_OPTIONS.map((platform) => (
<option key={platform} value={platform}>
{platform === "all" ? "All Platforms" : platform.charAt(0).toUpperCase() + platform.slice(1)}
</option>
))}
</select>
{hasActiveFilters && (
<button
onClick={clearFilters}
className="flex items-center gap-1 rounded-lg border border-gray-700 bg-gray-800 px-3 py-1.5 text-sm text-gray-400 hover:border-red-500/50 hover:text-red-400"
>
<X className="h-3.5 w-3.5" />
Clear
</button>
)}
<div className="ml-auto text-sm text-gray-500">
Showing {filteredTechniques.length} of {techniques?.length || 0} techniques
</div>
</div>
{/* Matrix */}
<AttackMatrix techniques={filteredTechniques} />
{/* Legend */}
<div className="flex flex-wrap items-center gap-4 rounded-xl border border-gray-800 bg-gray-900 p-4">
<span className="text-sm font-medium text-gray-400">Legend:</span>
{STATUS_OPTIONS.filter((s) => s.value !== "all").map((status) => (
<div key={status.value} className="flex items-center gap-2">
<div
className={`h-3 w-3 rounded ${
status.value === "validated"
? "bg-green-500"
: status.value === "partial"
? "bg-yellow-500"
: status.value === "in_progress"
? "bg-blue-500"
: status.value === "not_covered"
? "bg-red-500"
: "bg-gray-600"
}`}
/>
<span className="text-xs text-gray-400">{status.label}</span>
</div>
))}
</div>
</div>
);
}

View File

@@ -1,8 +1,370 @@
import { useState } from "react";
import { useQuery, useMutation, useQueryClient } from "@tanstack/react-query";
import {
Loader2,
AlertCircle,
RefreshCw,
Server,
Database,
HardDrive,
Clock,
CheckCircle,
XCircle,
Shield,
Search,
} from "lucide-react";
import {
triggerMitreSync,
triggerIntelScan,
getSchedulerStatus,
type SyncMitreResponse,
type IntelScanResponse,
} from "../api/system";
export default function SystemPage() {
const queryClient = useQueryClient();
const [syncResult, setSyncResult] = useState<SyncMitreResponse | null>(null);
const [intelResult, setIntelResult] = useState<IntelScanResponse | null>(null);
const {
data: schedulerStatus,
isLoading: statusLoading,
error: statusError,
} = useQuery({
queryKey: ["scheduler-status"],
queryFn: getSchedulerStatus,
refetchInterval: 30000, // Refresh every 30 seconds
});
const mitreSyncMutation = useMutation({
mutationFn: triggerMitreSync,
onSuccess: (data) => {
setSyncResult(data);
queryClient.invalidateQueries({ queryKey: ["techniques"] });
queryClient.invalidateQueries({ queryKey: ["metrics"] });
},
});
const intelScanMutation = useMutation({
mutationFn: triggerIntelScan,
onSuccess: (data) => {
setIntelResult(data);
queryClient.invalidateQueries({ queryKey: ["techniques"] });
},
});
const formatNextRun = (dateStr: string | null) => {
if (!dateStr) return "Not scheduled";
const date = new Date(dateStr);
return date.toLocaleString("en-US", {
dateStyle: "medium",
timeStyle: "short",
});
};
return (
<div className="space-y-4">
<h1 className="text-2xl font-bold text-white">System</h1>
<p className="text-gray-400">System administration panel coming soon.</p>
<div className="space-y-6">
{/* Header */}
<div>
<h1 className="text-2xl font-bold text-white">System Administration</h1>
<p className="mt-1 text-sm text-gray-400">
Manage synchronization jobs and system status
</p>
</div>
{/* Actions Grid */}
<div className="grid gap-6 lg:grid-cols-2">
{/* MITRE Sync */}
<div className="rounded-xl border border-gray-800 bg-gray-900 p-6">
<div className="flex items-start gap-4">
<div className="rounded-lg bg-cyan-500/10 p-3">
<Shield className="h-6 w-6 text-cyan-400" />
</div>
<div className="flex-1">
<h2 className="text-lg font-semibold text-white">MITRE ATT&CK Sync</h2>
<p className="mt-1 text-sm text-gray-400">
Synchronize techniques from the MITRE ATT&CK framework via TAXII or GitHub fallback.
</p>
{/* Status */}
{schedulerStatus && (
<div className="mt-4 flex items-center gap-2 text-sm">
<Clock className="h-4 w-4 text-gray-500" />
<span className="text-gray-400">Next automatic sync:</span>
<span className="text-gray-300">
{formatNextRun(
schedulerStatus.jobs.find((j) => j.id === "mitre_sync")?.next_run_time || null
)}
</span>
</div>
)}
{/* Result */}
{syncResult && (
<div className="mt-4 rounded-lg border border-green-500/30 bg-green-900/20 p-3">
<div className="flex items-center gap-2">
<CheckCircle className="h-4 w-4 text-green-400" />
<span className="text-sm font-medium text-green-400">Sync Complete</span>
</div>
<div className="mt-2 grid grid-cols-2 gap-2 text-sm">
<div>
<span className="text-gray-400">New techniques:</span>
<span className="ml-2 font-medium text-white">{syncResult.new}</span>
</div>
<div>
<span className="text-gray-400">Updated:</span>
<span className="ml-2 font-medium text-white">{syncResult.updated}</span>
</div>
</div>
</div>
)}
{mitreSyncMutation.isError && (
<div className="mt-4 rounded-lg border border-red-500/30 bg-red-900/20 p-3">
<div className="flex items-center gap-2">
<XCircle className="h-4 w-4 text-red-400" />
<span className="text-sm text-red-400">
Sync failed: {(mitreSyncMutation.error as Error)?.message}
</span>
</div>
</div>
)}
<button
onClick={() => mitreSyncMutation.mutate()}
disabled={mitreSyncMutation.isPending}
className="mt-4 flex items-center gap-2 rounded-lg bg-cyan-600 px-4 py-2 text-sm font-medium text-white hover:bg-cyan-500 disabled:opacity-50 transition-colors"
>
{mitreSyncMutation.isPending ? (
<Loader2 className="h-4 w-4 animate-spin" />
) : (
<RefreshCw className="h-4 w-4" />
)}
{mitreSyncMutation.isPending ? "Syncing..." : "Sync Now"}
</button>
</div>
</div>
</div>
{/* Intel Scan */}
<div className="rounded-xl border border-gray-800 bg-gray-900 p-6">
<div className="flex items-start gap-4">
<div className="rounded-lg bg-purple-500/10 p-3">
<Search className="h-6 w-6 text-purple-400" />
</div>
<div className="flex-1">
<h2 className="text-lg font-semibold text-white">Threat Intel Scan</h2>
<p className="mt-1 text-sm text-gray-400">
Scan RSS feeds and security blogs for new threat intelligence related to techniques.
</p>
{/* Status */}
{schedulerStatus && (
<div className="mt-4 flex items-center gap-2 text-sm">
<Clock className="h-4 w-4 text-gray-500" />
<span className="text-gray-400">Next automatic scan:</span>
<span className="text-gray-300">
{formatNextRun(
schedulerStatus.jobs.find((j) => j.id === "intel_scan")?.next_run_time || null
)}
</span>
</div>
)}
{/* Result */}
{intelResult && (
<div className="mt-4 rounded-lg border border-green-500/30 bg-green-900/20 p-3">
<div className="flex items-center gap-2">
<CheckCircle className="h-4 w-4 text-green-400" />
<span className="text-sm font-medium text-green-400">Scan Complete</span>
</div>
<div className="mt-2 text-sm">
<span className="text-gray-400">New intel items:</span>
<span className="ml-2 font-medium text-white">{intelResult.new_items}</span>
</div>
</div>
)}
{intelScanMutation.isError && (
<div className="mt-4 rounded-lg border border-red-500/30 bg-red-900/20 p-3">
<div className="flex items-center gap-2">
<XCircle className="h-4 w-4 text-red-400" />
<span className="text-sm text-red-400">
Scan failed: {(intelScanMutation.error as Error)?.message}
</span>
</div>
</div>
)}
<button
onClick={() => intelScanMutation.mutate()}
disabled={intelScanMutation.isPending}
className="mt-4 flex items-center gap-2 rounded-lg bg-purple-600 px-4 py-2 text-sm font-medium text-white hover:bg-purple-500 disabled:opacity-50 transition-colors"
>
{intelScanMutation.isPending ? (
<Loader2 className="h-4 w-4 animate-spin" />
) : (
<Search className="h-4 w-4" />
)}
{intelScanMutation.isPending ? "Scanning..." : "Scan Now"}
</button>
</div>
</div>
</div>
</div>
{/* System Information */}
<div className="rounded-xl border border-gray-800 bg-gray-900 p-6">
<h2 className="mb-4 text-lg font-semibold text-white">System Information</h2>
<div className="grid gap-4 sm:grid-cols-2 lg:grid-cols-4">
{/* Backend Status */}
<div className="rounded-lg border border-gray-800 bg-gray-800/50 p-4">
<div className="flex items-center gap-3">
<Server className="h-5 w-5 text-green-400" />
<div>
<p className="text-xs font-medium uppercase text-gray-500">Backend</p>
<p className="text-sm font-medium text-green-400">Online</p>
</div>
</div>
</div>
{/* Database Status */}
<div className="rounded-lg border border-gray-800 bg-gray-800/50 p-4">
<div className="flex items-center gap-3">
<Database className="h-5 w-5 text-green-400" />
<div>
<p className="text-xs font-medium uppercase text-gray-500">PostgreSQL</p>
<p className="text-sm font-medium text-green-400">Connected</p>
</div>
</div>
</div>
{/* MinIO Status */}
<div className="rounded-lg border border-gray-800 bg-gray-800/50 p-4">
<div className="flex items-center gap-3">
<HardDrive className="h-5 w-5 text-green-400" />
<div>
<p className="text-xs font-medium uppercase text-gray-500">MinIO Storage</p>
<p className="text-sm font-medium text-green-400">Available</p>
</div>
</div>
</div>
{/* Scheduler Status */}
<div className="rounded-lg border border-gray-800 bg-gray-800/50 p-4">
<div className="flex items-center gap-3">
<Clock
className={`h-5 w-5 ${
schedulerStatus?.running ? "text-green-400" : "text-yellow-400"
}`}
/>
<div>
<p className="text-xs font-medium uppercase text-gray-500">Scheduler</p>
<p
className={`text-sm font-medium ${
schedulerStatus?.running ? "text-green-400" : "text-yellow-400"
}`}
>
{statusLoading
? "Checking..."
: schedulerStatus?.running
? "Running"
: "Stopped"}
</p>
</div>
</div>
</div>
</div>
</div>
{/* Scheduled Jobs */}
<div className="rounded-xl border border-gray-800 bg-gray-900 p-6">
<h2 className="mb-4 text-lg font-semibold text-white">Scheduled Jobs</h2>
{statusLoading ? (
<div className="flex items-center justify-center py-8">
<Loader2 className="h-6 w-6 animate-spin text-cyan-400" />
</div>
) : statusError ? (
<div className="flex items-center justify-center gap-2 py-8 text-red-400">
<AlertCircle className="h-5 w-5" />
<span>Failed to load scheduler status</span>
</div>
) : (
<div className="overflow-x-auto">
<table className="w-full text-left text-sm">
<thead>
<tr className="border-b border-gray-800">
<th className="pb-3 pr-4 font-medium text-gray-400">Job ID</th>
<th className="pb-3 px-4 font-medium text-gray-400">Name</th>
<th className="pb-3 px-4 font-medium text-gray-400">Next Run</th>
<th className="pb-3 pl-4 font-medium text-gray-400">Status</th>
</tr>
</thead>
<tbody>
{schedulerStatus?.jobs.map((job) => (
<tr
key={job.id}
className="border-b border-gray-800/50 hover:bg-gray-800/30 transition-colors"
>
<td className="py-3 pr-4">
<code className="rounded bg-gray-800 px-2 py-0.5 text-xs text-cyan-400">
{job.id}
</code>
</td>
<td className="py-3 px-4 text-gray-200">{job.name}</td>
<td className="py-3 px-4 text-gray-400">
{formatNextRun(job.next_run_time)}
</td>
<td className="py-3 pl-4">
{job.next_run_time ? (
<span className="inline-flex items-center gap-1 text-green-400">
<CheckCircle className="h-3.5 w-3.5" />
Scheduled
</span>
) : (
<span className="inline-flex items-center gap-1 text-yellow-400">
<Clock className="h-3.5 w-3.5" />
Pending
</span>
)}
</td>
</tr>
))}
{(!schedulerStatus?.jobs || schedulerStatus.jobs.length === 0) && (
<tr>
<td colSpan={4} className="py-8 text-center text-gray-400">
No scheduled jobs found
</td>
</tr>
)}
</tbody>
</table>
</div>
)}
</div>
{/* Version Info */}
<div className="rounded-xl border border-gray-800 bg-gray-900 p-6">
<h2 className="mb-4 text-lg font-semibold text-white">Version Information</h2>
<dl className="grid gap-4 sm:grid-cols-2 lg:grid-cols-4">
<div>
<dt className="text-xs font-medium uppercase text-gray-500">Platform</dt>
<dd className="mt-1 text-sm text-gray-300">Aegis v0.1.0</dd>
</div>
<div>
<dt className="text-xs font-medium uppercase text-gray-500">Backend</dt>
<dd className="mt-1 text-sm text-gray-300">FastAPI + Python 3.11</dd>
</div>
<div>
<dt className="text-xs font-medium uppercase text-gray-500">Frontend</dt>
<dd className="mt-1 text-sm text-gray-300">React 19 + TypeScript</dd>
</div>
<div>
<dt className="text-xs font-medium uppercase text-gray-500">Database</dt>
<dd className="mt-1 text-sm text-gray-300">PostgreSQL 15</dd>
</div>
</dl>
</div>
</div>
);
}

View File

@@ -0,0 +1,363 @@
import { useParams, useNavigate } from "react-router-dom";
import { useQuery, useMutation, useQueryClient } from "@tanstack/react-query";
import {
Loader2,
AlertCircle,
ArrowLeft,
CheckCircle,
Clock,
Shield,
FileText,
ExternalLink,
Plus,
Check,
X,
AlertTriangle,
} from "lucide-react";
import { getTechniqueByMitreId, markTechniqueReviewed } from "../api/techniques";
import { useAuth } from "../context/AuthContext";
import type { TechniqueStatus, TestState, TestResult } from "../types/models";
const statusBadgeColors: Record<TechniqueStatus, string> = {
validated: "bg-green-900/50 text-green-400 border-green-500/30",
partial: "bg-yellow-900/50 text-yellow-400 border-yellow-500/30",
in_progress: "bg-blue-900/50 text-blue-400 border-blue-500/30",
not_covered: "bg-red-900/50 text-red-400 border-red-500/30",
not_evaluated: "bg-gray-800/50 text-gray-400 border-gray-600/30",
review_required: "bg-orange-900/50 text-orange-400 border-orange-500/30",
};
const testStateBadgeColors: Record<TestState, string> = {
draft: "bg-gray-800/50 text-gray-400 border-gray-600/30",
in_review: "bg-blue-900/50 text-blue-400 border-blue-500/30",
validated: "bg-green-900/50 text-green-400 border-green-500/30",
rejected: "bg-red-900/50 text-red-400 border-red-500/30",
};
const testResultBadgeColors: Record<TestResult, string> = {
detected: "bg-green-900/50 text-green-400 border-green-500/30",
not_detected: "bg-red-900/50 text-red-400 border-red-500/30",
partially_detected: "bg-yellow-900/50 text-yellow-400 border-yellow-500/30",
};
export default function TechniqueDetailPage() {
const { mitreId } = useParams<{ mitreId: string }>();
const navigate = useNavigate();
const queryClient = useQueryClient();
const { user } = useAuth();
const canReview =
user?.role === "admin" || user?.role === "red_lead" || user?.role === "blue_lead";
const {
data: technique,
isLoading,
error,
} = useQuery({
queryKey: ["technique", mitreId],
queryFn: () => getTechniqueByMitreId(mitreId!),
enabled: !!mitreId,
});
const reviewMutation = useMutation({
mutationFn: () => markTechniqueReviewed(mitreId!),
onSuccess: () => {
queryClient.invalidateQueries({ queryKey: ["technique", mitreId] });
queryClient.invalidateQueries({ queryKey: ["techniques"] });
},
});
if (isLoading) {
return (
<div className="flex h-64 items-center justify-center">
<Loader2 className="h-8 w-8 animate-spin text-cyan-400" />
</div>
);
}
if (error || !technique) {
return (
<div className="flex h-64 flex-col items-center justify-center gap-2">
<AlertCircle className="h-10 w-10 text-red-400" />
<p className="text-red-400">Failed to load technique</p>
<button
onClick={() => navigate("/techniques")}
className="mt-2 flex items-center gap-1 text-sm text-cyan-400 hover:underline"
>
<ArrowLeft className="h-4 w-4" />
Back to techniques
</button>
</div>
);
}
const formatDate = (dateStr: string | null) => {
if (!dateStr) return "—";
return new Date(dateStr).toLocaleDateString("en-US", {
year: "numeric",
month: "short",
day: "numeric",
hour: "2-digit",
minute: "2-digit",
});
};
return (
<div className="space-y-6">
{/* Back button */}
<button
onClick={() => navigate("/techniques")}
className="flex items-center gap-1 text-sm text-gray-400 hover:text-cyan-400 transition-colors"
>
<ArrowLeft className="h-4 w-4" />
Back to techniques
</button>
{/* Header */}
<div className="rounded-xl border border-gray-800 bg-gray-900 p-6">
<div className="flex items-start justify-between">
<div className="flex items-start gap-4">
<div className="rounded-lg bg-cyan-500/10 p-3">
<Shield className="h-8 w-8 text-cyan-400" />
</div>
<div>
<div className="flex items-center gap-3">
<h1 className="text-2xl font-bold text-white">{technique.mitre_id}</h1>
<span
className={`inline-flex rounded-full border px-2.5 py-0.5 text-xs font-medium ${
statusBadgeColors[technique.status_global]
}`}
>
{technique.status_global.replace(/_/g, " ")}
</span>
{technique.review_required && (
<span className="inline-flex items-center gap-1 rounded-full border border-orange-500/30 bg-orange-900/50 px-2.5 py-0.5 text-xs font-medium text-orange-400">
<AlertTriangle className="h-3 w-3" />
Review Required
</span>
)}
</div>
<p className="mt-1 text-lg text-gray-300">{technique.name}</p>
</div>
</div>
{canReview && technique.review_required && (
<button
onClick={() => reviewMutation.mutate()}
disabled={reviewMutation.isPending}
className="flex items-center gap-2 rounded-lg bg-cyan-600 px-4 py-2 text-sm font-medium text-white hover:bg-cyan-500 disabled:opacity-50 transition-colors"
>
{reviewMutation.isPending ? (
<Loader2 className="h-4 w-4 animate-spin" />
) : (
<CheckCircle className="h-4 w-4" />
)}
Mark as Reviewed
</button>
)}
</div>
</div>
{/* Info Section */}
<div className="grid gap-6 lg:grid-cols-3">
{/* Description */}
<div className="lg:col-span-2 rounded-xl border border-gray-800 bg-gray-900 p-6">
<h2 className="mb-3 text-lg font-semibold text-white">Description</h2>
<p className="text-sm text-gray-400 leading-relaxed whitespace-pre-wrap">
{technique.description || "No description available."}
</p>
</div>
{/* Metadata */}
<div className="rounded-xl border border-gray-800 bg-gray-900 p-6">
<h2 className="mb-4 text-lg font-semibold text-white">Details</h2>
<dl className="space-y-4">
<div>
<dt className="text-xs font-medium uppercase text-gray-500">Tactic</dt>
<dd className="mt-1 text-sm text-gray-300 capitalize">
{technique.tactic?.replace(/-/g, " ") || "—"}
</dd>
</div>
<div>
<dt className="text-xs font-medium uppercase text-gray-500">Platforms</dt>
<dd className="mt-1 flex flex-wrap gap-1">
{technique.platforms && technique.platforms.length > 0 ? (
technique.platforms.map((p) => (
<span
key={p}
className="rounded-full bg-gray-800 px-2 py-0.5 text-xs text-gray-300"
>
{p}
</span>
))
) : (
<span className="text-sm text-gray-500"></span>
)}
</dd>
</div>
<div>
<dt className="text-xs font-medium uppercase text-gray-500">Subtechnique</dt>
<dd className="mt-1 text-sm text-gray-300">
{technique.is_subtechnique ? `Yes (${technique.parent_mitre_id})` : "No"}
</dd>
</div>
<div>
<dt className="text-xs font-medium uppercase text-gray-500">Last Review</dt>
<dd className="mt-1 text-sm text-gray-300">
{formatDate(technique.last_review_date)}
</dd>
</div>
<div>
<dt className="text-xs font-medium uppercase text-gray-500">MITRE Version</dt>
<dd className="mt-1 text-sm text-gray-300">{technique.mitre_version || "—"}</dd>
</div>
</dl>
</div>
</div>
{/* Tests Section */}
<div className="rounded-xl border border-gray-800 bg-gray-900 p-6">
<div className="mb-4 flex items-center justify-between">
<h2 className="text-lg font-semibold text-white">Associated Tests</h2>
<button
onClick={() => navigate(`/tests/new?technique=${technique.id}`)}
className="flex items-center gap-1.5 rounded-lg border border-cyan-500/30 bg-cyan-500/10 px-3 py-1.5 text-sm font-medium text-cyan-400 hover:bg-cyan-500/20 transition-colors"
>
<Plus className="h-4 w-4" />
New Test
</button>
</div>
{technique.tests && technique.tests.length > 0 ? (
<div className="overflow-x-auto">
<table className="w-full text-left text-sm">
<thead>
<tr className="border-b border-gray-800">
<th className="pb-3 pr-4 font-medium text-gray-400">Name</th>
<th className="pb-3 px-4 font-medium text-gray-400">State</th>
<th className="pb-3 px-4 font-medium text-gray-400">Result</th>
<th className="pb-3 px-4 font-medium text-gray-400">Platform</th>
<th className="pb-3 px-4 font-medium text-gray-400">Created</th>
<th className="pb-3 pl-4 font-medium text-gray-400">Actions</th>
</tr>
</thead>
<tbody>
{technique.tests.map((test) => (
<tr
key={test.id}
className="border-b border-gray-800/50 hover:bg-gray-800/30 transition-colors"
>
<td className="py-3 pr-4">
<span className="font-medium text-gray-200">{test.name}</span>
</td>
<td className="py-3 px-4">
<span
className={`inline-flex rounded-full border px-2 py-0.5 text-xs font-medium ${
testStateBadgeColors[test.state]
}`}
>
{test.state.replace(/_/g, " ")}
</span>
</td>
<td className="py-3 px-4">
{test.result ? (
<span
className={`inline-flex rounded-full border px-2 py-0.5 text-xs font-medium ${
testResultBadgeColors[test.result]
}`}
>
{test.result.replace(/_/g, " ")}
</span>
) : (
<span className="text-gray-600"></span>
)}
</td>
<td className="py-3 px-4">
<span className="text-gray-400 capitalize">{test.platform || "—"}</span>
</td>
<td className="py-3 px-4">
<span className="text-gray-400">{formatDate(test.created_at)}</span>
</td>
<td className="py-3 pl-4">
<div className="flex items-center gap-2">
<button
onClick={() => navigate(`/tests/${test.id}`)}
className="rounded p-1 text-gray-400 hover:bg-gray-800 hover:text-cyan-400"
title="View Details"
>
<FileText className="h-4 w-4" />
</button>
{canReview && test.state === "draft" && (
<>
<button
onClick={() => navigate(`/tests/${test.id}/validate`)}
className="rounded p-1 text-gray-400 hover:bg-green-900/50 hover:text-green-400"
title="Validate"
>
<Check className="h-4 w-4" />
</button>
<button
onClick={() => navigate(`/tests/${test.id}/reject`)}
className="rounded p-1 text-gray-400 hover:bg-red-900/50 hover:text-red-400"
title="Reject"
>
<X className="h-4 w-4" />
</button>
</>
)}
</div>
</td>
</tr>
))}
</tbody>
</table>
</div>
) : (
<div className="flex flex-col items-center justify-center py-8 text-gray-400">
<Clock className="mb-2 h-8 w-8" />
<p>No tests have been created for this technique yet.</p>
<button
onClick={() => navigate(`/tests/new?technique=${technique.id}`)}
className="mt-3 flex items-center gap-1 text-sm text-cyan-400 hover:underline"
>
<Plus className="h-4 w-4" />
Create the first test
</button>
</div>
)}
</div>
{/* Intel Items Section */}
{technique.intel_items && technique.intel_items.length > 0 && (
<div className="rounded-xl border border-gray-800 bg-gray-900 p-6">
<h2 className="mb-4 text-lg font-semibold text-white">Threat Intelligence</h2>
<div className="space-y-3">
{technique.intel_items.map((intel) => (
<div
key={intel.id}
className="flex items-center justify-between rounded-lg border border-gray-800 bg-gray-800/30 p-4"
>
<div>
<p className="font-medium text-gray-200">{intel.title || intel.url}</p>
<p className="mt-0.5 text-xs text-gray-500">
{intel.source && <span>{intel.source} </span>}
Detected {formatDate(intel.detected_at)}
</p>
</div>
<a
href={intel.url}
target="_blank"
rel="noopener noreferrer"
className="flex items-center gap-1 text-sm text-cyan-400 hover:underline"
>
View
<ExternalLink className="h-3.5 w-3.5" />
</a>
</div>
))}
</div>
</div>
)}
</div>
);
}

View File

@@ -1,8 +1,280 @@
import { useState, useMemo } from "react";
import { useQuery } from "@tanstack/react-query";
import { Loader2, AlertCircle, Filter, X, Grid3X3, List } from "lucide-react";
import { getTechniques, type TechniqueSummary } from "../api/techniques";
import AttackMatrix from "../components/AttackMatrix";
import type { TechniqueStatus } from "../types/models";
import { useNavigate } from "react-router-dom";
const STATUS_OPTIONS: { value: TechniqueStatus | "all"; label: string; color: string }[] = [
{ value: "all", label: "All Statuses", color: "text-gray-400" },
{ value: "validated", label: "Validated", color: "text-green-400" },
{ value: "partial", label: "Partial", color: "text-yellow-400" },
{ value: "in_progress", label: "In Progress", color: "text-blue-400" },
{ value: "not_covered", label: "Not Covered", color: "text-red-400" },
{ value: "not_evaluated", label: "Not Evaluated", color: "text-gray-400" },
];
const PLATFORM_OPTIONS = ["all", "windows", "linux", "macos", "cloud", "network"] as const;
const statusBadgeColors: Record<TechniqueStatus, string> = {
validated: "bg-green-900/50 text-green-400 border-green-500/30",
partial: "bg-yellow-900/50 text-yellow-400 border-yellow-500/30",
in_progress: "bg-blue-900/50 text-blue-400 border-blue-500/30",
not_covered: "bg-red-900/50 text-red-400 border-red-500/30",
not_evaluated: "bg-gray-800/50 text-gray-400 border-gray-600/30",
review_required: "bg-orange-900/50 text-orange-400 border-orange-500/30",
};
export default function TechniquesPage() {
const navigate = useNavigate();
const [viewMode, setViewMode] = useState<"matrix" | "list">("matrix");
const [statusFilter, setStatusFilter] = useState<TechniqueStatus | "all">("all");
const [platformFilter, setPlatformFilter] = useState<string>("all");
const [tacticFilter, setTacticFilter] = useState<string>("all");
const {
data: techniques,
isLoading,
error,
} = useQuery({
queryKey: ["techniques"],
queryFn: () => getTechniques(),
});
// Extract unique tactics from techniques
const availableTactics = useMemo(() => {
if (!techniques) return [];
const tactics = new Set<string>();
for (const tech of techniques) {
if (tech.tactic) {
tech.tactic.split(",").forEach((t) => tactics.add(t.trim().toLowerCase()));
}
}
return Array.from(tactics).sort();
}, [techniques]);
// Apply filters
const filteredTechniques = useMemo(() => {
if (!techniques) return [];
return techniques.filter((tech: TechniqueSummary) => {
// Status filter
if (statusFilter !== "all" && tech.status_global !== statusFilter) {
return false;
}
// Tactic filter
if (tacticFilter !== "all") {
const techTactics = tech.tactic?.split(",").map((t) => t.trim().toLowerCase()) || [];
if (!techTactics.includes(tacticFilter)) {
return false;
}
}
return true;
});
}, [techniques, statusFilter, tacticFilter]);
const hasActiveFilters = statusFilter !== "all" || tacticFilter !== "all" || platformFilter !== "all";
const clearFilters = () => {
setStatusFilter("all");
setPlatformFilter("all");
setTacticFilter("all");
};
if (isLoading) {
return (
<div className="flex h-64 items-center justify-center">
<Loader2 className="h-8 w-8 animate-spin text-cyan-400" />
</div>
);
}
if (error) {
return (
<div className="flex h-64 flex-col items-center justify-center gap-2">
<AlertCircle className="h-10 w-10 text-red-400" />
<p className="text-red-400">Failed to load techniques</p>
</div>
);
}
return (
<div className="space-y-4">
<h1 className="text-2xl font-bold text-white">Techniques</h1>
<p className="text-gray-400">MITRE ATT&CK technique listing coming soon.</p>
<div className="space-y-6">
{/* Header */}
<div className="flex items-center justify-between">
<div>
<h1 className="text-2xl font-bold text-white">ATT&CK Techniques</h1>
<p className="mt-1 text-sm text-gray-400">
MITRE ATT&CK coverage matrix click any technique for details
</p>
</div>
<div className="flex items-center gap-2 rounded-lg border border-gray-700 bg-gray-800 p-1">
<button
onClick={() => setViewMode("matrix")}
className={`flex items-center gap-1.5 rounded px-3 py-1.5 text-sm transition-colors ${
viewMode === "matrix"
? "bg-cyan-500/20 text-cyan-400"
: "text-gray-400 hover:text-gray-200"
}`}
>
<Grid3X3 className="h-4 w-4" />
Matrix
</button>
<button
onClick={() => setViewMode("list")}
className={`flex items-center gap-1.5 rounded px-3 py-1.5 text-sm transition-colors ${
viewMode === "list"
? "bg-cyan-500/20 text-cyan-400"
: "text-gray-400 hover:text-gray-200"
}`}
>
<List className="h-4 w-4" />
List
</button>
</div>
</div>
{/* Filters */}
<div className="flex flex-wrap items-center gap-4 rounded-xl border border-gray-800 bg-gray-900 p-4">
<div className="flex items-center gap-2">
<Filter className="h-4 w-4 text-gray-400" />
<span className="text-sm font-medium text-gray-400">Filters:</span>
</div>
{/* Status filter */}
<select
value={statusFilter}
onChange={(e) => setStatusFilter(e.target.value as TechniqueStatus | "all")}
className="rounded-lg border border-gray-700 bg-gray-800 px-3 py-1.5 text-sm text-gray-200 focus:border-cyan-500 focus:outline-none"
>
{STATUS_OPTIONS.map((opt) => (
<option key={opt.value} value={opt.value}>
{opt.label}
</option>
))}
</select>
{/* Tactic filter */}
<select
value={tacticFilter}
onChange={(e) => setTacticFilter(e.target.value)}
className="rounded-lg border border-gray-700 bg-gray-800 px-3 py-1.5 text-sm text-gray-200 focus:border-cyan-500 focus:outline-none"
>
<option value="all">All Tactics</option>
{availableTactics.map((tactic) => (
<option key={tactic} value={tactic}>
{tactic
.split("-")
.map((w) => w.charAt(0).toUpperCase() + w.slice(1))
.join(" ")}
</option>
))}
</select>
{/* Platform filter */}
<select
value={platformFilter}
onChange={(e) => setPlatformFilter(e.target.value)}
className="rounded-lg border border-gray-700 bg-gray-800 px-3 py-1.5 text-sm text-gray-200 focus:border-cyan-500 focus:outline-none"
>
{PLATFORM_OPTIONS.map((platform) => (
<option key={platform} value={platform}>
{platform === "all" ? "All Platforms" : platform.charAt(0).toUpperCase() + platform.slice(1)}
</option>
))}
</select>
{hasActiveFilters && (
<button
onClick={clearFilters}
className="flex items-center gap-1 rounded-lg border border-gray-700 bg-gray-800 px-3 py-1.5 text-sm text-gray-400 hover:border-red-500/50 hover:text-red-400"
>
<X className="h-3.5 w-3.5" />
Clear
</button>
)}
<div className="ml-auto text-sm text-gray-500">
Showing {filteredTechniques.length} of {techniques?.length || 0} techniques
</div>
</div>
{/* Matrix or List View */}
{viewMode === "matrix" ? (
<AttackMatrix techniques={filteredTechniques} />
) : (
<div className="rounded-xl border border-gray-800 bg-gray-900">
<table className="w-full text-left text-sm">
<thead>
<tr className="border-b border-gray-800">
<th className="px-4 py-3 font-medium text-gray-400">MITRE ID</th>
<th className="px-4 py-3 font-medium text-gray-400">Name</th>
<th className="px-4 py-3 font-medium text-gray-400">Tactic</th>
<th className="px-4 py-3 font-medium text-gray-400">Status</th>
</tr>
</thead>
<tbody>
{filteredTechniques.map((tech) => (
<tr
key={tech.id}
onClick={() => navigate(`/techniques/${tech.mitre_id}`)}
className="cursor-pointer border-b border-gray-800/50 hover:bg-gray-800/50 transition-colors"
>
<td className="px-4 py-3">
<span className="font-mono text-cyan-400">{tech.mitre_id}</span>
</td>
<td className="px-4 py-3 text-gray-200">{tech.name}</td>
<td className="px-4 py-3">
<span className="text-gray-400 capitalize">
{tech.tactic?.replace(/-/g, " ") || "—"}
</span>
</td>
<td className="px-4 py-3">
<span
className={`inline-flex rounded-full border px-2 py-0.5 text-xs font-medium ${
statusBadgeColors[tech.status_global]
}`}
>
{tech.status_global.replace(/_/g, " ")}
</span>
</td>
</tr>
))}
</tbody>
</table>
{filteredTechniques.length === 0 && (
<div className="p-8 text-center text-gray-400">
No techniques found matching your filters
</div>
)}
</div>
)}
{/* Legend */}
<div className="flex flex-wrap items-center gap-4 rounded-xl border border-gray-800 bg-gray-900 p-4">
<span className="text-sm font-medium text-gray-400">Legend:</span>
{STATUS_OPTIONS.filter((s) => s.value !== "all").map((status) => (
<div key={status.value} className="flex items-center gap-2">
<div
className={`h-3 w-3 rounded ${
status.value === "validated"
? "bg-green-500"
: status.value === "partial"
? "bg-yellow-500"
: status.value === "in_progress"
? "bg-blue-500"
: status.value === "not_covered"
? "bg-red-500"
: "bg-gray-600"
}`}
/>
<span className="text-xs text-gray-400">{status.label}</span>
</div>
))}
</div>
</div>
);
}

View File

@@ -0,0 +1,109 @@
import { useNavigate, useSearchParams } from "react-router-dom";
import { useMutation, useQueryClient, useQuery } from "@tanstack/react-query";
import { ArrowLeft, FlaskConical } from "lucide-react";
import TestForm, { type TestFormData } from "../components/TestForm";
import { createTest } from "../api/tests";
import { getTechniqueByMitreId } from "../api/techniques";
export default function TestCreatePage() {
const navigate = useNavigate();
const queryClient = useQueryClient();
const [searchParams] = useSearchParams();
// Get technique ID from URL query param (UUID format)
const techniqueId = searchParams.get("technique");
// If we have a technique ID, try to get its mitre_id for the back link
const { data: technique } = useQuery({
queryKey: ["techniqueById", techniqueId],
queryFn: async () => {
// We need to find the mitre_id from the technique list
// This is a workaround since we get UUID but need mitre_id
const response = await fetch(`http://localhost:8000/api/v1/techniques`, {
headers: {
Authorization: `Bearer ${localStorage.getItem("token")}`,
},
});
const techniques = await response.json();
return techniques.find((t: { id: string }) => t.id === techniqueId);
},
enabled: !!techniqueId,
});
const createMutation = useMutation({
mutationFn: createTest,
onSuccess: (data) => {
queryClient.invalidateQueries({ queryKey: ["techniques"] });
queryClient.invalidateQueries({ queryKey: ["technique"] });
// Navigate back to the technique detail if we came from there
if (technique?.mitre_id) {
navigate(`/techniques/${technique.mitre_id}`);
} else {
navigate("/tests");
}
},
});
const handleSubmit = (data: TestFormData) => {
createMutation.mutate({
technique_id: data.technique_id,
name: data.name,
description: data.description || undefined,
platform: data.platform || undefined,
procedure_text: data.procedure_text || undefined,
tool_used: data.tool_used || undefined,
});
};
const handleBack = () => {
if (technique?.mitre_id) {
navigate(`/techniques/${technique.mitre_id}`);
} else {
navigate("/tests");
}
};
return (
<div className="space-y-6">
{/* Back button */}
<button
onClick={handleBack}
className="flex items-center gap-1 text-sm text-gray-400 hover:text-cyan-400 transition-colors"
>
<ArrowLeft className="h-4 w-4" />
{technique ? `Back to ${technique.mitre_id}` : "Back to tests"}
</button>
{/* Header */}
<div className="flex items-center gap-4">
<div className="rounded-lg bg-cyan-500/10 p-3">
<FlaskConical className="h-8 w-8 text-cyan-400" />
</div>
<div>
<h1 className="text-2xl font-bold text-white">Create New Test</h1>
<p className="mt-1 text-sm text-gray-400">
Document a security test for technique validation
</p>
</div>
</div>
{/* Error display */}
{createMutation.isError && (
<div className="rounded-lg border border-red-500/30 bg-red-900/20 p-4">
<p className="text-sm text-red-400">
Failed to create test: {(createMutation.error as Error)?.message || "Unknown error"}
</p>
</div>
)}
{/* Form */}
<div className="rounded-xl border border-gray-800 bg-gray-900 p-6">
<TestForm
preselectedTechniqueId={techniqueId || undefined}
onSubmit={handleSubmit}
isSubmitting={createMutation.isPending}
/>
</div>
</div>
);
}

View File

@@ -0,0 +1,355 @@
import { useParams, useNavigate } from "react-router-dom";
import { useQuery, useMutation, useQueryClient } from "@tanstack/react-query";
import {
Loader2,
AlertCircle,
ArrowLeft,
FlaskConical,
CheckCircle,
XCircle,
Clock,
FileText,
} from "lucide-react";
import { getTestById, validateTest, rejectTest } from "../api/tests";
import { uploadEvidence, getEvidence } from "../api/evidence";
import EvidenceUpload from "../components/EvidenceUpload";
import EvidenceList from "../components/EvidenceList";
import { useAuth } from "../context/AuthContext";
import type { TestState, TestResult } from "../types/models";
import { useState } from "react";
const testStateBadgeColors: Record<TestState, string> = {
draft: "bg-gray-800/50 text-gray-400 border-gray-600/30",
in_review: "bg-blue-900/50 text-blue-400 border-blue-500/30",
validated: "bg-green-900/50 text-green-400 border-green-500/30",
rejected: "bg-red-900/50 text-red-400 border-red-500/30",
};
const testResultBadgeColors: Record<TestResult, string> = {
detected: "bg-green-900/50 text-green-400 border-green-500/30",
not_detected: "bg-red-900/50 text-red-400 border-red-500/30",
partially_detected: "bg-yellow-900/50 text-yellow-400 border-yellow-500/30",
};
const RESULTS: { value: TestResult; label: string }[] = [
{ value: "detected", label: "Detected" },
{ value: "not_detected", label: "Not Detected" },
{ value: "partially_detected", label: "Partially Detected" },
];
export default function TestDetailPage() {
const { testId } = useParams<{ testId: string }>();
const navigate = useNavigate();
const queryClient = useQueryClient();
const { user } = useAuth();
const [showValidateModal, setShowValidateModal] = useState(false);
const [selectedResult, setSelectedResult] = useState<TestResult>("detected");
const canValidate =
user?.role === "admin" || user?.role === "red_lead" || user?.role === "blue_lead";
const {
data: test,
isLoading,
error,
} = useQuery({
queryKey: ["test", testId],
queryFn: () => getTestById(testId!),
enabled: !!testId,
});
const uploadMutation = useMutation({
mutationFn: (file: File) => uploadEvidence(testId!, file),
onSuccess: () => {
queryClient.invalidateQueries({ queryKey: ["test", testId] });
},
});
const validateMutation = useMutation({
mutationFn: () => validateTest(testId!, { result: selectedResult }),
onSuccess: () => {
queryClient.invalidateQueries({ queryKey: ["test", testId] });
queryClient.invalidateQueries({ queryKey: ["techniques"] });
setShowValidateModal(false);
},
});
const rejectMutation = useMutation({
mutationFn: () => rejectTest(testId!),
onSuccess: () => {
queryClient.invalidateQueries({ queryKey: ["test", testId] });
},
});
const handleDownload = async (evidenceId: string) => {
try {
const evidence = await getEvidence(evidenceId);
if (evidence.download_url) {
window.open(evidence.download_url, "_blank");
}
} catch (err) {
console.error("Failed to get download URL:", err);
}
};
if (isLoading) {
return (
<div className="flex h-64 items-center justify-center">
<Loader2 className="h-8 w-8 animate-spin text-cyan-400" />
</div>
);
}
if (error || !test) {
return (
<div className="flex h-64 flex-col items-center justify-center gap-2">
<AlertCircle className="h-10 w-10 text-red-400" />
<p className="text-red-400">Failed to load test</p>
<button
onClick={() => navigate("/tests")}
className="mt-2 flex items-center gap-1 text-sm text-cyan-400 hover:underline"
>
<ArrowLeft className="h-4 w-4" />
Back to tests
</button>
</div>
);
}
const formatDate = (dateStr: string | null) => {
if (!dateStr) return "—";
return new Date(dateStr).toLocaleDateString("en-US", {
year: "numeric",
month: "short",
day: "numeric",
hour: "2-digit",
minute: "2-digit",
});
};
const canEdit = test.state === "draft" || test.state === "rejected";
return (
<div className="space-y-6">
{/* Back button */}
<button
onClick={() => navigate(-1)}
className="flex items-center gap-1 text-sm text-gray-400 hover:text-cyan-400 transition-colors"
>
<ArrowLeft className="h-4 w-4" />
Back
</button>
{/* Header */}
<div className="rounded-xl border border-gray-800 bg-gray-900 p-6">
<div className="flex items-start justify-between">
<div className="flex items-start gap-4">
<div className="rounded-lg bg-cyan-500/10 p-3">
<FlaskConical className="h-8 w-8 text-cyan-400" />
</div>
<div>
<div className="flex items-center gap-3">
<h1 className="text-2xl font-bold text-white">{test.name}</h1>
<span
className={`inline-flex rounded-full border px-2.5 py-0.5 text-xs font-medium ${
testStateBadgeColors[test.state]
}`}
>
{test.state.replace(/_/g, " ")}
</span>
{test.result && (
<span
className={`inline-flex rounded-full border px-2.5 py-0.5 text-xs font-medium ${
testResultBadgeColors[test.result]
}`}
>
{test.result.replace(/_/g, " ")}
</span>
)}
</div>
<p className="mt-1 text-sm text-gray-400">
Created {formatDate(test.created_at)}
</p>
</div>
</div>
{/* Actions */}
{canValidate && canEdit && (
<div className="flex items-center gap-2">
<button
onClick={() => setShowValidateModal(true)}
className="flex items-center gap-1.5 rounded-lg bg-green-600 px-4 py-2 text-sm font-medium text-white hover:bg-green-500 transition-colors"
>
<CheckCircle className="h-4 w-4" />
Validate
</button>
<button
onClick={() => rejectMutation.mutate()}
disabled={rejectMutation.isPending}
className="flex items-center gap-1.5 rounded-lg border border-red-500/30 bg-red-900/20 px-4 py-2 text-sm font-medium text-red-400 hover:bg-red-900/40 transition-colors"
>
{rejectMutation.isPending ? (
<Loader2 className="h-4 w-4 animate-spin" />
) : (
<XCircle className="h-4 w-4" />
)}
Reject
</button>
</div>
)}
</div>
</div>
{/* Content Grid */}
<div className="grid gap-6 lg:grid-cols-3">
{/* Main Content */}
<div className="lg:col-span-2 space-y-6">
{/* Description */}
<div className="rounded-xl border border-gray-800 bg-gray-900 p-6">
<h2 className="mb-3 text-lg font-semibold text-white">Description</h2>
<p className="text-sm text-gray-400 leading-relaxed">
{test.description || "No description provided."}
</p>
</div>
{/* Procedure */}
<div className="rounded-xl border border-gray-800 bg-gray-900 p-6">
<h2 className="mb-3 text-lg font-semibold text-white">Procedure</h2>
{test.procedure_text ? (
<pre className="whitespace-pre-wrap rounded-lg bg-gray-800 p-4 font-mono text-sm text-gray-300">
{test.procedure_text}
</pre>
) : (
<p className="text-sm text-gray-500">No procedure documented.</p>
)}
</div>
{/* Evidence Section */}
<div className="rounded-xl border border-gray-800 bg-gray-900 p-6">
<h2 className="mb-4 text-lg font-semibold text-white">Evidence</h2>
{/* Upload */}
<EvidenceUpload
onUpload={async (file) => {
await uploadMutation.mutateAsync(file);
}}
isUploading={uploadMutation.isPending}
/>
{uploadMutation.isError && (
<div className="mt-3 rounded-lg border border-red-500/30 bg-red-900/20 p-3">
<p className="text-sm text-red-400">
Upload failed: {(uploadMutation.error as Error)?.message}
</p>
</div>
)}
{/* Evidence List */}
<div className="mt-6">
<EvidenceList
evidences={test.evidences || []}
onDownload={handleDownload}
/>
</div>
</div>
</div>
{/* Sidebar */}
<div className="space-y-6">
{/* Metadata */}
<div className="rounded-xl border border-gray-800 bg-gray-900 p-6">
<h2 className="mb-4 text-lg font-semibold text-white">Details</h2>
<dl className="space-y-4">
<div>
<dt className="text-xs font-medium uppercase text-gray-500">Platform</dt>
<dd className="mt-1 text-sm text-gray-300 capitalize">
{test.platform || "—"}
</dd>
</div>
<div>
<dt className="text-xs font-medium uppercase text-gray-500">Tool Used</dt>
<dd className="mt-1 text-sm text-gray-300">{test.tool_used || "—"}</dd>
</div>
<div>
<dt className="text-xs font-medium uppercase text-gray-500">Execution Date</dt>
<dd className="mt-1 text-sm text-gray-300">
{formatDate(test.execution_date)}
</dd>
</div>
{test.validated_at && (
<div>
<dt className="text-xs font-medium uppercase text-gray-500">Validated</dt>
<dd className="mt-1 text-sm text-gray-300">
{formatDate(test.validated_at)}
</dd>
</div>
)}
</dl>
</div>
</div>
</div>
{/* Validate Modal */}
{showValidateModal && (
<div className="fixed inset-0 z-50 flex items-center justify-center bg-black/50 backdrop-blur-sm">
<div className="w-full max-w-md rounded-xl border border-gray-800 bg-gray-900 p-6">
<h3 className="text-lg font-semibold text-white">Validate Test</h3>
<p className="mt-1 text-sm text-gray-400">
Select the detection result for this test.
</p>
<div className="mt-4 space-y-2">
{RESULTS.map((r) => (
<label
key={r.value}
className={`flex cursor-pointer items-center gap-3 rounded-lg border p-3 transition-colors ${
selectedResult === r.value
? "border-cyan-500 bg-cyan-500/10"
: "border-gray-700 bg-gray-800 hover:border-gray-600"
}`}
>
<input
type="radio"
name="result"
value={r.value}
checked={selectedResult === r.value}
onChange={(e) => setSelectedResult(e.target.value as TestResult)}
className="sr-only"
/>
<div
className={`h-4 w-4 rounded-full border-2 ${
selectedResult === r.value
? "border-cyan-500 bg-cyan-500"
: "border-gray-600"
}`}
/>
<span className="text-sm text-gray-200">{r.label}</span>
</label>
))}
</div>
<div className="mt-6 flex justify-end gap-3">
<button
onClick={() => setShowValidateModal(false)}
className="rounded-lg border border-gray-700 px-4 py-2 text-sm text-gray-400 hover:bg-gray-800"
>
Cancel
</button>
<button
onClick={() => validateMutation.mutate()}
disabled={validateMutation.isPending}
className="flex items-center gap-1.5 rounded-lg bg-green-600 px-4 py-2 text-sm font-medium text-white hover:bg-green-500 disabled:opacity-50"
>
{validateMutation.isPending && (
<Loader2 className="h-4 w-4 animate-spin" />
)}
Confirm Validation
</button>
</div>
</div>
</div>
)}
</div>
);
}

View File

@@ -1,8 +1,229 @@
import { useQuery } from "@tanstack/react-query";
import { useNavigate } from "react-router-dom";
import { Loader2, AlertCircle, FlaskConical, Plus } from "lucide-react";
import { getTechniques, type TechniqueSummary } from "../api/techniques";
import type { TestState, TestResult } from "../types/models";
import { useAuth } from "../context/AuthContext";
const testStateBadgeColors: Record<TestState, string> = {
draft: "bg-gray-800/50 text-gray-400 border-gray-600/30",
in_review: "bg-blue-900/50 text-blue-400 border-blue-500/30",
validated: "bg-green-900/50 text-green-400 border-green-500/30",
rejected: "bg-red-900/50 text-red-400 border-red-500/30",
};
const testResultBadgeColors: Record<TestResult, string> = {
detected: "bg-green-900/50 text-green-400 border-green-500/30",
not_detected: "bg-red-900/50 text-red-400 border-red-500/30",
partially_detected: "bg-yellow-900/50 text-yellow-400 border-yellow-500/30",
};
interface TestSummary {
id: string;
technique_id: string;
technique_mitre_id: string;
technique_name: string;
name: string;
state: TestState;
result: TestResult | null;
platform: string | null;
created_at: string;
}
export default function TestsPage() {
const navigate = useNavigate();
const { user } = useAuth();
const canCreate =
user?.role === "admin" || user?.role === "red_tech" || user?.role === "blue_tech";
// For now, we'll fetch techniques to get their tests
// In a production app, you'd want a dedicated /tests endpoint
const {
data: techniques,
isLoading,
error,
} = useQuery({
queryKey: ["techniques"],
queryFn: () => getTechniques(),
});
// Note: Since we don't have a direct /tests list endpoint, we're showing
// a message to navigate through techniques. In a full implementation,
// you'd add a /tests endpoint to the backend.
const formatDate = (dateStr: string) => {
return new Date(dateStr).toLocaleDateString("en-US", {
year: "numeric",
month: "short",
day: "numeric",
});
};
if (isLoading) {
return (
<div className="flex h-64 items-center justify-center">
<Loader2 className="h-8 w-8 animate-spin text-cyan-400" />
</div>
);
}
if (error) {
return (
<div className="flex h-64 flex-col items-center justify-center gap-2">
<AlertCircle className="h-10 w-10 text-red-400" />
<p className="text-red-400">Failed to load data</p>
</div>
);
}
return (
<div className="space-y-4">
<h1 className="text-2xl font-bold text-white">Tests</h1>
<p className="text-gray-400">Security test management coming soon.</p>
<div className="space-y-6">
{/* Header */}
<div className="flex items-center justify-between">
<div>
<h1 className="text-2xl font-bold text-white">Tests</h1>
<p className="mt-1 text-sm text-gray-400">
Security tests for technique validation
</p>
</div>
{canCreate && (
<button
onClick={() => navigate("/tests/new")}
className="flex items-center gap-1.5 rounded-lg bg-cyan-600 px-4 py-2 text-sm font-medium text-white hover:bg-cyan-500 transition-colors"
>
<Plus className="h-4 w-4" />
New Test
</button>
)}
</div>
{/* Info Card */}
<div className="rounded-xl border border-cyan-500/30 bg-cyan-500/10 p-6">
<div className="flex items-start gap-4">
<div className="rounded-lg bg-cyan-500/20 p-3">
<FlaskConical className="h-6 w-6 text-cyan-400" />
</div>
<div>
<h2 className="text-lg font-semibold text-white">Browse Tests by Technique</h2>
<p className="mt-1 text-sm text-gray-400">
Tests are organized by MITRE ATT&CK technique. Navigate to a technique from the{" "}
<button
onClick={() => navigate("/techniques")}
className="text-cyan-400 hover:underline"
>
Techniques page
</button>{" "}
to view and manage its associated tests.
</p>
<div className="mt-4 flex flex-wrap gap-3">
<button
onClick={() => navigate("/techniques")}
className="rounded-lg border border-cyan-500/30 bg-cyan-500/10 px-4 py-2 text-sm font-medium text-cyan-400 hover:bg-cyan-500/20 transition-colors"
>
Browse Techniques Matrix
</button>
{canCreate && (
<button
onClick={() => navigate("/tests/new")}
className="rounded-lg border border-gray-700 bg-gray-800 px-4 py-2 text-sm font-medium text-gray-300 hover:border-gray-600 hover:text-white transition-colors"
>
Create Standalone Test
</button>
)}
</div>
</div>
</div>
</div>
{/* Quick Stats */}
<div className="grid gap-4 sm:grid-cols-2 lg:grid-cols-4">
<div className="rounded-xl border border-gray-800 bg-gray-900 p-5">
<p className="text-sm text-gray-400">Techniques with Tests</p>
<p className="mt-1 text-2xl font-bold text-cyan-400">
{techniques?.filter((t: TechniqueSummary) => t.status_global !== "not_evaluated").length || 0}
</p>
</div>
<div className="rounded-xl border border-gray-800 bg-gray-900 p-5">
<p className="text-sm text-gray-400">Validated</p>
<p className="mt-1 text-2xl font-bold text-green-400">
{techniques?.filter((t: TechniqueSummary) => t.status_global === "validated").length || 0}
</p>
</div>
<div className="rounded-xl border border-gray-800 bg-gray-900 p-5">
<p className="text-sm text-gray-400">In Progress</p>
<p className="mt-1 text-2xl font-bold text-blue-400">
{techniques?.filter((t: TechniqueSummary) => t.status_global === "in_progress").length || 0}
</p>
</div>
<div className="rounded-xl border border-gray-800 bg-gray-900 p-5">
<p className="text-sm text-gray-400">Pending Evaluation</p>
<p className="mt-1 text-2xl font-bold text-gray-400">
{techniques?.filter((t: TechniqueSummary) => t.status_global === "not_evaluated").length || 0}
</p>
</div>
</div>
{/* Techniques with Recent Activity */}
<div className="rounded-xl border border-gray-800 bg-gray-900 p-6">
<h2 className="mb-4 text-lg font-semibold text-white">Techniques Being Tested</h2>
<div className="overflow-x-auto">
<table className="w-full text-left text-sm">
<thead>
<tr className="border-b border-gray-800">
<th className="pb-3 pr-4 font-medium text-gray-400">Technique</th>
<th className="pb-3 px-4 font-medium text-gray-400">Name</th>
<th className="pb-3 px-4 font-medium text-gray-400">Status</th>
<th className="pb-3 pl-4 font-medium text-gray-400">Action</th>
</tr>
</thead>
<tbody>
{techniques
?.filter((t: TechniqueSummary) => t.status_global !== "not_evaluated")
.slice(0, 10)
.map((tech: TechniqueSummary) => (
<tr
key={tech.id}
className="border-b border-gray-800/50 hover:bg-gray-800/30 transition-colors"
>
<td className="py-3 pr-4">
<span className="font-mono text-cyan-400">{tech.mitre_id}</span>
</td>
<td className="py-3 px-4 text-gray-200">{tech.name}</td>
<td className="py-3 px-4">
<span
className={`inline-flex rounded-full border px-2 py-0.5 text-xs font-medium ${
tech.status_global === "validated"
? testStateBadgeColors.validated
: tech.status_global === "in_progress"
? testStateBadgeColors.in_review
: tech.status_global === "partial"
? "bg-yellow-900/50 text-yellow-400 border-yellow-500/30"
: testStateBadgeColors.draft
}`}
>
{tech.status_global.replace(/_/g, " ")}
</span>
</td>
<td className="py-3 pl-4">
<button
onClick={() => navigate(`/techniques/${tech.mitre_id}`)}
className="text-sm text-cyan-400 hover:underline"
>
View Tests
</button>
</td>
</tr>
))}
</tbody>
</table>
{techniques?.filter((t: TechniqueSummary) => t.status_global !== "not_evaluated").length === 0 && (
<div className="py-8 text-center text-gray-400">
No techniques have been tested yet. Create your first test to get started.
</div>
)}
</div>
</div>
</div>
);
}