2025-06-21 16:49:16 +01:00
|
|
|
import { useState, useEffect } from "react";
|
2025-06-21 17:36:07 +01:00
|
|
|
import {
|
|
|
|
|
X,
|
|
|
|
|
History,
|
|
|
|
|
Send,
|
|
|
|
|
Clock,
|
|
|
|
|
FileText,
|
|
|
|
|
Code,
|
|
|
|
|
} from "lucide-react";
|
2025-06-21 16:49:16 +01:00
|
|
|
import { Button } from "@/components/ui/button";
|
|
|
|
|
import { Card, CardContent } from "@/components/ui/card";
|
|
|
|
|
import { Label } from "@/components/ui/label";
|
|
|
|
|
import { Textarea } from "@/components/ui/textarea";
|
|
|
|
|
import {
|
|
|
|
|
Select,
|
|
|
|
|
SelectContent,
|
|
|
|
|
SelectItem,
|
|
|
|
|
SelectTrigger,
|
|
|
|
|
SelectValue,
|
|
|
|
|
} from "@/components/ui/select";
|
|
|
|
|
|
|
|
|
|
import { makeRequest } from "@/lib/api";
|
2025-06-21 18:00:28 +01:00
|
|
|
import { getTaskPanelClasses, getBackdropClasses } from "@/lib/responsive-config";
|
2025-06-21 16:49:16 +01:00
|
|
|
import type {
|
|
|
|
|
TaskStatus,
|
|
|
|
|
TaskAttempt,
|
|
|
|
|
TaskAttemptActivity,
|
|
|
|
|
TaskAttemptStatus,
|
|
|
|
|
ExecutionProcess,
|
|
|
|
|
ExecutionProcessStatus,
|
|
|
|
|
ExecutionProcessType,
|
|
|
|
|
ApiResponse,
|
|
|
|
|
TaskWithAttemptStatus,
|
|
|
|
|
} from "shared/types";
|
|
|
|
|
|
|
|
|
|
interface TaskDetailsPanelProps {
|
|
|
|
|
task: TaskWithAttemptStatus | null;
|
|
|
|
|
projectId: string;
|
|
|
|
|
isOpen: boolean;
|
|
|
|
|
onClose: () => void;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const statusLabels: Record<TaskStatus, string> = {
|
|
|
|
|
todo: "To Do",
|
2025-06-21 17:36:07 +01:00
|
|
|
inprogress: "In Progress",
|
2025-06-21 16:49:16 +01:00
|
|
|
inreview: "In Review",
|
|
|
|
|
done: "Done",
|
|
|
|
|
cancelled: "Cancelled",
|
|
|
|
|
};
|
|
|
|
|
|
2025-06-21 17:36:07 +01:00
|
|
|
const getAttemptStatusDisplay = (
|
|
|
|
|
status: TaskAttemptStatus
|
|
|
|
|
): { label: string; className: string } => {
|
2025-06-21 16:49:16 +01:00
|
|
|
switch (status) {
|
|
|
|
|
case "init":
|
2025-06-21 17:36:07 +01:00
|
|
|
return {
|
|
|
|
|
label: "Init",
|
|
|
|
|
className: "bg-status-init text-status-init-foreground",
|
|
|
|
|
};
|
2025-06-21 16:49:16 +01:00
|
|
|
case "setuprunning":
|
2025-06-21 17:36:07 +01:00
|
|
|
return {
|
|
|
|
|
label: "Setup Running",
|
|
|
|
|
className: "bg-status-running text-status-running-foreground",
|
|
|
|
|
};
|
2025-06-21 16:49:16 +01:00
|
|
|
case "setupcomplete":
|
2025-06-21 17:36:07 +01:00
|
|
|
return {
|
|
|
|
|
label: "Setup Complete",
|
|
|
|
|
className: "bg-status-complete text-status-complete-foreground",
|
|
|
|
|
};
|
2025-06-21 16:49:16 +01:00
|
|
|
case "setupfailed":
|
2025-06-21 17:36:07 +01:00
|
|
|
return {
|
|
|
|
|
label: "Setup Failed",
|
|
|
|
|
className: "bg-status-failed text-status-failed-foreground",
|
|
|
|
|
};
|
2025-06-21 16:49:16 +01:00
|
|
|
case "executorrunning":
|
2025-06-21 17:36:07 +01:00
|
|
|
return {
|
|
|
|
|
label: "Executor Running",
|
|
|
|
|
className: "bg-status-running text-status-running-foreground",
|
|
|
|
|
};
|
2025-06-21 16:49:16 +01:00
|
|
|
case "executorcomplete":
|
2025-06-21 17:36:07 +01:00
|
|
|
return {
|
|
|
|
|
label: "Executor Complete",
|
|
|
|
|
className: "bg-status-complete text-status-complete-foreground",
|
|
|
|
|
};
|
2025-06-21 16:49:16 +01:00
|
|
|
case "executorfailed":
|
2025-06-21 17:36:07 +01:00
|
|
|
return {
|
|
|
|
|
label: "Executor Failed",
|
|
|
|
|
className: "bg-status-failed text-status-failed-foreground",
|
|
|
|
|
};
|
2025-06-21 16:49:16 +01:00
|
|
|
case "paused":
|
2025-06-21 17:36:07 +01:00
|
|
|
return {
|
|
|
|
|
label: "Paused",
|
|
|
|
|
className: "bg-status-paused text-status-paused-foreground",
|
|
|
|
|
};
|
2025-06-21 16:49:16 +01:00
|
|
|
default:
|
2025-06-21 17:36:07 +01:00
|
|
|
return {
|
|
|
|
|
label: "Unknown",
|
|
|
|
|
className: "bg-status-init text-status-init-foreground",
|
|
|
|
|
};
|
2025-06-21 16:49:16 +01:00
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2025-06-21 17:36:07 +01:00
|
|
|
const getProcessStatusDisplay = (
|
|
|
|
|
status: ExecutionProcessStatus
|
|
|
|
|
): { label: string; className: string } => {
|
2025-06-21 16:49:16 +01:00
|
|
|
switch (status) {
|
|
|
|
|
case "running":
|
2025-06-21 17:36:07 +01:00
|
|
|
return {
|
|
|
|
|
label: "Running",
|
|
|
|
|
className: "bg-status-running text-status-running-foreground",
|
|
|
|
|
};
|
2025-06-21 16:49:16 +01:00
|
|
|
case "completed":
|
2025-06-21 17:36:07 +01:00
|
|
|
return {
|
|
|
|
|
label: "Completed",
|
|
|
|
|
className: "bg-status-complete text-status-complete-foreground",
|
|
|
|
|
};
|
2025-06-21 16:49:16 +01:00
|
|
|
case "failed":
|
2025-06-21 17:36:07 +01:00
|
|
|
return {
|
|
|
|
|
label: "Failed",
|
|
|
|
|
className: "bg-status-failed text-status-failed-foreground",
|
|
|
|
|
};
|
2025-06-21 16:49:16 +01:00
|
|
|
case "killed":
|
2025-06-21 17:36:07 +01:00
|
|
|
return {
|
|
|
|
|
label: "Killed",
|
|
|
|
|
className: "bg-status-failed text-status-failed-foreground",
|
|
|
|
|
};
|
2025-06-21 16:49:16 +01:00
|
|
|
default:
|
2025-06-21 17:36:07 +01:00
|
|
|
return {
|
|
|
|
|
label: "Unknown",
|
|
|
|
|
className: "bg-status-init text-status-init-foreground",
|
|
|
|
|
};
|
2025-06-21 16:49:16 +01:00
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const getProcessTypeDisplay = (type: ExecutionProcessType): string => {
|
|
|
|
|
switch (type) {
|
|
|
|
|
case "setupscript":
|
|
|
|
|
return "Setup Script";
|
|
|
|
|
case "codingagent":
|
|
|
|
|
return "Coding Agent";
|
|
|
|
|
case "devserver":
|
|
|
|
|
return "Dev Server";
|
|
|
|
|
default:
|
|
|
|
|
return "Unknown";
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2025-06-21 17:36:07 +01:00
|
|
|
export function TaskDetailsPanel({
|
|
|
|
|
task,
|
|
|
|
|
projectId,
|
|
|
|
|
isOpen,
|
|
|
|
|
onClose,
|
|
|
|
|
}: TaskDetailsPanelProps) {
|
2025-06-21 16:49:16 +01:00
|
|
|
const [taskAttempts, setTaskAttempts] = useState<TaskAttempt[]>([]);
|
2025-06-21 17:36:07 +01:00
|
|
|
const [selectedAttempt, setSelectedAttempt] = useState<TaskAttempt | null>(
|
|
|
|
|
null
|
|
|
|
|
);
|
|
|
|
|
const [attemptActivities, setAttemptActivities] = useState<
|
|
|
|
|
TaskAttemptActivity[]
|
|
|
|
|
>([]);
|
|
|
|
|
const [executionProcesses, setExecutionProcesses] = useState<
|
|
|
|
|
ExecutionProcess[]
|
|
|
|
|
>([]);
|
2025-06-21 16:49:16 +01:00
|
|
|
const [loading, setLoading] = useState(false);
|
|
|
|
|
const [followUpMessage, setFollowUpMessage] = useState("");
|
|
|
|
|
const [showAttemptHistory, setShowAttemptHistory] = useState(false);
|
|
|
|
|
|
|
|
|
|
// Check if the selected attempt is active (not in a final state)
|
|
|
|
|
const isAttemptRunning =
|
|
|
|
|
selectedAttempt &&
|
|
|
|
|
attemptActivities.length > 0 &&
|
|
|
|
|
(attemptActivities[0].status === "init" ||
|
|
|
|
|
attemptActivities[0].status === "setuprunning" ||
|
|
|
|
|
attemptActivities[0].status === "setupcomplete" ||
|
|
|
|
|
attemptActivities[0].status === "executorrunning");
|
|
|
|
|
|
|
|
|
|
// Polling for updates when attempt is running
|
|
|
|
|
useEffect(() => {
|
|
|
|
|
if (!isAttemptRunning || !task) return;
|
|
|
|
|
|
|
|
|
|
const interval = setInterval(() => {
|
|
|
|
|
if (selectedAttempt) {
|
|
|
|
|
fetchAttemptActivities(selectedAttempt.id, true);
|
|
|
|
|
fetchExecutionProcesses(selectedAttempt.id, true);
|
|
|
|
|
}
|
|
|
|
|
}, 2000);
|
|
|
|
|
|
|
|
|
|
return () => clearInterval(interval);
|
|
|
|
|
}, [isAttemptRunning, task?.id, selectedAttempt?.id]);
|
|
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
|
if (task && isOpen) {
|
|
|
|
|
fetchTaskAttempts();
|
|
|
|
|
}
|
|
|
|
|
}, [task, isOpen]);
|
|
|
|
|
|
|
|
|
|
const fetchTaskAttempts = async () => {
|
|
|
|
|
if (!task) return;
|
|
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
setLoading(true);
|
|
|
|
|
const response = await makeRequest(
|
|
|
|
|
`/api/projects/${projectId}/tasks/${task.id}/attempts`
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
if (response.ok) {
|
|
|
|
|
const result: ApiResponse<TaskAttempt[]> = await response.json();
|
|
|
|
|
if (result.success && result.data) {
|
|
|
|
|
setTaskAttempts(result.data);
|
2025-06-21 17:36:07 +01:00
|
|
|
|
2025-06-21 16:49:16 +01:00
|
|
|
// Auto-select latest attempt
|
|
|
|
|
if (result.data.length > 0) {
|
|
|
|
|
const latestAttempt = result.data.reduce((latest, current) =>
|
|
|
|
|
new Date(current.created_at) > new Date(latest.created_at)
|
|
|
|
|
? current
|
|
|
|
|
: latest
|
|
|
|
|
);
|
|
|
|
|
setSelectedAttempt(latestAttempt);
|
|
|
|
|
fetchAttemptActivities(latestAttempt.id);
|
|
|
|
|
fetchExecutionProcesses(latestAttempt.id);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} catch (err) {
|
|
|
|
|
console.error("Failed to fetch task attempts:", err);
|
|
|
|
|
} finally {
|
|
|
|
|
setLoading(false);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2025-06-21 17:36:07 +01:00
|
|
|
const fetchAttemptActivities = async (
|
|
|
|
|
attemptId: string,
|
|
|
|
|
_isBackgroundUpdate = false
|
|
|
|
|
) => {
|
2025-06-21 16:49:16 +01:00
|
|
|
if (!task) return;
|
|
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
const response = await makeRequest(
|
|
|
|
|
`/api/projects/${projectId}/tasks/${task.id}/attempts/${attemptId}/activities`
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
if (response.ok) {
|
2025-06-21 17:36:07 +01:00
|
|
|
const result: ApiResponse<TaskAttemptActivity[]> =
|
|
|
|
|
await response.json();
|
2025-06-21 16:49:16 +01:00
|
|
|
if (result.success && result.data) {
|
|
|
|
|
setAttemptActivities(result.data);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} catch (err) {
|
|
|
|
|
console.error("Failed to fetch attempt activities:", err);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2025-06-21 17:36:07 +01:00
|
|
|
const fetchExecutionProcesses = async (
|
|
|
|
|
attemptId: string,
|
|
|
|
|
_isBackgroundUpdate = false
|
|
|
|
|
) => {
|
2025-06-21 16:49:16 +01:00
|
|
|
if (!task) return;
|
|
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
const response = await makeRequest(
|
|
|
|
|
`/api/projects/${projectId}/tasks/${task.id}/attempts/${attemptId}/execution-processes`
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
if (response.ok) {
|
|
|
|
|
const result: ApiResponse<ExecutionProcess[]> = await response.json();
|
|
|
|
|
if (result.success && result.data) {
|
|
|
|
|
setExecutionProcesses(result.data);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} catch (err) {
|
|
|
|
|
console.error("Failed to fetch execution processes:", err);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const handleAttemptChange = (attemptId: string) => {
|
2025-06-21 17:36:07 +01:00
|
|
|
const attempt = taskAttempts.find((a) => a.id === attemptId);
|
2025-06-21 16:49:16 +01:00
|
|
|
if (attempt) {
|
|
|
|
|
setSelectedAttempt(attempt);
|
|
|
|
|
fetchAttemptActivities(attempt.id);
|
|
|
|
|
fetchExecutionProcesses(attempt.id);
|
|
|
|
|
setShowAttemptHistory(false);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const handleSendFollowUp = () => {
|
|
|
|
|
// TODO: Implement follow-up message API
|
|
|
|
|
console.log("Follow-up message:", followUpMessage);
|
|
|
|
|
setFollowUpMessage("");
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const stopExecutionProcess = async (processId: string) => {
|
|
|
|
|
if (!task || !selectedAttempt) return;
|
|
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
const response = await makeRequest(
|
|
|
|
|
`/api/projects/${projectId}/tasks/${task.id}/attempts/${selectedAttempt.id}/execution-processes/${processId}/stop`,
|
|
|
|
|
{
|
|
|
|
|
method: "POST",
|
|
|
|
|
headers: {
|
|
|
|
|
"Content-Type": "application/json",
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
if (response.ok) {
|
|
|
|
|
// Refresh the execution processes
|
|
|
|
|
fetchExecutionProcesses(selectedAttempt.id);
|
|
|
|
|
fetchAttemptActivities(selectedAttempt.id);
|
|
|
|
|
}
|
|
|
|
|
} catch (err) {
|
|
|
|
|
console.error("Failed to stop execution process:", err);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const openInEditor = async () => {
|
|
|
|
|
if (!task || !selectedAttempt) return;
|
|
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
await makeRequest(
|
|
|
|
|
`/api/projects/${projectId}/tasks/${task.id}/attempts/${selectedAttempt.id}/open-editor`,
|
|
|
|
|
{
|
|
|
|
|
method: "POST",
|
|
|
|
|
headers: {
|
|
|
|
|
"Content-Type": "application/json",
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
);
|
|
|
|
|
} catch (err) {
|
|
|
|
|
console.error("Failed to open editor:", err);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
if (!task) return null;
|
|
|
|
|
|
|
|
|
|
return (
|
|
|
|
|
<>
|
|
|
|
|
{isOpen && (
|
|
|
|
|
<>
|
2025-06-21 18:00:28 +01:00
|
|
|
{/* Backdrop - only on smaller screens (overlay mode) */}
|
|
|
|
|
<div
|
|
|
|
|
className={getBackdropClasses()}
|
|
|
|
|
onClick={onClose}
|
|
|
|
|
/>
|
2025-06-21 17:36:07 +01:00
|
|
|
|
2025-06-21 16:49:16 +01:00
|
|
|
{/* Panel */}
|
2025-06-21 17:36:07 +01:00
|
|
|
<div
|
2025-06-21 18:00:28 +01:00
|
|
|
className={getTaskPanelClasses()}
|
2025-06-21 17:36:07 +01:00
|
|
|
>
|
2025-06-21 16:49:16 +01:00
|
|
|
<div className="flex flex-col h-full">
|
2025-06-21 17:36:07 +01:00
|
|
|
{/* Header */}
|
|
|
|
|
<div className="p-6 border-b space-y-4">
|
|
|
|
|
<div className="flex items-start justify-between">
|
|
|
|
|
<div className="flex-1 min-w-0">
|
|
|
|
|
<h2 className="text-xl font-bold mb-2 line-clamp-2">
|
|
|
|
|
{task.title}
|
|
|
|
|
</h2>
|
|
|
|
|
<div className="flex items-center gap-2 text-sm text-muted-foreground">
|
|
|
|
|
<span
|
|
|
|
|
className={`px-2 py-1 rounded-full text-xs font-medium ${
|
|
|
|
|
task.status === "todo"
|
|
|
|
|
? "bg-neutral text-neutral-foreground"
|
|
|
|
|
: task.status === "inprogress"
|
|
|
|
|
? "bg-info text-info-foreground"
|
|
|
|
|
: task.status === "inreview"
|
|
|
|
|
? "bg-warning text-warning-foreground"
|
|
|
|
|
: task.status === "done"
|
|
|
|
|
? "bg-success text-success-foreground"
|
|
|
|
|
: "bg-destructive text-destructive-foreground"
|
|
|
|
|
}`}
|
|
|
|
|
>
|
|
|
|
|
{statusLabels[task.status]}
|
|
|
|
|
</span>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
<div className="flex items-center gap-1">
|
|
|
|
|
<Button variant="ghost" size="icon" onClick={onClose}>
|
|
|
|
|
<X className="h-4 w-4" />
|
2025-06-21 16:49:16 +01:00
|
|
|
</Button>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
|
2025-06-21 17:36:07 +01:00
|
|
|
{/* Attempt Selection */}
|
|
|
|
|
<div className="flex items-center gap-2">
|
|
|
|
|
{selectedAttempt && !showAttemptHistory ? (
|
|
|
|
|
<div className="flex items-center gap-2 flex-1">
|
|
|
|
|
<span className="text-sm text-muted-foreground">
|
|
|
|
|
Current attempt:
|
|
|
|
|
</span>
|
|
|
|
|
<span className="text-sm font-medium">
|
|
|
|
|
{new Date(
|
|
|
|
|
selectedAttempt.created_at
|
|
|
|
|
).toLocaleDateString()}{" "}
|
|
|
|
|
{new Date(
|
|
|
|
|
selectedAttempt.created_at
|
|
|
|
|
).toLocaleTimeString()}
|
|
|
|
|
</span>
|
|
|
|
|
{taskAttempts.length > 1 && (
|
|
|
|
|
<Button
|
|
|
|
|
variant="outline"
|
|
|
|
|
size="sm"
|
|
|
|
|
onClick={() => setShowAttemptHistory(true)}
|
|
|
|
|
>
|
|
|
|
|
<History className="h-4 w-4 mr-1" />
|
|
|
|
|
History
|
|
|
|
|
</Button>
|
|
|
|
|
)}
|
|
|
|
|
</div>
|
|
|
|
|
) : (
|
|
|
|
|
<div className="flex items-center gap-2 flex-1">
|
|
|
|
|
<Select
|
|
|
|
|
value={selectedAttempt?.id || ""}
|
|
|
|
|
onValueChange={handleAttemptChange}
|
|
|
|
|
>
|
|
|
|
|
<SelectTrigger className="flex-1">
|
|
|
|
|
<SelectValue placeholder="Select an attempt..." />
|
|
|
|
|
</SelectTrigger>
|
|
|
|
|
<SelectContent>
|
|
|
|
|
{taskAttempts.map((attempt) => (
|
|
|
|
|
<SelectItem key={attempt.id} value={attempt.id}>
|
|
|
|
|
<div className="flex flex-col">
|
|
|
|
|
<span className="font-medium">
|
|
|
|
|
{new Date(
|
|
|
|
|
attempt.created_at
|
|
|
|
|
).toLocaleDateString()}{" "}
|
|
|
|
|
{new Date(
|
|
|
|
|
attempt.created_at
|
|
|
|
|
).toLocaleTimeString()}
|
2025-06-21 16:49:16 +01:00
|
|
|
</span>
|
2025-06-21 17:36:07 +01:00
|
|
|
<span className="text-xs text-muted-foreground text-left">
|
|
|
|
|
{attempt.executor || "executor"}
|
2025-06-21 16:49:16 +01:00
|
|
|
</span>
|
|
|
|
|
</div>
|
2025-06-21 17:36:07 +01:00
|
|
|
</SelectItem>
|
|
|
|
|
))}
|
|
|
|
|
</SelectContent>
|
|
|
|
|
</Select>
|
|
|
|
|
<Button
|
|
|
|
|
variant="outline"
|
|
|
|
|
size="sm"
|
|
|
|
|
onClick={() => setShowAttemptHistory(false)}
|
|
|
|
|
>
|
|
|
|
|
Close
|
|
|
|
|
</Button>
|
|
|
|
|
</div>
|
|
|
|
|
)}
|
|
|
|
|
|
|
|
|
|
{selectedAttempt && (
|
|
|
|
|
<div className="flex gap-1">
|
|
|
|
|
<Button
|
|
|
|
|
variant="outline"
|
|
|
|
|
size="sm"
|
|
|
|
|
onClick={openInEditor}
|
|
|
|
|
>
|
|
|
|
|
<Code className="h-4 w-4 mr-1" />
|
|
|
|
|
Editor
|
|
|
|
|
</Button>
|
|
|
|
|
<Button
|
|
|
|
|
variant="outline"
|
|
|
|
|
size="sm"
|
|
|
|
|
onClick={() =>
|
|
|
|
|
window.open(
|
|
|
|
|
`/projects/${projectId}/tasks/${task.id}/attempts/${selectedAttempt.id}/compare`,
|
|
|
|
|
"_blank"
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
>
|
|
|
|
|
<FileText className="h-4 w-4 mr-1" />
|
|
|
|
|
Changes
|
|
|
|
|
</Button>
|
|
|
|
|
</div>
|
|
|
|
|
)}
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
{/* Content */}
|
|
|
|
|
<div className="flex-1 overflow-y-auto p-6 space-y-6">
|
|
|
|
|
{loading ? (
|
|
|
|
|
<div className="text-center py-8">
|
|
|
|
|
<div className="animate-spin rounded-full h-8 w-8 border-b-2 border-foreground mx-auto mb-4"></div>
|
|
|
|
|
<p className="text-muted-foreground">Loading...</p>
|
|
|
|
|
</div>
|
|
|
|
|
) : (
|
|
|
|
|
<>
|
|
|
|
|
{/* Description */}
|
|
|
|
|
<div>
|
|
|
|
|
<Label className="text-sm font-medium mb-2 block">
|
|
|
|
|
Description
|
|
|
|
|
</Label>
|
|
|
|
|
<div className="p-3 bg-muted rounded-md min-h-[60px]">
|
|
|
|
|
{task.description ? (
|
|
|
|
|
<p className="text-sm whitespace-pre-wrap">
|
|
|
|
|
{task.description}
|
|
|
|
|
</p>
|
|
|
|
|
) : (
|
|
|
|
|
<p className="text-sm text-muted-foreground italic">
|
|
|
|
|
No description provided
|
|
|
|
|
</p>
|
|
|
|
|
)}
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
{/* Execution Processes */}
|
|
|
|
|
{selectedAttempt && executionProcesses.length > 0 && (
|
|
|
|
|
<div>
|
|
|
|
|
<Label className="text-sm font-medium mb-3 block">
|
|
|
|
|
Execution Processes
|
|
|
|
|
</Label>
|
|
|
|
|
<div className="space-y-3">
|
|
|
|
|
{executionProcesses.map((process) => (
|
|
|
|
|
<Card key={process.id} className="border">
|
|
|
|
|
<CardContent className="p-4 space-y-3">
|
|
|
|
|
<div className="flex items-center justify-between">
|
|
|
|
|
<div className="flex items-center gap-3">
|
|
|
|
|
<span
|
|
|
|
|
className={`px-2 py-1 rounded-full text-xs font-medium ${
|
|
|
|
|
getProcessStatusDisplay(process.status)
|
|
|
|
|
.className
|
|
|
|
|
}`}
|
2025-06-21 16:49:16 +01:00
|
|
|
>
|
2025-06-21 17:36:07 +01:00
|
|
|
{
|
|
|
|
|
getProcessStatusDisplay(process.status)
|
|
|
|
|
.label
|
|
|
|
|
}
|
|
|
|
|
</span>
|
|
|
|
|
<span className="font-medium text-sm">
|
|
|
|
|
{getProcessTypeDisplay(
|
|
|
|
|
process.process_type
|
|
|
|
|
)}
|
|
|
|
|
</span>
|
|
|
|
|
</div>
|
|
|
|
|
<div className="flex items-center gap-2">
|
|
|
|
|
<span className="text-xs text-muted-foreground">
|
|
|
|
|
{new Date(
|
|
|
|
|
process.started_at
|
|
|
|
|
).toLocaleTimeString()}
|
|
|
|
|
</span>
|
|
|
|
|
{process.status === "running" && (
|
|
|
|
|
<Button
|
|
|
|
|
onClick={() =>
|
|
|
|
|
stopExecutionProcess(process.id)
|
|
|
|
|
}
|
|
|
|
|
size="sm"
|
|
|
|
|
variant="destructive"
|
|
|
|
|
>
|
|
|
|
|
Stop
|
|
|
|
|
</Button>
|
|
|
|
|
)}
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
{(process.stdout || process.stderr) && (
|
|
|
|
|
<div className="space-y-2">
|
|
|
|
|
{process.stdout && (
|
|
|
|
|
<div>
|
|
|
|
|
<Label className="text-xs text-muted-foreground mb-1 block">
|
|
|
|
|
STDOUT
|
|
|
|
|
</Label>
|
|
|
|
|
<div
|
|
|
|
|
className="bg-black text-green-400 border border-green-400 rounded-md p-2 font-mono text-xs max-h-32 overflow-y-auto whitespace-pre-wrap"
|
|
|
|
|
style={{
|
|
|
|
|
fontFamily:
|
|
|
|
|
'ui-monospace, SFMono-Regular, "SF Mono", Monaco, Consolas, "Liberation Mono", "Courier New", monospace',
|
|
|
|
|
}}
|
|
|
|
|
>
|
|
|
|
|
{process.stdout}
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
)}
|
|
|
|
|
{process.stderr && (
|
|
|
|
|
<div>
|
|
|
|
|
<Label className="text-xs text-muted-foreground mb-1 block">
|
|
|
|
|
STDERR
|
|
|
|
|
</Label>
|
|
|
|
|
<div
|
|
|
|
|
className="bg-black text-red-400 border border-red-400 rounded-md p-2 font-mono text-xs max-h-32 overflow-y-auto whitespace-pre-wrap"
|
|
|
|
|
style={{
|
|
|
|
|
fontFamily:
|
|
|
|
|
'ui-monospace, SFMono-Regular, "SF Mono", Monaco, Consolas, "Liberation Mono", "Courier New", monospace',
|
|
|
|
|
}}
|
|
|
|
|
>
|
|
|
|
|
{process.stderr}
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
)}
|
2025-06-21 16:49:16 +01:00
|
|
|
</div>
|
|
|
|
|
)}
|
2025-06-21 17:36:07 +01:00
|
|
|
</CardContent>
|
|
|
|
|
</Card>
|
|
|
|
|
))}
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
)}
|
|
|
|
|
|
|
|
|
|
{/* Activity History */}
|
|
|
|
|
{selectedAttempt && (
|
|
|
|
|
<div>
|
|
|
|
|
<Label className="text-sm font-medium mb-3 block">
|
|
|
|
|
Activity History
|
|
|
|
|
</Label>
|
|
|
|
|
{attemptActivities.length === 0 ? (
|
|
|
|
|
<div className="text-center py-4 text-muted-foreground">
|
|
|
|
|
No activities found
|
|
|
|
|
</div>
|
|
|
|
|
) : (
|
|
|
|
|
<div className="space-y-3">
|
|
|
|
|
{attemptActivities.map((activity) => (
|
|
|
|
|
<Card key={activity.id} className="border">
|
|
|
|
|
<CardContent className="p-4">
|
|
|
|
|
<div className="flex items-center justify-between mb-2">
|
|
|
|
|
<span
|
|
|
|
|
className={`px-2 py-1 rounded-full text-xs font-medium ${
|
|
|
|
|
getAttemptStatusDisplay(activity.status)
|
|
|
|
|
.className
|
|
|
|
|
}`}
|
2025-06-21 16:49:16 +01:00
|
|
|
>
|
2025-06-21 17:36:07 +01:00
|
|
|
{
|
|
|
|
|
getAttemptStatusDisplay(activity.status)
|
|
|
|
|
.label
|
|
|
|
|
}
|
|
|
|
|
</span>
|
|
|
|
|
<div className="flex items-center gap-1 text-xs text-muted-foreground">
|
|
|
|
|
<Clock className="h-3 w-3" />
|
|
|
|
|
{new Date(
|
|
|
|
|
activity.created_at
|
|
|
|
|
).toLocaleString()}
|
2025-06-21 16:49:16 +01:00
|
|
|
</div>
|
|
|
|
|
</div>
|
2025-06-21 17:36:07 +01:00
|
|
|
{activity.note && (
|
|
|
|
|
<p className="text-sm text-muted-foreground">
|
|
|
|
|
{activity.note}
|
|
|
|
|
</p>
|
|
|
|
|
)}
|
|
|
|
|
</CardContent>
|
|
|
|
|
</Card>
|
|
|
|
|
))}
|
|
|
|
|
</div>
|
|
|
|
|
)}
|
2025-06-21 16:49:16 +01:00
|
|
|
</div>
|
|
|
|
|
)}
|
2025-06-21 17:36:07 +01:00
|
|
|
</>
|
2025-06-21 16:49:16 +01:00
|
|
|
)}
|
2025-06-21 17:36:07 +01:00
|
|
|
</div>
|
2025-06-21 16:49:16 +01:00
|
|
|
|
2025-06-21 17:36:07 +01:00
|
|
|
{/* Footer */}
|
|
|
|
|
<div className="border-t p-4">
|
|
|
|
|
<div className="space-y-2">
|
|
|
|
|
<Label className="text-sm font-medium">
|
|
|
|
|
Follow-up question
|
|
|
|
|
</Label>
|
|
|
|
|
<div className="flex gap-2">
|
|
|
|
|
<Textarea
|
|
|
|
|
placeholder="Ask a follow-up question about this task..."
|
|
|
|
|
value={followUpMessage}
|
|
|
|
|
onChange={(e) => setFollowUpMessage(e.target.value)}
|
|
|
|
|
className="flex-1 min-h-[60px] resize-none"
|
|
|
|
|
/>
|
|
|
|
|
<Button
|
|
|
|
|
onClick={handleSendFollowUp}
|
|
|
|
|
disabled={!followUpMessage.trim()}
|
|
|
|
|
className="self-end"
|
|
|
|
|
>
|
|
|
|
|
<Send className="h-4 w-4" />
|
|
|
|
|
</Button>
|
|
|
|
|
</div>
|
|
|
|
|
<p className="text-xs text-muted-foreground">
|
|
|
|
|
Follow-up functionality coming soon
|
|
|
|
|
</p>
|
|
|
|
|
</div>
|
2025-06-21 16:49:16 +01:00
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</>
|
|
|
|
|
)}
|
|
|
|
|
</>
|
|
|
|
|
);
|
|
|
|
|
}
|