agent-relay-server 0.35.1 → 0.35.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/docs/openapi.json +1 -1
- package/package.json +1 -1
- package/public/assets/{activity-B0_uE6Yh.js → activity-WjOShx3N.js} +2 -2
- package/public/assets/{activity-B0_uE6Yh.js.map → activity-WjOShx3N.js.map} +1 -1
- package/public/assets/{agent-profiles-Rwxrcf9F.js → agent-profiles-Bs8MAW6j.js} +2 -2
- package/public/assets/{agent-profiles-Rwxrcf9F.js.map → agent-profiles-Bs8MAW6j.js.map} +1 -1
- package/public/assets/{agents-Dp1EXJc8.js → agents-Cihuuoxa.js} +2 -2
- package/public/assets/{agents-Dp1EXJc8.js.map → agents-Cihuuoxa.js.map} +1 -1
- package/public/assets/{analytics-D5OT5ajj.js → analytics-B5_HS2Lt.js} +2 -2
- package/public/assets/{analytics-D5OT5ajj.js.map → analytics-B5_HS2Lt.js.map} +1 -1
- package/public/assets/{automation-Dm6rXNxK.js → automation-CoOe2nEA.js} +2 -2
- package/public/assets/{automation-Dm6rXNxK.js.map → automation-CoOe2nEA.js.map} +1 -1
- package/public/assets/{branch-state-badge-FX5Yww2s.js → branch-state-badge-EliCEAQI.js} +2 -2
- package/public/assets/{branch-state-badge-FX5Yww2s.js.map → branch-state-badge-EliCEAQI.js.map} +1 -1
- package/public/assets/{channels--rdAiX17.js → channels-BzSpsE5h.js} +2 -2
- package/public/assets/{channels--rdAiX17.js.map → channels-BzSpsE5h.js.map} +1 -1
- package/public/assets/{chat-JZAEDGfX.js → chat-0ZbxHlDu.js} +2 -2
- package/public/assets/{chat-JZAEDGfX.js.map → chat-0ZbxHlDu.js.map} +1 -1
- package/public/assets/{connectors-Bx4gzvNf.js → connectors-DoUgct8f.js} +2 -2
- package/public/assets/{connectors-Bx4gzvNf.js.map → connectors-DoUgct8f.js.map} +1 -1
- package/public/assets/{formatted-body-impl-CVq4qHix.js → formatted-body-impl-DhCblnWM.js} +2 -2
- package/public/assets/{formatted-body-impl-CVq4qHix.js.map → formatted-body-impl-DhCblnWM.js.map} +1 -1
- package/public/assets/{index-BHRtR4q7.js → index-CvSlyTSI.js} +6 -6
- package/public/assets/{index-BHRtR4q7.js.map → index-CvSlyTSI.js.map} +1 -1
- package/public/assets/{integrations-k1HIONjo.js → integrations-CUv4i_4i.js} +2 -2
- package/public/assets/{integrations-k1HIONjo.js.map → integrations-CUv4i_4i.js.map} +1 -1
- package/public/assets/{maintenance-CsoOFBXx.js → maintenance-CUxxVXsc.js} +2 -2
- package/public/assets/{maintenance-CsoOFBXx.js.map → maintenance-CUxxVXsc.js.map} +1 -1
- package/public/assets/{managed-agents-Q3HuVjGg.js → managed-agents-VZEeMMSP.js} +2 -2
- package/public/assets/{managed-agents-Q3HuVjGg.js.map → managed-agents-VZEeMMSP.js.map} +1 -1
- package/public/assets/{markdown-preview-impl-CnsMjrnu.js → markdown-preview-impl-Bbwmbxvn.js} +2 -2
- package/public/assets/{markdown-preview-impl-CnsMjrnu.js.map → markdown-preview-impl-Bbwmbxvn.js.map} +1 -1
- package/public/assets/{memory-D3-K5eJS.js → memory-BQAC3YmR.js} +2 -2
- package/public/assets/{memory-D3-K5eJS.js.map → memory-BQAC3YmR.js.map} +1 -1
- package/public/assets/{messages-B4lCP5rS.js → messages-_sdHV42k.js} +2 -2
- package/public/assets/{messages-B4lCP5rS.js.map → messages-_sdHV42k.js.map} +1 -1
- package/public/assets/{orchestrators-CRoZtLeQ.js → orchestrators-BkHHgd83.js} +2 -2
- package/public/assets/{orchestrators-CRoZtLeQ.js.map → orchestrators-BkHHgd83.js.map} +1 -1
- package/public/assets/{overview-CxCU2fOF.js → overview-CYAHOUyA.js} +2 -2
- package/public/assets/{overview-CxCU2fOF.js.map → overview-CYAHOUyA.js.map} +1 -1
- package/public/assets/{pairs-unqjPlmq.js → pairs-BvMH1CS7.js} +2 -2
- package/public/assets/{pairs-unqjPlmq.js.map → pairs-BvMH1CS7.js.map} +1 -1
- package/public/assets/{security-B7HhSYNy.js → security-Dsdr3lSX.js} +2 -2
- package/public/assets/{security-B7HhSYNy.js.map → security-Dsdr3lSX.js.map} +1 -1
- package/public/assets/{settings-B9NDhsAb.js → settings-DlovIWdE.js} +2 -2
- package/public/assets/{settings-B9NDhsAb.js.map → settings-DlovIWdE.js.map} +1 -1
- package/public/assets/store-CICRhg1m.js +9 -0
- package/public/assets/store-CICRhg1m.js.map +1 -0
- package/public/assets/{tasks-CIQolvNm.js → tasks-Cgan8Oqb.js} +2 -2
- package/public/assets/{tasks-CIQolvNm.js.map → tasks-Cgan8Oqb.js.map} +1 -1
- package/public/assets/{terminal-viewer-impl-DCifVqFR.js → terminal-viewer-impl-DVW-LRbz.js} +2 -2
- package/public/assets/{terminal-viewer-impl-DCifVqFR.js.map → terminal-viewer-impl-DVW-LRbz.js.map} +1 -1
- package/public/assets/{work-queue-Dr3c1V6O.js → work-queue-B4CifZKH.js} +2 -2
- package/public/assets/{work-queue-Dr3c1V6O.js.map → work-queue-B4CifZKH.js.map} +1 -1
- package/public/assets/{workspaces-B1Jxop7h.js → workspaces-D7lsWShY.js} +2 -2
- package/public/assets/{workspaces-B1Jxop7h.js.map → workspaces-D7lsWShY.js.map} +1 -1
- package/public/index.html +2 -2
- package/src/db/workspaces.ts +3 -2
- package/public/assets/store-DiSzYHj9.js +0 -9
- package/public/assets/store-DiSzYHj9.js.map +0 -1
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"memory-D3-K5eJS.js","names":[],"sources":["../../dashboard/src/components/views/memory.tsx"],"sourcesContent":["import { useEffect, useMemo, useState } from 'react'\nimport { useRelayStore, useNow } from '@/store'\nimport { Badge } from '@/components/ui/badge'\nimport { Button } from '@/components/ui/button'\nimport { Card, CardContent, CardHeader, CardTitle } from '@/components/ui/card'\nimport { Input } from '@/components/ui/input'\nimport { Textarea } from '@/components/ui/textarea'\nimport { ScrollArea } from '@/components/ui/scroll-area'\nimport { Separator } from '@/components/ui/separator'\nimport { BrainCircuit, Database, Edit3, Plus, RefreshCw, Search, Send, Trash2 } from 'lucide-react'\nimport { displayName, fmtTime, timeAgo } from '@/lib/display'\nimport type { Agent, CreateMemoryInput, Memory, MemoryFilterState, MemoryQuery, MemoryType, UpdateMemoryInput } from '@/types'\n\nconst MEMORY_TYPES: MemoryType[] = ['organization', 'role', 'project', 'task', 'interaction', 'agent']\nconst VISIBILITIES = ['private', 'project', 'org', 'public'] as const\nconst SENSITIVITIES = ['public', 'normal', 'sensitive', 'secret'] as const\nconst CONFIDENCES = ['reported', 'inferred', 'verified'] as const\n\ntype MemoryForm = {\n type: MemoryType\n scope: string\n title: string\n content: string\n tags: string\n visibility: (typeof VISIBILITIES)[number]\n sensitivity: (typeof SENSITIVITIES)[number]\n confidence: (typeof CONFIDENCES)[number]\n relevanceScore: string\n}\n\nfunction blankForm(): MemoryForm {\n return {\n type: 'project',\n scope: 'default',\n title: '',\n content: '',\n tags: '',\n visibility: 'project',\n sensitivity: 'normal',\n confidence: 'reported',\n relevanceScore: '0.7',\n }\n}\n\nfunction formFromMemory(memory: Memory): MemoryForm {\n return {\n type: memory.type,\n scope: memory.scope,\n title: memory.title,\n content: memory.content,\n tags: memory.tags.join(', '),\n visibility: memory.visibility,\n sensitivity: memory.sensitivity,\n confidence: memory.confidence,\n relevanceScore: String(memory.relevanceScore),\n }\n}\n\nfunction csv(value: string): string[] {\n return value.split(',').map((item) => item.trim()).filter(Boolean)\n}\n\nfunction queryFromFilters(filters: MemoryFilterState): MemoryQuery {\n const query: MemoryQuery = { limit: 100 }\n if (filters.type) query.type = filters.type as MemoryQuery['type']\n if (filters.scope.trim()) query.scope = filters.scope.trim()\n if (filters.visibility) query.visibility = filters.visibility as MemoryQuery['visibility']\n const tags = csv(filters.tags)\n if (tags.length) query.tags = tags\n if (filters.minRelevance.trim()) {\n const minRelevance = Number(filters.minRelevance)\n if (Number.isFinite(minRelevance)) query.minRelevance = Math.max(0, Math.min(1, minRelevance))\n }\n return query\n}\n\nfunction sensitivityClass(value: string) {\n if (value === 'secret') return 'bg-red-500/10 text-red-400 border-red-500/30'\n if (value === 'sensitive') return 'bg-yellow-500/10 text-yellow-400 border-yellow-500/30'\n if (value === 'public') return 'bg-blue-500/10 text-blue-400 border-blue-500/30'\n return 'bg-zinc-500/10 text-zinc-400 border-zinc-500/30'\n}\n\nfunction scoreClass(score: number) {\n if (score >= 0.8) return 'text-emerald-400'\n if (score >= 0.5) return 'text-yellow-400'\n return 'text-zinc-400'\n}\n\nfunction compactConfig(config: Record<string, unknown> | undefined): string {\n if (!config) return 'default'\n return Object.entries(config)\n .map(([key, value]) => `${key}=${Array.isArray(value) ? value.join(' ') : String(value)}`)\n .join(' ')\n}\n\nfunction agentOptions(agents: Agent[]) {\n return agents.filter((agent) => !['system', 'user', 'channel'].includes(agent.kind || ''))\n}\n\nexport function MemoryView() {\n const now = useNow()\n const agents = useRelayStore((s) => s.agents)\n const memories = useRelayStore((s) => s.memories)\n const memoryTotal = useRelayStore((s) => s.memoryTotal)\n const memoryStats = useRelayStore((s) => s.memoryStats)\n const memoryBrokerInfo = useRelayStore((s) => s.memoryBrokerInfo)\n const memoryFilters = useRelayStore((s) => s.memoryFilters)\n const selectedMemoryId = useRelayStore((s) => s.selectedMemoryId)\n const memoryLoading = useRelayStore((s) => s.memoryLoading)\n const set = useRelayStore((s) => s.set)\n const fetchMemories = useRelayStore((s) => s.fetchMemories)\n const refreshMemory = useRelayStore((s) => s.refreshMemory)\n const saveMemory = useRelayStore((s) => s.saveMemory)\n const deleteMemory = useRelayStore((s) => s.deleteMemory)\n const injectMemory = useRelayStore((s) => s.injectMemory)\n\n const selectedMemory = memories.find((memory) => memory.id === selectedMemoryId)\n const [formMode, setFormMode] = useState<'create' | 'edit'>('create')\n const [form, setForm] = useState<MemoryForm>(blankForm)\n const [injectAgent, setInjectAgent] = useState('')\n const [injectReason, setInjectReason] = useState('manual')\n const [maxTokens, setMaxTokens] = useState('1600')\n const [maxMemories, setMaxMemories] = useState('8')\n const [lastCommand, setLastCommand] = useState('')\n\n const candidates = useMemo(() => agentOptions(agents), [agents])\n\n useEffect(() => {\n if (!injectAgent && candidates[0]) setInjectAgent(candidates[0].id)\n }, [candidates, injectAgent])\n\n useEffect(() => {\n if (selectedMemory && formMode === 'edit') setForm(formFromMemory(selectedMemory))\n }, [selectedMemory?.id, formMode])\n\n const visibleMemories = useMemo(() => {\n const needle = memoryFilters.search.trim().toLowerCase()\n if (!needle) return memories\n return memories.filter((memory) => [\n memory.id, memory.type, memory.scope, memory.title, memory.content, ...memory.tags,\n ].some((value) => String(value).toLowerCase().includes(needle)))\n }, [memories, memoryFilters.search])\n\n function updateFilter(partial: Partial<MemoryFilterState>) {\n set({ memoryFilters: { ...memoryFilters, ...partial } })\n }\n\n async function submitMemory() {\n const score = Number(form.relevanceScore)\n const common = {\n title: form.title.trim(),\n content: form.content.trim(),\n tags: csv(form.tags),\n visibility: form.visibility,\n sensitivity: form.sensitivity,\n confidence: form.confidence,\n relevanceScore: Number.isFinite(score) ? Math.max(0, Math.min(1, score)) : 0.7,\n }\n if (formMode === 'edit' && selectedMemory) {\n await saveMemory(common satisfies UpdateMemoryInput, selectedMemory.id)\n return\n }\n const input: CreateMemoryInput = {\n ...common,\n type: form.type,\n scope: form.scope.trim(),\n }\n await saveMemory(input)\n setFormMode('create')\n setForm(blankForm())\n }\n\n async function runInject(memoryIds?: string[]) {\n if (!injectAgent) return\n const result = await injectMemory({\n agentId: injectAgent,\n memoryIds,\n query: memoryIds?.length ? undefined : queryFromFilters(memoryFilters),\n reason: injectReason.trim() || 'manual',\n budget: {\n maxTokens: Math.max(200, Number(maxTokens) || 1600),\n maxMemories: Math.max(1, Number(maxMemories) || 8),\n priorityCutoff: 3,\n },\n })\n if (result?.command?.id) setLastCommand(result.command.id)\n }\n\n function startCreate() {\n setFormMode('create')\n setForm(blankForm())\n }\n\n function startEdit() {\n if (!selectedMemory) return\n setFormMode('edit')\n setForm(formFromMemory(selectedMemory))\n }\n\n const canSave = form.title.trim() && form.content.trim() && (formMode === 'edit' || form.scope.trim())\n const configText = compactConfig(memoryBrokerInfo?.config)\n\n return (\n <div className=\"space-y-4\">\n <div className=\"flex items-center justify-between gap-3 flex-wrap\">\n <div className=\"flex items-center gap-2\">\n <BrainCircuit className=\"w-5 h-5 text-muted-foreground\" />\n <h2 className=\"text-lg font-semibold\">Memory</h2>\n <Badge variant=\"secondary\">{memoryStats?.total ?? memoryTotal}</Badge>\n {memoryBrokerInfo && (\n <Badge className={memoryBrokerInfo.external ? 'bg-blue-500/15 text-blue-400 border-blue-500/30' : 'bg-emerald-500/15 text-emerald-400 border-emerald-500/30'}>\n {memoryBrokerInfo.type}\n </Badge>\n )}\n </div>\n <Button size=\"sm\" variant=\"outline\" className=\"gap-1.5\" onClick={() => refreshMemory()}>\n <RefreshCw className={`w-3.5 h-3.5 ${memoryLoading ? 'animate-spin' : ''}`} />\n Refresh\n </Button>\n </div>\n\n <div className=\"grid gap-3 xl:grid-cols-[1.2fr_0.8fr]\">\n <Card>\n <CardHeader className=\"py-3 px-4\">\n <div className=\"flex items-center justify-between gap-2\">\n <CardTitle className=\"text-sm font-medium flex items-center gap-2\">\n <Database className=\"w-4 h-4 text-muted-foreground\" />\n Broker\n </CardTitle>\n {memoryBrokerInfo?.capabilities && (\n <div className=\"flex flex-wrap gap-1 justify-end\">\n {Object.entries(memoryBrokerInfo.capabilities).filter(([, enabled]) => Boolean(enabled)).map(([key]) => (\n <Badge key={key} variant=\"outline\" className=\"text-[10px] px-1.5 py-0\">{key}</Badge>\n ))}\n </div>\n )}\n </div>\n </CardHeader>\n <CardContent className=\"px-4 pb-4 space-y-3\">\n <div className=\"grid grid-cols-2 md:grid-cols-5 gap-2\">\n <Metric label=\"total\" value={memoryStats?.total ?? memoryTotal} />\n <Metric label=\"project\" value={memoryStats?.byType?.project ?? 0} />\n <Metric label=\"task\" value={memoryStats?.byType?.task ?? 0} />\n <Metric label=\"agent\" value={memoryStats?.byType?.agent ?? 0} />\n <Metric label=\"sensitive\" value={(memoryStats?.bySensitivity?.sensitive ?? 0) + (memoryStats?.bySensitivity?.secret ?? 0)} />\n </div>\n <div className=\"font-mono text-xs text-muted-foreground truncate\" title={configText}>{configText}</div>\n </CardContent>\n </Card>\n\n <Card>\n <CardHeader className=\"py-3 px-4\">\n <CardTitle className=\"text-sm font-medium\">Inject</CardTitle>\n </CardHeader>\n <CardContent className=\"px-4 pb-4 space-y-2\">\n <div className=\"grid grid-cols-1 sm:grid-cols-2 gap-2\">\n <select className=\"rounded-md border border-input bg-background px-2 py-1.5 text-sm\" value={injectAgent} onChange={(e) => setInjectAgent(e.target.value)}>\n <option value=\"\">Agent</option>\n {candidates.map((agent) => <option key={agent.id} value={agent.id}>{displayName(agent)}</option>)}\n </select>\n <Input value={injectReason} onChange={(e) => setInjectReason(e.target.value)} placeholder=\"reason\" />\n <Input value={maxTokens} onChange={(e) => setMaxTokens(e.target.value)} inputMode=\"numeric\" placeholder=\"tokens\" />\n <Input value={maxMemories} onChange={(e) => setMaxMemories(e.target.value)} inputMode=\"numeric\" placeholder=\"memories\" />\n </div>\n <div className=\"flex flex-wrap gap-2\">\n <Button size=\"sm\" className=\"gap-1.5\" disabled={!injectAgent || !selectedMemory} onClick={() => selectedMemory && runInject([selectedMemory.id])}>\n <Send className=\"w-3.5 h-3.5\" />\n Selected\n </Button>\n <Button size=\"sm\" variant=\"outline\" className=\"gap-1.5\" disabled={!injectAgent} onClick={() => runInject()}>\n <Search className=\"w-3.5 h-3.5\" />\n Filtered\n </Button>\n {lastCommand && <Badge variant=\"outline\" className=\"font-mono text-[10px]\">{lastCommand}</Badge>}\n </div>\n </CardContent>\n </Card>\n </div>\n\n <div className=\"grid gap-4 xl:grid-cols-[minmax(0,1fr)_420px]\">\n <div className=\"space-y-3\">\n <Card>\n <CardContent className=\"p-3\">\n <div className=\"grid grid-cols-2 lg:grid-cols-6 gap-2\">\n <Input className=\"lg:col-span-2\" value={memoryFilters.search} onChange={(e) => updateFilter({ search: e.target.value })} placeholder=\"Search visible memories\" />\n <select className=\"rounded-md border border-input bg-background px-2 py-1.5 text-sm\" value={memoryFilters.type} onChange={(e) => updateFilter({ type: e.target.value })}>\n <option value=\"\">All types</option>\n {MEMORY_TYPES.map((type) => <option key={type} value={type}>{type}</option>)}\n </select>\n <Input value={memoryFilters.scope} onChange={(e) => updateFilter({ scope: e.target.value })} placeholder=\"scope\" />\n <Input value={memoryFilters.tags} onChange={(e) => updateFilter({ tags: e.target.value })} placeholder=\"tags\" />\n <div className=\"flex gap-2\">\n <Input value={memoryFilters.minRelevance} onChange={(e) => updateFilter({ minRelevance: e.target.value })} placeholder=\"score\" />\n <Button size=\"sm\" variant=\"outline\" className=\"h-8 px-2\" onClick={() => fetchMemories()}>\n <Search className=\"w-4 h-4\" />\n </Button>\n </div>\n </div>\n </CardContent>\n </Card>\n\n <ScrollArea className=\"h-[calc(100dvh-24rem)] min-h-[360px]\">\n <div className=\"space-y-2 pr-2\">\n {visibleMemories.length === 0 && (\n <div className=\"text-center text-sm text-muted-foreground py-16\">No memories</div>\n )}\n {visibleMemories.map((memory) => (\n <MemoryRow\n key={memory.id}\n memory={memory}\n selected={memory.id === selectedMemoryId}\n now={now}\n onSelect={() => set({ selectedMemoryId: memory.id })}\n />\n ))}\n </div>\n </ScrollArea>\n </div>\n\n <Card>\n <CardHeader className=\"py-3 px-4\">\n <div className=\"flex items-center justify-between gap-2\">\n <CardTitle className=\"text-sm font-medium\">{formMode === 'edit' ? 'Edit Memory' : 'New Memory'}</CardTitle>\n <div className=\"flex gap-1\">\n <Button size=\"sm\" variant=\"ghost\" className=\"h-7 px-2\" onClick={startCreate} title=\"New memory\">\n <Plus className=\"w-3.5 h-3.5\" />\n </Button>\n <Button size=\"sm\" variant=\"ghost\" className=\"h-7 px-2\" onClick={startEdit} disabled={!selectedMemory} title=\"Edit selected\">\n <Edit3 className=\"w-3.5 h-3.5\" />\n </Button>\n <Button size=\"sm\" variant=\"ghost\" className=\"h-7 px-2 text-red-400 hover:text-red-300\" disabled={!selectedMemory} onClick={() => selectedMemory && deleteMemory(selectedMemory.id)} title=\"Delete selected\">\n <Trash2 className=\"w-3.5 h-3.5\" />\n </Button>\n </div>\n </div>\n </CardHeader>\n <CardContent className=\"px-4 pb-4 space-y-3\">\n {selectedMemory && (\n <>\n <div className=\"space-y-2\">\n <div className=\"flex flex-wrap gap-1\">\n <Badge variant=\"secondary\">{selectedMemory.type}</Badge>\n <Badge variant=\"outline\">{selectedMemory.scope}</Badge>\n <Badge className={sensitivityClass(selectedMemory.sensitivity)}>{selectedMemory.sensitivity}</Badge>\n <Badge variant=\"outline\" className={scoreClass(selectedMemory.relevanceScore)}>{selectedMemory.relevanceScore.toFixed(2)}</Badge>\n </div>\n <div className=\"text-xs text-muted-foreground\" title={fmtTime(selectedMemory.updatedAt)}>\n updated {timeAgo(now, selectedMemory.updatedAt)}\n </div>\n <p className=\"text-sm font-medium\">{selectedMemory.title}</p>\n <p className=\"text-xs text-muted-foreground whitespace-pre-wrap max-h-24 overflow-auto\">{selectedMemory.content}</p>\n </div>\n <Separator />\n </>\n )}\n\n <div className=\"grid grid-cols-2 gap-2\">\n <select className=\"rounded-md border border-input bg-background px-2 py-1.5 text-sm\" disabled={formMode === 'edit'} value={form.type} onChange={(e) => setForm({ ...form, type: e.target.value as MemoryType })}>\n {MEMORY_TYPES.map((type) => <option key={type} value={type}>{type}</option>)}\n </select>\n <Input disabled={formMode === 'edit'} value={form.scope} onChange={(e) => setForm({ ...form, scope: e.target.value })} placeholder=\"scope\" />\n <Input className=\"col-span-2\" value={form.title} onChange={(e) => setForm({ ...form, title: e.target.value })} placeholder=\"title\" />\n <Textarea className=\"col-span-2 min-h-32\" value={form.content} onChange={(e) => setForm({ ...form, content: e.target.value })} placeholder=\"content\" />\n <Input className=\"col-span-2\" value={form.tags} onChange={(e) => setForm({ ...form, tags: e.target.value })} placeholder=\"tags\" />\n <select className=\"rounded-md border border-input bg-background px-2 py-1.5 text-sm\" value={form.visibility} onChange={(e) => setForm({ ...form, visibility: e.target.value as MemoryForm['visibility'] })}>\n {VISIBILITIES.map((value) => <option key={value} value={value}>{value}</option>)}\n </select>\n <select className=\"rounded-md border border-input bg-background px-2 py-1.5 text-sm\" value={form.sensitivity} onChange={(e) => setForm({ ...form, sensitivity: e.target.value as MemoryForm['sensitivity'] })}>\n {SENSITIVITIES.map((value) => <option key={value} value={value}>{value}</option>)}\n </select>\n <select className=\"rounded-md border border-input bg-background px-2 py-1.5 text-sm\" value={form.confidence} onChange={(e) => setForm({ ...form, confidence: e.target.value as MemoryForm['confidence'] })}>\n {CONFIDENCES.map((value) => <option key={value} value={value}>{value}</option>)}\n </select>\n <Input value={form.relevanceScore} onChange={(e) => setForm({ ...form, relevanceScore: e.target.value })} inputMode=\"decimal\" placeholder=\"score\" />\n </div>\n <Button size=\"sm\" className=\"w-full\" disabled={!canSave} onClick={submitMemory}>\n {formMode === 'edit' ? 'Save memory' : 'Create memory'}\n </Button>\n </CardContent>\n </Card>\n </div>\n </div>\n )\n}\n\nfunction Metric({ label, value }: { label: string; value: number }) {\n return (\n <div className=\"rounded-md border border-border bg-muted/20 px-3 py-2\">\n <div className=\"text-lg font-semibold tabular-nums\">{value}</div>\n <div className=\"text-xs text-muted-foreground\">{label}</div>\n </div>\n )\n}\n\nfunction MemoryRow({ memory, selected, now, onSelect }: { memory: Memory; selected: boolean; now: number; onSelect: () => void }) {\n return (\n <button\n onClick={onSelect}\n className={`w-full text-left rounded-md border p-3 transition-colors ${selected ? 'border-primary bg-primary/5' : 'border-border bg-card hover:border-zinc-600'}`}\n >\n <div className=\"flex items-start justify-between gap-3\">\n <div className=\"min-w-0\">\n <div className=\"flex items-center gap-1.5 flex-wrap\">\n <Badge variant=\"secondary\" className=\"text-[10px]\">{memory.type}</Badge>\n <Badge variant=\"outline\" className=\"text-[10px] font-mono\">{memory.scope}</Badge>\n <Badge className={`text-[10px] ${sensitivityClass(memory.sensitivity)}`}>{memory.sensitivity}</Badge>\n </div>\n <div className=\"font-medium text-sm mt-2 truncate\">{memory.title}</div>\n <div className=\"text-xs text-muted-foreground line-clamp-2 mt-1\">{memory.content}</div>\n {memory.tags.length > 0 && (\n <div className=\"flex gap-1 flex-wrap mt-2\">\n {memory.tags.slice(0, 8).map((tag) => <Badge key={tag} variant=\"outline\" className=\"text-[10px] text-zinc-400\">#{tag}</Badge>)}\n </div>\n )}\n </div>\n <div className=\"text-right shrink-0 space-y-1\">\n <div className={`text-sm font-semibold tabular-nums ${scoreClass(memory.relevanceScore)}`}>{memory.relevanceScore.toFixed(2)}</div>\n <div className=\"text-[10px] text-muted-foreground\" title={fmtTime(memory.updatedAt)}>{timeAgo(now, memory.updatedAt)}</div>\n {memory.accessCount > 0 && <div className=\"text-[10px] text-muted-foreground\">{memory.accessCount} hits</div>}\n </div>\n </div>\n </button>\n )\n}\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"memory-BQAC3YmR.js","names":[],"sources":["../../dashboard/src/components/views/memory.tsx"],"sourcesContent":["import { useEffect, useMemo, useState } from 'react'\nimport { useRelayStore, useNow } from '@/store'\nimport { Badge } from '@/components/ui/badge'\nimport { Button } from '@/components/ui/button'\nimport { Card, CardContent, CardHeader, CardTitle } from '@/components/ui/card'\nimport { Input } from '@/components/ui/input'\nimport { Textarea } from '@/components/ui/textarea'\nimport { ScrollArea } from '@/components/ui/scroll-area'\nimport { Separator } from '@/components/ui/separator'\nimport { BrainCircuit, Database, Edit3, Plus, RefreshCw, Search, Send, Trash2 } from 'lucide-react'\nimport { displayName, fmtTime, timeAgo } from '@/lib/display'\nimport type { Agent, CreateMemoryInput, Memory, MemoryFilterState, MemoryQuery, MemoryType, UpdateMemoryInput } from '@/types'\n\nconst MEMORY_TYPES: MemoryType[] = ['organization', 'role', 'project', 'task', 'interaction', 'agent']\nconst VISIBILITIES = ['private', 'project', 'org', 'public'] as const\nconst SENSITIVITIES = ['public', 'normal', 'sensitive', 'secret'] as const\nconst CONFIDENCES = ['reported', 'inferred', 'verified'] as const\n\ntype MemoryForm = {\n type: MemoryType\n scope: string\n title: string\n content: string\n tags: string\n visibility: (typeof VISIBILITIES)[number]\n sensitivity: (typeof SENSITIVITIES)[number]\n confidence: (typeof CONFIDENCES)[number]\n relevanceScore: string\n}\n\nfunction blankForm(): MemoryForm {\n return {\n type: 'project',\n scope: 'default',\n title: '',\n content: '',\n tags: '',\n visibility: 'project',\n sensitivity: 'normal',\n confidence: 'reported',\n relevanceScore: '0.7',\n }\n}\n\nfunction formFromMemory(memory: Memory): MemoryForm {\n return {\n type: memory.type,\n scope: memory.scope,\n title: memory.title,\n content: memory.content,\n tags: memory.tags.join(', '),\n visibility: memory.visibility,\n sensitivity: memory.sensitivity,\n confidence: memory.confidence,\n relevanceScore: String(memory.relevanceScore),\n }\n}\n\nfunction csv(value: string): string[] {\n return value.split(',').map((item) => item.trim()).filter(Boolean)\n}\n\nfunction queryFromFilters(filters: MemoryFilterState): MemoryQuery {\n const query: MemoryQuery = { limit: 100 }\n if (filters.type) query.type = filters.type as MemoryQuery['type']\n if (filters.scope.trim()) query.scope = filters.scope.trim()\n if (filters.visibility) query.visibility = filters.visibility as MemoryQuery['visibility']\n const tags = csv(filters.tags)\n if (tags.length) query.tags = tags\n if (filters.minRelevance.trim()) {\n const minRelevance = Number(filters.minRelevance)\n if (Number.isFinite(minRelevance)) query.minRelevance = Math.max(0, Math.min(1, minRelevance))\n }\n return query\n}\n\nfunction sensitivityClass(value: string) {\n if (value === 'secret') return 'bg-red-500/10 text-red-400 border-red-500/30'\n if (value === 'sensitive') return 'bg-yellow-500/10 text-yellow-400 border-yellow-500/30'\n if (value === 'public') return 'bg-blue-500/10 text-blue-400 border-blue-500/30'\n return 'bg-zinc-500/10 text-zinc-400 border-zinc-500/30'\n}\n\nfunction scoreClass(score: number) {\n if (score >= 0.8) return 'text-emerald-400'\n if (score >= 0.5) return 'text-yellow-400'\n return 'text-zinc-400'\n}\n\nfunction compactConfig(config: Record<string, unknown> | undefined): string {\n if (!config) return 'default'\n return Object.entries(config)\n .map(([key, value]) => `${key}=${Array.isArray(value) ? value.join(' ') : String(value)}`)\n .join(' ')\n}\n\nfunction agentOptions(agents: Agent[]) {\n return agents.filter((agent) => !['system', 'user', 'channel'].includes(agent.kind || ''))\n}\n\nexport function MemoryView() {\n const now = useNow()\n const agents = useRelayStore((s) => s.agents)\n const memories = useRelayStore((s) => s.memories)\n const memoryTotal = useRelayStore((s) => s.memoryTotal)\n const memoryStats = useRelayStore((s) => s.memoryStats)\n const memoryBrokerInfo = useRelayStore((s) => s.memoryBrokerInfo)\n const memoryFilters = useRelayStore((s) => s.memoryFilters)\n const selectedMemoryId = useRelayStore((s) => s.selectedMemoryId)\n const memoryLoading = useRelayStore((s) => s.memoryLoading)\n const set = useRelayStore((s) => s.set)\n const fetchMemories = useRelayStore((s) => s.fetchMemories)\n const refreshMemory = useRelayStore((s) => s.refreshMemory)\n const saveMemory = useRelayStore((s) => s.saveMemory)\n const deleteMemory = useRelayStore((s) => s.deleteMemory)\n const injectMemory = useRelayStore((s) => s.injectMemory)\n\n const selectedMemory = memories.find((memory) => memory.id === selectedMemoryId)\n const [formMode, setFormMode] = useState<'create' | 'edit'>('create')\n const [form, setForm] = useState<MemoryForm>(blankForm)\n const [injectAgent, setInjectAgent] = useState('')\n const [injectReason, setInjectReason] = useState('manual')\n const [maxTokens, setMaxTokens] = useState('1600')\n const [maxMemories, setMaxMemories] = useState('8')\n const [lastCommand, setLastCommand] = useState('')\n\n const candidates = useMemo(() => agentOptions(agents), [agents])\n\n useEffect(() => {\n if (!injectAgent && candidates[0]) setInjectAgent(candidates[0].id)\n }, [candidates, injectAgent])\n\n useEffect(() => {\n if (selectedMemory && formMode === 'edit') setForm(formFromMemory(selectedMemory))\n }, [selectedMemory?.id, formMode])\n\n const visibleMemories = useMemo(() => {\n const needle = memoryFilters.search.trim().toLowerCase()\n if (!needle) return memories\n return memories.filter((memory) => [\n memory.id, memory.type, memory.scope, memory.title, memory.content, ...memory.tags,\n ].some((value) => String(value).toLowerCase().includes(needle)))\n }, [memories, memoryFilters.search])\n\n function updateFilter(partial: Partial<MemoryFilterState>) {\n set({ memoryFilters: { ...memoryFilters, ...partial } })\n }\n\n async function submitMemory() {\n const score = Number(form.relevanceScore)\n const common = {\n title: form.title.trim(),\n content: form.content.trim(),\n tags: csv(form.tags),\n visibility: form.visibility,\n sensitivity: form.sensitivity,\n confidence: form.confidence,\n relevanceScore: Number.isFinite(score) ? Math.max(0, Math.min(1, score)) : 0.7,\n }\n if (formMode === 'edit' && selectedMemory) {\n await saveMemory(common satisfies UpdateMemoryInput, selectedMemory.id)\n return\n }\n const input: CreateMemoryInput = {\n ...common,\n type: form.type,\n scope: form.scope.trim(),\n }\n await saveMemory(input)\n setFormMode('create')\n setForm(blankForm())\n }\n\n async function runInject(memoryIds?: string[]) {\n if (!injectAgent) return\n const result = await injectMemory({\n agentId: injectAgent,\n memoryIds,\n query: memoryIds?.length ? undefined : queryFromFilters(memoryFilters),\n reason: injectReason.trim() || 'manual',\n budget: {\n maxTokens: Math.max(200, Number(maxTokens) || 1600),\n maxMemories: Math.max(1, Number(maxMemories) || 8),\n priorityCutoff: 3,\n },\n })\n if (result?.command?.id) setLastCommand(result.command.id)\n }\n\n function startCreate() {\n setFormMode('create')\n setForm(blankForm())\n }\n\n function startEdit() {\n if (!selectedMemory) return\n setFormMode('edit')\n setForm(formFromMemory(selectedMemory))\n }\n\n const canSave = form.title.trim() && form.content.trim() && (formMode === 'edit' || form.scope.trim())\n const configText = compactConfig(memoryBrokerInfo?.config)\n\n return (\n <div className=\"space-y-4\">\n <div className=\"flex items-center justify-between gap-3 flex-wrap\">\n <div className=\"flex items-center gap-2\">\n <BrainCircuit className=\"w-5 h-5 text-muted-foreground\" />\n <h2 className=\"text-lg font-semibold\">Memory</h2>\n <Badge variant=\"secondary\">{memoryStats?.total ?? memoryTotal}</Badge>\n {memoryBrokerInfo && (\n <Badge className={memoryBrokerInfo.external ? 'bg-blue-500/15 text-blue-400 border-blue-500/30' : 'bg-emerald-500/15 text-emerald-400 border-emerald-500/30'}>\n {memoryBrokerInfo.type}\n </Badge>\n )}\n </div>\n <Button size=\"sm\" variant=\"outline\" className=\"gap-1.5\" onClick={() => refreshMemory()}>\n <RefreshCw className={`w-3.5 h-3.5 ${memoryLoading ? 'animate-spin' : ''}`} />\n Refresh\n </Button>\n </div>\n\n <div className=\"grid gap-3 xl:grid-cols-[1.2fr_0.8fr]\">\n <Card>\n <CardHeader className=\"py-3 px-4\">\n <div className=\"flex items-center justify-between gap-2\">\n <CardTitle className=\"text-sm font-medium flex items-center gap-2\">\n <Database className=\"w-4 h-4 text-muted-foreground\" />\n Broker\n </CardTitle>\n {memoryBrokerInfo?.capabilities && (\n <div className=\"flex flex-wrap gap-1 justify-end\">\n {Object.entries(memoryBrokerInfo.capabilities).filter(([, enabled]) => Boolean(enabled)).map(([key]) => (\n <Badge key={key} variant=\"outline\" className=\"text-[10px] px-1.5 py-0\">{key}</Badge>\n ))}\n </div>\n )}\n </div>\n </CardHeader>\n <CardContent className=\"px-4 pb-4 space-y-3\">\n <div className=\"grid grid-cols-2 md:grid-cols-5 gap-2\">\n <Metric label=\"total\" value={memoryStats?.total ?? memoryTotal} />\n <Metric label=\"project\" value={memoryStats?.byType?.project ?? 0} />\n <Metric label=\"task\" value={memoryStats?.byType?.task ?? 0} />\n <Metric label=\"agent\" value={memoryStats?.byType?.agent ?? 0} />\n <Metric label=\"sensitive\" value={(memoryStats?.bySensitivity?.sensitive ?? 0) + (memoryStats?.bySensitivity?.secret ?? 0)} />\n </div>\n <div className=\"font-mono text-xs text-muted-foreground truncate\" title={configText}>{configText}</div>\n </CardContent>\n </Card>\n\n <Card>\n <CardHeader className=\"py-3 px-4\">\n <CardTitle className=\"text-sm font-medium\">Inject</CardTitle>\n </CardHeader>\n <CardContent className=\"px-4 pb-4 space-y-2\">\n <div className=\"grid grid-cols-1 sm:grid-cols-2 gap-2\">\n <select className=\"rounded-md border border-input bg-background px-2 py-1.5 text-sm\" value={injectAgent} onChange={(e) => setInjectAgent(e.target.value)}>\n <option value=\"\">Agent</option>\n {candidates.map((agent) => <option key={agent.id} value={agent.id}>{displayName(agent)}</option>)}\n </select>\n <Input value={injectReason} onChange={(e) => setInjectReason(e.target.value)} placeholder=\"reason\" />\n <Input value={maxTokens} onChange={(e) => setMaxTokens(e.target.value)} inputMode=\"numeric\" placeholder=\"tokens\" />\n <Input value={maxMemories} onChange={(e) => setMaxMemories(e.target.value)} inputMode=\"numeric\" placeholder=\"memories\" />\n </div>\n <div className=\"flex flex-wrap gap-2\">\n <Button size=\"sm\" className=\"gap-1.5\" disabled={!injectAgent || !selectedMemory} onClick={() => selectedMemory && runInject([selectedMemory.id])}>\n <Send className=\"w-3.5 h-3.5\" />\n Selected\n </Button>\n <Button size=\"sm\" variant=\"outline\" className=\"gap-1.5\" disabled={!injectAgent} onClick={() => runInject()}>\n <Search className=\"w-3.5 h-3.5\" />\n Filtered\n </Button>\n {lastCommand && <Badge variant=\"outline\" className=\"font-mono text-[10px]\">{lastCommand}</Badge>}\n </div>\n </CardContent>\n </Card>\n </div>\n\n <div className=\"grid gap-4 xl:grid-cols-[minmax(0,1fr)_420px]\">\n <div className=\"space-y-3\">\n <Card>\n <CardContent className=\"p-3\">\n <div className=\"grid grid-cols-2 lg:grid-cols-6 gap-2\">\n <Input className=\"lg:col-span-2\" value={memoryFilters.search} onChange={(e) => updateFilter({ search: e.target.value })} placeholder=\"Search visible memories\" />\n <select className=\"rounded-md border border-input bg-background px-2 py-1.5 text-sm\" value={memoryFilters.type} onChange={(e) => updateFilter({ type: e.target.value })}>\n <option value=\"\">All types</option>\n {MEMORY_TYPES.map((type) => <option key={type} value={type}>{type}</option>)}\n </select>\n <Input value={memoryFilters.scope} onChange={(e) => updateFilter({ scope: e.target.value })} placeholder=\"scope\" />\n <Input value={memoryFilters.tags} onChange={(e) => updateFilter({ tags: e.target.value })} placeholder=\"tags\" />\n <div className=\"flex gap-2\">\n <Input value={memoryFilters.minRelevance} onChange={(e) => updateFilter({ minRelevance: e.target.value })} placeholder=\"score\" />\n <Button size=\"sm\" variant=\"outline\" className=\"h-8 px-2\" onClick={() => fetchMemories()}>\n <Search className=\"w-4 h-4\" />\n </Button>\n </div>\n </div>\n </CardContent>\n </Card>\n\n <ScrollArea className=\"h-[calc(100dvh-24rem)] min-h-[360px]\">\n <div className=\"space-y-2 pr-2\">\n {visibleMemories.length === 0 && (\n <div className=\"text-center text-sm text-muted-foreground py-16\">No memories</div>\n )}\n {visibleMemories.map((memory) => (\n <MemoryRow\n key={memory.id}\n memory={memory}\n selected={memory.id === selectedMemoryId}\n now={now}\n onSelect={() => set({ selectedMemoryId: memory.id })}\n />\n ))}\n </div>\n </ScrollArea>\n </div>\n\n <Card>\n <CardHeader className=\"py-3 px-4\">\n <div className=\"flex items-center justify-between gap-2\">\n <CardTitle className=\"text-sm font-medium\">{formMode === 'edit' ? 'Edit Memory' : 'New Memory'}</CardTitle>\n <div className=\"flex gap-1\">\n <Button size=\"sm\" variant=\"ghost\" className=\"h-7 px-2\" onClick={startCreate} title=\"New memory\">\n <Plus className=\"w-3.5 h-3.5\" />\n </Button>\n <Button size=\"sm\" variant=\"ghost\" className=\"h-7 px-2\" onClick={startEdit} disabled={!selectedMemory} title=\"Edit selected\">\n <Edit3 className=\"w-3.5 h-3.5\" />\n </Button>\n <Button size=\"sm\" variant=\"ghost\" className=\"h-7 px-2 text-red-400 hover:text-red-300\" disabled={!selectedMemory} onClick={() => selectedMemory && deleteMemory(selectedMemory.id)} title=\"Delete selected\">\n <Trash2 className=\"w-3.5 h-3.5\" />\n </Button>\n </div>\n </div>\n </CardHeader>\n <CardContent className=\"px-4 pb-4 space-y-3\">\n {selectedMemory && (\n <>\n <div className=\"space-y-2\">\n <div className=\"flex flex-wrap gap-1\">\n <Badge variant=\"secondary\">{selectedMemory.type}</Badge>\n <Badge variant=\"outline\">{selectedMemory.scope}</Badge>\n <Badge className={sensitivityClass(selectedMemory.sensitivity)}>{selectedMemory.sensitivity}</Badge>\n <Badge variant=\"outline\" className={scoreClass(selectedMemory.relevanceScore)}>{selectedMemory.relevanceScore.toFixed(2)}</Badge>\n </div>\n <div className=\"text-xs text-muted-foreground\" title={fmtTime(selectedMemory.updatedAt)}>\n updated {timeAgo(now, selectedMemory.updatedAt)}\n </div>\n <p className=\"text-sm font-medium\">{selectedMemory.title}</p>\n <p className=\"text-xs text-muted-foreground whitespace-pre-wrap max-h-24 overflow-auto\">{selectedMemory.content}</p>\n </div>\n <Separator />\n </>\n )}\n\n <div className=\"grid grid-cols-2 gap-2\">\n <select className=\"rounded-md border border-input bg-background px-2 py-1.5 text-sm\" disabled={formMode === 'edit'} value={form.type} onChange={(e) => setForm({ ...form, type: e.target.value as MemoryType })}>\n {MEMORY_TYPES.map((type) => <option key={type} value={type}>{type}</option>)}\n </select>\n <Input disabled={formMode === 'edit'} value={form.scope} onChange={(e) => setForm({ ...form, scope: e.target.value })} placeholder=\"scope\" />\n <Input className=\"col-span-2\" value={form.title} onChange={(e) => setForm({ ...form, title: e.target.value })} placeholder=\"title\" />\n <Textarea className=\"col-span-2 min-h-32\" value={form.content} onChange={(e) => setForm({ ...form, content: e.target.value })} placeholder=\"content\" />\n <Input className=\"col-span-2\" value={form.tags} onChange={(e) => setForm({ ...form, tags: e.target.value })} placeholder=\"tags\" />\n <select className=\"rounded-md border border-input bg-background px-2 py-1.5 text-sm\" value={form.visibility} onChange={(e) => setForm({ ...form, visibility: e.target.value as MemoryForm['visibility'] })}>\n {VISIBILITIES.map((value) => <option key={value} value={value}>{value}</option>)}\n </select>\n <select className=\"rounded-md border border-input bg-background px-2 py-1.5 text-sm\" value={form.sensitivity} onChange={(e) => setForm({ ...form, sensitivity: e.target.value as MemoryForm['sensitivity'] })}>\n {SENSITIVITIES.map((value) => <option key={value} value={value}>{value}</option>)}\n </select>\n <select className=\"rounded-md border border-input bg-background px-2 py-1.5 text-sm\" value={form.confidence} onChange={(e) => setForm({ ...form, confidence: e.target.value as MemoryForm['confidence'] })}>\n {CONFIDENCES.map((value) => <option key={value} value={value}>{value}</option>)}\n </select>\n <Input value={form.relevanceScore} onChange={(e) => setForm({ ...form, relevanceScore: e.target.value })} inputMode=\"decimal\" placeholder=\"score\" />\n </div>\n <Button size=\"sm\" className=\"w-full\" disabled={!canSave} onClick={submitMemory}>\n {formMode === 'edit' ? 'Save memory' : 'Create memory'}\n </Button>\n </CardContent>\n </Card>\n </div>\n </div>\n )\n}\n\nfunction Metric({ label, value }: { label: string; value: number }) {\n return (\n <div className=\"rounded-md border border-border bg-muted/20 px-3 py-2\">\n <div className=\"text-lg font-semibold tabular-nums\">{value}</div>\n <div className=\"text-xs text-muted-foreground\">{label}</div>\n </div>\n )\n}\n\nfunction MemoryRow({ memory, selected, now, onSelect }: { memory: Memory; selected: boolean; now: number; onSelect: () => void }) {\n return (\n <button\n onClick={onSelect}\n className={`w-full text-left rounded-md border p-3 transition-colors ${selected ? 'border-primary bg-primary/5' : 'border-border bg-card hover:border-zinc-600'}`}\n >\n <div className=\"flex items-start justify-between gap-3\">\n <div className=\"min-w-0\">\n <div className=\"flex items-center gap-1.5 flex-wrap\">\n <Badge variant=\"secondary\" className=\"text-[10px]\">{memory.type}</Badge>\n <Badge variant=\"outline\" className=\"text-[10px] font-mono\">{memory.scope}</Badge>\n <Badge className={`text-[10px] ${sensitivityClass(memory.sensitivity)}`}>{memory.sensitivity}</Badge>\n </div>\n <div className=\"font-medium text-sm mt-2 truncate\">{memory.title}</div>\n <div className=\"text-xs text-muted-foreground line-clamp-2 mt-1\">{memory.content}</div>\n {memory.tags.length > 0 && (\n <div className=\"flex gap-1 flex-wrap mt-2\">\n {memory.tags.slice(0, 8).map((tag) => <Badge key={tag} variant=\"outline\" className=\"text-[10px] text-zinc-400\">#{tag}</Badge>)}\n </div>\n )}\n </div>\n <div className=\"text-right shrink-0 space-y-1\">\n <div className={`text-sm font-semibold tabular-nums ${scoreClass(memory.relevanceScore)}`}>{memory.relevanceScore.toFixed(2)}</div>\n <div className=\"text-[10px] text-muted-foreground\" title={fmtTime(memory.updatedAt)}>{timeAgo(now, memory.updatedAt)}</div>\n {memory.accessCount > 0 && <div className=\"text-[10px] text-muted-foreground\">{memory.accessCount} hits</div>}\n </div>\n </div>\n </button>\n )\n}\n"],"mappings":"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"}
|
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
import{r as e}from"./chunk-CilyBKbf.js";import{$ as t,A as n,Vn as r,Y as i,Zt as a,_n as o,_t as s,i as c,kn as l,m as u,p as d,z as f}from"./lucide-react-CD8Xl2U3.js";import{i as p,t as m}from"./store-
|
|
2
|
-
//# sourceMappingURL=messages-
|
|
1
|
+
import{r as e}from"./chunk-CilyBKbf.js";import{$ as t,A as n,Vn as r,Y as i,Zt as a,_n as o,_t as s,i as c,kn as l,m as u,p as d,z as f}from"./lucide-react-CD8Xl2U3.js";import{i as p,t as m}from"./store-CICRhg1m.js";import{d as h,m as g,u as _,w as v,z as y}from"./display-Bebqs1qu.js";import{t as b}from"./badge-t8zAwHW9.js";import{t as x}from"./button-DDA5P2YQ.js";import{L as S,M as C,t as w}from"./index-CvSlyTSI.js";import{n as T,t as E}from"./card-CggxP1h9.js";var D=e(r(),1),O=l();function k(e,t){if(t.startsWith(`/`))return t;let n=`${e.replace(/\/+$/,``)}/${t}`.split(`/`),r=[];for(let e of n)!e||e===`.`||(e===`..`?r.pop():r.push(e));return`/`+r.join(`/`)}function A(e){let t=e.replace(/\/+$/,``),n=t.lastIndexOf(`/`);return n>0?t.slice(0,n):`/`}function j(){let e=p(),n=m(e=>e.messages),r=m(e=>e.agentsById),a=m(e=>e.selectedAgent),o=m(e=>e.channelFilter),s=m(e=>e.tagFilter),l=m(e=>e.replyTo),u=m(e=>e.set),d=m(e=>e.openCompose),h=m(e=>e.startReply),g=m(e=>e.doClaim),v=m(e=>e.doDeliveryAction),y=m(e=>e.doDeleteMessage),w=m(e=>e.openThread),T=m(e=>e.cancelReply),E=C(),k=S(),A=(0,D.useMemo)(()=>{let e=[...n].sort((e,t)=>t.id-e.id);return a&&(e=e.filter(e=>e.from===a||e.to===a)),o&&(e=e.filter(e=>e.channel===o)),s&&(e=e.filter(e=>((r[e.from]||r[e.to])?.tags||[]).includes(s))),e},[n,a,o,s,r]),j=(0,D.useMemo)(()=>{let e=[],t=new Map;for(let n of A){let r=n.threadId??n.id;t.has(r)||t.set(r,[]),t.get(r).push(n),n.id===r&&e.push(n)}let n=new Set(e.map(e=>e.id));for(let t of A)!n.has(t.threadId??t.id)&&!e.find(e=>e.id===t.id)&&e.push(t);return e.sort((e,t)=>t.id-e.id),e.map(e=>({root:e,replies:(t.get(e.id)||[]).filter(t=>t.id!==e.id).sort((e,t)=>e.id-t.id)}))},[A]);function N(e){return r[e]?_(r[e]):e.slice(-10)}let P=(0,D.useMemo)(()=>[...new Set(n.map(e=>e.channel).filter(Boolean))],[n]);return(0,O.jsxs)(`div`,{className:`space-y-4`,children:[(0,O.jsxs)(`div`,{className:`flex items-center justify-between flex-wrap gap-2`,children:[(0,O.jsxs)(`div`,{className:`flex items-center gap-2`,children:[(0,O.jsx)(t,{className:`w-5 h-5 text-muted-foreground`}),(0,O.jsx)(`h2`,{className:`text-lg font-semibold`,children:`Messages`}),(0,O.jsx)(b,{variant:`secondary`,children:A.length})]}),(0,O.jsxs)(x,{size:`sm`,onClick:d,children:[(0,O.jsx)(f,{className:`w-4 h-4 mr-1`}),` Compose`]})]}),(0,O.jsxs)(`div`,{className:`flex flex-wrap gap-2`,children:[(0,O.jsxs)(`select`,{className:`rounded-md border border-input bg-background px-3 py-1.5 text-sm`,value:a,onChange:e=>u({selectedAgent:e.target.value}),children:[(0,O.jsx)(`option`,{value:``,children:`All agents`}),E.map(e=>(0,O.jsx)(`option`,{value:e.id,children:_(e)},e.id))]}),(0,O.jsxs)(`select`,{className:`rounded-md border border-input bg-background px-3 py-1.5 text-sm`,value:s,onChange:e=>u({tagFilter:e.target.value}),children:[(0,O.jsx)(`option`,{value:``,children:`All tags`}),k.map(e=>(0,O.jsxs)(`option`,{value:e,children:[`#`,e]},e))]}),(0,O.jsxs)(`select`,{className:`rounded-md border border-input bg-background px-3 py-1.5 text-sm`,value:o,onChange:e=>u({channelFilter:e.target.value}),children:[(0,O.jsx)(`option`,{value:``,children:`All channels`}),P.map(e=>(0,O.jsx)(`option`,{value:e,children:e},e))]})]}),l&&(0,O.jsxs)(`div`,{className:`flex items-center gap-2 rounded-md bg-blue-500/10 border border-blue-500/20 px-3 py-2 text-sm text-blue-300`,children:[(0,O.jsx)(i,{className:`w-4 h-4 shrink-0`}),(0,O.jsxs)(`span`,{children:[`Replying to message #`,l.id,` from `,(0,O.jsx)(`strong`,{children:N(l.from)})]}),(0,O.jsx)(x,{variant:`ghost`,size:`sm`,className:`ml-auto h-6 px-1.5`,onClick:T,children:(0,O.jsx)(c,{className:`w-3.5 h-3.5`})})]}),(0,O.jsx)(`div`,{className:`h-[calc(100dvh-15rem)] overflow-y-auto`,children:(0,O.jsxs)(`div`,{className:`space-y-2 pr-2`,children:[j.length===0&&(0,O.jsx)(`div`,{className:`text-center text-muted-foreground py-16 text-sm`,children:`No messages`}),j.map(({root:t,replies:n})=>(0,O.jsxs)(`div`,{className:`space-y-1`,children:[(0,O.jsx)(M,{msg:t,now:e,agentLabel:N,agentsById:r,onReply:()=>h(t),onClaim:()=>g(t.id),onRetry:()=>v(t.id,`retry-now`),onMarkDead:()=>v(t.id,`mark-dead`,`Marked dead from dashboard`),onClear:()=>v(t.id,`clear`,`Cleared after manual handling`),onDelete:()=>y(t.id),onThread:t.threadId?()=>w(t.threadId):void 0,replyCount:n.length}),n.map(t=>(0,O.jsx)(`div`,{className:`ml-6`,children:(0,O.jsx)(M,{msg:t,now:e,agentLabel:N,agentsById:r,onReply:()=>h(t),onClaim:()=>g(t.id),onRetry:()=>v(t.id,`retry-now`),onMarkDead:()=>v(t.id,`mark-dead`,`Marked dead from dashboard`),onClear:()=>v(t.id,`clear`,`Cleared after manual handling`),onDelete:()=>y(t.id)})},t.id))]},t.id))]})})]})}function M({msg:e,now:t,agentLabel:r,agentsById:c,onReply:l,onClaim:d,onRetry:f,onMarkDead:p,onClear:_,onDelete:S,onThread:C,replyCount:j}){let[M,P]=(0,D.useState)(!1),F=v(e),I=m(e=>e.openFilesAt),L=e.deliveryStatus===`queued`||e.deliveryStatus===`failed`||e.deliveryStatus===`dead`||!!(e.deliveryLastError||e.deliveryPoisonReason),R=c[e.from===`user`?e.to:e.from];function z(e){return k((typeof R?.meta?.cwd==`string`?R.meta.cwd:``)||`/`,e)}function B(e,t){let n=z(e);I({path:A(n),selectedPath:n,line:t,overlay:!0})}return(0,O.jsx)(E,{className:`group relative`,children:(0,O.jsxs)(T,{className:`p-3`,children:[(0,O.jsxs)(`div`,{className:`min-w-0`,children:[(0,O.jsxs)(`div`,{className:`flex items-center gap-2 flex-wrap text-xs text-muted-foreground mb-1`,children:[(0,O.jsx)(`span`,{className:`font-medium text-foreground`,children:r(e.from)}),(0,O.jsx)(`span`,{children:`→`}),(0,O.jsx)(`span`,{children:h(e.to,c)}),e.channel&&(0,O.jsxs)(b,{variant:`outline`,className:`text-xs px-1.5 py-0 gap-0.5`,children:[(0,O.jsx)(s,{className:`w-2.5 h-2.5`}),e.channel]}),e.claimedBy&&(0,O.jsxs)(b,{variant:`outline`,className:`text-xs px-1.5 py-0 text-emerald-400 border-emerald-500/30`,children:[(0,O.jsx)(a,{className:`w-2.5 h-2.5 mr-0.5`}),`claimed`]}),L&&(0,O.jsx)(N,{msg:e,now:t}),e.subject&&(0,O.jsx)(`span`,{className:`font-medium text-foreground`,children:e.subject}),(0,O.jsxs)(`span`,{className:`ml-auto shrink-0`,title:g(e.createdAt),children:[(0,O.jsxs)(`span`,{className:`opacity-40 mr-1`,children:[`#`,e.id]}),y(t,e.createdAt)]})]}),F&&(0,O.jsx)(`div`,{className:`cursor-pointer`,onClick:()=>P(e=>!e),children:(0,O.jsx)(w,{text:F,rawBody:e.body,className:`text-sm text-muted-foreground ${M?``:`max-h-[3.75rem] overflow-hidden`}`,onOpenPath:B,resolvePathTitle:(e,t)=>`${z(e)}${t?`:${t}`:``}`})})]}),(0,O.jsxs)(`div`,{className:`absolute top-2 right-2 flex items-center gap-1 opacity-0 group-hover:opacity-100 transition-opacity bg-card rounded-md shadow-sm border border-border px-1 py-0.5`,children:[(0,O.jsx)(x,{variant:`ghost`,size:`sm`,className:`h-7 px-2 text-xs`,onClick:l,title:`Reply`,children:(0,O.jsx)(i,{className:`w-3.5 h-3.5`})}),e.claimable&&!e.claimedBy&&(0,O.jsx)(x,{variant:`ghost`,size:`sm`,className:`h-7 px-2 text-xs text-orange-400 hover:text-orange-300`,onClick:d,title:`Claim`,children:(0,O.jsx)(a,{className:`w-3.5 h-3.5`})}),L&&(0,O.jsx)(x,{variant:`ghost`,size:`sm`,className:`h-7 px-2 text-xs text-blue-400 hover:text-blue-300`,onClick:f,title:`Retry delivery now`,children:(0,O.jsx)(n,{className:`w-3.5 h-3.5`})}),e.deliveryStatus!==`dead`&&L&&(0,O.jsx)(x,{variant:`ghost`,size:`sm`,className:`h-7 px-2 text-xs text-red-400 hover:text-red-300`,onClick:p,title:`Mark dead`,children:(0,O.jsx)(o,{className:`w-3.5 h-3.5`})}),(e.deliveryStatus===`failed`||e.deliveryStatus===`dead`)&&(0,O.jsx)(x,{variant:`ghost`,size:`sm`,className:`h-7 px-2 text-xs text-emerald-400 hover:text-emerald-300`,onClick:_,title:`Clear after manual handling`,children:(0,O.jsx)(a,{className:`w-3.5 h-3.5`})}),C&&(j??0)>0&&(0,O.jsxs)(x,{variant:`ghost`,size:`sm`,className:`h-7 px-2 text-xs`,onClick:C,title:`Thread`,children:[`+`,j]}),(0,O.jsx)(x,{variant:`ghost`,size:`sm`,className:`h-7 px-2 text-xs text-red-400 hover:text-red-300`,onClick:S,title:`Delete`,children:(0,O.jsx)(u,{className:`w-3.5 h-3.5`})})]})]})})}function N({msg:e,now:t}){let n=e.deliveryStatus||`pending`,r=e.deliveryPoisonReason||e.deliveryLastError||(e.deliveryNextRetryAt?`retry ${y(t,e.deliveryNextRetryAt)}`:void 0);return(0,O.jsxs)(b,{variant:`outline`,className:`text-xs px-1.5 py-0 ${n===`dead`?`text-red-300 border-red-500/30 bg-red-500/10`:n===`failed`?`text-amber-300 border-amber-500/30 bg-amber-500/10`:`text-blue-300 border-blue-500/30 bg-blue-500/10`}`,title:r||n,children:[(0,O.jsx)(d,{className:`w-2.5 h-2.5 mr-0.5`}),n,e.deliveryAttempts?` ${e.deliveryAttempts}x`:``]})}export{j as MessagesView};
|
|
2
|
+
//# sourceMappingURL=messages-_sdHV42k.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"messages-B4lCP5rS.js","names":[],"sources":["../../dashboard/src/components/views/messages.tsx"],"sourcesContent":["import { useMemo, useState } from 'react'\nimport { useRelayStore, useNow } from '@/store'\nimport { useComposeAgents, useUniqueTags } from '@/hooks/use-selectors'\nimport { Card, CardContent } from '@/components/ui/card'\nimport { Badge } from '@/components/ui/badge'\nimport { Button } from '@/components/ui/button'\nimport { Input } from '@/components/ui/input'\nimport { FormattedBody } from '@/components/shared/formatted-body'\nimport {\n Mail, Pencil, MessageSquareReply, CheckCircle2, Trash2, X, Hash, AlertTriangle, RotateCcw, Ban,\n} from 'lucide-react'\nimport { displayName, displayTarget, messageBody, timeAgo, fmtTime } from '@/lib/display'\nimport type { Message } from '@/types'\n\nfunction resolveRelativePath(cwd: string, path: string): string {\n if (path.startsWith('/')) return path\n const parts = `${cwd.replace(/\\/+$/, '')}/${path}`.split('/')\n const stack: string[] = []\n for (const part of parts) {\n if (!part || part === '.') continue\n if (part === '..') stack.pop()\n else stack.push(part)\n }\n return '/' + stack.join('/')\n}\n\nfunction parentPath(path: string): string {\n const trimmed = path.replace(/\\/+$/, '')\n const index = trimmed.lastIndexOf('/')\n return index > 0 ? trimmed.slice(0, index) : '/'\n}\n\nexport function MessagesView() {\n const now = useNow()\n const messages = useRelayStore((s) => s.messages)\n const agentsById = useRelayStore((s) => s.agentsById)\n const selectedAgent = useRelayStore((s) => s.selectedAgent)\n const channelFilter = useRelayStore((s) => s.channelFilter)\n const tagFilter = useRelayStore((s) => s.tagFilter)\n const replyTo = useRelayStore((s) => s.replyTo)\n const set = useRelayStore((s) => s.set)\n const openCompose = useRelayStore((s) => s.openCompose)\n const startReply = useRelayStore((s) => s.startReply)\n const doClaim = useRelayStore((s) => s.doClaim)\n const doDeliveryAction = useRelayStore((s) => s.doDeliveryAction)\n const doDeleteMessage = useRelayStore((s) => s.doDeleteMessage)\n const openThread = useRelayStore((s) => s.openThread)\n const cancelReply = useRelayStore((s) => s.cancelReply)\n\n const composeAgents = useComposeAgents()\n const uniqueTags = useUniqueTags()\n\n const filtered = useMemo(() => {\n let list = [...messages].sort((a, b) => b.id - a.id)\n if (selectedAgent) list = list.filter((m) => m.from === selectedAgent || m.to === selectedAgent)\n if (channelFilter) list = list.filter((m) => m.channel === channelFilter)\n if (tagFilter) list = list.filter((m) => {\n const agent = agentsById[m.from] || agentsById[m.to]\n return (agent?.tags || []).includes(tagFilter)\n })\n return list\n }, [messages, selectedAgent, channelFilter, tagFilter, agentsById])\n\n // Group by threadId — show root messages, collapse replies\n const grouped = useMemo(() => {\n const roots: Message[] = []\n const threads = new Map<number, Message[]>()\n for (const msg of filtered) {\n const tid = msg.threadId ?? msg.id\n if (!threads.has(tid)) threads.set(tid, [])\n threads.get(tid)!.push(msg)\n if (msg.id === tid) roots.push(msg)\n }\n // messages without a known root\n const seenRoots = new Set(roots.map((r) => r.id))\n for (const msg of filtered) {\n if (!seenRoots.has(msg.threadId ?? msg.id) && !roots.find((r) => r.id === msg.id)) {\n roots.push(msg)\n }\n }\n roots.sort((a, b) => b.id - a.id)\n return roots.map((r) => ({ root: r, replies: (threads.get(r.id) || []).filter((m) => m.id !== r.id).sort((a, b) => a.id - b.id) }))\n }, [filtered])\n\n function agentLabel(id: string) {\n return agentsById[id] ? displayName(agentsById[id]) : id.slice(-10)\n }\n\n const uniqueChannels = useMemo(() => {\n return [...new Set(messages.map((m) => m.channel).filter(Boolean) as string[])]\n }, [messages])\n\n return (\n <div className=\"space-y-4\">\n <div className=\"flex items-center justify-between flex-wrap gap-2\">\n <div className=\"flex items-center gap-2\">\n <Mail className=\"w-5 h-5 text-muted-foreground\" />\n <h2 className=\"text-lg font-semibold\">Messages</h2>\n <Badge variant=\"secondary\">{filtered.length}</Badge>\n </div>\n <Button size=\"sm\" onClick={openCompose}>\n <Pencil className=\"w-4 h-4 mr-1\" /> Compose\n </Button>\n </div>\n\n <div className=\"flex flex-wrap gap-2\">\n <select\n className=\"rounded-md border border-input bg-background px-3 py-1.5 text-sm\"\n value={selectedAgent}\n onChange={(e) => set({ selectedAgent: e.target.value })}\n >\n <option value=\"\">All agents</option>\n {composeAgents.map((a) => <option key={a.id} value={a.id}>{displayName(a)}</option>)}\n </select>\n <select\n className=\"rounded-md border border-input bg-background px-3 py-1.5 text-sm\"\n value={tagFilter}\n onChange={(e) => set({ tagFilter: e.target.value })}\n >\n <option value=\"\">All tags</option>\n {uniqueTags.map((t) => <option key={t} value={t}>#{t}</option>)}\n </select>\n <select\n className=\"rounded-md border border-input bg-background px-3 py-1.5 text-sm\"\n value={channelFilter}\n onChange={(e) => set({ channelFilter: e.target.value })}\n >\n <option value=\"\">All channels</option>\n {uniqueChannels.map((c) => <option key={c} value={c}>{c}</option>)}\n </select>\n </div>\n\n {replyTo && (\n <div className=\"flex items-center gap-2 rounded-md bg-blue-500/10 border border-blue-500/20 px-3 py-2 text-sm text-blue-300\">\n <MessageSquareReply className=\"w-4 h-4 shrink-0\" />\n <span>Replying to message #{replyTo.id} from <strong>{agentLabel(replyTo.from)}</strong></span>\n <Button variant=\"ghost\" size=\"sm\" className=\"ml-auto h-6 px-1.5\" onClick={cancelReply}>\n <X className=\"w-3.5 h-3.5\" />\n </Button>\n </div>\n )}\n\n <div className=\"h-[calc(100dvh-15rem)] overflow-y-auto\">\n <div className=\"space-y-2 pr-2\">\n {grouped.length === 0 && (\n <div className=\"text-center text-muted-foreground py-16 text-sm\">No messages</div>\n )}\n {grouped.map(({ root, replies }) => (\n <div key={root.id} className=\"space-y-1\">\n <MessageRow\n msg={root}\n now={now}\n agentLabel={agentLabel}\n agentsById={agentsById}\n onReply={() => startReply(root)}\n onClaim={() => doClaim(root.id)}\n onRetry={() => doDeliveryAction(root.id, 'retry-now')}\n onMarkDead={() => doDeliveryAction(root.id, 'mark-dead', 'Marked dead from dashboard')}\n onClear={() => doDeliveryAction(root.id, 'clear', 'Cleared after manual handling')}\n onDelete={() => doDeleteMessage(root.id)}\n onThread={root.threadId ? () => openThread(root.threadId!) : undefined}\n replyCount={replies.length}\n />\n {replies.map((r) => (\n <div key={r.id} className=\"ml-6\">\n <MessageRow\n msg={r}\n now={now}\n agentLabel={agentLabel}\n agentsById={agentsById}\n onReply={() => startReply(r)}\n onClaim={() => doClaim(r.id)}\n onRetry={() => doDeliveryAction(r.id, 'retry-now')}\n onMarkDead={() => doDeliveryAction(r.id, 'mark-dead', 'Marked dead from dashboard')}\n onClear={() => doDeliveryAction(r.id, 'clear', 'Cleared after manual handling')}\n onDelete={() => doDeleteMessage(r.id)}\n />\n </div>\n ))}\n </div>\n ))}\n </div>\n </div>\n </div>\n )\n}\n\nfunction MessageRow({\n msg, now, agentLabel, agentsById, onReply, onClaim, onRetry, onMarkDead, onClear, onDelete, onThread, replyCount,\n}: {\n msg: Message\n now: number\n agentLabel: (id: string) => string\n agentsById: Record<string, import('@/types').Agent>\n onReply: () => void\n onClaim: () => void\n onRetry: () => void\n onMarkDead: () => void\n onClear: () => void\n onDelete: () => void\n onThread?: () => void\n replyCount?: number\n}) {\n const [expanded, setExpanded] = useState(false)\n const body = messageBody(msg)\n const openFilesAt = useRelayStore((s) => s.openFilesAt)\n const deliveryProblem = msg.deliveryStatus === 'queued' || msg.deliveryStatus === 'failed' || msg.deliveryStatus === 'dead' || Boolean(msg.deliveryLastError || msg.deliveryPoisonReason)\n const peer = msg.from === 'user' ? msg.to : msg.from\n const peerAgent = agentsById[peer]\n\n function resolvedReferencedPath(path: string): string {\n const cwd = typeof peerAgent?.meta?.cwd === 'string' ? peerAgent.meta.cwd : ''\n return resolveRelativePath(cwd || '/', path)\n }\n\n function openReferencedPath(path: string, line?: number) {\n const absolute = resolvedReferencedPath(path)\n void openFilesAt({ path: parentPath(absolute), selectedPath: absolute, line, overlay: true })\n }\n\n return (\n <Card className=\"group relative\">\n <CardContent className=\"p-3\">\n <div className=\"min-w-0\">\n <div className=\"flex items-center gap-2 flex-wrap text-xs text-muted-foreground mb-1\">\n <span className=\"font-medium text-foreground\">{agentLabel(msg.from)}</span>\n <span>→</span>\n <span>{displayTarget(msg.to, agentsById)}</span>\n {msg.channel && (\n <Badge variant=\"outline\" className=\"text-xs px-1.5 py-0 gap-0.5\">\n <Hash className=\"w-2.5 h-2.5\" />{msg.channel}\n </Badge>\n )}\n {msg.claimedBy && (\n <Badge variant=\"outline\" className=\"text-xs px-1.5 py-0 text-emerald-400 border-emerald-500/30\">\n <CheckCircle2 className=\"w-2.5 h-2.5 mr-0.5\" />claimed\n </Badge>\n )}\n {deliveryProblem && <DeliveryBadge msg={msg} now={now} />}\n {msg.subject && <span className=\"font-medium text-foreground\">{msg.subject}</span>}\n <span className=\"ml-auto shrink-0\" title={fmtTime(msg.createdAt)}><span className=\"opacity-40 mr-1\">#{msg.id}</span>{timeAgo(now, msg.createdAt)}</span>\n </div>\n {body && (\n <div className=\"cursor-pointer\" onClick={() => setExpanded((v) => !v)}>\n <FormattedBody text={body} rawBody={msg.body} className={`text-sm text-muted-foreground ${expanded ? '' : 'max-h-[3.75rem] overflow-hidden'}`} onOpenPath={openReferencedPath} resolvePathTitle={(path, line) => `${resolvedReferencedPath(path)}${line ? `:${line}` : ''}`} />\n </div>\n )}\n </div>\n <div className=\"absolute top-2 right-2 flex items-center gap-1 opacity-0 group-hover:opacity-100 transition-opacity bg-card rounded-md shadow-sm border border-border px-1 py-0.5\">\n <Button variant=\"ghost\" size=\"sm\" className=\"h-7 px-2 text-xs\" onClick={onReply} title=\"Reply\">\n <MessageSquareReply className=\"w-3.5 h-3.5\" />\n </Button>\n {msg.claimable && !msg.claimedBy && (\n <Button variant=\"ghost\" size=\"sm\" className=\"h-7 px-2 text-xs text-orange-400 hover:text-orange-300\" onClick={onClaim} title=\"Claim\">\n <CheckCircle2 className=\"w-3.5 h-3.5\" />\n </Button>\n )}\n {deliveryProblem && (\n <Button variant=\"ghost\" size=\"sm\" className=\"h-7 px-2 text-xs text-blue-400 hover:text-blue-300\" onClick={onRetry} title=\"Retry delivery now\">\n <RotateCcw className=\"w-3.5 h-3.5\" />\n </Button>\n )}\n {msg.deliveryStatus !== 'dead' && deliveryProblem && (\n <Button variant=\"ghost\" size=\"sm\" className=\"h-7 px-2 text-xs text-red-400 hover:text-red-300\" onClick={onMarkDead} title=\"Mark dead\">\n <Ban className=\"w-3.5 h-3.5\" />\n </Button>\n )}\n {(msg.deliveryStatus === 'failed' || msg.deliveryStatus === 'dead') && (\n <Button variant=\"ghost\" size=\"sm\" className=\"h-7 px-2 text-xs text-emerald-400 hover:text-emerald-300\" onClick={onClear} title=\"Clear after manual handling\">\n <CheckCircle2 className=\"w-3.5 h-3.5\" />\n </Button>\n )}\n {onThread && (replyCount ?? 0) > 0 && (\n <Button variant=\"ghost\" size=\"sm\" className=\"h-7 px-2 text-xs\" onClick={onThread} title=\"Thread\">\n +{replyCount}\n </Button>\n )}\n <Button variant=\"ghost\" size=\"sm\" className=\"h-7 px-2 text-xs text-red-400 hover:text-red-300\" onClick={onDelete} title=\"Delete\">\n <Trash2 className=\"w-3.5 h-3.5\" />\n </Button>\n </div>\n </CardContent>\n </Card>\n )\n}\n\nfunction DeliveryBadge({ msg, now }: { msg: Message; now: number }) {\n const status = msg.deliveryStatus || 'pending'\n const detail = msg.deliveryPoisonReason || msg.deliveryLastError || (msg.deliveryNextRetryAt ? `retry ${timeAgo(now, msg.deliveryNextRetryAt)}` : undefined)\n const className = status === 'dead'\n ? 'text-red-300 border-red-500/30 bg-red-500/10'\n : status === 'failed'\n ? 'text-amber-300 border-amber-500/30 bg-amber-500/10'\n : 'text-blue-300 border-blue-500/30 bg-blue-500/10'\n return (\n <Badge variant=\"outline\" className={`text-xs px-1.5 py-0 ${className}`} title={detail || status}>\n <AlertTriangle className=\"w-2.5 h-2.5 mr-0.5\" />\n {status}{msg.deliveryAttempts ? ` ${msg.deliveryAttempts}x` : ''}\n </Badge>\n )\n}\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"messages-_sdHV42k.js","names":[],"sources":["../../dashboard/src/components/views/messages.tsx"],"sourcesContent":["import { useMemo, useState } from 'react'\nimport { useRelayStore, useNow } from '@/store'\nimport { useComposeAgents, useUniqueTags } from '@/hooks/use-selectors'\nimport { Card, CardContent } from '@/components/ui/card'\nimport { Badge } from '@/components/ui/badge'\nimport { Button } from '@/components/ui/button'\nimport { Input } from '@/components/ui/input'\nimport { FormattedBody } from '@/components/shared/formatted-body'\nimport {\n Mail, Pencil, MessageSquareReply, CheckCircle2, Trash2, X, Hash, AlertTriangle, RotateCcw, Ban,\n} from 'lucide-react'\nimport { displayName, displayTarget, messageBody, timeAgo, fmtTime } from '@/lib/display'\nimport type { Message } from '@/types'\n\nfunction resolveRelativePath(cwd: string, path: string): string {\n if (path.startsWith('/')) return path\n const parts = `${cwd.replace(/\\/+$/, '')}/${path}`.split('/')\n const stack: string[] = []\n for (const part of parts) {\n if (!part || part === '.') continue\n if (part === '..') stack.pop()\n else stack.push(part)\n }\n return '/' + stack.join('/')\n}\n\nfunction parentPath(path: string): string {\n const trimmed = path.replace(/\\/+$/, '')\n const index = trimmed.lastIndexOf('/')\n return index > 0 ? trimmed.slice(0, index) : '/'\n}\n\nexport function MessagesView() {\n const now = useNow()\n const messages = useRelayStore((s) => s.messages)\n const agentsById = useRelayStore((s) => s.agentsById)\n const selectedAgent = useRelayStore((s) => s.selectedAgent)\n const channelFilter = useRelayStore((s) => s.channelFilter)\n const tagFilter = useRelayStore((s) => s.tagFilter)\n const replyTo = useRelayStore((s) => s.replyTo)\n const set = useRelayStore((s) => s.set)\n const openCompose = useRelayStore((s) => s.openCompose)\n const startReply = useRelayStore((s) => s.startReply)\n const doClaim = useRelayStore((s) => s.doClaim)\n const doDeliveryAction = useRelayStore((s) => s.doDeliveryAction)\n const doDeleteMessage = useRelayStore((s) => s.doDeleteMessage)\n const openThread = useRelayStore((s) => s.openThread)\n const cancelReply = useRelayStore((s) => s.cancelReply)\n\n const composeAgents = useComposeAgents()\n const uniqueTags = useUniqueTags()\n\n const filtered = useMemo(() => {\n let list = [...messages].sort((a, b) => b.id - a.id)\n if (selectedAgent) list = list.filter((m) => m.from === selectedAgent || m.to === selectedAgent)\n if (channelFilter) list = list.filter((m) => m.channel === channelFilter)\n if (tagFilter) list = list.filter((m) => {\n const agent = agentsById[m.from] || agentsById[m.to]\n return (agent?.tags || []).includes(tagFilter)\n })\n return list\n }, [messages, selectedAgent, channelFilter, tagFilter, agentsById])\n\n // Group by threadId — show root messages, collapse replies\n const grouped = useMemo(() => {\n const roots: Message[] = []\n const threads = new Map<number, Message[]>()\n for (const msg of filtered) {\n const tid = msg.threadId ?? msg.id\n if (!threads.has(tid)) threads.set(tid, [])\n threads.get(tid)!.push(msg)\n if (msg.id === tid) roots.push(msg)\n }\n // messages without a known root\n const seenRoots = new Set(roots.map((r) => r.id))\n for (const msg of filtered) {\n if (!seenRoots.has(msg.threadId ?? msg.id) && !roots.find((r) => r.id === msg.id)) {\n roots.push(msg)\n }\n }\n roots.sort((a, b) => b.id - a.id)\n return roots.map((r) => ({ root: r, replies: (threads.get(r.id) || []).filter((m) => m.id !== r.id).sort((a, b) => a.id - b.id) }))\n }, [filtered])\n\n function agentLabel(id: string) {\n return agentsById[id] ? displayName(agentsById[id]) : id.slice(-10)\n }\n\n const uniqueChannels = useMemo(() => {\n return [...new Set(messages.map((m) => m.channel).filter(Boolean) as string[])]\n }, [messages])\n\n return (\n <div className=\"space-y-4\">\n <div className=\"flex items-center justify-between flex-wrap gap-2\">\n <div className=\"flex items-center gap-2\">\n <Mail className=\"w-5 h-5 text-muted-foreground\" />\n <h2 className=\"text-lg font-semibold\">Messages</h2>\n <Badge variant=\"secondary\">{filtered.length}</Badge>\n </div>\n <Button size=\"sm\" onClick={openCompose}>\n <Pencil className=\"w-4 h-4 mr-1\" /> Compose\n </Button>\n </div>\n\n <div className=\"flex flex-wrap gap-2\">\n <select\n className=\"rounded-md border border-input bg-background px-3 py-1.5 text-sm\"\n value={selectedAgent}\n onChange={(e) => set({ selectedAgent: e.target.value })}\n >\n <option value=\"\">All agents</option>\n {composeAgents.map((a) => <option key={a.id} value={a.id}>{displayName(a)}</option>)}\n </select>\n <select\n className=\"rounded-md border border-input bg-background px-3 py-1.5 text-sm\"\n value={tagFilter}\n onChange={(e) => set({ tagFilter: e.target.value })}\n >\n <option value=\"\">All tags</option>\n {uniqueTags.map((t) => <option key={t} value={t}>#{t}</option>)}\n </select>\n <select\n className=\"rounded-md border border-input bg-background px-3 py-1.5 text-sm\"\n value={channelFilter}\n onChange={(e) => set({ channelFilter: e.target.value })}\n >\n <option value=\"\">All channels</option>\n {uniqueChannels.map((c) => <option key={c} value={c}>{c}</option>)}\n </select>\n </div>\n\n {replyTo && (\n <div className=\"flex items-center gap-2 rounded-md bg-blue-500/10 border border-blue-500/20 px-3 py-2 text-sm text-blue-300\">\n <MessageSquareReply className=\"w-4 h-4 shrink-0\" />\n <span>Replying to message #{replyTo.id} from <strong>{agentLabel(replyTo.from)}</strong></span>\n <Button variant=\"ghost\" size=\"sm\" className=\"ml-auto h-6 px-1.5\" onClick={cancelReply}>\n <X className=\"w-3.5 h-3.5\" />\n </Button>\n </div>\n )}\n\n <div className=\"h-[calc(100dvh-15rem)] overflow-y-auto\">\n <div className=\"space-y-2 pr-2\">\n {grouped.length === 0 && (\n <div className=\"text-center text-muted-foreground py-16 text-sm\">No messages</div>\n )}\n {grouped.map(({ root, replies }) => (\n <div key={root.id} className=\"space-y-1\">\n <MessageRow\n msg={root}\n now={now}\n agentLabel={agentLabel}\n agentsById={agentsById}\n onReply={() => startReply(root)}\n onClaim={() => doClaim(root.id)}\n onRetry={() => doDeliveryAction(root.id, 'retry-now')}\n onMarkDead={() => doDeliveryAction(root.id, 'mark-dead', 'Marked dead from dashboard')}\n onClear={() => doDeliveryAction(root.id, 'clear', 'Cleared after manual handling')}\n onDelete={() => doDeleteMessage(root.id)}\n onThread={root.threadId ? () => openThread(root.threadId!) : undefined}\n replyCount={replies.length}\n />\n {replies.map((r) => (\n <div key={r.id} className=\"ml-6\">\n <MessageRow\n msg={r}\n now={now}\n agentLabel={agentLabel}\n agentsById={agentsById}\n onReply={() => startReply(r)}\n onClaim={() => doClaim(r.id)}\n onRetry={() => doDeliveryAction(r.id, 'retry-now')}\n onMarkDead={() => doDeliveryAction(r.id, 'mark-dead', 'Marked dead from dashboard')}\n onClear={() => doDeliveryAction(r.id, 'clear', 'Cleared after manual handling')}\n onDelete={() => doDeleteMessage(r.id)}\n />\n </div>\n ))}\n </div>\n ))}\n </div>\n </div>\n </div>\n )\n}\n\nfunction MessageRow({\n msg, now, agentLabel, agentsById, onReply, onClaim, onRetry, onMarkDead, onClear, onDelete, onThread, replyCount,\n}: {\n msg: Message\n now: number\n agentLabel: (id: string) => string\n agentsById: Record<string, import('@/types').Agent>\n onReply: () => void\n onClaim: () => void\n onRetry: () => void\n onMarkDead: () => void\n onClear: () => void\n onDelete: () => void\n onThread?: () => void\n replyCount?: number\n}) {\n const [expanded, setExpanded] = useState(false)\n const body = messageBody(msg)\n const openFilesAt = useRelayStore((s) => s.openFilesAt)\n const deliveryProblem = msg.deliveryStatus === 'queued' || msg.deliveryStatus === 'failed' || msg.deliveryStatus === 'dead' || Boolean(msg.deliveryLastError || msg.deliveryPoisonReason)\n const peer = msg.from === 'user' ? msg.to : msg.from\n const peerAgent = agentsById[peer]\n\n function resolvedReferencedPath(path: string): string {\n const cwd = typeof peerAgent?.meta?.cwd === 'string' ? peerAgent.meta.cwd : ''\n return resolveRelativePath(cwd || '/', path)\n }\n\n function openReferencedPath(path: string, line?: number) {\n const absolute = resolvedReferencedPath(path)\n void openFilesAt({ path: parentPath(absolute), selectedPath: absolute, line, overlay: true })\n }\n\n return (\n <Card className=\"group relative\">\n <CardContent className=\"p-3\">\n <div className=\"min-w-0\">\n <div className=\"flex items-center gap-2 flex-wrap text-xs text-muted-foreground mb-1\">\n <span className=\"font-medium text-foreground\">{agentLabel(msg.from)}</span>\n <span>→</span>\n <span>{displayTarget(msg.to, agentsById)}</span>\n {msg.channel && (\n <Badge variant=\"outline\" className=\"text-xs px-1.5 py-0 gap-0.5\">\n <Hash className=\"w-2.5 h-2.5\" />{msg.channel}\n </Badge>\n )}\n {msg.claimedBy && (\n <Badge variant=\"outline\" className=\"text-xs px-1.5 py-0 text-emerald-400 border-emerald-500/30\">\n <CheckCircle2 className=\"w-2.5 h-2.5 mr-0.5\" />claimed\n </Badge>\n )}\n {deliveryProblem && <DeliveryBadge msg={msg} now={now} />}\n {msg.subject && <span className=\"font-medium text-foreground\">{msg.subject}</span>}\n <span className=\"ml-auto shrink-0\" title={fmtTime(msg.createdAt)}><span className=\"opacity-40 mr-1\">#{msg.id}</span>{timeAgo(now, msg.createdAt)}</span>\n </div>\n {body && (\n <div className=\"cursor-pointer\" onClick={() => setExpanded((v) => !v)}>\n <FormattedBody text={body} rawBody={msg.body} className={`text-sm text-muted-foreground ${expanded ? '' : 'max-h-[3.75rem] overflow-hidden'}`} onOpenPath={openReferencedPath} resolvePathTitle={(path, line) => `${resolvedReferencedPath(path)}${line ? `:${line}` : ''}`} />\n </div>\n )}\n </div>\n <div className=\"absolute top-2 right-2 flex items-center gap-1 opacity-0 group-hover:opacity-100 transition-opacity bg-card rounded-md shadow-sm border border-border px-1 py-0.5\">\n <Button variant=\"ghost\" size=\"sm\" className=\"h-7 px-2 text-xs\" onClick={onReply} title=\"Reply\">\n <MessageSquareReply className=\"w-3.5 h-3.5\" />\n </Button>\n {msg.claimable && !msg.claimedBy && (\n <Button variant=\"ghost\" size=\"sm\" className=\"h-7 px-2 text-xs text-orange-400 hover:text-orange-300\" onClick={onClaim} title=\"Claim\">\n <CheckCircle2 className=\"w-3.5 h-3.5\" />\n </Button>\n )}\n {deliveryProblem && (\n <Button variant=\"ghost\" size=\"sm\" className=\"h-7 px-2 text-xs text-blue-400 hover:text-blue-300\" onClick={onRetry} title=\"Retry delivery now\">\n <RotateCcw className=\"w-3.5 h-3.5\" />\n </Button>\n )}\n {msg.deliveryStatus !== 'dead' && deliveryProblem && (\n <Button variant=\"ghost\" size=\"sm\" className=\"h-7 px-2 text-xs text-red-400 hover:text-red-300\" onClick={onMarkDead} title=\"Mark dead\">\n <Ban className=\"w-3.5 h-3.5\" />\n </Button>\n )}\n {(msg.deliveryStatus === 'failed' || msg.deliveryStatus === 'dead') && (\n <Button variant=\"ghost\" size=\"sm\" className=\"h-7 px-2 text-xs text-emerald-400 hover:text-emerald-300\" onClick={onClear} title=\"Clear after manual handling\">\n <CheckCircle2 className=\"w-3.5 h-3.5\" />\n </Button>\n )}\n {onThread && (replyCount ?? 0) > 0 && (\n <Button variant=\"ghost\" size=\"sm\" className=\"h-7 px-2 text-xs\" onClick={onThread} title=\"Thread\">\n +{replyCount}\n </Button>\n )}\n <Button variant=\"ghost\" size=\"sm\" className=\"h-7 px-2 text-xs text-red-400 hover:text-red-300\" onClick={onDelete} title=\"Delete\">\n <Trash2 className=\"w-3.5 h-3.5\" />\n </Button>\n </div>\n </CardContent>\n </Card>\n )\n}\n\nfunction DeliveryBadge({ msg, now }: { msg: Message; now: number }) {\n const status = msg.deliveryStatus || 'pending'\n const detail = msg.deliveryPoisonReason || msg.deliveryLastError || (msg.deliveryNextRetryAt ? `retry ${timeAgo(now, msg.deliveryNextRetryAt)}` : undefined)\n const className = status === 'dead'\n ? 'text-red-300 border-red-500/30 bg-red-500/10'\n : status === 'failed'\n ? 'text-amber-300 border-amber-500/30 bg-amber-500/10'\n : 'text-blue-300 border-blue-500/30 bg-blue-500/10'\n return (\n <Badge variant=\"outline\" className={`text-xs px-1.5 py-0 ${className}`} title={detail || status}>\n <AlertTriangle className=\"w-2.5 h-2.5 mr-0.5\" />\n {status}{msg.deliveryAttempts ? ` ${msg.deliveryAttempts}x` : ''}\n </Badge>\n )\n}\n"],"mappings":"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"}
|
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
import{I as e,P as t,Qt as n,T as r,k as i,kn as a,p as o,zt as s}from"./lucide-react-CD8Xl2U3.js";import{i as c,t as l}from"./store-
|
|
2
|
-
//# sourceMappingURL=orchestrators-
|
|
1
|
+
import{I as e,P as t,Qt as n,T as r,k as i,kn as a,p as o,zt as s}from"./lucide-react-CD8Xl2U3.js";import{i as c,t as l}from"./store-CICRhg1m.js";import{A as u,k as d,z as f}from"./display-Bebqs1qu.js";import{t as p}from"./badge-t8zAwHW9.js";import{t as m}from"./button-DDA5P2YQ.js";import{D as h}from"./index-CvSlyTSI.js";import{i as g,n as _,r as v,t as y}from"./card-CggxP1h9.js";var b=a();function x({agent:e,orch:n}){let r=l(e=>e.orchestratorAction),a=l(e=>e.openFilesAt);return(0,b.jsxs)(`div`,{className:`flex items-center gap-2 py-1.5`,children:[(0,b.jsx)(s,{className:`w-3.5 h-3.5 text-muted-foreground shrink-0`}),(0,b.jsxs)(`div`,{className:`flex-1 min-w-0`,children:[(0,b.jsx)(`div`,{className:`text-xs font-medium truncate`,children:e.label||e.agentId.slice(-12)}),(0,b.jsxs)(`button`,{type:`button`,className:`max-w-full truncate text-left text-[10px] text-muted-foreground hover:text-foreground`,title:`Open ${e.cwd||n.baseDir} in Files`,onClick:t=>{t.stopPropagation(),a({orchestratorId:n.id,path:e.cwd||n.baseDir})},children:[e.provider,` · `,e.cwd||`~`]})]}),(0,b.jsxs)(`div`,{className:`flex gap-1`,children:[(0,b.jsx)(m,{size:`icon`,variant:`ghost`,className:`h-6 w-6`,title:`Restart agent`,onClick:t=>{t.stopPropagation(),r(n.id,`restart`,e.agentId,`orchestrators`)},children:(0,b.jsx)(i,{className:`w-3 h-3`})}),(0,b.jsx)(m,{size:`icon`,variant:`ghost`,className:`h-6 w-6 text-red-400 hover:text-red-300`,title:`Shutdown agent`,onClick:t=>{t.stopPropagation(),r(n.id,`shutdown`,e.agentId,`orchestrators`)},children:(0,b.jsx)(t,{className:`w-3 h-3`})})]})]})}function S({orch:a}){let s=c(),S=l(e=>e.orchestratorAction),C=l(e=>e.upgradeOrchestrator),w=l(e=>e.stats.version),T=l(e=>e.openOrchestratorSpawnFor),E=l(e=>e.openFilesAt),D=l(e=>e.deleteOrchestrator),O=l(e=>e.refreshOrchestratorProviders),k=u(a),A=d(a),j=a.status===`online`,M=a.package?.version||a.version,N=a.upgrade,P=N?.status===`pending`,F=j&&!!a.version&&!!w&&a.version!==w,I=j&&!P&&(F||N?.status===`failed`),L=Object.entries(a.contracts||{}).filter(([,e])=>typeof e==`number`),R=a.providerStatus||a.providers.map(e=>({name:e,available:!0,checkedAt:Number(a.lastSeen)||Date.now()})),z=a.managedAgents.map(e=>e.workspace?{agent:e,workspace:e.workspace}:null).filter(e=>!!(e?.workspace?.id||e?.workspace?.worktreePath));return(0,b.jsxs)(y,{className:`${j?``:`opacity-60`}`,children:[(0,b.jsx)(v,{className:`pb-2 pt-4 px-4`,children:(0,b.jsxs)(`div`,{className:`flex items-start gap-2`,children:[(0,b.jsx)(`div`,{className:`p-1.5 rounded-md ${j?`bg-emerald-500/10`:`bg-zinc-800`}`,children:(0,b.jsx)(r,{className:`w-4 h-4 ${j?`text-emerald-400`:`text-zinc-500`}`})}),(0,b.jsxs)(`div`,{className:`flex-1 min-w-0`,children:[(0,b.jsx)(g,{className:`text-sm font-medium truncate`,children:a.hostname}),(0,b.jsxs)(`div`,{className:`text-xs text-muted-foreground mt-0.5`,children:[a.id.slice(0,16),`…`]})]}),(0,b.jsx)(p,{className:`text-[10px] shrink-0 ${A}`,children:k})]})}),(0,b.jsxs)(_,{className:`px-4 pb-4 space-y-3`,children:[(0,b.jsxs)(`div`,{className:`flex flex-wrap gap-x-4 gap-y-1 text-xs text-muted-foreground`,children:[(0,b.jsx)(`span`,{className:j?`text-emerald-400`:`text-zinc-500`,children:a.status}),M&&(0,b.jsxs)(`span`,{children:[a.package?.name||`orchestrator`,` v`,M]}),P&&(0,b.jsxs)(p,{className:`text-[10px] bg-amber-500/10 text-amber-400 border-amber-500/20 gap-1`,children:[(0,b.jsx)(n,{className:`w-3 h-3 animate-pulse`}),`upgrading → v`,N.desiredVersion]}),!P&&N?.status===`failed`&&(0,b.jsxs)(p,{className:`text-[10px] bg-red-500/10 text-red-400 border-red-500/20 gap-1`,title:N.error,children:[(0,b.jsx)(o,{className:`w-3 h-3`}),`upgrade failed`]}),!P&&F&&(0,b.jsxs)(p,{className:`text-[10px] bg-blue-500/10 text-blue-400 border-blue-500/20`,children:[`→ v`,w,` available`]}),(0,b.jsx)(`span`,{className:`ml-auto`,children:f(s,a.lastSeen)})]}),R.length>0&&(0,b.jsx)(`div`,{className:`flex flex-wrap gap-1`,children:R.map(e=>(0,b.jsx)(p,{variant:`secondary`,title:e.available?`version`in e&&e.version?e.version:e.name:`reason`in e&&e.reason?e.reason:`${e.name} unavailable`,className:`text-[10px] ${e.available?`bg-emerald-500/10 text-emerald-400 border-emerald-500/20`:`bg-red-500/10 text-red-400 border-red-500/20`}`,children:e.name},e.name))}),L.length>0&&(0,b.jsx)(`div`,{className:`flex flex-wrap gap-1`,children:L.map(([e,t])=>(0,b.jsxs)(p,{variant:`secondary`,className:`text-[10px]`,children:[e,`=`,t]},e))}),a.managedAgents.length>0&&(0,b.jsxs)(b.Fragment,{children:[(0,b.jsx)(h,{}),(0,b.jsxs)(`div`,{children:[(0,b.jsxs)(`div`,{className:`text-xs font-medium text-muted-foreground mb-1.5`,children:[`Managed agents (`,a.managedAgents.length,`)`]}),(0,b.jsx)(`div`,{className:`space-y-0`,children:a.managedAgents.map(e=>(0,b.jsx)(x,{agent:e,orch:a},e.agentId))})]})]}),z.length>0&&(0,b.jsxs)(b.Fragment,{children:[(0,b.jsx)(h,{}),(0,b.jsxs)(`div`,{children:[(0,b.jsxs)(`div`,{className:`text-xs font-medium text-muted-foreground mb-1.5`,children:[`Workspaces (`,z.length,`)`]}),(0,b.jsx)(`div`,{className:`space-y-1`,children:z.map(({agent:e,workspace:t})=>(0,b.jsxs)(`button`,{type:`button`,className:`block w-full min-w-0 rounded border border-border/60 px-2 py-1.5 text-left hover:bg-accent/40`,onClick:()=>t.worktreePath&&void E({orchestratorId:a.id,path:t.worktreePath}),children:[(0,b.jsxs)(`div`,{className:`flex items-center gap-2 text-xs`,children:[(0,b.jsx)(`span`,{className:`truncate font-medium`,children:t.branch||e.label||e.agentId.slice(-12)}),t.status&&(0,b.jsx)(p,{variant:`outline`,className:`h-4 shrink-0 text-[10px]`,children:t.status})]}),(0,b.jsx)(`div`,{className:`truncate text-[10px] text-muted-foreground`,children:t.worktreePath||t.repoRoot})]},t.id||t.worktreePath))})]})]}),(0,b.jsx)(h,{}),(0,b.jsxs)(`div`,{className:`flex flex-wrap gap-1.5`,children:[j&&(0,b.jsxs)(m,{size:`sm`,variant:`outline`,className:`h-7 text-xs gap-1`,onClick:()=>T(a.id),children:[(0,b.jsx)(e,{className:`w-3 h-3`}),`Spawn agent`]}),j&&(0,b.jsxs)(m,{size:`sm`,variant:`outline`,className:`h-7 text-xs gap-1`,onClick:()=>O(a.id),children:[(0,b.jsx)(i,{className:`w-3 h-3`}),`Refresh providers`]}),(I||P)&&(0,b.jsxs)(m,{size:`sm`,variant:`outline`,disabled:P,className:`h-7 text-xs gap-1 text-blue-400 hover:text-blue-300 border-blue-500/30 disabled:opacity-60`,title:P?`Upgrading to v${N.desiredVersion}…`:`Upgrade to v${w}`,onClick:()=>C(a.id),children:[(0,b.jsx)(n,{className:`w-3 h-3 ${P?`animate-pulse`:``}`}),P?`Upgrading…`:`Upgrade`]}),j&&(0,b.jsxs)(m,{size:`sm`,variant:`outline`,className:`h-7 text-xs gap-1`,onClick:()=>S(a.id,`restart`,void 0,`orchestrators`),children:[(0,b.jsx)(i,{className:`w-3 h-3`}),`Restart all`]}),j&&(0,b.jsxs)(m,{size:`sm`,variant:`outline`,className:`h-7 text-xs gap-1 text-red-400 hover:text-red-300 border-red-500/30`,onClick:()=>S(a.id,`shutdown`,void 0,`orchestrators`),children:[(0,b.jsx)(t,{className:`w-3 h-3`}),`Shutdown all`]}),(0,b.jsx)(m,{size:`sm`,variant:`ghost`,className:`h-7 text-xs text-muted-foreground hover:text-red-400`,onClick:()=>D(a.id),children:`Remove`})]})]})]})}function C(){let t=l(e=>e.orchestrators),n=l(e=>e.openOrchestratorSpawn),i=l(e=>e.openOrchestratorInstall),a=t.filter(e=>e.status===`online`).length;return(0,b.jsxs)(`div`,{className:`space-y-4`,children:[(0,b.jsxs)(`div`,{className:`flex items-center gap-3`,children:[(0,b.jsxs)(`div`,{children:[(0,b.jsx)(`h2`,{className:`text-base font-semibold`,children:`Orchestrators`}),(0,b.jsx)(`p`,{className:`text-xs text-muted-foreground`,children:`Multi-host agent lifecycle managers`})]}),(0,b.jsxs)(`div`,{className:`flex items-center gap-2 ml-auto`,children:[(0,b.jsxs)(p,{variant:`outline`,className:`text-xs`,children:[a,` online`]}),(0,b.jsxs)(m,{size:`sm`,variant:`outline`,className:`gap-1`,onClick:i,children:[(0,b.jsx)(r,{className:`w-3.5 h-3.5`}),`Add`]}),(0,b.jsxs)(m,{size:`sm`,className:`gap-1`,onClick:n,children:[(0,b.jsx)(e,{className:`w-3.5 h-3.5`}),`Spawn`]})]})]}),t.length===0?(0,b.jsx)(`div`,{className:`text-center py-16 text-muted-foreground text-sm`,children:`No orchestrators registered.`}):(0,b.jsx)(`div`,{className:`grid grid-cols-1 md:grid-cols-2 xl:grid-cols-3 gap-4`,children:t.map(e=>(0,b.jsx)(S,{orch:e},e.id))})]})}export{C as OrchestratorsView};
|
|
2
|
+
//# sourceMappingURL=orchestrators-BkHHgd83.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"orchestrators-CRoZtLeQ.js","names":[],"sources":["../../dashboard/src/components/views/orchestrators.tsx"],"sourcesContent":["import { useRelayStore, useNow } from '@/store'\nimport { Card, CardContent, CardHeader, CardTitle } from '@/components/ui/card'\nimport { Badge } from '@/components/ui/badge'\nimport { Button } from '@/components/ui/button'\nimport { Separator } from '@/components/ui/separator'\nimport { displayName, orchestratorHealthLabel, orchestratorHealthClass, timeAgo } from '@/lib/display'\nimport { Server, Plus, RotateCw, Power, Cpu, ArrowUpCircle, AlertTriangle } from 'lucide-react'\nimport type { Orchestrator, ManagedAgent } from '@/types'\n\nfunction ManagedAgentRow({ agent, orch }: { agent: ManagedAgent; orch: Orchestrator }) {\n const orchestratorAction = useRelayStore((s) => s.orchestratorAction)\n const openFilesAt = useRelayStore((s) => s.openFilesAt)\n\n return (\n <div className=\"flex items-center gap-2 py-1.5\">\n <Cpu className=\"w-3.5 h-3.5 text-muted-foreground shrink-0\" />\n <div className=\"flex-1 min-w-0\">\n <div className=\"text-xs font-medium truncate\">\n {agent.label || agent.agentId.slice(-12)}\n </div>\n <button\n type=\"button\"\n className=\"max-w-full truncate text-left text-[10px] text-muted-foreground hover:text-foreground\"\n title={`Open ${agent.cwd || orch.baseDir} in Files`}\n onClick={(e) => { e.stopPropagation(); void openFilesAt({ orchestratorId: orch.id, path: agent.cwd || orch.baseDir }) }}\n >\n {agent.provider} · {agent.cwd || '~'}\n </button>\n </div>\n <div className=\"flex gap-1\">\n <Button\n size=\"icon\"\n variant=\"ghost\"\n className=\"h-6 w-6\"\n title=\"Restart agent\"\n onClick={(e) => { e.stopPropagation(); orchestratorAction(orch.id, 'restart', agent.agentId, 'orchestrators') }}\n >\n <RotateCw className=\"w-3 h-3\" />\n </Button>\n <Button\n size=\"icon\"\n variant=\"ghost\"\n className=\"h-6 w-6 text-red-400 hover:text-red-300\"\n title=\"Shutdown agent\"\n onClick={(e) => { e.stopPropagation(); orchestratorAction(orch.id, 'shutdown', agent.agentId, 'orchestrators') }}\n >\n <Power className=\"w-3 h-3\" />\n </Button>\n </div>\n </div>\n )\n}\n\nfunction OrchestratorCard({ orch }: { orch: Orchestrator }) {\n const now = useNow()\n const orchestratorAction = useRelayStore((s) => s.orchestratorAction)\n const upgradeOrchestrator = useRelayStore((s) => s.upgradeOrchestrator)\n const fleetTarget = useRelayStore((s) => s.stats.version)\n const openOrchestratorSpawnFor = useRelayStore((s) => s.openOrchestratorSpawnFor)\n const openFilesAt = useRelayStore((s) => s.openFilesAt)\n const deleteOrchestrator = useRelayStore((s) => s.deleteOrchestrator)\n const refreshOrchestratorProviders = useRelayStore((s) => s.refreshOrchestratorProviders)\n\n const healthLabel = orchestratorHealthLabel(orch)\n const healthClass = orchestratorHealthClass(orch)\n const online = orch.status === 'online'\n const packageVersion = orch.package?.version || orch.version\n const upgrade = orch.upgrade\n const upgrading = upgrade?.status === 'pending'\n // Drift = reported version differs from the relay's own version (fleet target).\n const driftAvailable = online && !!orch.version && !!fleetTarget && orch.version !== fleetTarget\n const canUpgrade = online && !upgrading && (driftAvailable || upgrade?.status === 'failed')\n const contractEntries = Object.entries(orch.contracts || {}).filter(([, value]) => typeof value === 'number')\n const providerStatus = orch.providerStatus || orch.providers.map((provider) => ({ name: provider, available: true, checkedAt: Number(orch.lastSeen) || Date.now() }))\n const workspaces = orch.managedAgents\n .map((agent) => agent.workspace ? { agent, workspace: agent.workspace } : null)\n .filter((item): item is { agent: ManagedAgent; workspace: NonNullable<ManagedAgent['workspace']> } => Boolean(item?.workspace?.id || item?.workspace?.worktreePath))\n\n return (\n <Card className={`${!online ? 'opacity-60' : ''}`}>\n <CardHeader className=\"pb-2 pt-4 px-4\">\n <div className=\"flex items-start gap-2\">\n <div className={`p-1.5 rounded-md ${online ? 'bg-emerald-500/10' : 'bg-zinc-800'}`}>\n <Server className={`w-4 h-4 ${online ? 'text-emerald-400' : 'text-zinc-500'}`} />\n </div>\n <div className=\"flex-1 min-w-0\">\n <CardTitle className=\"text-sm font-medium truncate\">{orch.hostname}</CardTitle>\n <div className=\"text-xs text-muted-foreground mt-0.5\">{orch.id.slice(0, 16)}…</div>\n </div>\n <Badge className={`text-[10px] shrink-0 ${healthClass}`}>{healthLabel}</Badge>\n </div>\n </CardHeader>\n\n <CardContent className=\"px-4 pb-4 space-y-3\">\n {/* Meta row */}\n <div className=\"flex flex-wrap gap-x-4 gap-y-1 text-xs text-muted-foreground\">\n <span className={online ? 'text-emerald-400' : 'text-zinc-500'}>{orch.status}</span>\n {packageVersion && <span>{orch.package?.name || 'orchestrator'} v{packageVersion}</span>}\n {upgrading && (\n <Badge className=\"text-[10px] bg-amber-500/10 text-amber-400 border-amber-500/20 gap-1\">\n <ArrowUpCircle className=\"w-3 h-3 animate-pulse\" />upgrading → v{upgrade!.desiredVersion}\n </Badge>\n )}\n {!upgrading && upgrade?.status === 'failed' && (\n <Badge className=\"text-[10px] bg-red-500/10 text-red-400 border-red-500/20 gap-1\" title={upgrade.error}>\n <AlertTriangle className=\"w-3 h-3\" />upgrade failed\n </Badge>\n )}\n {!upgrading && driftAvailable && (\n <Badge className=\"text-[10px] bg-blue-500/10 text-blue-400 border-blue-500/20\">→ v{fleetTarget} available</Badge>\n )}\n <span className=\"ml-auto\">{timeAgo(now, orch.lastSeen)}</span>\n </div>\n\n {providerStatus.length > 0 && (\n <div className=\"flex flex-wrap gap-1\">\n {providerStatus.map((status) => (\n <Badge\n key={status.name}\n variant=\"secondary\"\n title={status.available ? ('version' in status && status.version ? status.version : status.name) : ('reason' in status && status.reason ? status.reason : `${status.name} unavailable`)}\n className={`text-[10px] ${status.available ? 'bg-emerald-500/10 text-emerald-400 border-emerald-500/20' : 'bg-red-500/10 text-red-400 border-red-500/20'}`}\n >\n {status.name}\n </Badge>\n ))}\n </div>\n )}\n\n {contractEntries.length > 0 && (\n <div className=\"flex flex-wrap gap-1\">\n {contractEntries.map(([name, version]) => (\n <Badge key={name} variant=\"secondary\" className=\"text-[10px]\">{name}={version}</Badge>\n ))}\n </div>\n )}\n\n {/* Managed agents */}\n {orch.managedAgents.length > 0 && (\n <>\n <Separator />\n <div>\n <div className=\"text-xs font-medium text-muted-foreground mb-1.5\">\n Managed agents ({orch.managedAgents.length})\n </div>\n <div className=\"space-y-0\">\n {orch.managedAgents.map((a) => (\n <ManagedAgentRow key={a.agentId} agent={a} orch={orch} />\n ))}\n </div>\n </div>\n </>\n )}\n\n {workspaces.length > 0 && (\n <>\n <Separator />\n <div>\n <div className=\"text-xs font-medium text-muted-foreground mb-1.5\">\n Workspaces ({workspaces.length})\n </div>\n <div className=\"space-y-1\">\n {workspaces.map(({ agent, workspace }) => (\n <button\n key={workspace.id || workspace.worktreePath}\n type=\"button\"\n className=\"block w-full min-w-0 rounded border border-border/60 px-2 py-1.5 text-left hover:bg-accent/40\"\n onClick={() => workspace.worktreePath && void openFilesAt({ orchestratorId: orch.id, path: workspace.worktreePath })}\n >\n <div className=\"flex items-center gap-2 text-xs\">\n <span className=\"truncate font-medium\">{workspace.branch || agent.label || agent.agentId.slice(-12)}</span>\n {workspace.status && <Badge variant=\"outline\" className=\"h-4 shrink-0 text-[10px]\">{workspace.status}</Badge>}\n </div>\n <div className=\"truncate text-[10px] text-muted-foreground\">{workspace.worktreePath || workspace.repoRoot}</div>\n </button>\n ))}\n </div>\n </div>\n </>\n )}\n\n {/* Actions */}\n <Separator />\n <div className=\"flex flex-wrap gap-1.5\">\n {online && (\n <Button size=\"sm\" variant=\"outline\" className=\"h-7 text-xs gap-1\" onClick={() => openOrchestratorSpawnFor(orch.id)}>\n <Plus className=\"w-3 h-3\" />\n Spawn agent\n </Button>\n )}\n {online && (\n <Button\n size=\"sm\"\n variant=\"outline\"\n className=\"h-7 text-xs gap-1\"\n onClick={() => refreshOrchestratorProviders(orch.id)}\n >\n <RotateCw className=\"w-3 h-3\" />\n Refresh providers\n </Button>\n )}\n {(canUpgrade || upgrading) && (\n <Button\n size=\"sm\"\n variant=\"outline\"\n disabled={upgrading}\n className=\"h-7 text-xs gap-1 text-blue-400 hover:text-blue-300 border-blue-500/30 disabled:opacity-60\"\n title={upgrading ? `Upgrading to v${upgrade!.desiredVersion}…` : `Upgrade to v${fleetTarget}`}\n onClick={() => upgradeOrchestrator(orch.id)}\n >\n <ArrowUpCircle className={`w-3 h-3 ${upgrading ? 'animate-pulse' : ''}`} />\n {upgrading ? 'Upgrading…' : 'Upgrade'}\n </Button>\n )}\n {online && (\n <Button\n size=\"sm\"\n variant=\"outline\"\n className=\"h-7 text-xs gap-1\"\n onClick={() => orchestratorAction(orch.id, 'restart', undefined, 'orchestrators')}\n >\n <RotateCw className=\"w-3 h-3\" />\n Restart all\n </Button>\n )}\n {online && (\n <Button\n size=\"sm\"\n variant=\"outline\"\n className=\"h-7 text-xs gap-1 text-red-400 hover:text-red-300 border-red-500/30\"\n onClick={() => orchestratorAction(orch.id, 'shutdown', undefined, 'orchestrators')}\n >\n <Power className=\"w-3 h-3\" />\n Shutdown all\n </Button>\n )}\n <Button\n size=\"sm\"\n variant=\"ghost\"\n className=\"h-7 text-xs text-muted-foreground hover:text-red-400\"\n onClick={() => deleteOrchestrator(orch.id)}\n >\n Remove\n </Button>\n </div>\n </CardContent>\n </Card>\n )\n}\n\nexport function OrchestratorsView() {\n const orchestrators = useRelayStore((s) => s.orchestrators)\n const openOrchestratorSpawn = useRelayStore((s) => s.openOrchestratorSpawn)\n const openOrchestratorInstall = useRelayStore((s) => s.openOrchestratorInstall)\n\n const onlineCount = orchestrators.filter((o) => o.status === 'online').length\n\n return (\n <div className=\"space-y-4\">\n {/* Header */}\n <div className=\"flex items-center gap-3\">\n <div>\n <h2 className=\"text-base font-semibold\">Orchestrators</h2>\n <p className=\"text-xs text-muted-foreground\">Multi-host agent lifecycle managers</p>\n </div>\n <div className=\"flex items-center gap-2 ml-auto\">\n <Badge variant=\"outline\" className=\"text-xs\">\n {onlineCount} online\n </Badge>\n <Button size=\"sm\" variant=\"outline\" className=\"gap-1\" onClick={openOrchestratorInstall}>\n <Server className=\"w-3.5 h-3.5\" />\n Add\n </Button>\n <Button size=\"sm\" className=\"gap-1\" onClick={openOrchestratorSpawn}>\n <Plus className=\"w-3.5 h-3.5\" />\n Spawn\n </Button>\n </div>\n </div>\n\n {/* Cards */}\n {orchestrators.length === 0 ? (\n <div className=\"text-center py-16 text-muted-foreground text-sm\">\n No orchestrators registered.\n </div>\n ) : (\n <div className=\"grid grid-cols-1 md:grid-cols-2 xl:grid-cols-3 gap-4\">\n {orchestrators.map((orch) => (\n <OrchestratorCard key={orch.id} orch={orch} />\n ))}\n </div>\n )}\n </div>\n )\n}\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"orchestrators-BkHHgd83.js","names":[],"sources":["../../dashboard/src/components/views/orchestrators.tsx"],"sourcesContent":["import { useRelayStore, useNow } from '@/store'\nimport { Card, CardContent, CardHeader, CardTitle } from '@/components/ui/card'\nimport { Badge } from '@/components/ui/badge'\nimport { Button } from '@/components/ui/button'\nimport { Separator } from '@/components/ui/separator'\nimport { displayName, orchestratorHealthLabel, orchestratorHealthClass, timeAgo } from '@/lib/display'\nimport { Server, Plus, RotateCw, Power, Cpu, ArrowUpCircle, AlertTriangle } from 'lucide-react'\nimport type { Orchestrator, ManagedAgent } from '@/types'\n\nfunction ManagedAgentRow({ agent, orch }: { agent: ManagedAgent; orch: Orchestrator }) {\n const orchestratorAction = useRelayStore((s) => s.orchestratorAction)\n const openFilesAt = useRelayStore((s) => s.openFilesAt)\n\n return (\n <div className=\"flex items-center gap-2 py-1.5\">\n <Cpu className=\"w-3.5 h-3.5 text-muted-foreground shrink-0\" />\n <div className=\"flex-1 min-w-0\">\n <div className=\"text-xs font-medium truncate\">\n {agent.label || agent.agentId.slice(-12)}\n </div>\n <button\n type=\"button\"\n className=\"max-w-full truncate text-left text-[10px] text-muted-foreground hover:text-foreground\"\n title={`Open ${agent.cwd || orch.baseDir} in Files`}\n onClick={(e) => { e.stopPropagation(); void openFilesAt({ orchestratorId: orch.id, path: agent.cwd || orch.baseDir }) }}\n >\n {agent.provider} · {agent.cwd || '~'}\n </button>\n </div>\n <div className=\"flex gap-1\">\n <Button\n size=\"icon\"\n variant=\"ghost\"\n className=\"h-6 w-6\"\n title=\"Restart agent\"\n onClick={(e) => { e.stopPropagation(); orchestratorAction(orch.id, 'restart', agent.agentId, 'orchestrators') }}\n >\n <RotateCw className=\"w-3 h-3\" />\n </Button>\n <Button\n size=\"icon\"\n variant=\"ghost\"\n className=\"h-6 w-6 text-red-400 hover:text-red-300\"\n title=\"Shutdown agent\"\n onClick={(e) => { e.stopPropagation(); orchestratorAction(orch.id, 'shutdown', agent.agentId, 'orchestrators') }}\n >\n <Power className=\"w-3 h-3\" />\n </Button>\n </div>\n </div>\n )\n}\n\nfunction OrchestratorCard({ orch }: { orch: Orchestrator }) {\n const now = useNow()\n const orchestratorAction = useRelayStore((s) => s.orchestratorAction)\n const upgradeOrchestrator = useRelayStore((s) => s.upgradeOrchestrator)\n const fleetTarget = useRelayStore((s) => s.stats.version)\n const openOrchestratorSpawnFor = useRelayStore((s) => s.openOrchestratorSpawnFor)\n const openFilesAt = useRelayStore((s) => s.openFilesAt)\n const deleteOrchestrator = useRelayStore((s) => s.deleteOrchestrator)\n const refreshOrchestratorProviders = useRelayStore((s) => s.refreshOrchestratorProviders)\n\n const healthLabel = orchestratorHealthLabel(orch)\n const healthClass = orchestratorHealthClass(orch)\n const online = orch.status === 'online'\n const packageVersion = orch.package?.version || orch.version\n const upgrade = orch.upgrade\n const upgrading = upgrade?.status === 'pending'\n // Drift = reported version differs from the relay's own version (fleet target).\n const driftAvailable = online && !!orch.version && !!fleetTarget && orch.version !== fleetTarget\n const canUpgrade = online && !upgrading && (driftAvailable || upgrade?.status === 'failed')\n const contractEntries = Object.entries(orch.contracts || {}).filter(([, value]) => typeof value === 'number')\n const providerStatus = orch.providerStatus || orch.providers.map((provider) => ({ name: provider, available: true, checkedAt: Number(orch.lastSeen) || Date.now() }))\n const workspaces = orch.managedAgents\n .map((agent) => agent.workspace ? { agent, workspace: agent.workspace } : null)\n .filter((item): item is { agent: ManagedAgent; workspace: NonNullable<ManagedAgent['workspace']> } => Boolean(item?.workspace?.id || item?.workspace?.worktreePath))\n\n return (\n <Card className={`${!online ? 'opacity-60' : ''}`}>\n <CardHeader className=\"pb-2 pt-4 px-4\">\n <div className=\"flex items-start gap-2\">\n <div className={`p-1.5 rounded-md ${online ? 'bg-emerald-500/10' : 'bg-zinc-800'}`}>\n <Server className={`w-4 h-4 ${online ? 'text-emerald-400' : 'text-zinc-500'}`} />\n </div>\n <div className=\"flex-1 min-w-0\">\n <CardTitle className=\"text-sm font-medium truncate\">{orch.hostname}</CardTitle>\n <div className=\"text-xs text-muted-foreground mt-0.5\">{orch.id.slice(0, 16)}…</div>\n </div>\n <Badge className={`text-[10px] shrink-0 ${healthClass}`}>{healthLabel}</Badge>\n </div>\n </CardHeader>\n\n <CardContent className=\"px-4 pb-4 space-y-3\">\n {/* Meta row */}\n <div className=\"flex flex-wrap gap-x-4 gap-y-1 text-xs text-muted-foreground\">\n <span className={online ? 'text-emerald-400' : 'text-zinc-500'}>{orch.status}</span>\n {packageVersion && <span>{orch.package?.name || 'orchestrator'} v{packageVersion}</span>}\n {upgrading && (\n <Badge className=\"text-[10px] bg-amber-500/10 text-amber-400 border-amber-500/20 gap-1\">\n <ArrowUpCircle className=\"w-3 h-3 animate-pulse\" />upgrading → v{upgrade!.desiredVersion}\n </Badge>\n )}\n {!upgrading && upgrade?.status === 'failed' && (\n <Badge className=\"text-[10px] bg-red-500/10 text-red-400 border-red-500/20 gap-1\" title={upgrade.error}>\n <AlertTriangle className=\"w-3 h-3\" />upgrade failed\n </Badge>\n )}\n {!upgrading && driftAvailable && (\n <Badge className=\"text-[10px] bg-blue-500/10 text-blue-400 border-blue-500/20\">→ v{fleetTarget} available</Badge>\n )}\n <span className=\"ml-auto\">{timeAgo(now, orch.lastSeen)}</span>\n </div>\n\n {providerStatus.length > 0 && (\n <div className=\"flex flex-wrap gap-1\">\n {providerStatus.map((status) => (\n <Badge\n key={status.name}\n variant=\"secondary\"\n title={status.available ? ('version' in status && status.version ? status.version : status.name) : ('reason' in status && status.reason ? status.reason : `${status.name} unavailable`)}\n className={`text-[10px] ${status.available ? 'bg-emerald-500/10 text-emerald-400 border-emerald-500/20' : 'bg-red-500/10 text-red-400 border-red-500/20'}`}\n >\n {status.name}\n </Badge>\n ))}\n </div>\n )}\n\n {contractEntries.length > 0 && (\n <div className=\"flex flex-wrap gap-1\">\n {contractEntries.map(([name, version]) => (\n <Badge key={name} variant=\"secondary\" className=\"text-[10px]\">{name}={version}</Badge>\n ))}\n </div>\n )}\n\n {/* Managed agents */}\n {orch.managedAgents.length > 0 && (\n <>\n <Separator />\n <div>\n <div className=\"text-xs font-medium text-muted-foreground mb-1.5\">\n Managed agents ({orch.managedAgents.length})\n </div>\n <div className=\"space-y-0\">\n {orch.managedAgents.map((a) => (\n <ManagedAgentRow key={a.agentId} agent={a} orch={orch} />\n ))}\n </div>\n </div>\n </>\n )}\n\n {workspaces.length > 0 && (\n <>\n <Separator />\n <div>\n <div className=\"text-xs font-medium text-muted-foreground mb-1.5\">\n Workspaces ({workspaces.length})\n </div>\n <div className=\"space-y-1\">\n {workspaces.map(({ agent, workspace }) => (\n <button\n key={workspace.id || workspace.worktreePath}\n type=\"button\"\n className=\"block w-full min-w-0 rounded border border-border/60 px-2 py-1.5 text-left hover:bg-accent/40\"\n onClick={() => workspace.worktreePath && void openFilesAt({ orchestratorId: orch.id, path: workspace.worktreePath })}\n >\n <div className=\"flex items-center gap-2 text-xs\">\n <span className=\"truncate font-medium\">{workspace.branch || agent.label || agent.agentId.slice(-12)}</span>\n {workspace.status && <Badge variant=\"outline\" className=\"h-4 shrink-0 text-[10px]\">{workspace.status}</Badge>}\n </div>\n <div className=\"truncate text-[10px] text-muted-foreground\">{workspace.worktreePath || workspace.repoRoot}</div>\n </button>\n ))}\n </div>\n </div>\n </>\n )}\n\n {/* Actions */}\n <Separator />\n <div className=\"flex flex-wrap gap-1.5\">\n {online && (\n <Button size=\"sm\" variant=\"outline\" className=\"h-7 text-xs gap-1\" onClick={() => openOrchestratorSpawnFor(orch.id)}>\n <Plus className=\"w-3 h-3\" />\n Spawn agent\n </Button>\n )}\n {online && (\n <Button\n size=\"sm\"\n variant=\"outline\"\n className=\"h-7 text-xs gap-1\"\n onClick={() => refreshOrchestratorProviders(orch.id)}\n >\n <RotateCw className=\"w-3 h-3\" />\n Refresh providers\n </Button>\n )}\n {(canUpgrade || upgrading) && (\n <Button\n size=\"sm\"\n variant=\"outline\"\n disabled={upgrading}\n className=\"h-7 text-xs gap-1 text-blue-400 hover:text-blue-300 border-blue-500/30 disabled:opacity-60\"\n title={upgrading ? `Upgrading to v${upgrade!.desiredVersion}…` : `Upgrade to v${fleetTarget}`}\n onClick={() => upgradeOrchestrator(orch.id)}\n >\n <ArrowUpCircle className={`w-3 h-3 ${upgrading ? 'animate-pulse' : ''}`} />\n {upgrading ? 'Upgrading…' : 'Upgrade'}\n </Button>\n )}\n {online && (\n <Button\n size=\"sm\"\n variant=\"outline\"\n className=\"h-7 text-xs gap-1\"\n onClick={() => orchestratorAction(orch.id, 'restart', undefined, 'orchestrators')}\n >\n <RotateCw className=\"w-3 h-3\" />\n Restart all\n </Button>\n )}\n {online && (\n <Button\n size=\"sm\"\n variant=\"outline\"\n className=\"h-7 text-xs gap-1 text-red-400 hover:text-red-300 border-red-500/30\"\n onClick={() => orchestratorAction(orch.id, 'shutdown', undefined, 'orchestrators')}\n >\n <Power className=\"w-3 h-3\" />\n Shutdown all\n </Button>\n )}\n <Button\n size=\"sm\"\n variant=\"ghost\"\n className=\"h-7 text-xs text-muted-foreground hover:text-red-400\"\n onClick={() => deleteOrchestrator(orch.id)}\n >\n Remove\n </Button>\n </div>\n </CardContent>\n </Card>\n )\n}\n\nexport function OrchestratorsView() {\n const orchestrators = useRelayStore((s) => s.orchestrators)\n const openOrchestratorSpawn = useRelayStore((s) => s.openOrchestratorSpawn)\n const openOrchestratorInstall = useRelayStore((s) => s.openOrchestratorInstall)\n\n const onlineCount = orchestrators.filter((o) => o.status === 'online').length\n\n return (\n <div className=\"space-y-4\">\n {/* Header */}\n <div className=\"flex items-center gap-3\">\n <div>\n <h2 className=\"text-base font-semibold\">Orchestrators</h2>\n <p className=\"text-xs text-muted-foreground\">Multi-host agent lifecycle managers</p>\n </div>\n <div className=\"flex items-center gap-2 ml-auto\">\n <Badge variant=\"outline\" className=\"text-xs\">\n {onlineCount} online\n </Badge>\n <Button size=\"sm\" variant=\"outline\" className=\"gap-1\" onClick={openOrchestratorInstall}>\n <Server className=\"w-3.5 h-3.5\" />\n Add\n </Button>\n <Button size=\"sm\" className=\"gap-1\" onClick={openOrchestratorSpawn}>\n <Plus className=\"w-3.5 h-3.5\" />\n Spawn\n </Button>\n </div>\n </div>\n\n {/* Cards */}\n {orchestrators.length === 0 ? (\n <div className=\"text-center py-16 text-muted-foreground text-sm\">\n No orchestrators registered.\n </div>\n ) : (\n <div className=\"grid grid-cols-1 md:grid-cols-2 xl:grid-cols-3 gap-4\">\n {orchestrators.map((orch) => (\n <OrchestratorCard key={orch.id} orch={orch} />\n ))}\n </div>\n )}\n </div>\n )\n}\n"],"mappings":"yYASA,SAAS,EAAgB,CAAE,QAAO,QAAqD,CACrF,IAAM,EAAqB,EAAe,GAAM,EAAE,mBAAmB,CAC/D,EAAc,EAAe,GAAM,EAAE,YAAY,CAEvD,OACE,EAAA,EAAA,MAAC,MAAD,CAAK,UAAU,0CAAf,EACE,EAAA,EAAA,KAAC,EAAD,CAAK,UAAU,6CAA+C,CAAA,EAC9D,EAAA,EAAA,MAAC,MAAD,CAAK,UAAU,0BAAf,EACE,EAAA,EAAA,KAAC,MAAD,CAAK,UAAU,wCACZ,EAAM,OAAS,EAAM,QAAQ,MAAM,IAAI,CACpC,CAAA,EACN,EAAA,EAAA,MAAC,SAAD,CACE,KAAK,SACL,UAAU,wFACV,MAAO,QAAQ,EAAM,KAAO,EAAK,QAAQ,WACzC,QAAU,GAAM,CAAE,EAAE,iBAAiB,CAAE,EAAiB,CAAE,eAAgB,EAAK,GAAI,KAAM,EAAM,KAAO,EAAK,QAAS,CAAC,WAJvH,CAMG,EAAM,SAAS,MAAI,EAAM,KAAO,IAC1B,GACL,IACN,EAAA,EAAA,MAAC,MAAD,CAAK,UAAU,sBAAf,EACE,EAAA,EAAA,KAAC,EAAD,CACE,KAAK,OACL,QAAQ,QACR,UAAU,UACV,MAAM,gBACN,QAAU,GAAM,CAAE,EAAE,iBAAiB,CAAE,EAAmB,EAAK,GAAI,UAAW,EAAM,QAAS,gBAAgB,YAE7G,EAAA,EAAA,KAAC,EAAD,CAAU,UAAU,UAAY,CAAA,CACzB,CAAA,EACT,EAAA,EAAA,KAAC,EAAD,CACE,KAAK,OACL,QAAQ,QACR,UAAU,0CACV,MAAM,iBACN,QAAU,GAAM,CAAE,EAAE,iBAAiB,CAAE,EAAmB,EAAK,GAAI,WAAY,EAAM,QAAS,gBAAgB,YAE9G,EAAA,EAAA,KAAC,EAAD,CAAO,UAAU,UAAY,CAAA,CACtB,CAAA,CACL,GACF,GAIV,SAAS,EAAiB,CAAE,QAAgC,CAC1D,IAAM,EAAM,GAAQ,CACd,EAAqB,EAAe,GAAM,EAAE,mBAAmB,CAC/D,EAAsB,EAAe,GAAM,EAAE,oBAAoB,CACjE,EAAc,EAAe,GAAM,EAAE,MAAM,QAAQ,CACnD,EAA2B,EAAe,GAAM,EAAE,yBAAyB,CAC3E,EAAc,EAAe,GAAM,EAAE,YAAY,CACjD,EAAqB,EAAe,GAAM,EAAE,mBAAmB,CAC/D,EAA+B,EAAe,GAAM,EAAE,6BAA6B,CAEnF,EAAc,EAAwB,EAAK,CAC3C,EAAc,EAAwB,EAAK,CAC3C,EAAS,EAAK,SAAW,SACzB,EAAiB,EAAK,SAAS,SAAW,EAAK,QAC/C,EAAU,EAAK,QACf,EAAY,GAAS,SAAW,UAEhC,EAAiB,GAAU,CAAC,CAAC,EAAK,SAAW,CAAC,CAAC,GAAe,EAAK,UAAY,EAC/E,EAAa,GAAU,CAAC,IAAc,GAAkB,GAAS,SAAW,UAC5E,EAAkB,OAAO,QAAQ,EAAK,WAAa,EAAE,CAAC,CAAC,QAAQ,EAAG,KAAW,OAAO,GAAU,SAAS,CACvG,EAAiB,EAAK,gBAAkB,EAAK,UAAU,IAAK,IAAc,CAAE,KAAM,EAAU,UAAW,GAAM,UAAW,OAAO,EAAK,SAAS,EAAI,KAAK,KAAK,CAAE,EAAE,CAC/J,EAAa,EAAK,cACrB,IAAK,GAAU,EAAM,UAAY,CAAE,QAAO,UAAW,EAAM,UAAW,CAAG,KAAK,CAC9E,OAAQ,GAA6F,GAAQ,GAAM,WAAW,IAAM,GAAM,WAAW,cAAc,CAEtK,OACE,EAAA,EAAA,MAAC,EAAD,CAAM,UAAW,GAAI,EAAwB,GAAf,wBAA9B,EACE,EAAA,EAAA,KAAC,EAAD,CAAY,UAAU,2BACpB,EAAA,EAAA,MAAC,MAAD,CAAK,UAAU,kCAAf,EACE,EAAA,EAAA,KAAC,MAAD,CAAK,UAAW,oBAAoB,EAAS,oBAAsB,0BACjE,EAAA,EAAA,KAAC,EAAD,CAAQ,UAAW,WAAW,EAAS,mBAAqB,kBAAqB,CAAA,CAC7E,CAAA,EACN,EAAA,EAAA,MAAC,MAAD,CAAK,UAAU,0BAAf,EACE,EAAA,EAAA,KAAC,EAAD,CAAW,UAAU,wCAAgC,EAAK,SAAqB,CAAA,EAC/E,EAAA,EAAA,MAAC,MAAD,CAAK,UAAU,gDAAf,CAAuD,EAAK,GAAG,MAAM,EAAG,GAAG,CAAC,IAAO,GAC/E,IACN,EAAA,EAAA,KAAC,EAAD,CAAO,UAAW,wBAAwB,aAAgB,EAAoB,CAAA,CAC1E,GACK,CAAA,EAEb,EAAA,EAAA,MAAC,EAAD,CAAa,UAAU,+BAAvB,EAEE,EAAA,EAAA,MAAC,MAAD,CAAK,UAAU,wEAAf,EACE,EAAA,EAAA,KAAC,OAAD,CAAM,UAAW,EAAS,mBAAqB,yBAAkB,EAAK,OAAc,CAAA,CACnF,IAAkB,EAAA,EAAA,MAAC,OAAD,CAAA,SAAA,CAAO,EAAK,SAAS,MAAQ,eAAe,KAAG,EAAsB,CAAA,CAAA,CACvF,IACC,EAAA,EAAA,MAAC,EAAD,CAAO,UAAU,gFAAjB,EACE,EAAA,EAAA,KAAC,EAAD,CAAe,UAAU,wBAA0B,CAAA,iBAAc,EAAS,eACpE,GAET,CAAC,GAAa,GAAS,SAAW,WACjC,EAAA,EAAA,MAAC,EAAD,CAAO,UAAU,iEAAiE,MAAO,EAAQ,eAAjG,EACE,EAAA,EAAA,KAAC,EAAD,CAAe,UAAU,UAAY,CAAA,CAAA,iBAC/B,GAET,CAAC,GAAa,IACb,EAAA,EAAA,MAAC,EAAD,CAAO,UAAU,uEAAjB,CAA+E,MAAI,EAAY,aAAkB,IAEnH,EAAA,EAAA,KAAC,OAAD,CAAM,UAAU,mBAAW,EAAQ,EAAK,EAAK,SAAS,CAAQ,CAAA,CAC1D,GAEL,EAAe,OAAS,IACvB,EAAA,EAAA,KAAC,MAAD,CAAK,UAAU,gCACZ,EAAe,IAAK,IACnB,EAAA,EAAA,KAAC,EAAD,CAEE,QAAQ,YACR,MAAO,EAAO,UAAa,YAAa,GAAU,EAAO,QAAU,EAAO,QAAU,EAAO,KAAS,WAAY,GAAU,EAAO,OAAS,EAAO,OAAS,GAAG,EAAO,KAAK,cACzK,UAAW,eAAe,EAAO,UAAY,2DAA6D,0DAEzG,EAAO,KACF,CAND,EAAO,KAMN,CACR,CACE,CAAA,CAGP,EAAgB,OAAS,IACxB,EAAA,EAAA,KAAC,MAAD,CAAK,UAAU,gCACZ,EAAgB,KAAK,CAAC,EAAM,MAC3B,EAAA,EAAA,MAAC,EAAD,CAAkB,QAAQ,YAAY,UAAU,uBAAhD,CAA+D,EAAK,IAAE,EAAgB,EAA1E,EAA0E,CACtF,CACE,CAAA,CAIP,EAAK,cAAc,OAAS,IAC3B,EAAA,EAAA,MAAA,EAAA,SAAA,CAAA,SAAA,EACE,EAAA,EAAA,KAAC,EAAD,EAAa,CAAA,EACb,EAAA,EAAA,MAAC,MAAD,CAAA,SAAA,EACE,EAAA,EAAA,MAAC,MAAD,CAAK,UAAU,4DAAf,CAAkE,mBAC/C,EAAK,cAAc,OAAO,IACvC,IACN,EAAA,EAAA,KAAC,MAAD,CAAK,UAAU,qBACZ,EAAK,cAAc,IAAK,IACvB,EAAA,EAAA,KAAC,EAAD,CAAiC,MAAO,EAAS,OAAQ,CAAnC,EAAE,QAAiC,CACzD,CACE,CAAA,CACF,CAAA,CAAA,CACL,CAAA,CAAA,CAGJ,EAAW,OAAS,IACnB,EAAA,EAAA,MAAA,EAAA,SAAA,CAAA,SAAA,EACE,EAAA,EAAA,KAAC,EAAD,EAAa,CAAA,EACb,EAAA,EAAA,MAAC,MAAD,CAAA,SAAA,EACE,EAAA,EAAA,MAAC,MAAD,CAAK,UAAU,4DAAf,CAAkE,eACnD,EAAW,OAAO,IAC3B,IACN,EAAA,EAAA,KAAC,MAAD,CAAK,UAAU,qBACZ,EAAW,KAAK,CAAE,QAAO,gBACxB,EAAA,EAAA,MAAC,SAAD,CAEE,KAAK,SACL,UAAU,gGACV,YAAe,EAAU,cAAgB,KAAK,EAAY,CAAE,eAAgB,EAAK,GAAI,KAAM,EAAU,aAAc,CAAC,UAJtH,EAME,EAAA,EAAA,MAAC,MAAD,CAAK,UAAU,2CAAf,EACE,EAAA,EAAA,KAAC,OAAD,CAAM,UAAU,gCAAwB,EAAU,QAAU,EAAM,OAAS,EAAM,QAAQ,MAAM,IAAI,CAAQ,CAAA,CAC1G,EAAU,SAAU,EAAA,EAAA,KAAC,EAAD,CAAO,QAAQ,UAAU,UAAU,oCAA4B,EAAU,OAAe,CAAA,CACzG,IACN,EAAA,EAAA,KAAC,MAAD,CAAK,UAAU,sDAA8C,EAAU,cAAgB,EAAU,SAAe,CAAA,CACzG,EAVF,EAAU,IAAM,EAAU,aAUxB,CACT,CACE,CAAA,CACF,CAAA,CAAA,CACL,CAAA,CAAA,EAIL,EAAA,EAAA,KAAC,EAAD,EAAa,CAAA,EACb,EAAA,EAAA,MAAC,MAAD,CAAK,UAAU,kCAAf,CACG,IACC,EAAA,EAAA,MAAC,EAAD,CAAQ,KAAK,KAAK,QAAQ,UAAU,UAAU,oBAAoB,YAAe,EAAyB,EAAK,GAAG,UAAlH,EACE,EAAA,EAAA,KAAC,EAAD,CAAM,UAAU,UAAY,CAAA,CAAA,cAErB,GAEV,IACC,EAAA,EAAA,MAAC,EAAD,CACE,KAAK,KACL,QAAQ,UACR,UAAU,oBACV,YAAe,EAA6B,EAAK,GAAG,UAJtD,EAME,EAAA,EAAA,KAAC,EAAD,CAAU,UAAU,UAAY,CAAA,CAAA,oBAEzB,IAET,GAAc,KACd,EAAA,EAAA,MAAC,EAAD,CACE,KAAK,KACL,QAAQ,UACR,SAAU,EACV,UAAU,6FACV,MAAO,EAAY,iBAAiB,EAAS,eAAe,GAAK,eAAe,IAChF,YAAe,EAAoB,EAAK,GAAG,UAN7C,EAQE,EAAA,EAAA,KAAC,EAAD,CAAe,UAAW,WAAW,EAAY,gBAAkB,KAAQ,CAAA,CAC1E,EAAY,aAAe,UACrB,GAEV,IACC,EAAA,EAAA,MAAC,EAAD,CACE,KAAK,KACL,QAAQ,UACR,UAAU,oBACV,YAAe,EAAmB,EAAK,GAAI,UAAW,IAAA,GAAW,gBAAgB,UAJnF,EAME,EAAA,EAAA,KAAC,EAAD,CAAU,UAAU,UAAY,CAAA,CAAA,cAEzB,GAEV,IACC,EAAA,EAAA,MAAC,EAAD,CACE,KAAK,KACL,QAAQ,UACR,UAAU,sEACV,YAAe,EAAmB,EAAK,GAAI,WAAY,IAAA,GAAW,gBAAgB,UAJpF,EAME,EAAA,EAAA,KAAC,EAAD,CAAO,UAAU,UAAY,CAAA,CAAA,eAEtB,IAEX,EAAA,EAAA,KAAC,EAAD,CACE,KAAK,KACL,QAAQ,QACR,UAAU,uDACV,YAAe,EAAmB,EAAK,GAAG,UAC3C,SAEQ,CAAA,CACL,GACM,GACT,GAIX,SAAgB,GAAoB,CAClC,IAAM,EAAgB,EAAe,GAAM,EAAE,cAAc,CACrD,EAAwB,EAAe,GAAM,EAAE,sBAAsB,CACrE,EAA0B,EAAe,GAAM,EAAE,wBAAwB,CAEzE,EAAc,EAAc,OAAQ,GAAM,EAAE,SAAW,SAAS,CAAC,OAEvE,OACE,EAAA,EAAA,MAAC,MAAD,CAAK,UAAU,qBAAf,EAEE,EAAA,EAAA,MAAC,MAAD,CAAK,UAAU,mCAAf,EACE,EAAA,EAAA,MAAC,MAAD,CAAA,SAAA,EACE,EAAA,EAAA,KAAC,KAAD,CAAI,UAAU,mCAA0B,gBAAkB,CAAA,EAC1D,EAAA,EAAA,KAAC,IAAD,CAAG,UAAU,yCAAgC,sCAAuC,CAAA,CAChF,CAAA,CAAA,EACN,EAAA,EAAA,MAAC,MAAD,CAAK,UAAU,2CAAf,EACE,EAAA,EAAA,MAAC,EAAD,CAAO,QAAQ,UAAU,UAAU,mBAAnC,CACG,EAAY,UACP,IACR,EAAA,EAAA,MAAC,EAAD,CAAQ,KAAK,KAAK,QAAQ,UAAU,UAAU,QAAQ,QAAS,WAA/D,EACE,EAAA,EAAA,KAAC,EAAD,CAAQ,UAAU,cAAgB,CAAA,CAAA,MAE3B,IACT,EAAA,EAAA,MAAC,EAAD,CAAQ,KAAK,KAAK,UAAU,QAAQ,QAAS,WAA7C,EACE,EAAA,EAAA,KAAC,EAAD,CAAM,UAAU,cAAgB,CAAA,CAAA,QAEzB,GACL,GACF,GAGL,EAAc,SAAW,GACxB,EAAA,EAAA,KAAC,MAAD,CAAK,UAAU,2DAAkD,+BAE3D,CAAA,EAEN,EAAA,EAAA,KAAC,MAAD,CAAK,UAAU,gEACZ,EAAc,IAAK,IAClB,EAAA,EAAA,KAAC,EAAD,CAAsC,OAAQ,CAAvB,EAAK,GAAkB,CAC9C,CACE,CAAA,CAEJ"}
|
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
import{r as e}from"./chunk-CilyBKbf.js";import{$ as t,En as n,J as r,Vn as i,at as a,kn as o,l as s,mn as c,mt as l,ot as u,p as d}from"./lucide-react-CD8Xl2U3.js";import{i as f,t as p}from"./store-
|
|
2
|
-
//# sourceMappingURL=overview-
|
|
1
|
+
import{r as e}from"./chunk-CilyBKbf.js";import{$ as t,En as n,J as r,Vn as i,at as a,kn as o,l as s,mn as c,mt as l,ot as u,p as d}from"./lucide-react-CD8Xl2U3.js";import{i as f,t as p}from"./store-CICRhg1m.js";import{B as m,E as h,U as g,d as _,h as v,ot as y,u as b,z as x}from"./display-Bebqs1qu.js";import{t as S}from"./badge-t8zAwHW9.js";import{A as C,D as w,N as T,_ as E,g as D,j as O,s as k}from"./index-CvSlyTSI.js";import{i as A,n as j,r as M,t as N}from"./card-CggxP1h9.js";var P=e(i(),1),F=o();function I(){let e=p(e=>e.stats),i=p(e=>e.agents),o=p(e=>e.messages),I=p(e=>e.health),L=f(),R=p(e=>e.agentsById),z=p(e=>e.switchView),B=p(e=>e.runHealthAction),V=p(e=>e.managedPolicies),H=p(e=>e.showOffline),U=T(),W=O(),G=C(),K=(0,P.useMemo)(()=>{let e=g(i,!1,V);return H||(e=e.filter(e=>e.status!==`offline`)),[...e].sort((e,t)=>(y[e.status]??9)-(y[t.status]??9)||m(t.lastSeen)-m(e.lastSeen))},[i,H,V]),q=K.slice(0,20),J=[...o].sort((e,t)=>t.id-e.id).slice(0,15),Y=I&&I.status!==`ok`?v(I.checks||[]):[];function X(e){z(e)}return(0,F.jsxs)(`div`,{className:`space-y-6`,children:[(0,F.jsxs)(`div`,{className:`grid grid-cols-2 md:grid-cols-4 gap-4`,children:[(0,F.jsxs)(N,{className:`cursor-pointer hover:border-zinc-600 transition-colors`,onClick:()=>X(`agents`),children:[(0,F.jsx)(M,{className:`pb-1 pt-4 px-4`,children:(0,F.jsxs)(A,{className:`text-xs font-medium text-muted-foreground flex items-center gap-1.5`,children:[(0,F.jsx)(s,{className:`w-3.5 h-3.5`}),`Agents`]})}),(0,F.jsxs)(j,{className:`px-4 pb-4`,children:[(0,F.jsx)(`div`,{className:`text-2xl font-bold`,children:K.length}),(0,F.jsxs)(`div`,{className:`text-xs text-muted-foreground mt-0.5`,children:[U,` online`]})]})]}),(0,F.jsxs)(N,{className:`cursor-pointer hover:border-zinc-600 transition-colors`,onClick:()=>X(`agents`),children:[(0,F.jsx)(M,{className:`pb-1 pt-4 px-4`,children:(0,F.jsxs)(A,{className:`text-xs font-medium text-muted-foreground flex items-center gap-1.5`,children:[(0,F.jsx)(n,{className:`w-3.5 h-3.5`}),`Busy`]})}),(0,F.jsxs)(j,{className:`px-4 pb-4`,children:[(0,F.jsx)(`div`,{className:`text-2xl font-bold`,children:W}),(0,F.jsx)(`div`,{className:`text-xs text-muted-foreground mt-0.5`,children:`in a turn now`})]})]}),(0,F.jsxs)(N,{className:`cursor-pointer hover:border-zinc-600 transition-colors`,onClick:()=>X(`messages`),children:[(0,F.jsx)(M,{className:`pb-1 pt-4 px-4`,children:(0,F.jsxs)(A,{className:`text-xs font-medium text-muted-foreground flex items-center gap-1.5`,children:[(0,F.jsx)(r,{className:`w-3.5 h-3.5`}),`Messages 24h`]})}),(0,F.jsxs)(j,{className:`px-4 pb-4`,children:[(0,F.jsx)(`div`,{className:`text-2xl font-bold`,children:e.messagesLast24h??`—`}),(0,F.jsx)(`div`,{className:`text-xs text-muted-foreground mt-0.5`,children:`last 24 hours`})]})]}),(0,F.jsxs)(N,{className:`cursor-pointer hover:border-zinc-600 transition-colors`,onClick:()=>X(`messages`),children:[(0,F.jsx)(M,{className:`pb-1 pt-4 px-4`,children:(0,F.jsxs)(A,{className:`text-xs font-medium text-muted-foreground flex items-center gap-1.5`,children:[(0,F.jsx)(t,{className:`w-3.5 h-3.5`}),`Total Messages`]})}),(0,F.jsxs)(j,{className:`px-4 pb-4`,children:[(0,F.jsx)(`div`,{className:`text-2xl font-bold`,children:e.messages??`—`}),(0,F.jsx)(`div`,{className:`text-xs text-muted-foreground mt-0.5`,children:`all time`})]})]})]}),Y.length>0&&(0,F.jsxs)(N,{className:`border-yellow-500/40 bg-yellow-500/5`,children:[(0,F.jsx)(M,{className:`pb-2 pt-4 px-4`,children:(0,F.jsxs)(A,{className:`text-sm font-medium text-yellow-400 flex items-center gap-2`,children:[(0,F.jsx)(d,{className:`w-4 h-4`}),`Health Alerts (`,Y.length,`)`]})}),(0,F.jsx)(j,{className:`px-4 pb-4 space-y-2`,children:Y.map(e=>(0,F.jsxs)(`div`,{className:`text-sm space-y-1`,children:[(0,F.jsxs)(`div`,{className:`flex items-center gap-2`,children:[(0,F.jsx)(S,{variant:`outline`,className:e.status===`error`?`border-red-500/40 text-red-400`:`border-yellow-500/40 text-yellow-400`,children:e.status}),(0,F.jsx)(`span`,{className:`font-medium`,children:e.name})]}),(0,F.jsx)(`p`,{className:`text-xs text-muted-foreground`,children:e.detail}),e.subjects.length>0&&(0,F.jsx)(`div`,{className:`flex flex-wrap gap-1.5`,children:e.subjects.map(e=>(0,F.jsxs)(S,{variant:`secondary`,className:`max-w-full truncate text-[11px]`,title:[e.id,e.status,e.detail].filter(Boolean).join(` · `),children:[e.label||e.id,e.status?` · ${e.status}`:``,e.detail?` · ${e.detail}`:``]},e.id))}),e.actions.length>0&&(0,F.jsx)(`div`,{className:`flex flex-wrap gap-1.5 pt-0.5`,children:e.actions.slice(0,3).map(e=>(0,F.jsx)(`button`,{onClick:()=>B(e),className:`text-xs text-blue-400 hover:text-blue-300 underline underline-offset-2`,children:e.label},e.label))})]},e.name))})]}),(0,F.jsxs)(`div`,{className:`grid grid-cols-1 sm:grid-cols-3 gap-4`,children:[(0,F.jsx)(N,{className:`cursor-pointer hover:border-zinc-600 transition-colors ${G.unreadInbox>0?`border-red-500/40`:``}`,onClick:()=>X(`chat`),children:(0,F.jsxs)(j,{className:`p-4 flex items-center gap-3`,children:[(0,F.jsx)(`div`,{className:`p-2 rounded-lg ${G.unreadInbox>0?`bg-red-500/15`:`bg-zinc-800`}`,children:(0,F.jsx)(l,{className:`w-4 h-4 ${G.unreadInbox>0?`text-red-400`:`text-zinc-400`}`})}),(0,F.jsxs)(`div`,{children:[(0,F.jsx)(`div`,{className:`text-lg font-bold`,children:G.unreadInbox}),(0,F.jsx)(`div`,{className:`text-xs text-muted-foreground`,children:`Unread`})]})]})}),(0,F.jsx)(N,{className:`cursor-pointer hover:border-zinc-600 transition-colors ${G.pendingPairInvites>0?`border-yellow-500/40`:``}`,onClick:()=>X(`pairs`),children:(0,F.jsxs)(j,{className:`p-4 flex items-center gap-3`,children:[(0,F.jsx)(`div`,{className:`p-2 rounded-lg ${G.pendingPairInvites>0?`bg-yellow-500/15`:`bg-zinc-800`}`,children:(0,F.jsx)(u,{className:`w-4 h-4 ${G.pendingPairInvites>0?`text-yellow-400`:`text-zinc-400`}`})}),(0,F.jsxs)(`div`,{children:[(0,F.jsx)(`div`,{className:`text-lg font-bold`,children:G.pendingPairInvites}),(0,F.jsx)(`div`,{className:`text-xs text-muted-foreground`,children:`Pair invites`})]})]})}),(0,F.jsx)(N,{className:`cursor-pointer hover:border-zinc-600 transition-colors ${G.claimableTasks>0?`border-orange-500/40`:``}`,onClick:()=>X(`work`),children:(0,F.jsxs)(j,{className:`p-4 flex items-center gap-3`,children:[(0,F.jsx)(`div`,{className:`p-2 rounded-lg ${G.claimableTasks>0?`bg-orange-500/15`:`bg-zinc-800`}`,children:(0,F.jsx)(a,{className:`w-4 h-4 ${G.claimableTasks>0?`text-orange-400`:`text-zinc-400`}`})}),(0,F.jsxs)(`div`,{children:[(0,F.jsx)(`div`,{className:`text-lg font-bold`,children:G.claimableTasks}),(0,F.jsx)(`div`,{className:`text-xs text-muted-foreground`,children:`Claimable waiting`})]})]})})]}),(0,F.jsxs)(`div`,{className:`grid grid-cols-1 md:grid-cols-2 gap-6`,children:[(0,F.jsxs)(N,{children:[(0,F.jsx)(M,{className:`pb-2 pt-4 px-4`,children:(0,F.jsxs)(A,{className:`text-sm font-medium flex items-center gap-2`,children:[(0,F.jsx)(c,{className:`w-4 h-4 text-muted-foreground`}),`Agents`,(0,F.jsx)(`span`,{className:`ml-auto text-xs text-muted-foreground font-normal`,children:q.length})]})}),(0,F.jsx)(j,{className:`px-0 pb-0`,children:(0,F.jsx)(k,{className:`h-[340px]`,children:q.length===0?(0,F.jsx)(`div`,{className:`px-4 pb-4 text-sm text-muted-foreground`,children:`No agents registered.`}):(0,F.jsx)(`div`,{className:`space-y-0`,children:q.map((e,t)=>(0,F.jsxs)(`div`,{children:[t>0&&(0,F.jsx)(w,{}),(0,F.jsxs)(`div`,{className:`flex items-center gap-2.5 px-4 py-2.5 hover:bg-zinc-800/50 transition-colors cursor-pointer`,onClick:()=>p.getState().openAgentDetail(e),children:[(0,F.jsx)(E,{agent:e,now:L}),(0,F.jsx)(D,{agent:e}),(0,F.jsxs)(`div`,{className:`flex-1 min-w-0`,children:[(0,F.jsx)(`div`,{className:`text-sm font-medium truncate`,children:b(e)}),e.tags.length>0&&(0,F.jsx)(`div`,{className:`flex gap-1 mt-0.5 flex-wrap`,children:e.tags.slice(0,2).map(e=>(0,F.jsx)(S,{variant:`outline`,className:`text-[9px] px-1 py-0 h-4`,children:e},e))})]}),(0,F.jsx)(`span`,{className:`text-xs text-muted-foreground shrink-0`,children:x(L,e.lastSeen)})]})]},e.id))})})})]}),(0,F.jsxs)(N,{children:[(0,F.jsx)(M,{className:`pb-2 pt-4 px-4`,children:(0,F.jsxs)(A,{className:`text-sm font-medium flex items-center gap-2`,children:[(0,F.jsx)(n,{className:`w-4 h-4 text-muted-foreground`}),`Recent Activity`,(0,F.jsx)(`span`,{className:`ml-auto text-xs text-muted-foreground font-normal`,children:J.length})]})}),(0,F.jsx)(j,{className:`px-0 pb-0`,children:(0,F.jsx)(k,{className:`h-[340px]`,children:J.length===0?(0,F.jsx)(`div`,{className:`px-4 pb-4 text-sm text-muted-foreground`,children:`No recent messages.`}):(0,F.jsx)(`div`,{className:`space-y-0`,children:J.map((e,t)=>(0,F.jsxs)(`div`,{children:[t>0&&(0,F.jsx)(w,{}),(0,F.jsxs)(`div`,{className:`px-4 py-2.5 hover:bg-zinc-800/50 transition-colors`,children:[(0,F.jsxs)(`div`,{className:`flex items-center gap-1 text-xs text-muted-foreground mb-0.5`,children:[(0,F.jsx)(`span`,{className:`text-foreground font-medium truncate max-w-[100px]`,children:_(e.from,R)}),(0,F.jsx)(`span`,{children:`→`}),(0,F.jsx)(`span`,{className:`truncate max-w-[100px]`,children:_(e.to,R)}),(0,F.jsxs)(`span`,{className:`ml-auto shrink-0 tabular-nums`,children:[(0,F.jsxs)(`span`,{className:`opacity-40 mr-1`,children:[`#`,e.id]}),x(L,e.createdAt)]})]}),(0,F.jsx)(`div`,{className:`text-xs text-foreground line-clamp-2 leading-snug`,children:h(e)})]})]},e.id))})})})]})]})]})}export{I as OverviewView};
|
|
2
|
+
//# sourceMappingURL=overview-CYAHOUyA.js.map
|