Files
openclaw-taskboard/app/agents/[slug]/page.tsx

272 lines
11 KiB
TypeScript

import Link from "next/link";
import { notFound } from "next/navigation";
import { Badge } from "@/components/ui/badge";
import { Card, CardContent, CardDescription, CardHeader, CardTitle } from "@/components/ui/card";
import { findAgentBySlug } from "@/lib/agents";
import { formatDateTime } from "@/lib/utils";
export const dynamic = "force-dynamic";
function familyVariant(family: string) {
if (family === "zeroclaw") {
return "success";
}
if (family === "direct") {
return "warning";
}
return "default";
}
function dispatchVariant(state: string) {
return state === "failed" ? "warning" : state === "completed" ? "success" : "secondary";
}
export default async function AgentDetailPage({
params,
}: {
params: Promise<{ slug: string }>;
}) {
const { slug } = await params;
const agent = await findAgentBySlug(slug);
if (!agent) {
notFound();
}
return (
<div className="space-y-6">
<div className="flex flex-wrap items-start justify-between gap-4">
<div>
<Link className="text-sm text-cyan-300/80 hover:text-cyan-200" href="/agents">
Back to agents
</Link>
<h1 className="mt-2 flex items-center gap-3 text-3xl font-semibold text-white">
<span>{agent.emoji}</span>
<span>{agent.name}</span>
</h1>
<p className="mt-2 max-w-3xl text-sm text-slate-300">{agent.role}</p>
</div>
<div className="flex flex-wrap gap-2">
<Badge variant={familyVariant(agent.family)}>{agent.family}</Badge>
<Badge variant="secondary">{agent.status}</Badge>
<Badge variant="outline">{agent.host}</Badge>
</div>
</div>
<div className="grid gap-4 xl:grid-cols-[minmax(0,1.2fr)_380px]">
<Card>
<CardHeader>
<CardTitle>Overview</CardTitle>
<CardDescription>Runtime, routing, heartbeat, and capability details for this agent.</CardDescription>
</CardHeader>
<CardContent className="grid gap-5 md:grid-cols-2">
<dl className="grid gap-3 text-sm text-slate-300">
<div>
<dt className="text-xs uppercase tracking-[0.2em] text-slate-500">Host</dt>
<dd>{agent.host}</dd>
</div>
<div>
<dt className="text-xs uppercase tracking-[0.2em] text-slate-500">Model</dt>
<dd>{agent.model || "Host-local/runtime-defined"}</dd>
</div>
<div>
<dt className="text-xs uppercase tracking-[0.2em] text-slate-500">Dispatch</dt>
<dd>{agent.defaultDispatchMethod}</dd>
</div>
<div>
<dt className="text-xs uppercase tracking-[0.2em] text-slate-500">Heartbeat</dt>
<dd>{agent.heartbeatAt ? formatDateTime(agent.heartbeatAt) : "No heartbeat"}</dd>
</div>
<div className="md:col-span-2">
<dt className="text-xs uppercase tracking-[0.2em] text-slate-500">Runtime Path</dt>
<dd className="break-all font-mono text-xs text-cyan-100">{agent.runtimePath}</dd>
</div>
{agent.configPath ? (
<div className="md:col-span-2">
<dt className="text-xs uppercase tracking-[0.2em] text-slate-500">Config Path</dt>
<dd className="break-all font-mono text-xs text-cyan-100">{agent.configPath}</dd>
</div>
) : null}
<div>
<dt className="text-xs uppercase tracking-[0.2em] text-slate-500">Current Task</dt>
<dd>{agent.currentTask || "No heartbeat task"}</dd>
</div>
<div>
<dt className="text-xs uppercase tracking-[0.2em] text-slate-500">Failure Count</dt>
<dd>{agent.failureStreak}</dd>
</div>
</dl>
<div className="space-y-5">
<div>
<p className="mb-2 text-xs uppercase tracking-[0.2em] text-slate-500">Channels</p>
<div className="flex flex-wrap gap-2">
{agent.channels.map((channel) => (
<Badge key={`${agent.slug}-${channel.label}`} variant="outline">
{channel.label}: {channel.value}
</Badge>
))}
</div>
</div>
<div>
<p className="mb-2 text-xs uppercase tracking-[0.2em] text-slate-500">Tools</p>
<div className="flex flex-wrap gap-2">
{agent.tools.length ? (
agent.tools.map((tool) => (
<Badge key={tool} variant="secondary">
{tool}
</Badge>
))
) : (
<p className="text-sm text-slate-400">No parsed tools.</p>
)}
</div>
</div>
<div>
<p className="mb-2 text-xs uppercase tracking-[0.2em] text-slate-500">Capabilities</p>
<div className="flex flex-wrap gap-2">
{agent.capabilities.length ? (
agent.capabilities.map((capability) => (
<Badge key={capability} variant="outline">
{capability}
</Badge>
))
) : (
<p className="text-sm text-slate-400">No parsed capabilities.</p>
)}
</div>
</div>
</div>
</CardContent>
</Card>
<Card>
<CardHeader>
<CardTitle>Activity Snapshot</CardTitle>
<CardDescription>Recent event and assignment health for this agent.</CardDescription>
</CardHeader>
<CardContent className="space-y-4">
<div className="grid grid-cols-2 gap-3">
<div className="rounded-xl border border-white/10 bg-slate-950/40 p-4">
<p className="text-xs uppercase tracking-[0.2em] text-slate-500">Active</p>
<p className="mt-2 text-2xl font-semibold text-white">{agent.activeTasks.length}</p>
</div>
<div className="rounded-xl border border-white/10 bg-slate-950/40 p-4">
<p className="text-xs uppercase tracking-[0.2em] text-slate-500">Completed</p>
<p className="mt-2 text-2xl font-semibold text-white">{agent.completedTasks.length}</p>
</div>
</div>
<div>
<p className="mb-2 text-xs uppercase tracking-[0.2em] text-slate-500">Last Event</p>
{agent.lastEvent ? (
<div className="rounded-xl border border-white/10 bg-slate-950/40 p-4">
<div className="flex flex-wrap gap-2">
<Badge variant={agent.lastEvent.event_type === "dispatch_failed" ? "warning" : "secondary"}>
{agent.lastEvent.event_type}
</Badge>
<Badge variant={dispatchVariant(agent.lastEvent.state || "planned")}>
{agent.lastEvent.state || "n/a"}
</Badge>
</div>
<p className="mt-3 font-medium text-white">{agent.lastEvent.summary}</p>
<p className="mt-2 break-words text-sm text-slate-300">
{agent.lastEvent.detail || "No detail captured."}
</p>
<p className="mt-2 text-xs uppercase tracking-[0.2em] text-slate-500">
{formatDateTime(agent.lastEvent.created_at)}
</p>
</div>
) : (
<p className="text-sm text-slate-400">No audit events recorded yet.</p>
)}
</div>
{agent.notes.length ? (
<div>
<p className="mb-2 text-xs uppercase tracking-[0.2em] text-slate-500">Notes</p>
<div className="space-y-2">
{agent.notes.map((note) => (
<div className="rounded-xl border border-white/10 bg-slate-950/40 p-3 text-sm text-slate-300" key={note}>
{note}
</div>
))}
</div>
</div>
) : null}
</CardContent>
</Card>
</div>
<div className="grid gap-4 xl:grid-cols-2">
<Card>
<CardHeader>
<CardTitle>Assigned Tasks</CardTitle>
<CardDescription>Current active work assigned to this agent.</CardDescription>
</CardHeader>
<CardContent className="space-y-3">
{agent.activeTasks.length ? (
agent.activeTasks.map((task) => (
<div className="rounded-xl border border-white/10 bg-slate-950/40 p-4" key={task.id}>
<div className="flex items-start justify-between gap-3">
<div className="min-w-0">
<p className="font-medium text-white">{task.title}</p>
<p className="mt-1 break-words text-sm text-slate-300">{task.description}</p>
</div>
<Badge variant={dispatchVariant(task.dispatch_state)}>{task.dispatch_state}</Badge>
</div>
<div className="mt-3 flex flex-wrap gap-2">
<Badge variant="secondary">{task.status}</Badge>
<Badge variant="outline">{task.priority}</Badge>
{task.tags.slice(0, 4).map((tag) => (
<Badge key={tag} variant="outline">
{tag}
</Badge>
))}
</div>
</div>
))
) : (
<p className="text-sm text-slate-400">No active tasks assigned.</p>
)}
</CardContent>
</Card>
<Card>
<CardHeader>
<CardTitle>Recently Completed</CardTitle>
<CardDescription>Latest finished work attributed to this agent.</CardDescription>
</CardHeader>
<CardContent className="space-y-3">
{agent.completedTasks.length ? (
agent.completedTasks.map((task) => (
<div className="rounded-xl border border-white/10 bg-slate-950/40 p-4" key={task.id}>
<div className="flex items-start justify-between gap-3">
<div className="min-w-0">
<p className="font-medium text-white">{task.title}</p>
{task.result_summary ? (
<p className="mt-1 break-words text-sm text-slate-300">{task.result_summary}</p>
) : (
<p className="mt-1 break-words text-sm text-slate-300">{task.description}</p>
)}
</div>
<Badge variant="success">done</Badge>
</div>
<p className="mt-2 text-xs uppercase tracking-[0.2em] text-slate-500">
{formatDateTime(task.completed_at || task.updated_at)}
</p>
</div>
))
) : (
<p className="text-sm text-slate-400">No recently completed tasks.</p>
)}
</CardContent>
</Card>
</div>
</div>
);
}