Remove Running Sessions components from CC Agents
- Deleted RunningSessionsView.tsx file - Removed import and usage in CCAgents.tsx - Removed 'Running Sessions' tab from navigation - Removed activeTab state management - Simplified UI to show only agents list
This commit is contained in:
@@ -38,7 +38,6 @@ import { Toast, ToastContainer } from "@/components/ui/toast";
|
|||||||
import { CreateAgent } from "./CreateAgent";
|
import { CreateAgent } from "./CreateAgent";
|
||||||
import { AgentExecution } from "./AgentExecution";
|
import { AgentExecution } from "./AgentExecution";
|
||||||
import { AgentRunsList } from "./AgentRunsList";
|
import { AgentRunsList } from "./AgentRunsList";
|
||||||
import { RunningSessionsView } from "./RunningSessionsView";
|
|
||||||
import { GitHubAgentBrowser } from "./GitHubAgentBrowser";
|
import { GitHubAgentBrowser } from "./GitHubAgentBrowser";
|
||||||
import { ICON_MAP } from "./IconPicker";
|
import { ICON_MAP } from "./IconPicker";
|
||||||
|
|
||||||
@@ -73,7 +72,6 @@ export const CCAgents: React.FC<CCAgentsProps> = ({ onBack, className }) => {
|
|||||||
const [toast, setToast] = useState<{ message: string; type: "success" | "error" } | null>(null);
|
const [toast, setToast] = useState<{ message: string; type: "success" | "error" } | null>(null);
|
||||||
const [currentPage, setCurrentPage] = useState(1);
|
const [currentPage, setCurrentPage] = useState(1);
|
||||||
const [view, setView] = useState<"list" | "create" | "edit" | "execute">("list");
|
const [view, setView] = useState<"list" | "create" | "edit" | "execute">("list");
|
||||||
const [activeTab, setActiveTab] = useState<"agents" | "running">("agents");
|
|
||||||
const [selectedAgent, setSelectedAgent] = useState<Agent | null>(null);
|
const [selectedAgent, setSelectedAgent] = useState<Agent | null>(null);
|
||||||
// const [selectedRunId, setSelectedRunId] = useState<number | null>(null);
|
// const [selectedRunId, setSelectedRunId] = useState<number | null>(null);
|
||||||
const [showGitHubBrowser, setShowGitHubBrowser] = useState(false);
|
const [showGitHubBrowser, setShowGitHubBrowser] = useState(false);
|
||||||
@@ -363,203 +361,154 @@ export const CCAgents: React.FC<CCAgentsProps> = ({ onBack, className }) => {
|
|||||||
</motion.div>
|
</motion.div>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{/* Tab Navigation */}
|
{/* Main Content */}
|
||||||
<div className="border-b border-border">
|
|
||||||
<nav className="flex space-x-8">
|
|
||||||
<button
|
|
||||||
onClick={() => setActiveTab("agents")}
|
|
||||||
className={cn(
|
|
||||||
"py-2 px-1 border-b-2 font-medium text-sm transition-colors",
|
|
||||||
activeTab === "agents"
|
|
||||||
? "border-primary text-primary"
|
|
||||||
: "border-transparent text-muted-foreground hover:text-foreground hover:border-muted-foreground"
|
|
||||||
)}
|
|
||||||
>
|
|
||||||
<div className="flex items-center gap-2">
|
|
||||||
<Bot className="h-4 w-4" />
|
|
||||||
Agents
|
|
||||||
</div>
|
|
||||||
</button>
|
|
||||||
<button
|
|
||||||
onClick={() => setActiveTab("running")}
|
|
||||||
className={cn(
|
|
||||||
"py-2 px-1 border-b-2 font-medium text-sm transition-colors",
|
|
||||||
activeTab === "running"
|
|
||||||
? "border-primary text-primary"
|
|
||||||
: "border-transparent text-muted-foreground hover:text-foreground hover:border-muted-foreground"
|
|
||||||
)}
|
|
||||||
>
|
|
||||||
<div className="flex items-center gap-2">
|
|
||||||
<Play className="h-4 w-4" />
|
|
||||||
Running Sessions
|
|
||||||
</div>
|
|
||||||
</button>
|
|
||||||
</nav>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
{/* Tab Content */}
|
|
||||||
<div className="flex-1 overflow-y-auto">
|
<div className="flex-1 overflow-y-auto">
|
||||||
<AnimatePresence mode="wait">
|
<AnimatePresence mode="wait">
|
||||||
{activeTab === "agents" && (
|
<motion.div
|
||||||
<motion.div
|
key="agents"
|
||||||
key="agents"
|
initial={{ opacity: 0, y: 20 }}
|
||||||
initial={{ opacity: 0, y: 20 }}
|
animate={{ opacity: 1, y: 0 }}
|
||||||
animate={{ opacity: 1, y: 0 }}
|
exit={{ opacity: 0, y: -20 }}
|
||||||
exit={{ opacity: 0, y: -20 }}
|
transition={{ duration: 0.2 }}
|
||||||
transition={{ duration: 0.2 }}
|
className="pt-6 space-y-8"
|
||||||
className="pt-6 space-y-8"
|
>
|
||||||
>
|
{/* Agents Grid */}
|
||||||
{/* Agents Grid */}
|
<div>
|
||||||
<div>
|
{loading ? (
|
||||||
{loading ? (
|
<div className="flex items-center justify-center h-64">
|
||||||
<div className="flex items-center justify-center h-64">
|
<div className="animate-spin rounded-full h-8 w-8 border-b-2 border-primary"></div>
|
||||||
<div className="animate-spin rounded-full h-8 w-8 border-b-2 border-primary"></div>
|
</div>
|
||||||
|
) : agents.length === 0 ? (
|
||||||
|
<div className="flex flex-col items-center justify-center h-64 text-center">
|
||||||
|
<Bot className="h-16 w-16 text-muted-foreground mb-4" />
|
||||||
|
<h3 className="text-lg font-medium mb-2">No agents yet</h3>
|
||||||
|
<p className="text-sm text-muted-foreground mb-4">
|
||||||
|
Create your first CC Agent to get started
|
||||||
|
</p>
|
||||||
|
<Button onClick={() => setView("create")} size="default">
|
||||||
|
<Plus className="h-4 w-4 mr-2" />
|
||||||
|
Create CC Agent
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
<>
|
||||||
|
<div className="grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-3 gap-4">
|
||||||
|
<AnimatePresence mode="popLayout">
|
||||||
|
{paginatedAgents.map((agent, index) => (
|
||||||
|
<motion.div
|
||||||
|
key={agent.id}
|
||||||
|
initial={{ opacity: 0, scale: 0.9 }}
|
||||||
|
animate={{ opacity: 1, scale: 1 }}
|
||||||
|
exit={{ opacity: 0, scale: 0.9 }}
|
||||||
|
transition={{ duration: 0.2, delay: index * 0.05 }}
|
||||||
|
>
|
||||||
|
<Card className="h-full hover:shadow-lg transition-shadow">
|
||||||
|
<CardContent className="p-6 flex flex-col items-center text-center">
|
||||||
|
<div className="mb-4 p-4 rounded-full bg-primary/10 text-primary">
|
||||||
|
{renderIcon(agent.icon)}
|
||||||
|
</div>
|
||||||
|
<h3 className="text-lg font-semibold mb-2">
|
||||||
|
{agent.name}
|
||||||
|
</h3>
|
||||||
|
<p className="text-xs text-muted-foreground">
|
||||||
|
Created: {new Date(agent.created_at).toLocaleDateString()}
|
||||||
|
</p>
|
||||||
|
</CardContent>
|
||||||
|
<CardFooter className="p-4 pt-0 flex justify-center gap-1 flex-wrap">
|
||||||
|
<Button
|
||||||
|
size="sm"
|
||||||
|
variant="ghost"
|
||||||
|
onClick={() => handleExecuteAgent(agent)}
|
||||||
|
className="flex items-center gap-1"
|
||||||
|
title="Execute agent"
|
||||||
|
>
|
||||||
|
<Play className="h-3 w-3" />
|
||||||
|
Execute
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
size="sm"
|
||||||
|
variant="ghost"
|
||||||
|
onClick={() => handleEditAgent(agent)}
|
||||||
|
className="flex items-center gap-1"
|
||||||
|
title="Edit agent"
|
||||||
|
>
|
||||||
|
<Edit className="h-3 w-3" />
|
||||||
|
Edit
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
size="sm"
|
||||||
|
variant="ghost"
|
||||||
|
onClick={() => handleExportAgent(agent)}
|
||||||
|
className="flex items-center gap-1"
|
||||||
|
title="Export agent to .claudia.json"
|
||||||
|
>
|
||||||
|
<Upload className="h-3 w-3" />
|
||||||
|
Export
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
size="sm"
|
||||||
|
variant="ghost"
|
||||||
|
onClick={() => handleDeleteAgent(agent)}
|
||||||
|
className="flex items-center gap-1 text-destructive hover:text-destructive"
|
||||||
|
title="Delete agent"
|
||||||
|
>
|
||||||
|
<Trash2 className="h-3 w-3" />
|
||||||
|
Delete
|
||||||
|
</Button>
|
||||||
|
</CardFooter>
|
||||||
|
</Card>
|
||||||
|
</motion.div>
|
||||||
|
))}
|
||||||
|
</AnimatePresence>
|
||||||
</div>
|
</div>
|
||||||
) : agents.length === 0 ? (
|
|
||||||
<div className="flex flex-col items-center justify-center h-64 text-center">
|
{/* Pagination */}
|
||||||
<Bot className="h-16 w-16 text-muted-foreground mb-4" />
|
{totalPages > 1 && (
|
||||||
<h3 className="text-lg font-medium mb-2">No agents yet</h3>
|
<div className="mt-6 flex justify-center gap-2">
|
||||||
<p className="text-sm text-muted-foreground mb-4">
|
<Button
|
||||||
Create your first CC Agent to get started
|
size="sm"
|
||||||
</p>
|
variant="outline"
|
||||||
<Button onClick={() => setView("create")} size="default">
|
onClick={() => setCurrentPage(p => Math.max(1, p - 1))}
|
||||||
<Plus className="h-4 w-4 mr-2" />
|
disabled={currentPage === 1}
|
||||||
Create CC Agent
|
>
|
||||||
</Button>
|
Previous
|
||||||
|
</Button>
|
||||||
|
<span className="flex items-center px-3 text-sm">
|
||||||
|
Page {currentPage} of {totalPages}
|
||||||
|
</span>
|
||||||
|
<Button
|
||||||
|
size="sm"
|
||||||
|
variant="outline"
|
||||||
|
onClick={() => setCurrentPage(p => Math.min(totalPages, p + 1))}
|
||||||
|
disabled={currentPage === totalPages}
|
||||||
|
>
|
||||||
|
Next
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Execution History */}
|
||||||
|
{!loading && agents.length > 0 && (
|
||||||
|
<div className="overflow-hidden">
|
||||||
|
<div className="flex items-center gap-2 mb-4">
|
||||||
|
<History className="h-5 w-5 text-muted-foreground" />
|
||||||
|
<h2 className="text-lg font-semibold">Recent Executions</h2>
|
||||||
|
</div>
|
||||||
|
{runsLoading ? (
|
||||||
|
<div className="flex items-center justify-center h-32">
|
||||||
|
<div className="animate-spin rounded-full h-6 w-6 border-b-2 border-primary"></div>
|
||||||
</div>
|
</div>
|
||||||
) : (
|
) : (
|
||||||
<>
|
<AgentRunsList
|
||||||
<div className="grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-3 gap-4">
|
runs={runs}
|
||||||
<AnimatePresence mode="popLayout">
|
/>
|
||||||
{paginatedAgents.map((agent, index) => (
|
|
||||||
<motion.div
|
|
||||||
key={agent.id}
|
|
||||||
initial={{ opacity: 0, scale: 0.9 }}
|
|
||||||
animate={{ opacity: 1, scale: 1 }}
|
|
||||||
exit={{ opacity: 0, scale: 0.9 }}
|
|
||||||
transition={{ duration: 0.2, delay: index * 0.05 }}
|
|
||||||
>
|
|
||||||
<Card className="h-full hover:shadow-lg transition-shadow">
|
|
||||||
<CardContent className="p-6 flex flex-col items-center text-center">
|
|
||||||
<div className="mb-4 p-4 rounded-full bg-primary/10 text-primary">
|
|
||||||
{renderIcon(agent.icon)}
|
|
||||||
</div>
|
|
||||||
<h3 className="text-lg font-semibold mb-2">
|
|
||||||
{agent.name}
|
|
||||||
</h3>
|
|
||||||
<p className="text-xs text-muted-foreground">
|
|
||||||
Created: {new Date(agent.created_at).toLocaleDateString()}
|
|
||||||
</p>
|
|
||||||
</CardContent>
|
|
||||||
<CardFooter className="p-4 pt-0 flex justify-center gap-1 flex-wrap">
|
|
||||||
<Button
|
|
||||||
size="sm"
|
|
||||||
variant="ghost"
|
|
||||||
onClick={() => handleExecuteAgent(agent)}
|
|
||||||
className="flex items-center gap-1"
|
|
||||||
title="Execute agent"
|
|
||||||
>
|
|
||||||
<Play className="h-3 w-3" />
|
|
||||||
Execute
|
|
||||||
</Button>
|
|
||||||
<Button
|
|
||||||
size="sm"
|
|
||||||
variant="ghost"
|
|
||||||
onClick={() => handleEditAgent(agent)}
|
|
||||||
className="flex items-center gap-1"
|
|
||||||
title="Edit agent"
|
|
||||||
>
|
|
||||||
<Edit className="h-3 w-3" />
|
|
||||||
Edit
|
|
||||||
</Button>
|
|
||||||
<Button
|
|
||||||
size="sm"
|
|
||||||
variant="ghost"
|
|
||||||
onClick={() => handleExportAgent(agent)}
|
|
||||||
className="flex items-center gap-1"
|
|
||||||
title="Export agent to .claudia.json"
|
|
||||||
>
|
|
||||||
<Upload className="h-3 w-3" />
|
|
||||||
Export
|
|
||||||
</Button>
|
|
||||||
<Button
|
|
||||||
size="sm"
|
|
||||||
variant="ghost"
|
|
||||||
onClick={() => handleDeleteAgent(agent)}
|
|
||||||
className="flex items-center gap-1 text-destructive hover:text-destructive"
|
|
||||||
title="Delete agent"
|
|
||||||
>
|
|
||||||
<Trash2 className="h-3 w-3" />
|
|
||||||
Delete
|
|
||||||
</Button>
|
|
||||||
</CardFooter>
|
|
||||||
</Card>
|
|
||||||
</motion.div>
|
|
||||||
))}
|
|
||||||
</AnimatePresence>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
{/* Pagination */}
|
|
||||||
{totalPages > 1 && (
|
|
||||||
<div className="mt-6 flex justify-center gap-2">
|
|
||||||
<Button
|
|
||||||
size="sm"
|
|
||||||
variant="outline"
|
|
||||||
onClick={() => setCurrentPage(p => Math.max(1, p - 1))}
|
|
||||||
disabled={currentPage === 1}
|
|
||||||
>
|
|
||||||
Previous
|
|
||||||
</Button>
|
|
||||||
<span className="flex items-center px-3 text-sm">
|
|
||||||
Page {currentPage} of {totalPages}
|
|
||||||
</span>
|
|
||||||
<Button
|
|
||||||
size="sm"
|
|
||||||
variant="outline"
|
|
||||||
onClick={() => setCurrentPage(p => Math.min(totalPages, p + 1))}
|
|
||||||
disabled={currentPage === totalPages}
|
|
||||||
>
|
|
||||||
Next
|
|
||||||
</Button>
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
</>
|
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
|
)}
|
||||||
{/* Execution History */}
|
</motion.div>
|
||||||
{!loading && agents.length > 0 && (
|
|
||||||
<div className="overflow-hidden">
|
|
||||||
<div className="flex items-center gap-2 mb-4">
|
|
||||||
<History className="h-5 w-5 text-muted-foreground" />
|
|
||||||
<h2 className="text-lg font-semibold">Recent Executions</h2>
|
|
||||||
</div>
|
|
||||||
{runsLoading ? (
|
|
||||||
<div className="flex items-center justify-center h-32">
|
|
||||||
<div className="animate-spin rounded-full h-6 w-6 border-b-2 border-primary"></div>
|
|
||||||
</div>
|
|
||||||
) : (
|
|
||||||
<AgentRunsList
|
|
||||||
runs={runs}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
</motion.div>
|
|
||||||
)}
|
|
||||||
|
|
||||||
{activeTab === "running" && (
|
|
||||||
<motion.div
|
|
||||||
key="running"
|
|
||||||
initial={{ opacity: 0, y: 20 }}
|
|
||||||
animate={{ opacity: 1, y: 0 }}
|
|
||||||
exit={{ opacity: 0, y: -20 }}
|
|
||||||
transition={{ duration: 0.2 }}
|
|
||||||
className="pt-6"
|
|
||||||
>
|
|
||||||
<RunningSessionsView />
|
|
||||||
</motion.div>
|
|
||||||
)}
|
|
||||||
</AnimatePresence>
|
</AnimatePresence>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@@ -1,281 +0,0 @@
|
|||||||
import { useState, useEffect } from 'react';
|
|
||||||
import { motion, AnimatePresence } from 'framer-motion';
|
|
||||||
import { Play, Square, Clock, Cpu, RefreshCw, Eye, ArrowLeft, Bot } from 'lucide-react';
|
|
||||||
import { Button } from '@/components/ui/button';
|
|
||||||
import { Card, CardContent, CardHeader, CardTitle } from '@/components/ui/card';
|
|
||||||
import { Badge } from '@/components/ui/badge';
|
|
||||||
import { Toast, ToastContainer } from '@/components/ui/toast';
|
|
||||||
import { SessionOutputViewer } from './SessionOutputViewer';
|
|
||||||
import { api } from '@/lib/api';
|
|
||||||
import type { AgentRun } from '@/lib/api';
|
|
||||||
|
|
||||||
interface RunningSessionsViewProps {
|
|
||||||
className?: string;
|
|
||||||
showBackButton?: boolean;
|
|
||||||
onBack?: () => void;
|
|
||||||
}
|
|
||||||
|
|
||||||
export function RunningSessionsView({ className, showBackButton = false, onBack }: RunningSessionsViewProps) {
|
|
||||||
const [runningSessions, setRunningSessions] = useState<AgentRun[]>([]);
|
|
||||||
const [loading, setLoading] = useState(true);
|
|
||||||
const [refreshing, setRefreshing] = useState(false);
|
|
||||||
const [selectedSession, setSelectedSession] = useState<AgentRun | null>(null);
|
|
||||||
const [toast, setToast] = useState<{ message: string; type: "success" | "error" } | null>(null);
|
|
||||||
|
|
||||||
const loadRunningSessions = async () => {
|
|
||||||
try {
|
|
||||||
const sessions = await api.listRunningAgentSessions();
|
|
||||||
setRunningSessions(sessions);
|
|
||||||
} catch (error) {
|
|
||||||
console.error('Failed to load running sessions:', error);
|
|
||||||
setToast({ message: 'Failed to load running sessions', type: 'error' });
|
|
||||||
} finally {
|
|
||||||
setLoading(false);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const refreshSessions = async () => {
|
|
||||||
setRefreshing(true);
|
|
||||||
try {
|
|
||||||
// First cleanup finished processes
|
|
||||||
await api.cleanupFinishedProcesses();
|
|
||||||
// Then reload the list
|
|
||||||
await loadRunningSessions();
|
|
||||||
setToast({ message: 'Running sessions list has been updated', type: 'success' });
|
|
||||||
} catch (error) {
|
|
||||||
console.error('Failed to refresh sessions:', error);
|
|
||||||
setToast({ message: 'Failed to refresh sessions', type: 'error' });
|
|
||||||
} finally {
|
|
||||||
setRefreshing(false);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const killSession = async (runId: number, agentName: string) => {
|
|
||||||
try {
|
|
||||||
const success = await api.killAgentSession(runId);
|
|
||||||
if (success) {
|
|
||||||
setToast({ message: `${agentName} session has been stopped`, type: 'success' });
|
|
||||||
// Refresh the list after killing
|
|
||||||
await loadRunningSessions();
|
|
||||||
} else {
|
|
||||||
setToast({ message: 'Session may have already finished', type: 'error' });
|
|
||||||
}
|
|
||||||
} catch (error) {
|
|
||||||
console.error('Failed to kill session:', error);
|
|
||||||
setToast({ message: 'Failed to terminate session', type: 'error' });
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const formatDuration = (startTime: string) => {
|
|
||||||
const start = new Date(startTime);
|
|
||||||
const now = new Date();
|
|
||||||
const durationMs = now.getTime() - start.getTime();
|
|
||||||
const minutes = Math.floor(durationMs / (1000 * 60));
|
|
||||||
const seconds = Math.floor((durationMs % (1000 * 60)) / 1000);
|
|
||||||
return `${minutes}m ${seconds}s`;
|
|
||||||
};
|
|
||||||
|
|
||||||
const getStatusBadge = (status: string) => {
|
|
||||||
switch (status) {
|
|
||||||
case 'running':
|
|
||||||
return <Badge variant="default" className="bg-green-100 text-green-800 border-green-200">Running</Badge>;
|
|
||||||
case 'pending':
|
|
||||||
return <Badge variant="secondary">Pending</Badge>;
|
|
||||||
default:
|
|
||||||
return <Badge variant="outline">{status}</Badge>;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
loadRunningSessions();
|
|
||||||
|
|
||||||
// Set up auto-refresh every 5 seconds
|
|
||||||
const interval = setInterval(() => {
|
|
||||||
if (!refreshing) {
|
|
||||||
loadRunningSessions();
|
|
||||||
}
|
|
||||||
}, 5000);
|
|
||||||
|
|
||||||
return () => clearInterval(interval);
|
|
||||||
}, [refreshing]);
|
|
||||||
|
|
||||||
if (loading) {
|
|
||||||
return (
|
|
||||||
<div className={`flex items-center justify-center p-8 ${className}`}>
|
|
||||||
<div className="flex items-center space-x-2">
|
|
||||||
<RefreshCw className="h-4 w-4 animate-spin" />
|
|
||||||
<span>Loading running sessions...</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
<div className={`space-y-4 ${className}`}>
|
|
||||||
<div className="flex items-center justify-between">
|
|
||||||
<div className="flex items-center space-x-2">
|
|
||||||
{showBackButton && onBack && (
|
|
||||||
<Button
|
|
||||||
variant="ghost"
|
|
||||||
size="icon"
|
|
||||||
onClick={onBack}
|
|
||||||
className="h-8 w-8"
|
|
||||||
>
|
|
||||||
<ArrowLeft className="h-4 w-4" />
|
|
||||||
</Button>
|
|
||||||
)}
|
|
||||||
<Play className="h-5 w-5" />
|
|
||||||
<h2 className="text-lg font-semibold">Running Agent Sessions</h2>
|
|
||||||
<Badge variant="secondary">{runningSessions.length}</Badge>
|
|
||||||
</div>
|
|
||||||
<Button
|
|
||||||
variant="outline"
|
|
||||||
size="sm"
|
|
||||||
onClick={refreshSessions}
|
|
||||||
disabled={refreshing}
|
|
||||||
className="flex items-center space-x-2"
|
|
||||||
>
|
|
||||||
<RefreshCw className={`h-4 w-4 ${refreshing ? 'animate-spin' : ''}`} />
|
|
||||||
<span>Refresh</span>
|
|
||||||
</Button>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
{runningSessions.length === 0 ? (
|
|
||||||
<Card>
|
|
||||||
<CardContent className="flex items-center justify-center p-8">
|
|
||||||
<div className="text-center space-y-2">
|
|
||||||
<Clock className="h-8 w-8 mx-auto text-muted-foreground" />
|
|
||||||
<p className="text-muted-foreground">No agent sessions are currently running</p>
|
|
||||||
</div>
|
|
||||||
</CardContent>
|
|
||||||
</Card>
|
|
||||||
) : (
|
|
||||||
<div className="space-y-3">
|
|
||||||
{runningSessions.map((session) => (
|
|
||||||
<motion.div
|
|
||||||
key={session.id}
|
|
||||||
initial={{ opacity: 0, y: 20 }}
|
|
||||||
animate={{ opacity: 1, y: 0 }}
|
|
||||||
exit={{ opacity: 0, y: -20 }}
|
|
||||||
transition={{ duration: 0.2 }}
|
|
||||||
>
|
|
||||||
<Card className="hover:shadow-md transition-shadow">
|
|
||||||
<CardHeader className="pb-3">
|
|
||||||
<div className="flex items-center justify-between">
|
|
||||||
<div className="flex items-center space-x-3">
|
|
||||||
<div className="flex items-center justify-center w-8 h-8 bg-blue-100 rounded-full">
|
|
||||||
<Bot className="h-5 w-5 text-blue-600" />
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<CardTitle className="text-base">{session.agent_name}</CardTitle>
|
|
||||||
<div className="flex items-center space-x-2 mt-1">
|
|
||||||
{getStatusBadge(session.status)}
|
|
||||||
{session.pid && (
|
|
||||||
<Badge variant="outline" className="text-xs">
|
|
||||||
<Cpu className="h-3 w-3 mr-1" />
|
|
||||||
PID {session.pid}
|
|
||||||
</Badge>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div className="flex items-center space-x-2">
|
|
||||||
<Button
|
|
||||||
variant="outline"
|
|
||||||
size="sm"
|
|
||||||
onClick={() => setSelectedSession(session)}
|
|
||||||
className="flex items-center space-x-2"
|
|
||||||
>
|
|
||||||
<Eye className="h-4 w-4" />
|
|
||||||
<span>View Output</span>
|
|
||||||
</Button>
|
|
||||||
<Button
|
|
||||||
variant="destructive"
|
|
||||||
size="sm"
|
|
||||||
onClick={() => session.id && killSession(session.id, session.agent_name)}
|
|
||||||
className="flex items-center space-x-2"
|
|
||||||
>
|
|
||||||
<Square className="h-4 w-4" />
|
|
||||||
<span>Stop</span>
|
|
||||||
</Button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</CardHeader>
|
|
||||||
<CardContent className="pt-0">
|
|
||||||
<div className="space-y-2">
|
|
||||||
<div>
|
|
||||||
<p className="text-sm text-muted-foreground">Task</p>
|
|
||||||
<p className="text-sm font-medium truncate">{session.task}</p>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div className="grid grid-cols-2 gap-4 text-sm">
|
|
||||||
<div>
|
|
||||||
<p className="text-muted-foreground">Model</p>
|
|
||||||
<p className="font-medium">{session.model}</p>
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<p className="text-muted-foreground">Duration</p>
|
|
||||||
<p className="font-medium">
|
|
||||||
{session.process_started_at
|
|
||||||
? formatDuration(session.process_started_at)
|
|
||||||
: 'Unknown'
|
|
||||||
}
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div>
|
|
||||||
<p className="text-sm text-muted-foreground">Project Path</p>
|
|
||||||
<p className="text-xs font-mono bg-muted px-2 py-1 rounded truncate">
|
|
||||||
{session.project_path}
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
{session.session_id && (
|
|
||||||
<div>
|
|
||||||
<p className="text-sm text-muted-foreground">Session ID</p>
|
|
||||||
<p className="text-xs font-mono bg-muted px-2 py-1 rounded truncate">
|
|
||||||
{session.session_id}
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
</CardContent>
|
|
||||||
</Card>
|
|
||||||
</motion.div>
|
|
||||||
))}
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
|
|
||||||
{/* Session Output Viewer */}
|
|
||||||
<AnimatePresence>
|
|
||||||
{selectedSession && (
|
|
||||||
<motion.div
|
|
||||||
initial={{ opacity: 0 }}
|
|
||||||
animate={{ opacity: 1 }}
|
|
||||||
exit={{ opacity: 0 }}
|
|
||||||
className="fixed inset-0 bg-background/80 backdrop-blur-sm z-50 flex items-center justify-center p-4"
|
|
||||||
>
|
|
||||||
<div className="w-full max-w-4xl h-full max-h-[90vh]">
|
|
||||||
<SessionOutputViewer
|
|
||||||
session={selectedSession}
|
|
||||||
onClose={() => setSelectedSession(null)}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</motion.div>
|
|
||||||
)}
|
|
||||||
</AnimatePresence>
|
|
||||||
|
|
||||||
{/* Toast Notification */}
|
|
||||||
<ToastContainer>
|
|
||||||
{toast && (
|
|
||||||
<Toast
|
|
||||||
message={toast.message}
|
|
||||||
type={toast.type}
|
|
||||||
onDismiss={() => setToast(null)}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
</ToastContainer>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
Reference in New Issue
Block a user