Files
beaver_project/app-instance/frontend/components/chat-workbench/ProcessLane.tsx
steven_li cdfc222c9f feat: 添加swarms团队编排功能并优化agent委派系统
- 引入AgentTeamOrchestrator支持多agent协同任务执行
- 增加第三方swarms库依赖并配置git协议替换以改善包管理
- 扩展DelegationManager支持团队任务调度和进度跟踪
- 实现中文bigram分词算法提升中文任务检索准确性
- 调整A2AClient和DelegationManager超时时间从30秒增至600秒
- 优化AgentRunResult状态判断逻辑增加有意义摘要检测
- 修改Dockerfile配置npm仓库镜像地址和git协议映射
- 更新CLI命令行接口支持网关端口配置传递
- 调整提供者超时配置机制增强请求稳定性
- 移除过时的support_group字段简化agent描述符结构
- 增强错误处理和进度事件报告机制改进用户体验
2026-04-14 14:34:23 +08:00

164 lines
7.4 KiB
TypeScript

'use client';
import { AlertCircle, Bot, BrainCircuit, Loader2, ServerCog, Square } from 'lucide-react';
import type { ProcessEvent, ProcessRun } from '@/types';
import { Button } from '@/components/ui/button';
import { Badge } from '@/components/ui/badge';
import { Card, CardContent, CardHeader, CardTitle } from '@/components/ui/card';
import { ScrollArea } from '@/components/ui/scroll-area';
import { appActorTypeLabel, appEventKindLabel, appStatusLabel } from '@/lib/i18n/common';
import { pickAppText } from '@/lib/i18n/core';
import { useAppI18n } from '@/lib/i18n/provider';
import { cn } from '@/lib/utils';
function statusTone(status: string) {
if (status === 'done') return 'bg-emerald-500/10 text-emerald-300 border-emerald-500/20';
if (status === 'error') return 'bg-rose-500/10 text-rose-300 border-rose-500/20';
if (status === 'cancelled') return 'bg-zinc-500/10 text-zinc-300 border-zinc-500/20';
if (status === 'waiting') return 'bg-amber-500/10 text-amber-300 border-amber-500/20';
return 'bg-sky-500/10 text-sky-300 border-sky-500/20';
}
function actorIcon(run: ProcessRun) {
if (run.actor_type === 'mcp') return <ServerCog className="w-4 h-4" />;
if (run.actor_type === 'system') return <BrainCircuit className="w-4 h-4" />;
return <Bot className="w-4 h-4" />;
}
export function ProcessLane({
runs,
events,
selectedRunId,
onSelectRun,
onCancelRun,
}: {
runs: ProcessRun[];
events: ProcessEvent[];
selectedRunId: string | null;
onSelectRun: (runId: string) => void;
onCancelRun: (runId: string) => void;
}) {
const { locale } = useAppI18n();
const sortedRuns = [...runs].sort((a, b) => {
const at = new Date(a.started_at).getTime();
const bt = new Date(b.started_at).getTime();
return bt - at;
});
if (sortedRuns.length === 0) {
return null;
}
return (
<div className="h-full flex flex-col bg-card/60 border-l border-border">
<div className="px-4 py-3 border-b border-border flex items-center justify-between">
<div>
<h2 className="text-sm font-semibold tracking-wide uppercase text-muted-foreground">{pickAppText(locale, '执行过程', 'Execution')}</h2>
<p className="text-xs text-muted-foreground mt-1">{pickAppText(locale, '智能体、A2A、MCP 的实时过程', 'Live process stream for agents, A2A, and MCP')}</p>
</div>
<Badge variant="outline" className="text-xs">
{pickAppText(locale, `${sortedRuns.length} 个任务`, `${sortedRuns.length} tasks`)}
</Badge>
</div>
<ScrollArea className="flex-1 px-4 py-4">
<div className="space-y-3">
{sortedRuns.map((run) => {
const runEvents = events
.filter((event) => event.run_id === run.run_id)
.slice(-5)
.reverse();
const isSelected = run.run_id === selectedRunId;
const canCancel =
!run.parent_run_id &&
run.actor_type !== 'mcp' &&
(run.status === 'running' || run.status === 'waiting');
return (
<Card
key={run.run_id}
className={cn(
'cursor-pointer transition-colors border-border/80 hover:border-primary/40',
isSelected && 'border-primary bg-primary/5'
)}
onClick={() => onSelectRun(run.run_id)}
>
<CardHeader className="pb-3">
<div className="flex items-start justify-between gap-3">
<div className="min-w-0 flex-1">
<div className="flex items-center gap-2 flex-wrap">
<div className="w-8 h-8 rounded-full bg-muted flex items-center justify-center text-muted-foreground">
{actorIcon(run)}
</div>
<div className="min-w-0">
<CardTitle className="text-sm leading-none truncate">{run.actor_name}</CardTitle>
<p className="text-xs text-muted-foreground mt-1 truncate">{run.title}</p>
</div>
</div>
</div>
<div className="flex items-center gap-2 shrink-0">
<Badge variant="outline" className={cn('text-[10px] border', statusTone(run.status))}>
{appStatusLabel(run.status, locale)}
</Badge>
{canCancel && (
<Button
variant="outline"
size="sm"
className="h-7 px-2"
onClick={(event) => {
event.stopPropagation();
onCancelRun(run.run_id);
}}
>
<Square className="w-3.5 h-3.5 mr-1" />
{pickAppText(locale, '取消', 'Cancel')}
</Button>
)}
</div>
</div>
</CardHeader>
<CardContent className="pt-0 space-y-2">
<div className="flex items-center gap-2 text-[11px] text-muted-foreground flex-wrap">
<span>{appActorTypeLabel(run.actor_type, locale)}</span>
{run.source && <span>{run.source}</span>}
{run.parent_run_id && <span>{pickAppText(locale, '子任务', 'Subtask')}</span>}
</div>
{run.summary && (
<div className="rounded-md bg-muted/40 px-3 py-2 text-xs text-muted-foreground whitespace-pre-wrap line-clamp-3">
{run.summary}
</div>
)}
<div className="space-y-1.5">
{runEvents.length === 0 && run.status === 'running' && (
<div className="flex items-center gap-2 text-xs text-muted-foreground">
<Loader2 className="w-3.5 h-3.5 animate-spin" />
{pickAppText(locale, '等待首个事件...', 'Waiting for the first event...')}
</div>
)}
{runEvents.map((event) => (
<div key={event.event_id} className="text-xs rounded-md border border-border/50 bg-background/60 px-3 py-2">
<div className="flex items-center gap-2 text-[10px] uppercase tracking-wide text-muted-foreground mb-1">
<span>{appEventKindLabel(event.kind, locale)}</span>
{event.status && <span>{appStatusLabel(event.status, locale)}</span>}
</div>
<div className="text-foreground/90 whitespace-pre-wrap break-words">
{event.text || pickAppText(locale, '结构化更新', 'Structured update')}
</div>
</div>
))}
{run.status === 'error' && (
<div className="flex items-center gap-2 text-xs text-rose-300">
<AlertCircle className="w-3.5 h-3.5" />
{pickAppText(locale, '此任务执行失败。', 'This task failed.')}
</div>
)}
</div>
</CardContent>
</Card>
);
})}
</div>
</ScrollArea>
</div>
);
}