agent-relay-server 0.37.0 → 0.38.0
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 +2 -2
- package/public/assets/{activity-BgkmA1lh.js → activity-ClpDglG8.js} +2 -2
- package/public/assets/{activity-BgkmA1lh.js.map → activity-ClpDglG8.js.map} +1 -1
- package/public/assets/{agent-profiles-DS4_jLPT.js → agent-profiles-kb5H23CF.js} +2 -2
- package/public/assets/{agent-profiles-DS4_jLPT.js.map → agent-profiles-kb5H23CF.js.map} +1 -1
- package/public/assets/{agents-B7HnuAXx.js → agents-CHmEJvqV.js} +2 -2
- package/public/assets/{agents-B7HnuAXx.js.map → agents-CHmEJvqV.js.map} +1 -1
- package/public/assets/{analytics-0-akxJCJ.js → analytics-2kTjXIj1.js} +3 -3
- package/public/assets/{analytics-0-akxJCJ.js.map → analytics-2kTjXIj1.js.map} +1 -1
- package/public/assets/{automation-CaE1z_-M.js → automation-B5U_g-1P.js} +2 -2
- package/public/assets/{automation-CaE1z_-M.js.map → automation-B5U_g-1P.js.map} +1 -1
- package/public/assets/{branch-state-badge-D8-T2c1K.js → branch-state-badge-B1K7aIzF.js} +2 -2
- package/public/assets/{branch-state-badge-D8-T2c1K.js.map → branch-state-badge-B1K7aIzF.js.map} +1 -1
- package/public/assets/{channels-ppN8k4hu.js → channels-DyPw9JsY.js} +2 -2
- package/public/assets/{channels-ppN8k4hu.js.map → channels-DyPw9JsY.js.map} +1 -1
- package/public/assets/chat-zPXWB-03.js +2 -0
- package/public/assets/chat-zPXWB-03.js.map +1 -0
- package/public/assets/{connectors-CL9BALhF.js → connectors-k7JYCrrl.js} +2 -2
- package/public/assets/{connectors-CL9BALhF.js.map → connectors-k7JYCrrl.js.map} +1 -1
- package/public/assets/display-ConJ9cJB.js.map +1 -1
- package/public/assets/{formatted-body-impl-DExNPNsL.js → formatted-body-impl-tmf8IBfr.js} +2 -2
- package/public/assets/{formatted-body-impl-DExNPNsL.js.map → formatted-body-impl-tmf8IBfr.js.map} +1 -1
- package/public/assets/index-B1QUkb_O.js +21 -0
- package/public/assets/index-B1QUkb_O.js.map +1 -0
- package/public/assets/index-Bins8N_5.css +2 -0
- package/public/assets/{integrations-DX55ARy0.js → integrations-BEkyjBAs.js} +2 -2
- package/public/assets/{integrations-DX55ARy0.js.map → integrations-BEkyjBAs.js.map} +1 -1
- package/public/assets/{maintenance-DpTdJxQp.js → maintenance-Tn23oWBF.js} +2 -2
- package/public/assets/{maintenance-DpTdJxQp.js.map → maintenance-Tn23oWBF.js.map} +1 -1
- package/public/assets/{managed-agents-B3df2xfk.js → managed-agents-CasacvJX.js} +2 -2
- package/public/assets/{managed-agents-B3df2xfk.js.map → managed-agents-CasacvJX.js.map} +1 -1
- package/public/assets/{markdown-preview-impl-D0Zj7c3T.js → markdown-preview-impl-D4UIjB3I.js} +2 -2
- package/public/assets/{markdown-preview-impl-D0Zj7c3T.js.map → markdown-preview-impl-D4UIjB3I.js.map} +1 -1
- package/public/assets/{memory-TATN2vZf.js → memory-SVCob0fo.js} +2 -2
- package/public/assets/{memory-TATN2vZf.js.map → memory-SVCob0fo.js.map} +1 -1
- package/public/assets/{messages-3rS1lxIf.js → messages-CHK24Uxx.js} +2 -2
- package/public/assets/{messages-3rS1lxIf.js.map → messages-CHK24Uxx.js.map} +1 -1
- package/public/assets/{orchestrators-CRIV0g5y.js → orchestrators-CQcJb6VE.js} +2 -2
- package/public/assets/{orchestrators-CRIV0g5y.js.map → orchestrators-CQcJb6VE.js.map} +1 -1
- package/public/assets/{overview-CmHC_5oM.js → overview-DbyX7k-7.js} +2 -2
- package/public/assets/{overview-CmHC_5oM.js.map → overview-DbyX7k-7.js.map} +1 -1
- package/public/assets/{pairs-DBPAhXTI.js → pairs-CaL0_ZfW.js} +2 -2
- package/public/assets/{pairs-DBPAhXTI.js.map → pairs-CaL0_ZfW.js.map} +1 -1
- package/public/assets/{security-572X5MNX.js → security-BogsfkbT.js} +2 -2
- package/public/assets/{security-572X5MNX.js.map → security-BogsfkbT.js.map} +1 -1
- package/public/assets/{settings-vTBu8w3O.js → settings-BOsnUh5f.js} +2 -2
- package/public/assets/{settings-vTBu8w3O.js.map → settings-BOsnUh5f.js.map} +1 -1
- package/public/assets/{store-DKVWC6Uh.js → store-Bo72e9My.js} +2 -2
- package/public/assets/{store-DKVWC6Uh.js.map → store-Bo72e9My.js.map} +1 -1
- package/public/assets/{tasks-C0bPrDgN.js → tasks-CCxQovOv.js} +2 -2
- package/public/assets/{tasks-C0bPrDgN.js.map → tasks-CCxQovOv.js.map} +1 -1
- package/public/assets/{terminal-viewer-impl-rVPA6Fsx.js → terminal-viewer-impl-BDikdsxs.js} +2 -2
- package/public/assets/{terminal-viewer-impl-rVPA6Fsx.js.map → terminal-viewer-impl-BDikdsxs.js.map} +1 -1
- package/public/assets/{work-queue-BxkpTt_A.js → work-queue-fM-tu0iP.js} +2 -2
- package/public/assets/{work-queue-BxkpTt_A.js.map → work-queue-fM-tu0iP.js.map} +1 -1
- package/public/assets/{workspaces-CoC2nflZ.js → workspaces-Df0xJuIo.js} +2 -2
- package/public/assets/{workspaces-CoC2nflZ.js.map → workspaces-Df0xJuIo.js.map} +1 -1
- package/public/index.html +3 -3
- package/src/automations.ts +17 -2
- package/src/cli/index.ts +1 -1
- package/src/cli/workspace.ts +36 -3
- package/src/config-store.ts +46 -0
- package/src/maintenance.ts +6 -1
- package/src/mcp.ts +7 -2
- package/src/routes/workspaces.ts +4 -1
- package/src/services/send-message.ts +19 -1
- package/src/services/spawn-agent.ts +6 -0
- package/src/workspace-actions.ts +5 -1
- package/src/workspace-merge.ts +102 -3
- package/src/workspace-phase.ts +15 -1
- package/public/assets/chat-BANKUW05.js +0 -2
- package/public/assets/chat-BANKUW05.js.map +0 -1
- package/public/assets/index-3pO43nJo.css +0 -2
- package/public/assets/index-DEZdON6c.js +0 -21
- package/public/assets/index-DEZdON6c.js.map +0 -1
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"managed-agents-B3df2xfk.js","names":[],"sources":["../../dashboard/src/components/shared/health-badge.tsx","../../dashboard/src/components/views/managed-agents.tsx"],"sourcesContent":["import { cn } from '@/lib/utils'\nimport type { ManagedAgentLifecycleStatus } from '@/types'\n\nconst styles: Record<ManagedAgentLifecycleStatus, string> = {\n running: 'bg-emerald-500/15 text-emerald-400 border-emerald-500/30',\n starting: 'bg-blue-500/15 text-blue-400 border-blue-500/30',\n stopping: 'bg-yellow-500/15 text-yellow-400 border-yellow-500/30',\n backoff: 'bg-red-500/15 text-red-400 border-red-500/30',\n stopped: 'bg-muted text-muted-foreground border-border',\n}\n\nexport function HealthBadge({ status }: { status: ManagedAgentLifecycleStatus }) {\n return (\n <span className={cn('inline-flex items-center rounded border px-2 py-0.5 text-xs font-medium', styles[status])}>\n {status}\n </span>\n )\n}\n","import { useEffect, useState } from 'react'\nimport { FileText, Pencil, Play, Plus, RotateCw, Square, Terminal, Trash2 } from 'lucide-react'\nimport { useRelayStore } from '@/store'\nimport { Button } from '@/components/ui/button'\nimport { HealthBadge } from '@/components/shared/health-badge'\nimport { LogViewer } from '@/components/shared/log-viewer'\nimport { TerminalDialog } from '@/components/shared/terminal-viewer'\nimport { fmtTime } from '@/lib/display'\nimport type { Agent, ManagedPolicyHealth } from '@/types'\n\nfunction terminalSessionFor(item: ManagedPolicyHealth, agent?: Agent): string | undefined {\n const agentSession = agent?.meta?.tmuxSession\n if (typeof agentSession === 'string' && agentSession.trim()) return agentSession\n return item.state.tmuxSession\n}\n\nfunction canReadTerminal(item: ManagedPolicyHealth, agent?: Agent): boolean {\n if (!terminalSessionFor(item, agent)) return false\n if (agent) return Boolean(agent.providerCapabilities?.terminal?.live?.read)\n return item.policy.provider === 'claude'\n}\n\nfunction canWriteTerminal(item: ManagedPolicyHealth, agent?: Agent): boolean {\n if (!canReadTerminal(item, agent)) return false\n if (agent) return Boolean(agent.providerCapabilities?.terminal?.live?.write || item.policy.provider === 'claude')\n return item.policy.provider === 'claude'\n}\n\nexport function ManagedAgentsView() {\n const policies = useRelayStore((s) => s.managedPolicies)\n const agentsById = useRelayStore((s) => s.agentsById)\n const fetchManagedPolicies = useRelayStore((s) => s.fetchManagedPolicies)\n const apiCall = useRelayStore((s) => s.apiCall)\n const openPolicyCreate = useRelayStore((s) => s.openPolicyCreate)\n const openPolicyEdit = useRelayStore((s) => s.openPolicyEdit)\n const deletePolicy = useRelayStore((s) => s.deletePolicy)\n const [logsFor, setLogsFor] = useState<ManagedPolicyHealth | null>(null)\n const [terminalFor, setTerminalFor] = useState<{ item: ManagedPolicyHealth; session: string } | null>(null)\n\n useEffect(() => {\n void fetchManagedPolicies()\n }, [])\n\n async function lifecycle(name: string, action: 'start' | 'stop' | 'restart') {\n await apiCall('POST', `/spawn-policy/${encodeURIComponent(name)}/${action}`, { surface: { component: 'managed-agents' } })\n await fetchManagedPolicies()\n }\n\n return (\n <div className=\"space-y-4\">\n <div className=\"flex items-center justify-between\">\n <div>\n <h1 className=\"text-2xl font-semibold\">Managed Agents</h1>\n <p className=\"text-sm text-muted-foreground\">Spawn policies, lifecycle state, and headless session logs.</p>\n </div>\n <Button onClick={openPolicyCreate} size=\"sm\">\n <Plus className=\"h-4 w-4 mr-1\" />\n Create Policy\n </Button>\n </div>\n\n {policies.length === 0 && (\n <div className=\"rounded-md border border-border px-3 py-8 text-center text-sm text-muted-foreground\">\n No managed agents configured\n </div>\n )}\n\n <div className=\"grid gap-3 sm:grid-cols-2 lg:grid-cols-3\">\n {policies.map((item) => {\n const agent = item.state.agentId ? agentsById[item.state.agentId] : undefined\n const terminalSession = terminalSessionFor(item, agent)\n return (\n <div key={item.policy.name} className=\"rounded-md border border-border bg-card p-3 space-y-2\">\n <div className=\"flex items-start justify-between gap-2\">\n <div className=\"min-w-0\">\n <div className=\"font-medium text-sm truncate\">{item.policy.label || item.policy.name}</div>\n <div className=\"text-xs text-muted-foreground truncate\">{item.policy.mode} · {item.policy.cwd}</div>\n </div>\n <HealthBadge status={item.state.status} />\n </div>\n\n <div className=\"grid grid-cols-2 gap-x-4 gap-y-1 text-xs\">\n <div className=\"text-muted-foreground\">Provider</div>\n <div>{item.policy.provider}</div>\n <div className=\"text-muted-foreground\">Agent</div>\n <div className=\"font-mono truncate\">{item.state.agentId || '-'}</div>\n <div className=\"text-muted-foreground\">Restarts</div>\n <div>{item.state.restartCount}</div>\n {(item.state.backoffUntil || item.state.lastError) && (\n <>\n <div className=\"text-muted-foreground\">{item.state.backoffUntil ? 'Backoff' : 'Error'}</div>\n <div className=\"truncate\">{item.state.backoffUntil ? fmtTime(item.state.backoffUntil) : item.state.lastError}</div>\n </>\n )}\n </div>\n\n <div className=\"flex flex-wrap gap-1 pt-1 border-t border-border\">\n <Button size=\"icon\" variant=\"outline\" className=\"h-7 w-7\" onClick={() => lifecycle(item.policy.name, 'start')} title=\"Start\"><Play className=\"h-3.5 w-3.5\" /></Button>\n <Button size=\"icon\" variant=\"outline\" className=\"h-7 w-7\" onClick={() => lifecycle(item.policy.name, 'stop')} title=\"Stop\"><Square className=\"h-3.5 w-3.5\" /></Button>\n <Button size=\"icon\" variant=\"outline\" className=\"h-7 w-7\" onClick={() => lifecycle(item.policy.name, 'restart')} title=\"Restart\"><RotateCw className=\"h-3.5 w-3.5\" /></Button>\n <Button size=\"icon\" variant=\"outline\" className=\"h-7 w-7\" onClick={() => setLogsFor(item)} disabled={!item.state.tmuxSession} title=\"Logs\"><FileText className=\"h-3.5 w-3.5\" /></Button>\n <Button size=\"icon\" variant=\"outline\" className=\"h-7 w-7\" onClick={() => terminalSession && setTerminalFor({ item, session: terminalSession })} disabled={!canReadTerminal(item, agent)} title=\"Terminal\"><Terminal className=\"h-3.5 w-3.5\" /></Button>\n <Button size=\"icon\" variant=\"outline\" className=\"h-7 w-7\" onClick={() => openPolicyEdit(item.policy)} title=\"Edit\"><Pencil className=\"h-3.5 w-3.5\" /></Button>\n <Button size=\"icon\" variant=\"outline\" className=\"h-7 w-7\" onClick={() => deletePolicy(item.policy.name)} title=\"Delete\"><Trash2 className=\"h-3.5 w-3.5\" /></Button>\n </div>\n </div>\n )\n })}\n </div>\n\n {logsFor?.state.tmuxSession && (\n <div className=\"space-y-2\">\n <div className=\"flex items-center justify-between\">\n <h2 className=\"text-lg font-semibold\">{logsFor.policy.name} Logs</h2>\n <Button variant=\"outline\" size=\"sm\" onClick={() => setLogsFor(null)}>Close</Button>\n </div>\n <LogViewer orchestratorId={logsFor.policy.orchestratorId} session={logsFor.state.tmuxSession} />\n </div>\n )}\n\n <TerminalDialog\n open={!!terminalFor}\n onOpenChange={(open) => { if (!open) setTerminalFor(null) }}\n orchestratorId={terminalFor?.item.policy.orchestratorId ?? ''}\n session={terminalFor?.session ?? ''}\n interactive={terminalFor ? canWriteTerminal(terminalFor.item, terminalFor.item.state.agentId ? agentsById[terminalFor.item.state.agentId] : undefined) : false}\n />\n </div>\n )\n}\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"managed-agents-CasacvJX.js","names":[],"sources":["../../dashboard/src/components/shared/health-badge.tsx","../../dashboard/src/components/views/managed-agents.tsx"],"sourcesContent":["import { cn } from '@/lib/utils'\nimport type { ManagedAgentLifecycleStatus } from '@/types'\n\nconst styles: Record<ManagedAgentLifecycleStatus, string> = {\n running: 'bg-emerald-500/15 text-emerald-400 border-emerald-500/30',\n starting: 'bg-blue-500/15 text-blue-400 border-blue-500/30',\n stopping: 'bg-yellow-500/15 text-yellow-400 border-yellow-500/30',\n backoff: 'bg-red-500/15 text-red-400 border-red-500/30',\n stopped: 'bg-muted text-muted-foreground border-border',\n}\n\nexport function HealthBadge({ status }: { status: ManagedAgentLifecycleStatus }) {\n return (\n <span className={cn('inline-flex items-center rounded border px-2 py-0.5 text-xs font-medium', styles[status])}>\n {status}\n </span>\n )\n}\n","import { useEffect, useState } from 'react'\nimport { FileText, Pencil, Play, Plus, RotateCw, Square, Terminal, Trash2 } from 'lucide-react'\nimport { useRelayStore } from '@/store'\nimport { Button } from '@/components/ui/button'\nimport { HealthBadge } from '@/components/shared/health-badge'\nimport { LogViewer } from '@/components/shared/log-viewer'\nimport { TerminalDialog } from '@/components/shared/terminal-viewer'\nimport { fmtTime } from '@/lib/display'\nimport type { Agent, ManagedPolicyHealth } from '@/types'\n\nfunction terminalSessionFor(item: ManagedPolicyHealth, agent?: Agent): string | undefined {\n const agentSession = agent?.meta?.tmuxSession\n if (typeof agentSession === 'string' && agentSession.trim()) return agentSession\n return item.state.tmuxSession\n}\n\nfunction canReadTerminal(item: ManagedPolicyHealth, agent?: Agent): boolean {\n if (!terminalSessionFor(item, agent)) return false\n if (agent) return Boolean(agent.providerCapabilities?.terminal?.live?.read)\n return item.policy.provider === 'claude'\n}\n\nfunction canWriteTerminal(item: ManagedPolicyHealth, agent?: Agent): boolean {\n if (!canReadTerminal(item, agent)) return false\n if (agent) return Boolean(agent.providerCapabilities?.terminal?.live?.write || item.policy.provider === 'claude')\n return item.policy.provider === 'claude'\n}\n\nexport function ManagedAgentsView() {\n const policies = useRelayStore((s) => s.managedPolicies)\n const agentsById = useRelayStore((s) => s.agentsById)\n const fetchManagedPolicies = useRelayStore((s) => s.fetchManagedPolicies)\n const apiCall = useRelayStore((s) => s.apiCall)\n const openPolicyCreate = useRelayStore((s) => s.openPolicyCreate)\n const openPolicyEdit = useRelayStore((s) => s.openPolicyEdit)\n const deletePolicy = useRelayStore((s) => s.deletePolicy)\n const [logsFor, setLogsFor] = useState<ManagedPolicyHealth | null>(null)\n const [terminalFor, setTerminalFor] = useState<{ item: ManagedPolicyHealth; session: string } | null>(null)\n\n useEffect(() => {\n void fetchManagedPolicies()\n }, [])\n\n async function lifecycle(name: string, action: 'start' | 'stop' | 'restart') {\n await apiCall('POST', `/spawn-policy/${encodeURIComponent(name)}/${action}`, { surface: { component: 'managed-agents' } })\n await fetchManagedPolicies()\n }\n\n return (\n <div className=\"space-y-4\">\n <div className=\"flex items-center justify-between\">\n <div>\n <h1 className=\"text-2xl font-semibold\">Managed Agents</h1>\n <p className=\"text-sm text-muted-foreground\">Spawn policies, lifecycle state, and headless session logs.</p>\n </div>\n <Button onClick={openPolicyCreate} size=\"sm\">\n <Plus className=\"h-4 w-4 mr-1\" />\n Create Policy\n </Button>\n </div>\n\n {policies.length === 0 && (\n <div className=\"rounded-md border border-border px-3 py-8 text-center text-sm text-muted-foreground\">\n No managed agents configured\n </div>\n )}\n\n <div className=\"grid gap-3 sm:grid-cols-2 lg:grid-cols-3\">\n {policies.map((item) => {\n const agent = item.state.agentId ? agentsById[item.state.agentId] : undefined\n const terminalSession = terminalSessionFor(item, agent)\n return (\n <div key={item.policy.name} className=\"rounded-md border border-border bg-card p-3 space-y-2\">\n <div className=\"flex items-start justify-between gap-2\">\n <div className=\"min-w-0\">\n <div className=\"font-medium text-sm truncate\">{item.policy.label || item.policy.name}</div>\n <div className=\"text-xs text-muted-foreground truncate\">{item.policy.mode} · {item.policy.cwd}</div>\n </div>\n <HealthBadge status={item.state.status} />\n </div>\n\n <div className=\"grid grid-cols-2 gap-x-4 gap-y-1 text-xs\">\n <div className=\"text-muted-foreground\">Provider</div>\n <div>{item.policy.provider}</div>\n <div className=\"text-muted-foreground\">Agent</div>\n <div className=\"font-mono truncate\">{item.state.agentId || '-'}</div>\n <div className=\"text-muted-foreground\">Restarts</div>\n <div>{item.state.restartCount}</div>\n {(item.state.backoffUntil || item.state.lastError) && (\n <>\n <div className=\"text-muted-foreground\">{item.state.backoffUntil ? 'Backoff' : 'Error'}</div>\n <div className=\"truncate\">{item.state.backoffUntil ? fmtTime(item.state.backoffUntil) : item.state.lastError}</div>\n </>\n )}\n </div>\n\n <div className=\"flex flex-wrap gap-1 pt-1 border-t border-border\">\n <Button size=\"icon\" variant=\"outline\" className=\"h-7 w-7\" onClick={() => lifecycle(item.policy.name, 'start')} title=\"Start\"><Play className=\"h-3.5 w-3.5\" /></Button>\n <Button size=\"icon\" variant=\"outline\" className=\"h-7 w-7\" onClick={() => lifecycle(item.policy.name, 'stop')} title=\"Stop\"><Square className=\"h-3.5 w-3.5\" /></Button>\n <Button size=\"icon\" variant=\"outline\" className=\"h-7 w-7\" onClick={() => lifecycle(item.policy.name, 'restart')} title=\"Restart\"><RotateCw className=\"h-3.5 w-3.5\" /></Button>\n <Button size=\"icon\" variant=\"outline\" className=\"h-7 w-7\" onClick={() => setLogsFor(item)} disabled={!item.state.tmuxSession} title=\"Logs\"><FileText className=\"h-3.5 w-3.5\" /></Button>\n <Button size=\"icon\" variant=\"outline\" className=\"h-7 w-7\" onClick={() => terminalSession && setTerminalFor({ item, session: terminalSession })} disabled={!canReadTerminal(item, agent)} title=\"Terminal\"><Terminal className=\"h-3.5 w-3.5\" /></Button>\n <Button size=\"icon\" variant=\"outline\" className=\"h-7 w-7\" onClick={() => openPolicyEdit(item.policy)} title=\"Edit\"><Pencil className=\"h-3.5 w-3.5\" /></Button>\n <Button size=\"icon\" variant=\"outline\" className=\"h-7 w-7\" onClick={() => deletePolicy(item.policy.name)} title=\"Delete\"><Trash2 className=\"h-3.5 w-3.5\" /></Button>\n </div>\n </div>\n )\n })}\n </div>\n\n {logsFor?.state.tmuxSession && (\n <div className=\"space-y-2\">\n <div className=\"flex items-center justify-between\">\n <h2 className=\"text-lg font-semibold\">{logsFor.policy.name} Logs</h2>\n <Button variant=\"outline\" size=\"sm\" onClick={() => setLogsFor(null)}>Close</Button>\n </div>\n <LogViewer orchestratorId={logsFor.policy.orchestratorId} session={logsFor.state.tmuxSession} />\n </div>\n )}\n\n <TerminalDialog\n open={!!terminalFor}\n onOpenChange={(open) => { if (!open) setTerminalFor(null) }}\n orchestratorId={terminalFor?.item.policy.orchestratorId ?? ''}\n session={terminalFor?.session ?? ''}\n interactive={terminalFor ? canWriteTerminal(terminalFor.item, terminalFor.item.state.agentId ? agentsById[terminalFor.item.state.agentId] : undefined) : false}\n />\n </div>\n )\n}\n"],"mappings":"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"}
|
package/public/assets/{markdown-preview-impl-D0Zj7c3T.js → markdown-preview-impl-D4UIjB3I.js}
RENAMED
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
import{An as e}from"./lucide-react-DLQFnqNm.js";import{
|
|
2
|
-
//# sourceMappingURL=markdown-preview-impl-
|
|
1
|
+
import{An as e}from"./lucide-react-DLQFnqNm.js";import{S as t,x as n}from"./index-B1QUkb_O.js";import{i as r,n as i,r as a,t as o}from"./lib-BAE--T4O.js";var s=e(),c={...a,attributes:{...a.attributes,code:[...a.attributes?.code||[],[`className`,/^language-[\w-]+$/]]}};function l(e){return!e||!/^https?:\/\//i.test(e)?{}:{target:`_blank`,rel:`noreferrer`}}function u({content:e,path:a}){return(0,s.jsx)(`div`,{className:`markdown-preview max-w-none overflow-hidden p-4 text-sm leading-6`,children:(0,s.jsx)(r,{remarkPlugins:[o],rehypePlugins:[[i,c]],components:{a({href:e,children:t}){return(0,s.jsx)(`a`,{href:e,...l(e),children:t})},img({src:e,alt:t}){return e?(0,s.jsx)(n,{src:e,alt:t,className:`max-w-full`}):null},code({className:e,children:n,...r}){let i=/language-([\w-]+)/.exec(e||``),o=String(n).replace(/\n$/,``);return i?(0,s.jsx)(t,{content:o,path:`${a}.${i[1]}`,languageHint:i[1],compact:!0}):(0,s.jsx)(`code`,{className:e,...r,children:n})}},children:e})})}export{u as MarkdownPreview};
|
|
2
|
+
//# sourceMappingURL=markdown-preview-impl-D4UIjB3I.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"markdown-preview-impl-
|
|
1
|
+
{"version":3,"file":"markdown-preview-impl-D4UIjB3I.js","names":[],"sources":["../../dashboard/src/components/shared/markdown-preview-impl.tsx"],"sourcesContent":["import ReactMarkdown from 'react-markdown'\nimport rehypeSanitize, { defaultSchema } from 'rehype-sanitize'\nimport remarkGfm from 'remark-gfm'\nimport { CodePreview } from './code-preview'\nimport { ZoomableImage } from './zoomable-image'\n\nconst markdownSchema = {\n ...defaultSchema,\n attributes: {\n ...defaultSchema.attributes,\n code: [\n ...(defaultSchema.attributes?.code || []),\n ['className', /^language-[\\w-]+$/],\n ],\n },\n}\n\nfunction externalLinkProps(href?: string) {\n if (!href || !/^https?:\\/\\//i.test(href)) return {}\n return { target: '_blank', rel: 'noreferrer' }\n}\n\nexport interface MarkdownPreviewProps { content: string; path: string }\n\nexport function MarkdownPreview({ content, path }: MarkdownPreviewProps) {\n return (\n <div className=\"markdown-preview max-w-none overflow-hidden p-4 text-sm leading-6\">\n <ReactMarkdown\n remarkPlugins={[remarkGfm]}\n rehypePlugins={[[rehypeSanitize, markdownSchema]]}\n components={{\n a({ href, children }) {\n return (\n <a href={href} {...externalLinkProps(href)}>\n {children}\n </a>\n )\n },\n img({ src, alt }) {\n if (!src) return null\n return <ZoomableImage src={src} alt={alt} className=\"max-w-full\" />\n },\n code({ className, children, ...props }) {\n const match = /language-([\\w-]+)/.exec(className || '')\n const code = String(children).replace(/\\n$/, '')\n if (match) {\n return (\n <CodePreview\n content={code}\n path={`${path}.${match[1]}`}\n languageHint={match[1]}\n compact\n />\n )\n }\n return (\n <code className={className} {...props}>\n {children}\n </code>\n )\n },\n }}\n >\n {content}\n </ReactMarkdown>\n </div>\n )\n}\n"],"mappings":"oKAMM,EAAiB,CACrB,GAAG,EACH,WAAY,CACV,GAAG,EAAc,WACjB,KAAM,CACJ,GAAI,EAAc,YAAY,MAAQ,EAAE,CACxC,CAAC,YAAa,oBAAoB,CACnC,CACF,CACF,CAED,SAAS,EAAkB,EAAe,CAExC,MADI,CAAC,GAAQ,CAAC,gBAAgB,KAAK,EAAK,CAAS,EAAE,CAC5C,CAAE,OAAQ,SAAU,IAAK,aAAc,CAKhD,SAAgB,EAAgB,CAAE,UAAS,QAA8B,CACvE,OACE,EAAA,EAAA,KAAC,MAAD,CAAK,UAAU,8EACb,EAAA,EAAA,KAAC,EAAD,CACE,cAAe,CAAC,EAAU,CAC1B,cAAe,CAAC,CAAC,EAAgB,EAAe,CAAC,CACjD,WAAY,CACV,EAAE,CAAE,OAAM,YAAY,CACpB,OACE,EAAA,EAAA,KAAC,IAAD,CAAS,OAAM,GAAI,EAAkB,EAAK,CACvC,WACC,CAAA,EAGR,IAAI,CAAE,MAAK,OAAO,CAEhB,OADK,GACE,EAAA,EAAA,KAAC,EAAD,CAAoB,MAAU,MAAK,UAAU,aAAe,CAAA,CADlD,MAGnB,KAAK,CAAE,YAAW,WAAU,GAAG,GAAS,CACtC,IAAM,EAAQ,oBAAoB,KAAK,GAAa,GAAG,CACjD,EAAO,OAAO,EAAS,CAAC,QAAQ,MAAO,GAAG,CAWhD,OAVI,GAEA,EAAA,EAAA,KAAC,EAAD,CACE,QAAS,EACT,KAAM,GAAG,EAAK,GAAG,EAAM,KACvB,aAAc,EAAM,GACpB,QAAA,GACA,CAAA,EAIJ,EAAA,EAAA,KAAC,OAAD,CAAiB,YAAW,GAAI,EAC7B,WACI,CAAA,EAGZ,UAEA,EACa,CAAA,CACZ,CAAA"}
|
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
import{r as e}from"./chunk-CilyBKbf.js";import{An as t,B as n,D as r,E as ee,Hn as i,I as te,M as ne,Rt as re,fn as ie,m as ae}from"./lucide-react-DLQFnqNm.js";import{i as oe,t as a}from"./store-DKVWC6Uh.js";import{H as o,d as se,h as s}from"./display-ConJ9cJB.js";import{t as c}from"./badge-JVybSpzR.js";import{t as l}from"./button-BsMqBNJb.js";import{t as u}from"./input-dPzf0luy.js";import{D as ce,r as le,s as d}from"./index-DEZdON6c.js";import{i as f,n as p,r as m,t as h}from"./card-I8w4U656.js";var g=e(i(),1),_=t(),v=[`organization`,`role`,`project`,`task`,`interaction`,`agent`],ue=[`private`,`project`,`org`,`public`],de=[`public`,`normal`,`sensitive`,`secret`],y=[`reported`,`inferred`,`verified`];function b(){return{type:`project`,scope:`default`,title:``,content:``,tags:``,visibility:`project`,sensitivity:`normal`,confidence:`reported`,relevanceScore:`0.7`}}function x(e){return{type:e.type,scope:e.scope,title:e.title,content:e.content,tags:e.tags.join(`, `),visibility:e.visibility,sensitivity:e.sensitivity,confidence:e.confidence,relevanceScore:String(e.relevanceScore)}}function S(e){return e.split(`,`).map(e=>e.trim()).filter(Boolean)}function fe(e){let t={limit:100};e.type&&(t.type=e.type),e.scope.trim()&&(t.scope=e.scope.trim()),e.visibility&&(t.visibility=e.visibility);let n=S(e.tags);if(n.length&&(t.tags=n),e.minRelevance.trim()){let n=Number(e.minRelevance);Number.isFinite(n)&&(t.minRelevance=Math.max(0,Math.min(1,n)))}return t}function C(e){return e===`secret`?`bg-red-500/10 text-red-400 border-red-500/30`:e===`sensitive`?`bg-yellow-500/10 text-yellow-400 border-yellow-500/30`:e===`public`?`bg-blue-500/10 text-blue-400 border-blue-500/30`:`bg-zinc-500/10 text-zinc-400 border-zinc-500/30`}function w(e){return e>=.8?`text-emerald-400`:e>=.5?`text-yellow-400`:`text-zinc-400`}function pe(e){return e?Object.entries(e).map(([e,t])=>`${e}=${Array.isArray(t)?t.join(` `):String(t)}`).join(` `):`default`}function T(e){return e.filter(e=>![`system`,`user`,`channel`].includes(e.kind||``))}function E(){let e=oe(),t=a(e=>e.agents),i=a(e=>e.memories),E=a(e=>e.memoryTotal),O=a(e=>e.memoryStats),k=a(e=>e.memoryBrokerInfo),A=a(e=>e.memoryFilters),j=a(e=>e.selectedMemoryId),he=a(e=>e.memoryLoading),M=a(e=>e.set),N=a(e=>e.fetchMemories),P=a(e=>e.refreshMemory),F=a(e=>e.saveMemory),I=a(e=>e.deleteMemory),L=a(e=>e.injectMemory),R=i.find(e=>e.id===j),[z,B]=(0,g.useState)(`create`),[V,H]=(0,g.useState)(b),[U,W]=(0,g.useState)(``),[G,ge]=(0,g.useState)(`manual`),[K,_e]=(0,g.useState)(`1600`),[q,ve]=(0,g.useState)(`8`),[J,ye]=(0,g.useState)(``),Y=(0,g.useMemo)(()=>T(t),[t]);(0,g.useEffect)(()=>{!U&&Y[0]&&W(Y[0].id)},[Y,U]),(0,g.useEffect)(()=>{R&&z===`edit`&&H(x(R))},[R?.id,z]);let X=(0,g.useMemo)(()=>{let e=A.search.trim().toLowerCase();return e?i.filter(t=>[t.id,t.type,t.scope,t.title,t.content,...t.tags].some(t=>String(t).toLowerCase().includes(e))):i},[i,A.search]);function Z(e){M({memoryFilters:{...A,...e}})}async function be(){let e=Number(V.relevanceScore),t={title:V.title.trim(),content:V.content.trim(),tags:S(V.tags),visibility:V.visibility,sensitivity:V.sensitivity,confidence:V.confidence,relevanceScore:Number.isFinite(e)?Math.max(0,Math.min(1,e)):.7};if(z===`edit`&&R){await F(t,R.id);return}await F({...t,type:V.type,scope:V.scope.trim()}),B(`create`),H(b())}async function Q(e){if(!U)return;let t=await L({agentId:U,memoryIds:e,query:e?.length?void 0:fe(A),reason:G.trim()||`manual`,budget:{maxTokens:Math.max(200,Number(K)||1600),maxMemories:Math.max(1,Number(q)||8),priorityCutoff:3}});t?.command?.id&&ye(t.command.id)}function xe(){B(`create`),H(b())}function Se(){R&&(B(`edit`),H(x(R)))}let Ce=V.title.trim()&&V.content.trim()&&(z===`edit`||V.scope.trim()),$=pe(k?.config);return(0,_.jsxs)(`div`,{className:`space-y-4`,children:[(0,_.jsxs)(`div`,{className:`flex items-center justify-between gap-3 flex-wrap`,children:[(0,_.jsxs)(`div`,{className:`flex items-center gap-2`,children:[(0,_.jsx)(ie,{className:`w-5 h-5 text-muted-foreground`}),(0,_.jsx)(`h2`,{className:`text-lg font-semibold`,children:`Memory`}),(0,_.jsx)(c,{variant:`secondary`,children:O?.total??E}),k&&(0,_.jsx)(c,{className:k.external?`bg-blue-500/15 text-blue-400 border-blue-500/30`:`bg-emerald-500/15 text-emerald-400 border-emerald-500/30`,children:k.type})]}),(0,_.jsxs)(l,{size:`sm`,variant:`outline`,className:`gap-1.5`,onClick:()=>P(),children:[(0,_.jsx)(ne,{className:`w-3.5 h-3.5 ${he?`animate-spin`:``}`}),`Refresh`]})]}),(0,_.jsxs)(`div`,{className:`grid gap-3 xl:grid-cols-[1.2fr_0.8fr]`,children:[(0,_.jsxs)(h,{children:[(0,_.jsx)(m,{className:`py-3 px-4`,children:(0,_.jsxs)(`div`,{className:`flex items-center justify-between gap-2`,children:[(0,_.jsxs)(f,{className:`text-sm font-medium flex items-center gap-2`,children:[(0,_.jsx)(re,{className:`w-4 h-4 text-muted-foreground`}),`Broker`]}),k?.capabilities&&(0,_.jsx)(`div`,{className:`flex flex-wrap gap-1 justify-end`,children:Object.entries(k.capabilities).filter(([,e])=>!!e).map(([e])=>(0,_.jsx)(c,{variant:`outline`,className:`text-[10px] px-1.5 py-0`,children:e},e))})]})}),(0,_.jsxs)(p,{className:`px-4 pb-4 space-y-3`,children:[(0,_.jsxs)(`div`,{className:`grid grid-cols-2 md:grid-cols-5 gap-2`,children:[(0,_.jsx)(D,{label:`total`,value:O?.total??E}),(0,_.jsx)(D,{label:`project`,value:O?.byType?.project??0}),(0,_.jsx)(D,{label:`task`,value:O?.byType?.task??0}),(0,_.jsx)(D,{label:`agent`,value:O?.byType?.agent??0}),(0,_.jsx)(D,{label:`sensitive`,value:(O?.bySensitivity?.sensitive??0)+(O?.bySensitivity?.secret??0)})]}),(0,_.jsx)(`div`,{className:`font-mono text-xs text-muted-foreground truncate`,title:$,children:$})]})]}),(0,_.jsxs)(h,{children:[(0,_.jsx)(m,{className:`py-3 px-4`,children:(0,_.jsx)(f,{className:`text-sm font-medium`,children:`Inject`})}),(0,_.jsxs)(p,{className:`px-4 pb-4 space-y-2`,children:[(0,_.jsxs)(`div`,{className:`grid grid-cols-1 sm:grid-cols-2 gap-2`,children:[(0,_.jsxs)(`select`,{className:`rounded-md border border-input bg-background px-2 py-1.5 text-sm`,value:U,onChange:e=>W(e.target.value),children:[(0,_.jsx)(`option`,{value:``,children:`Agent`}),Y.map(e=>(0,_.jsx)(`option`,{value:e.id,children:se(e)},e.id))]}),(0,_.jsx)(u,{value:G,onChange:e=>ge(e.target.value),placeholder:`reason`}),(0,_.jsx)(u,{value:K,onChange:e=>_e(e.target.value),inputMode:`numeric`,placeholder:`tokens`}),(0,_.jsx)(u,{value:q,onChange:e=>ve(e.target.value),inputMode:`numeric`,placeholder:`memories`})]}),(0,_.jsxs)(`div`,{className:`flex flex-wrap gap-2`,children:[(0,_.jsxs)(l,{size:`sm`,className:`gap-1.5`,disabled:!U||!R,onClick:()=>R&&Q([R.id]),children:[(0,_.jsx)(ee,{className:`w-3.5 h-3.5`}),`Selected`]}),(0,_.jsxs)(l,{size:`sm`,variant:`outline`,className:`gap-1.5`,disabled:!U,onClick:()=>Q(),children:[(0,_.jsx)(r,{className:`w-3.5 h-3.5`}),`Filtered`]}),J&&(0,_.jsx)(c,{variant:`outline`,className:`font-mono text-[10px]`,children:J})]})]})]})]}),(0,_.jsxs)(`div`,{className:`grid gap-4 xl:grid-cols-[minmax(0,1fr)_420px]`,children:[(0,_.jsxs)(`div`,{className:`space-y-3`,children:[(0,_.jsx)(h,{children:(0,_.jsx)(p,{className:`p-3`,children:(0,_.jsxs)(`div`,{className:`grid grid-cols-2 lg:grid-cols-6 gap-2`,children:[(0,_.jsx)(u,{className:`lg:col-span-2`,value:A.search,onChange:e=>Z({search:e.target.value}),placeholder:`Search visible memories`}),(0,_.jsxs)(`select`,{className:`rounded-md border border-input bg-background px-2 py-1.5 text-sm`,value:A.type,onChange:e=>Z({type:e.target.value}),children:[(0,_.jsx)(`option`,{value:``,children:`All types`}),v.map(e=>(0,_.jsx)(`option`,{value:e,children:e},e))]}),(0,_.jsx)(u,{value:A.scope,onChange:e=>Z({scope:e.target.value}),placeholder:`scope`}),(0,_.jsx)(u,{value:A.tags,onChange:e=>Z({tags:e.target.value}),placeholder:`tags`}),(0,_.jsxs)(`div`,{className:`flex gap-2`,children:[(0,_.jsx)(u,{value:A.minRelevance,onChange:e=>Z({minRelevance:e.target.value}),placeholder:`score`}),(0,_.jsx)(l,{size:`sm`,variant:`outline`,className:`h-8 px-2`,onClick:()=>N(),children:(0,_.jsx)(r,{className:`w-4 h-4`})})]})]})})}),(0,_.jsx)(d,{className:`h-[calc(100dvh-24rem)] min-h-[360px]`,children:(0,_.jsxs)(`div`,{className:`space-y-2 pr-2`,children:[X.length===0&&(0,_.jsx)(`div`,{className:`text-center text-sm text-muted-foreground py-16`,children:`No memories`}),X.map(t=>(0,_.jsx)(me,{memory:t,selected:t.id===j,now:e,onSelect:()=>M({selectedMemoryId:t.id})},t.id))]})})]}),(0,_.jsxs)(h,{children:[(0,_.jsx)(m,{className:`py-3 px-4`,children:(0,_.jsxs)(`div`,{className:`flex items-center justify-between gap-2`,children:[(0,_.jsx)(f,{className:`text-sm font-medium`,children:z===`edit`?`Edit Memory`:`New Memory`}),(0,_.jsxs)(`div`,{className:`flex gap-1`,children:[(0,_.jsx)(l,{size:`sm`,variant:`ghost`,className:`h-7 px-2`,onClick:xe,title:`New memory`,children:(0,_.jsx)(te,{className:`w-3.5 h-3.5`})}),(0,_.jsx)(l,{size:`sm`,variant:`ghost`,className:`h-7 px-2`,onClick:Se,disabled:!R,title:`Edit selected`,children:(0,_.jsx)(n,{className:`w-3.5 h-3.5`})}),(0,_.jsx)(l,{size:`sm`,variant:`ghost`,className:`h-7 px-2 text-red-400 hover:text-red-300`,disabled:!R,onClick:()=>R&&I(R.id),title:`Delete selected`,children:(0,_.jsx)(ae,{className:`w-3.5 h-3.5`})})]})]})}),(0,_.jsxs)(p,{className:`px-4 pb-4 space-y-3`,children:[R&&(0,_.jsxs)(_.Fragment,{children:[(0,_.jsxs)(`div`,{className:`space-y-2`,children:[(0,_.jsxs)(`div`,{className:`flex flex-wrap gap-1`,children:[(0,_.jsx)(c,{variant:`secondary`,children:R.type}),(0,_.jsx)(c,{variant:`outline`,children:R.scope}),(0,_.jsx)(c,{className:C(R.sensitivity),children:R.sensitivity}),(0,_.jsx)(c,{variant:`outline`,className:w(R.relevanceScore),children:R.relevanceScore.toFixed(2)})]}),(0,_.jsxs)(`div`,{className:`text-xs text-muted-foreground`,title:s(R.updatedAt),children:[`updated `,o(e,R.updatedAt)]}),(0,_.jsx)(`p`,{className:`text-sm font-medium`,children:R.title}),(0,_.jsx)(`p`,{className:`text-xs text-muted-foreground whitespace-pre-wrap max-h-24 overflow-auto`,children:R.content})]}),(0,_.jsx)(ce,{})]}),(0,_.jsxs)(`div`,{className:`grid grid-cols-2 gap-2`,children:[(0,_.jsx)(`select`,{className:`rounded-md border border-input bg-background px-2 py-1.5 text-sm`,disabled:z===`edit`,value:V.type,onChange:e=>H({...V,type:e.target.value}),children:v.map(e=>(0,_.jsx)(`option`,{value:e,children:e},e))}),(0,_.jsx)(u,{disabled:z===`edit`,value:V.scope,onChange:e=>H({...V,scope:e.target.value}),placeholder:`scope`}),(0,_.jsx)(u,{className:`col-span-2`,value:V.title,onChange:e=>H({...V,title:e.target.value}),placeholder:`title`}),(0,_.jsx)(le,{className:`col-span-2 min-h-32`,value:V.content,onChange:e=>H({...V,content:e.target.value}),placeholder:`content`}),(0,_.jsx)(u,{className:`col-span-2`,value:V.tags,onChange:e=>H({...V,tags:e.target.value}),placeholder:`tags`}),(0,_.jsx)(`select`,{className:`rounded-md border border-input bg-background px-2 py-1.5 text-sm`,value:V.visibility,onChange:e=>H({...V,visibility:e.target.value}),children:ue.map(e=>(0,_.jsx)(`option`,{value:e,children:e},e))}),(0,_.jsx)(`select`,{className:`rounded-md border border-input bg-background px-2 py-1.5 text-sm`,value:V.sensitivity,onChange:e=>H({...V,sensitivity:e.target.value}),children:de.map(e=>(0,_.jsx)(`option`,{value:e,children:e},e))}),(0,_.jsx)(`select`,{className:`rounded-md border border-input bg-background px-2 py-1.5 text-sm`,value:V.confidence,onChange:e=>H({...V,confidence:e.target.value}),children:y.map(e=>(0,_.jsx)(`option`,{value:e,children:e},e))}),(0,_.jsx)(u,{value:V.relevanceScore,onChange:e=>H({...V,relevanceScore:e.target.value}),inputMode:`decimal`,placeholder:`score`})]}),(0,_.jsx)(l,{size:`sm`,className:`w-full`,disabled:!Ce,onClick:be,children:z===`edit`?`Save memory`:`Create memory`})]})]})]})]})}function D({label:e,value:t}){return(0,_.jsxs)(`div`,{className:`rounded-md border border-border bg-muted/20 px-3 py-2`,children:[(0,_.jsx)(`div`,{className:`text-lg font-semibold tabular-nums`,children:t}),(0,_.jsx)(`div`,{className:`text-xs text-muted-foreground`,children:e})]})}function me({memory:e,selected:t,now:n,onSelect:r}){return(0,_.jsx)(`button`,{onClick:r,className:`w-full text-left rounded-md border p-3 transition-colors ${t?`border-primary bg-primary/5`:`border-border bg-card hover:border-zinc-600`}`,children:(0,_.jsxs)(`div`,{className:`flex items-start justify-between gap-3`,children:[(0,_.jsxs)(`div`,{className:`min-w-0`,children:[(0,_.jsxs)(`div`,{className:`flex items-center gap-1.5 flex-wrap`,children:[(0,_.jsx)(c,{variant:`secondary`,className:`text-[10px]`,children:e.type}),(0,_.jsx)(c,{variant:`outline`,className:`text-[10px] font-mono`,children:e.scope}),(0,_.jsx)(c,{className:`text-[10px] ${C(e.sensitivity)}`,children:e.sensitivity})]}),(0,_.jsx)(`div`,{className:`font-medium text-sm mt-2 truncate`,children:e.title}),(0,_.jsx)(`div`,{className:`text-xs text-muted-foreground line-clamp-2 mt-1`,children:e.content}),e.tags.length>0&&(0,_.jsx)(`div`,{className:`flex gap-1 flex-wrap mt-2`,children:e.tags.slice(0,8).map(e=>(0,_.jsxs)(c,{variant:`outline`,className:`text-[10px] text-zinc-400`,children:[`#`,e]},e))})]}),(0,_.jsxs)(`div`,{className:`text-right shrink-0 space-y-1`,children:[(0,_.jsx)(`div`,{className:`text-sm font-semibold tabular-nums ${w(e.relevanceScore)}`,children:e.relevanceScore.toFixed(2)}),(0,_.jsx)(`div`,{className:`text-[10px] text-muted-foreground`,title:s(e.updatedAt),children:o(n,e.updatedAt)}),e.accessCount>0&&(0,_.jsxs)(`div`,{className:`text-[10px] text-muted-foreground`,children:[e.accessCount,` hits`]})]})]})})}export{E as MemoryView};
|
|
2
|
-
//# sourceMappingURL=memory-
|
|
1
|
+
import{r as e}from"./chunk-CilyBKbf.js";import{An as t,B as n,D as r,E as ee,Hn as i,I as te,M as ne,Rt as re,fn as ie,m as ae}from"./lucide-react-DLQFnqNm.js";import{i as oe,t as a}from"./store-Bo72e9My.js";import{H as o,d as se,h as s}from"./display-ConJ9cJB.js";import{t as c}from"./badge-JVybSpzR.js";import{t as l}from"./button-BsMqBNJb.js";import{t as u}from"./input-dPzf0luy.js";import{O as ce,r as le,s as d}from"./index-B1QUkb_O.js";import{i as f,n as p,r as m,t as h}from"./card-I8w4U656.js";var g=e(i(),1),_=t(),v=[`organization`,`role`,`project`,`task`,`interaction`,`agent`],ue=[`private`,`project`,`org`,`public`],de=[`public`,`normal`,`sensitive`,`secret`],y=[`reported`,`inferred`,`verified`];function b(){return{type:`project`,scope:`default`,title:``,content:``,tags:``,visibility:`project`,sensitivity:`normal`,confidence:`reported`,relevanceScore:`0.7`}}function x(e){return{type:e.type,scope:e.scope,title:e.title,content:e.content,tags:e.tags.join(`, `),visibility:e.visibility,sensitivity:e.sensitivity,confidence:e.confidence,relevanceScore:String(e.relevanceScore)}}function S(e){return e.split(`,`).map(e=>e.trim()).filter(Boolean)}function fe(e){let t={limit:100};e.type&&(t.type=e.type),e.scope.trim()&&(t.scope=e.scope.trim()),e.visibility&&(t.visibility=e.visibility);let n=S(e.tags);if(n.length&&(t.tags=n),e.minRelevance.trim()){let n=Number(e.minRelevance);Number.isFinite(n)&&(t.minRelevance=Math.max(0,Math.min(1,n)))}return t}function C(e){return e===`secret`?`bg-red-500/10 text-red-400 border-red-500/30`:e===`sensitive`?`bg-yellow-500/10 text-yellow-400 border-yellow-500/30`:e===`public`?`bg-blue-500/10 text-blue-400 border-blue-500/30`:`bg-zinc-500/10 text-zinc-400 border-zinc-500/30`}function w(e){return e>=.8?`text-emerald-400`:e>=.5?`text-yellow-400`:`text-zinc-400`}function pe(e){return e?Object.entries(e).map(([e,t])=>`${e}=${Array.isArray(t)?t.join(` `):String(t)}`).join(` `):`default`}function T(e){return e.filter(e=>![`system`,`user`,`channel`].includes(e.kind||``))}function E(){let e=oe(),t=a(e=>e.agents),i=a(e=>e.memories),E=a(e=>e.memoryTotal),O=a(e=>e.memoryStats),k=a(e=>e.memoryBrokerInfo),A=a(e=>e.memoryFilters),j=a(e=>e.selectedMemoryId),he=a(e=>e.memoryLoading),M=a(e=>e.set),N=a(e=>e.fetchMemories),P=a(e=>e.refreshMemory),F=a(e=>e.saveMemory),I=a(e=>e.deleteMemory),L=a(e=>e.injectMemory),R=i.find(e=>e.id===j),[z,B]=(0,g.useState)(`create`),[V,H]=(0,g.useState)(b),[U,W]=(0,g.useState)(``),[G,ge]=(0,g.useState)(`manual`),[K,_e]=(0,g.useState)(`1600`),[q,ve]=(0,g.useState)(`8`),[J,ye]=(0,g.useState)(``),Y=(0,g.useMemo)(()=>T(t),[t]);(0,g.useEffect)(()=>{!U&&Y[0]&&W(Y[0].id)},[Y,U]),(0,g.useEffect)(()=>{R&&z===`edit`&&H(x(R))},[R?.id,z]);let X=(0,g.useMemo)(()=>{let e=A.search.trim().toLowerCase();return e?i.filter(t=>[t.id,t.type,t.scope,t.title,t.content,...t.tags].some(t=>String(t).toLowerCase().includes(e))):i},[i,A.search]);function Z(e){M({memoryFilters:{...A,...e}})}async function be(){let e=Number(V.relevanceScore),t={title:V.title.trim(),content:V.content.trim(),tags:S(V.tags),visibility:V.visibility,sensitivity:V.sensitivity,confidence:V.confidence,relevanceScore:Number.isFinite(e)?Math.max(0,Math.min(1,e)):.7};if(z===`edit`&&R){await F(t,R.id);return}await F({...t,type:V.type,scope:V.scope.trim()}),B(`create`),H(b())}async function Q(e){if(!U)return;let t=await L({agentId:U,memoryIds:e,query:e?.length?void 0:fe(A),reason:G.trim()||`manual`,budget:{maxTokens:Math.max(200,Number(K)||1600),maxMemories:Math.max(1,Number(q)||8),priorityCutoff:3}});t?.command?.id&&ye(t.command.id)}function xe(){B(`create`),H(b())}function Se(){R&&(B(`edit`),H(x(R)))}let Ce=V.title.trim()&&V.content.trim()&&(z===`edit`||V.scope.trim()),$=pe(k?.config);return(0,_.jsxs)(`div`,{className:`space-y-4`,children:[(0,_.jsxs)(`div`,{className:`flex items-center justify-between gap-3 flex-wrap`,children:[(0,_.jsxs)(`div`,{className:`flex items-center gap-2`,children:[(0,_.jsx)(ie,{className:`w-5 h-5 text-muted-foreground`}),(0,_.jsx)(`h2`,{className:`text-lg font-semibold`,children:`Memory`}),(0,_.jsx)(c,{variant:`secondary`,children:O?.total??E}),k&&(0,_.jsx)(c,{className:k.external?`bg-blue-500/15 text-blue-400 border-blue-500/30`:`bg-emerald-500/15 text-emerald-400 border-emerald-500/30`,children:k.type})]}),(0,_.jsxs)(l,{size:`sm`,variant:`outline`,className:`gap-1.5`,onClick:()=>P(),children:[(0,_.jsx)(ne,{className:`w-3.5 h-3.5 ${he?`animate-spin`:``}`}),`Refresh`]})]}),(0,_.jsxs)(`div`,{className:`grid gap-3 xl:grid-cols-[1.2fr_0.8fr]`,children:[(0,_.jsxs)(h,{children:[(0,_.jsx)(m,{className:`py-3 px-4`,children:(0,_.jsxs)(`div`,{className:`flex items-center justify-between gap-2`,children:[(0,_.jsxs)(f,{className:`text-sm font-medium flex items-center gap-2`,children:[(0,_.jsx)(re,{className:`w-4 h-4 text-muted-foreground`}),`Broker`]}),k?.capabilities&&(0,_.jsx)(`div`,{className:`flex flex-wrap gap-1 justify-end`,children:Object.entries(k.capabilities).filter(([,e])=>!!e).map(([e])=>(0,_.jsx)(c,{variant:`outline`,className:`text-[10px] px-1.5 py-0`,children:e},e))})]})}),(0,_.jsxs)(p,{className:`px-4 pb-4 space-y-3`,children:[(0,_.jsxs)(`div`,{className:`grid grid-cols-2 md:grid-cols-5 gap-2`,children:[(0,_.jsx)(D,{label:`total`,value:O?.total??E}),(0,_.jsx)(D,{label:`project`,value:O?.byType?.project??0}),(0,_.jsx)(D,{label:`task`,value:O?.byType?.task??0}),(0,_.jsx)(D,{label:`agent`,value:O?.byType?.agent??0}),(0,_.jsx)(D,{label:`sensitive`,value:(O?.bySensitivity?.sensitive??0)+(O?.bySensitivity?.secret??0)})]}),(0,_.jsx)(`div`,{className:`font-mono text-xs text-muted-foreground truncate`,title:$,children:$})]})]}),(0,_.jsxs)(h,{children:[(0,_.jsx)(m,{className:`py-3 px-4`,children:(0,_.jsx)(f,{className:`text-sm font-medium`,children:`Inject`})}),(0,_.jsxs)(p,{className:`px-4 pb-4 space-y-2`,children:[(0,_.jsxs)(`div`,{className:`grid grid-cols-1 sm:grid-cols-2 gap-2`,children:[(0,_.jsxs)(`select`,{className:`rounded-md border border-input bg-background px-2 py-1.5 text-sm`,value:U,onChange:e=>W(e.target.value),children:[(0,_.jsx)(`option`,{value:``,children:`Agent`}),Y.map(e=>(0,_.jsx)(`option`,{value:e.id,children:se(e)},e.id))]}),(0,_.jsx)(u,{value:G,onChange:e=>ge(e.target.value),placeholder:`reason`}),(0,_.jsx)(u,{value:K,onChange:e=>_e(e.target.value),inputMode:`numeric`,placeholder:`tokens`}),(0,_.jsx)(u,{value:q,onChange:e=>ve(e.target.value),inputMode:`numeric`,placeholder:`memories`})]}),(0,_.jsxs)(`div`,{className:`flex flex-wrap gap-2`,children:[(0,_.jsxs)(l,{size:`sm`,className:`gap-1.5`,disabled:!U||!R,onClick:()=>R&&Q([R.id]),children:[(0,_.jsx)(ee,{className:`w-3.5 h-3.5`}),`Selected`]}),(0,_.jsxs)(l,{size:`sm`,variant:`outline`,className:`gap-1.5`,disabled:!U,onClick:()=>Q(),children:[(0,_.jsx)(r,{className:`w-3.5 h-3.5`}),`Filtered`]}),J&&(0,_.jsx)(c,{variant:`outline`,className:`font-mono text-[10px]`,children:J})]})]})]})]}),(0,_.jsxs)(`div`,{className:`grid gap-4 xl:grid-cols-[minmax(0,1fr)_420px]`,children:[(0,_.jsxs)(`div`,{className:`space-y-3`,children:[(0,_.jsx)(h,{children:(0,_.jsx)(p,{className:`p-3`,children:(0,_.jsxs)(`div`,{className:`grid grid-cols-2 lg:grid-cols-6 gap-2`,children:[(0,_.jsx)(u,{className:`lg:col-span-2`,value:A.search,onChange:e=>Z({search:e.target.value}),placeholder:`Search visible memories`}),(0,_.jsxs)(`select`,{className:`rounded-md border border-input bg-background px-2 py-1.5 text-sm`,value:A.type,onChange:e=>Z({type:e.target.value}),children:[(0,_.jsx)(`option`,{value:``,children:`All types`}),v.map(e=>(0,_.jsx)(`option`,{value:e,children:e},e))]}),(0,_.jsx)(u,{value:A.scope,onChange:e=>Z({scope:e.target.value}),placeholder:`scope`}),(0,_.jsx)(u,{value:A.tags,onChange:e=>Z({tags:e.target.value}),placeholder:`tags`}),(0,_.jsxs)(`div`,{className:`flex gap-2`,children:[(0,_.jsx)(u,{value:A.minRelevance,onChange:e=>Z({minRelevance:e.target.value}),placeholder:`score`}),(0,_.jsx)(l,{size:`sm`,variant:`outline`,className:`h-8 px-2`,onClick:()=>N(),children:(0,_.jsx)(r,{className:`w-4 h-4`})})]})]})})}),(0,_.jsx)(d,{className:`h-[calc(100dvh-24rem)] min-h-[360px]`,children:(0,_.jsxs)(`div`,{className:`space-y-2 pr-2`,children:[X.length===0&&(0,_.jsx)(`div`,{className:`text-center text-sm text-muted-foreground py-16`,children:`No memories`}),X.map(t=>(0,_.jsx)(me,{memory:t,selected:t.id===j,now:e,onSelect:()=>M({selectedMemoryId:t.id})},t.id))]})})]}),(0,_.jsxs)(h,{children:[(0,_.jsx)(m,{className:`py-3 px-4`,children:(0,_.jsxs)(`div`,{className:`flex items-center justify-between gap-2`,children:[(0,_.jsx)(f,{className:`text-sm font-medium`,children:z===`edit`?`Edit Memory`:`New Memory`}),(0,_.jsxs)(`div`,{className:`flex gap-1`,children:[(0,_.jsx)(l,{size:`sm`,variant:`ghost`,className:`h-7 px-2`,onClick:xe,title:`New memory`,children:(0,_.jsx)(te,{className:`w-3.5 h-3.5`})}),(0,_.jsx)(l,{size:`sm`,variant:`ghost`,className:`h-7 px-2`,onClick:Se,disabled:!R,title:`Edit selected`,children:(0,_.jsx)(n,{className:`w-3.5 h-3.5`})}),(0,_.jsx)(l,{size:`sm`,variant:`ghost`,className:`h-7 px-2 text-red-400 hover:text-red-300`,disabled:!R,onClick:()=>R&&I(R.id),title:`Delete selected`,children:(0,_.jsx)(ae,{className:`w-3.5 h-3.5`})})]})]})}),(0,_.jsxs)(p,{className:`px-4 pb-4 space-y-3`,children:[R&&(0,_.jsxs)(_.Fragment,{children:[(0,_.jsxs)(`div`,{className:`space-y-2`,children:[(0,_.jsxs)(`div`,{className:`flex flex-wrap gap-1`,children:[(0,_.jsx)(c,{variant:`secondary`,children:R.type}),(0,_.jsx)(c,{variant:`outline`,children:R.scope}),(0,_.jsx)(c,{className:C(R.sensitivity),children:R.sensitivity}),(0,_.jsx)(c,{variant:`outline`,className:w(R.relevanceScore),children:R.relevanceScore.toFixed(2)})]}),(0,_.jsxs)(`div`,{className:`text-xs text-muted-foreground`,title:s(R.updatedAt),children:[`updated `,o(e,R.updatedAt)]}),(0,_.jsx)(`p`,{className:`text-sm font-medium`,children:R.title}),(0,_.jsx)(`p`,{className:`text-xs text-muted-foreground whitespace-pre-wrap max-h-24 overflow-auto`,children:R.content})]}),(0,_.jsx)(ce,{})]}),(0,_.jsxs)(`div`,{className:`grid grid-cols-2 gap-2`,children:[(0,_.jsx)(`select`,{className:`rounded-md border border-input bg-background px-2 py-1.5 text-sm`,disabled:z===`edit`,value:V.type,onChange:e=>H({...V,type:e.target.value}),children:v.map(e=>(0,_.jsx)(`option`,{value:e,children:e},e))}),(0,_.jsx)(u,{disabled:z===`edit`,value:V.scope,onChange:e=>H({...V,scope:e.target.value}),placeholder:`scope`}),(0,_.jsx)(u,{className:`col-span-2`,value:V.title,onChange:e=>H({...V,title:e.target.value}),placeholder:`title`}),(0,_.jsx)(le,{className:`col-span-2 min-h-32`,value:V.content,onChange:e=>H({...V,content:e.target.value}),placeholder:`content`}),(0,_.jsx)(u,{className:`col-span-2`,value:V.tags,onChange:e=>H({...V,tags:e.target.value}),placeholder:`tags`}),(0,_.jsx)(`select`,{className:`rounded-md border border-input bg-background px-2 py-1.5 text-sm`,value:V.visibility,onChange:e=>H({...V,visibility:e.target.value}),children:ue.map(e=>(0,_.jsx)(`option`,{value:e,children:e},e))}),(0,_.jsx)(`select`,{className:`rounded-md border border-input bg-background px-2 py-1.5 text-sm`,value:V.sensitivity,onChange:e=>H({...V,sensitivity:e.target.value}),children:de.map(e=>(0,_.jsx)(`option`,{value:e,children:e},e))}),(0,_.jsx)(`select`,{className:`rounded-md border border-input bg-background px-2 py-1.5 text-sm`,value:V.confidence,onChange:e=>H({...V,confidence:e.target.value}),children:y.map(e=>(0,_.jsx)(`option`,{value:e,children:e},e))}),(0,_.jsx)(u,{value:V.relevanceScore,onChange:e=>H({...V,relevanceScore:e.target.value}),inputMode:`decimal`,placeholder:`score`})]}),(0,_.jsx)(l,{size:`sm`,className:`w-full`,disabled:!Ce,onClick:be,children:z===`edit`?`Save memory`:`Create memory`})]})]})]})]})}function D({label:e,value:t}){return(0,_.jsxs)(`div`,{className:`rounded-md border border-border bg-muted/20 px-3 py-2`,children:[(0,_.jsx)(`div`,{className:`text-lg font-semibold tabular-nums`,children:t}),(0,_.jsx)(`div`,{className:`text-xs text-muted-foreground`,children:e})]})}function me({memory:e,selected:t,now:n,onSelect:r}){return(0,_.jsx)(`button`,{onClick:r,className:`w-full text-left rounded-md border p-3 transition-colors ${t?`border-primary bg-primary/5`:`border-border bg-card hover:border-zinc-600`}`,children:(0,_.jsxs)(`div`,{className:`flex items-start justify-between gap-3`,children:[(0,_.jsxs)(`div`,{className:`min-w-0`,children:[(0,_.jsxs)(`div`,{className:`flex items-center gap-1.5 flex-wrap`,children:[(0,_.jsx)(c,{variant:`secondary`,className:`text-[10px]`,children:e.type}),(0,_.jsx)(c,{variant:`outline`,className:`text-[10px] font-mono`,children:e.scope}),(0,_.jsx)(c,{className:`text-[10px] ${C(e.sensitivity)}`,children:e.sensitivity})]}),(0,_.jsx)(`div`,{className:`font-medium text-sm mt-2 truncate`,children:e.title}),(0,_.jsx)(`div`,{className:`text-xs text-muted-foreground line-clamp-2 mt-1`,children:e.content}),e.tags.length>0&&(0,_.jsx)(`div`,{className:`flex gap-1 flex-wrap mt-2`,children:e.tags.slice(0,8).map(e=>(0,_.jsxs)(c,{variant:`outline`,className:`text-[10px] text-zinc-400`,children:[`#`,e]},e))})]}),(0,_.jsxs)(`div`,{className:`text-right shrink-0 space-y-1`,children:[(0,_.jsx)(`div`,{className:`text-sm font-semibold tabular-nums ${w(e.relevanceScore)}`,children:e.relevanceScore.toFixed(2)}),(0,_.jsx)(`div`,{className:`text-[10px] text-muted-foreground`,title:s(e.updatedAt),children:o(n,e.updatedAt)}),e.accessCount>0&&(0,_.jsxs)(`div`,{className:`text-[10px] text-muted-foreground`,children:[e.accessCount,` hits`]})]})]})})}export{E as MemoryView};
|
|
2
|
+
//# sourceMappingURL=memory-SVCob0fo.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"memory-TATN2vZf.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-SVCob0fo.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,An as r,Hn as i,Y as a,Zt as o,_n as s,_t as c,i as l,m as u,p as d,z as f}from"./lucide-react-DLQFnqNm.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,An as r,Hn as i,Y as a,Zt as o,_n as s,_t as c,i as l,m as u,p as d,z as f}from"./lucide-react-DLQFnqNm.js";import{i as p,t as m}from"./store-Bo72e9My.js";import{D as h,H as g,d as _,f as v,h as y}from"./display-ConJ9cJB.js";import{t as b}from"./badge-JVybSpzR.js";import{t as x}from"./button-BsMqBNJb.js";import{N as S,R as C,t as w}from"./index-B1QUkb_O.js";import{n as T,t as E}from"./card-I8w4U656.js";var D=e(i(),1),O=r();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),i=m(e=>e.selectedAgent),o=m(e=>e.channelFilter),s=m(e=>e.tagFilter),c=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=S(),k=C(),A=(0,D.useMemo)(()=>{let e=[...n].sort((e,t)=>t.id-e.id);return i&&(e=e.filter(e=>e.from===i||e.to===i)),o&&(e=e.filter(e=>e.channel===o)),s&&(e=e.filter(e=>((r[e.from]||r[e.to])?.tags||[]).includes(s))),e},[n,i,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:i,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))]})]}),c&&(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)(a,{className:`w-4 h-4 shrink-0`}),(0,O.jsxs)(`span`,{children:[`Replying to message #`,c.id,` from `,(0,O.jsx)(`strong`,{children:N(c.from)})]}),(0,O.jsx)(x,{variant:`ghost`,size:`sm`,className:`ml-auto h-6 px-1.5`,onClick:T,children:(0,O.jsx)(l,{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:i,onReply:l,onClaim:d,onRetry:f,onMarkDead:p,onClear:_,onDelete:S,onThread:C,replyCount:j}){let[M,P]=(0,D.useState)(!1),F=h(e),I=m(e=>e.openFilesAt),L=e.deliveryStatus===`queued`||e.deliveryStatus===`failed`||e.deliveryStatus===`dead`||!!(e.deliveryLastError||e.deliveryPoisonReason),R=i[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:v(e.to,i)}),e.channel&&(0,O.jsxs)(b,{variant:`outline`,className:`text-xs px-1.5 py-0 gap-0.5`,children:[(0,O.jsx)(c,{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)(o,{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:y(e.createdAt),children:[(0,O.jsxs)(`span`,{className:`opacity-40 mr-1`,children:[`#`,e.id]}),g(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)(a,{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)(o,{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)(s,{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)(o,{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 ${g(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-CHK24Uxx.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"messages-3rS1lxIf.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-CHK24Uxx.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":"weAcA,SAAS,EAAoB,EAAa,EAAsB,CAC9D,GAAI,EAAK,WAAW,IAAI,CAAE,OAAO,EACjC,IAAM,EAAQ,GAAG,EAAI,QAAQ,OAAQ,GAAG,CAAC,GAAG,IAAO,MAAM,IAAI,CACvD,EAAkB,EAAE,CAC1B,IAAK,IAAM,KAAQ,EACb,CAAC,GAAQ,IAAS,MAClB,IAAS,KAAM,EAAM,KAAK,CACzB,EAAM,KAAK,EAAK,EAEvB,MAAO,IAAM,EAAM,KAAK,IAAI,CAG9B,SAAS,EAAW,EAAsB,CACxC,IAAM,EAAU,EAAK,QAAQ,OAAQ,GAAG,CAClC,EAAQ,EAAQ,YAAY,IAAI,CACtC,OAAO,EAAQ,EAAI,EAAQ,MAAM,EAAG,EAAM,CAAG,IAG/C,SAAgB,GAAe,CAC7B,IAAM,EAAM,GAAQ,CACd,EAAW,EAAe,GAAM,EAAE,SAAS,CAC3C,EAAa,EAAe,GAAM,EAAE,WAAW,CAC/C,EAAgB,EAAe,GAAM,EAAE,cAAc,CACrD,EAAgB,EAAe,GAAM,EAAE,cAAc,CACrD,EAAY,EAAe,GAAM,EAAE,UAAU,CAC7C,EAAU,EAAe,GAAM,EAAE,QAAQ,CACzC,EAAM,EAAe,GAAM,EAAE,IAAI,CACjC,EAAc,EAAe,GAAM,EAAE,YAAY,CACjD,EAAa,EAAe,GAAM,EAAE,WAAW,CAC/C,EAAU,EAAe,GAAM,EAAE,QAAQ,CACzC,EAAmB,EAAe,GAAM,EAAE,iBAAiB,CAC3D,EAAkB,EAAe,GAAM,EAAE,gBAAgB,CACzD,EAAa,EAAe,GAAM,EAAE,WAAW,CAC/C,EAAc,EAAe,GAAM,EAAE,YAAY,CAEjD,EAAgB,GAAkB,CAClC,EAAa,GAAe,CAE5B,GAAA,EAAA,EAAA,aAAyB,CAC7B,IAAI,EAAO,CAAC,GAAG,EAAS,CAAC,MAAM,EAAG,IAAM,EAAE,GAAK,EAAE,GAAG,CAOpD,OANI,IAAe,EAAO,EAAK,OAAQ,GAAM,EAAE,OAAS,GAAiB,EAAE,KAAO,EAAc,EAC5F,IAAe,EAAO,EAAK,OAAQ,GAAM,EAAE,UAAY,EAAc,EACrE,IAAW,EAAO,EAAK,OAAQ,KACnB,EAAW,EAAE,OAAS,EAAW,EAAE,MAClC,MAAQ,EAAE,EAAE,SAAS,EAAU,CAC9C,EACK,GACN,CAAC,EAAU,EAAe,EAAe,EAAW,EAAW,CAAC,CAG7D,GAAA,EAAA,EAAA,aAAwB,CAC5B,IAAM,EAAmB,EAAE,CACrB,EAAU,IAAI,IACpB,IAAK,IAAM,KAAO,EAAU,CAC1B,IAAM,EAAM,EAAI,UAAY,EAAI,GAC3B,EAAQ,IAAI,EAAI,EAAE,EAAQ,IAAI,EAAK,EAAE,CAAC,CAC3C,EAAQ,IAAI,EAAI,CAAE,KAAK,EAAI,CACvB,EAAI,KAAO,GAAK,EAAM,KAAK,EAAI,CAGrC,IAAM,EAAY,IAAI,IAAI,EAAM,IAAK,GAAM,EAAE,GAAG,CAAC,CACjD,IAAK,IAAM,KAAO,EACZ,CAAC,EAAU,IAAI,EAAI,UAAY,EAAI,GAAG,EAAI,CAAC,EAAM,KAAM,GAAM,EAAE,KAAO,EAAI,GAAG,EAC/E,EAAM,KAAK,EAAI,CAInB,OADA,EAAM,MAAM,EAAG,IAAM,EAAE,GAAK,EAAE,GAAG,CAC1B,EAAM,IAAK,IAAO,CAAE,KAAM,EAAG,SAAU,EAAQ,IAAI,EAAE,GAAG,EAAI,EAAE,EAAE,OAAQ,GAAM,EAAE,KAAO,EAAE,GAAG,CAAC,MAAM,EAAG,IAAM,EAAE,GAAK,EAAE,GAAG,CAAE,EAAE,EAClI,CAAC,EAAS,CAAC,CAEd,SAAS,EAAW,EAAY,CAC9B,OAAO,EAAW,GAAM,EAAY,EAAW,GAAI,CAAG,EAAG,MAAM,IAAI,CAGrE,IAAM,GAAA,EAAA,EAAA,aACG,CAAC,GAAG,IAAI,IAAI,EAAS,IAAK,GAAM,EAAE,QAAQ,CAAC,OAAO,QAAQ,CAAa,CAAC,CAC9E,CAAC,EAAS,CAAC,CAEd,OACE,EAAA,EAAA,MAAC,MAAD,CAAK,UAAU,qBAAf,EACE,EAAA,EAAA,MAAC,MAAD,CAAK,UAAU,6DAAf,EACE,EAAA,EAAA,MAAC,MAAD,CAAK,UAAU,mCAAf,EACE,EAAA,EAAA,KAAC,EAAD,CAAM,UAAU,gCAAkC,CAAA,EAClD,EAAA,EAAA,KAAC,KAAD,CAAI,UAAU,iCAAwB,WAAa,CAAA,EACnD,EAAA,EAAA,KAAC,EAAD,CAAO,QAAQ,qBAAa,EAAS,OAAe,CAAA,CAChD,IACN,EAAA,EAAA,MAAC,EAAD,CAAQ,KAAK,KAAK,QAAS,WAA3B,EACE,EAAA,EAAA,KAAC,EAAD,CAAQ,UAAU,eAAiB,CAAA,CAAA,WAC5B,GACL,IAEN,EAAA,EAAA,MAAC,MAAD,CAAK,UAAU,gCAAf,EACE,EAAA,EAAA,MAAC,SAAD,CACE,UAAU,mEACV,MAAO,EACP,SAAW,GAAM,EAAI,CAAE,cAAe,EAAE,OAAO,MAAO,CAAC,UAHzD,EAKE,EAAA,EAAA,KAAC,SAAD,CAAQ,MAAM,YAAG,aAAmB,CAAA,CACnC,EAAc,IAAK,IAAM,EAAA,EAAA,KAAC,SAAD,CAAmB,MAAO,EAAE,YAAK,EAAY,EAAE,CAAU,CAA5C,EAAE,GAA0C,CAAC,CAC7E,IACT,EAAA,EAAA,MAAC,SAAD,CACE,UAAU,mEACV,MAAO,EACP,SAAW,GAAM,EAAI,CAAE,UAAW,EAAE,OAAO,MAAO,CAAC,UAHrD,EAKE,EAAA,EAAA,KAAC,SAAD,CAAQ,MAAM,YAAG,WAAiB,CAAA,CACjC,EAAW,IAAK,IAAM,EAAA,EAAA,MAAC,SAAD,CAAgB,MAAO,WAAvB,CAA0B,IAAE,EAAW,EAA1B,EAA0B,CAAC,CACxD,IACT,EAAA,EAAA,MAAC,SAAD,CACE,UAAU,mEACV,MAAO,EACP,SAAW,GAAM,EAAI,CAAE,cAAe,EAAE,OAAO,MAAO,CAAC,UAHzD,EAKE,EAAA,EAAA,KAAC,SAAD,CAAQ,MAAM,YAAG,eAAqB,CAAA,CACrC,EAAe,IAAK,IAAM,EAAA,EAAA,KAAC,SAAD,CAAgB,MAAO,WAAI,EAAW,CAAzB,EAAyB,CAAC,CAC3D,GACL,GAEL,IACC,EAAA,EAAA,MAAC,MAAD,CAAK,UAAU,uHAAf,EACE,EAAA,EAAA,KAAC,EAAD,CAAoB,UAAU,mBAAqB,CAAA,EACnD,EAAA,EAAA,MAAC,OAAD,CAAA,SAAA,CAAM,wBAAsB,EAAQ,GAAG,UAAM,EAAA,EAAA,KAAC,SAAD,CAAA,SAAS,EAAW,EAAQ,KAAK,CAAU,CAAA,CAAO,CAAA,CAAA,EAC/F,EAAA,EAAA,KAAC,EAAD,CAAQ,QAAQ,QAAQ,KAAK,KAAK,UAAU,qBAAqB,QAAS,YACxE,EAAA,EAAA,KAAC,EAAD,CAAG,UAAU,cAAgB,CAAA,CACtB,CAAA,CACL,IAGR,EAAA,EAAA,KAAC,MAAD,CAAK,UAAU,mDACb,EAAA,EAAA,MAAC,MAAD,CAAK,UAAU,0BAAf,CACG,EAAQ,SAAW,IAClB,EAAA,EAAA,KAAC,MAAD,CAAK,UAAU,2DAAkD,cAAiB,CAAA,CAEnF,EAAQ,KAAK,CAAE,OAAM,cACpB,EAAA,EAAA,MAAC,MAAD,CAAmB,UAAU,qBAA7B,EACE,EAAA,EAAA,KAAC,EAAD,CACE,IAAK,EACA,MACO,aACA,aACZ,YAAe,EAAW,EAAK,CAC/B,YAAe,EAAQ,EAAK,GAAG,CAC/B,YAAe,EAAiB,EAAK,GAAI,YAAY,CACrD,eAAkB,EAAiB,EAAK,GAAI,YAAa,6BAA6B,CACtF,YAAe,EAAiB,EAAK,GAAI,QAAS,gCAAgC,CAClF,aAAgB,EAAgB,EAAK,GAAG,CACxC,SAAU,EAAK,aAAiB,EAAW,EAAK,SAAU,CAAG,IAAA,GAC7D,WAAY,EAAQ,OACpB,CAAA,CACD,EAAQ,IAAK,IACZ,EAAA,EAAA,KAAC,MAAD,CAAgB,UAAU,iBACxB,EAAA,EAAA,KAAC,EAAD,CACE,IAAK,EACA,MACO,aACA,aACZ,YAAe,EAAW,EAAE,CAC5B,YAAe,EAAQ,EAAE,GAAG,CAC5B,YAAe,EAAiB,EAAE,GAAI,YAAY,CAClD,eAAkB,EAAiB,EAAE,GAAI,YAAa,6BAA6B,CACnF,YAAe,EAAiB,EAAE,GAAI,QAAS,gCAAgC,CAC/E,aAAgB,EAAgB,EAAE,GAAG,CACrC,CAAA,CACE,CAbI,EAAE,GAaN,CACN,CACE,EA/BI,EAAK,GA+BT,CACN,CACE,GACF,CAAA,CACF,GAIV,SAAS,EAAW,CAClB,MAAK,MAAK,aAAY,aAAY,UAAS,UAAS,UAAS,aAAY,UAAS,WAAU,WAAU,cAcrG,CACD,GAAM,CAAC,EAAU,IAAA,EAAA,EAAA,UAAwB,GAAM,CACzC,EAAO,EAAY,EAAI,CACvB,EAAc,EAAe,GAAM,EAAE,YAAY,CACjD,EAAkB,EAAI,iBAAmB,UAAY,EAAI,iBAAmB,UAAY,EAAI,iBAAmB,QAAU,GAAQ,EAAI,mBAAqB,EAAI,sBAE9J,EAAY,EADL,EAAI,OAAS,OAAS,EAAI,GAAK,EAAI,MAGhD,SAAS,EAAuB,EAAsB,CAEpD,OAAO,GADK,OAAO,GAAW,MAAM,KAAQ,SAAW,EAAU,KAAK,IAAM,KAC1C,IAAK,EAAK,CAG9C,SAAS,EAAmB,EAAc,EAAe,CACvD,IAAM,EAAW,EAAuB,EAAK,CAC7C,EAAiB,CAAE,KAAM,EAAW,EAAS,CAAE,aAAc,EAAU,OAAM,QAAS,GAAM,CAAC,CAG/F,OACE,EAAA,EAAA,KAAC,EAAD,CAAM,UAAU,2BACd,EAAA,EAAA,MAAC,EAAD,CAAa,UAAU,eAAvB,EACE,EAAA,EAAA,MAAC,MAAD,CAAK,UAAU,mBAAf,EACE,EAAA,EAAA,MAAC,MAAD,CAAK,UAAU,gFAAf,EACE,EAAA,EAAA,KAAC,OAAD,CAAM,UAAU,uCAA+B,EAAW,EAAI,KAAK,CAAQ,CAAA,EAC3E,EAAA,EAAA,KAAC,OAAD,CAAA,SAAM,IAAQ,CAAA,EACd,EAAA,EAAA,KAAC,OAAD,CAAA,SAAO,EAAc,EAAI,GAAI,EAAW,CAAQ,CAAA,CAC/C,EAAI,UACH,EAAA,EAAA,MAAC,EAAD,CAAO,QAAQ,UAAU,UAAU,uCAAnC,EACE,EAAA,EAAA,KAAC,EAAD,CAAM,UAAU,cAAgB,CAAA,CAAC,EAAI,QAC/B,GAET,EAAI,YACH,EAAA,EAAA,MAAC,EAAD,CAAO,QAAQ,UAAU,UAAU,sEAAnC,EACE,EAAA,EAAA,KAAC,EAAD,CAAc,UAAU,qBAAuB,CAAA,CAAA,UACzC,GAET,IAAmB,EAAA,EAAA,KAAC,EAAD,CAAoB,MAAU,MAAO,CAAA,CACxD,EAAI,UAAW,EAAA,EAAA,KAAC,OAAD,CAAM,UAAU,uCAA+B,EAAI,QAAe,CAAA,EAClF,EAAA,EAAA,MAAC,OAAD,CAAM,UAAU,mBAAmB,MAAO,EAAQ,EAAI,UAAU,UAAhE,EAAkE,EAAA,EAAA,MAAC,OAAD,CAAM,UAAU,2BAAhB,CAAkC,IAAE,EAAI,GAAU,GAAC,EAAQ,EAAK,EAAI,UAAU,CAAQ,GACpJ,GACL,IACC,EAAA,EAAA,KAAC,MAAD,CAAK,UAAU,iBAAiB,YAAe,EAAa,GAAM,CAAC,EAAE,WACnE,EAAA,EAAA,KAAC,EAAD,CAAe,KAAM,EAAM,QAAS,EAAI,KAAM,UAAW,iCAAiC,EAAW,GAAK,oCAAqC,WAAY,EAAoB,kBAAmB,EAAM,IAAS,GAAG,EAAuB,EAAK,GAAG,EAAO,IAAI,IAAS,KAAQ,CAAA,CAC3Q,CAAA,CAEJ,IACN,EAAA,EAAA,MAAC,MAAD,CAAK,UAAU,6KAAf,EACE,EAAA,EAAA,KAAC,EAAD,CAAQ,QAAQ,QAAQ,KAAK,KAAK,UAAU,mBAAmB,QAAS,EAAS,MAAM,kBACrF,EAAA,EAAA,KAAC,EAAD,CAAoB,UAAU,cAAgB,CAAA,CACvC,CAAA,CACR,EAAI,WAAa,CAAC,EAAI,YACrB,EAAA,EAAA,KAAC,EAAD,CAAQ,QAAQ,QAAQ,KAAK,KAAK,UAAU,yDAAyD,QAAS,EAAS,MAAM,kBAC3H,EAAA,EAAA,KAAC,EAAD,CAAc,UAAU,cAAgB,CAAA,CACjC,CAAA,CAEV,IACC,EAAA,EAAA,KAAC,EAAD,CAAQ,QAAQ,QAAQ,KAAK,KAAK,UAAU,qDAAqD,QAAS,EAAS,MAAM,+BACvH,EAAA,EAAA,KAAC,EAAD,CAAW,UAAU,cAAgB,CAAA,CAC9B,CAAA,CAEV,EAAI,iBAAmB,QAAU,IAChC,EAAA,EAAA,KAAC,EAAD,CAAQ,QAAQ,QAAQ,KAAK,KAAK,UAAU,mDAAmD,QAAS,EAAY,MAAM,sBACxH,EAAA,EAAA,KAAC,EAAD,CAAK,UAAU,cAAgB,CAAA,CACxB,CAAA,EAET,EAAI,iBAAmB,UAAY,EAAI,iBAAmB,UAC1D,EAAA,EAAA,KAAC,EAAD,CAAQ,QAAQ,QAAQ,KAAK,KAAK,UAAU,2DAA2D,QAAS,EAAS,MAAM,wCAC7H,EAAA,EAAA,KAAC,EAAD,CAAc,UAAU,cAAgB,CAAA,CACjC,CAAA,CAEV,IAAa,GAAc,GAAK,IAC/B,EAAA,EAAA,MAAC,EAAD,CAAQ,QAAQ,QAAQ,KAAK,KAAK,UAAU,mBAAmB,QAAS,EAAU,MAAM,kBAAxF,CAAiG,IAC7F,EACK,IAEX,EAAA,EAAA,KAAC,EAAD,CAAQ,QAAQ,QAAQ,KAAK,KAAK,UAAU,mDAAmD,QAAS,EAAU,MAAM,mBACtH,EAAA,EAAA,KAAC,EAAD,CAAQ,UAAU,cAAgB,CAAA,CAC3B,CAAA,CACL,GACM,GACT,CAAA,CAIX,SAAS,EAAc,CAAE,MAAK,OAAsC,CAClE,IAAM,EAAS,EAAI,gBAAkB,UAC/B,EAAS,EAAI,sBAAwB,EAAI,oBAAsB,EAAI,oBAAsB,SAAS,EAAQ,EAAK,EAAI,oBAAoB,GAAK,IAAA,IAMlJ,OACE,EAAA,EAAA,MAAC,EAAD,CAAO,QAAQ,UAAU,UAAW,uBANpB,IAAW,OACzB,+CACA,IAAW,SACT,qDACA,oDAEoE,MAAO,GAAU,WAAzF,EACE,EAAA,EAAA,KAAC,EAAD,CAAe,UAAU,qBAAuB,CAAA,CAC/C,EAAQ,EAAI,iBAAmB,IAAI,EAAI,iBAAiB,GAAK,GACxD"}
|
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
import{An as e,I as t,P as n,Qt as r,T as i,k as a,p as o,zt as s}from"./lucide-react-DLQFnqNm.js";import{i as c,t as l}from"./store-
|
|
2
|
-
//# sourceMappingURL=orchestrators-
|
|
1
|
+
import{An as e,I as t,P as n,Qt as r,T as i,k as a,p as o,zt as s}from"./lucide-react-DLQFnqNm.js";import{i as c,t as l}from"./store-Bo72e9My.js";import{H as u,M as d,N as f}from"./display-ConJ9cJB.js";import{t as p}from"./badge-JVybSpzR.js";import{t as m}from"./button-BsMqBNJb.js";import{O as h}from"./index-B1QUkb_O.js";import{i as g,n as _,r as v,t as y}from"./card-I8w4U656.js";var b=e();function x({agent:e,orch:t}){let r=l(e=>e.orchestratorAction),i=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||t.baseDir} in Files`,onClick:n=>{n.stopPropagation(),i({orchestratorId:t.id,path:e.cwd||t.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:n=>{n.stopPropagation(),r(t.id,`restart`,e.agentId,`orchestrators`)},children:(0,b.jsx)(a,{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:n=>{n.stopPropagation(),r(t.id,`shutdown`,e.agentId,`orchestrators`)},children:(0,b.jsx)(n,{className:`w-3 h-3`})})]})]})}function S({orch:e}){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=f(e),A=d(e),j=e.status===`online`,M=e.package?.version||e.version,N=e.upgrade,P=N?.status===`pending`,F=j&&!!e.version&&!!w&&e.version!==w,I=j&&!P&&(F||N?.status===`failed`),L=Object.entries(e.contracts||{}).filter(([,e])=>typeof e==`number`),R=e.providerStatus||e.providers.map(t=>({name:t,available:!0,checkedAt:Number(e.lastSeen)||Date.now()})),z=e.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)(i,{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:e.hostname}),(0,b.jsxs)(`div`,{className:`text-xs text-muted-foreground mt-0.5`,children:[e.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:e.status}),M&&(0,b.jsxs)(`span`,{children:[e.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)(r,{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:u(s,e.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))}),e.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 (`,e.managedAgents.length,`)`]}),(0,b.jsx)(`div`,{className:`space-y-0`,children:e.managedAgents.map(t=>(0,b.jsx)(x,{agent:t,orch:e},t.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:t,workspace:n})=>(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:()=>n.worktreePath&&void E({orchestratorId:e.id,path:n.worktreePath}),children:[(0,b.jsxs)(`div`,{className:`flex items-center gap-2 text-xs`,children:[(0,b.jsx)(`span`,{className:`truncate font-medium`,children:n.branch||t.label||t.agentId.slice(-12)}),n.status&&(0,b.jsx)(p,{variant:`outline`,className:`h-4 shrink-0 text-[10px]`,children:n.status})]}),(0,b.jsx)(`div`,{className:`truncate text-[10px] text-muted-foreground`,children:n.worktreePath||n.repoRoot})]},n.id||n.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(e.id),children:[(0,b.jsx)(t,{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(e.id),children:[(0,b.jsx)(a,{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(e.id),children:[(0,b.jsx)(r,{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(e.id,`restart`,void 0,`orchestrators`),children:[(0,b.jsx)(a,{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(e.id,`shutdown`,void 0,`orchestrators`),children:[(0,b.jsx)(n,{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(e.id),children:`Remove`})]})]})]})}function C(){let e=l(e=>e.orchestrators),n=l(e=>e.openOrchestratorSpawn),r=l(e=>e.openOrchestratorInstall),a=e.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:r,children:[(0,b.jsx)(i,{className:`w-3.5 h-3.5`}),`Add`]}),(0,b.jsxs)(m,{size:`sm`,className:`gap-1`,onClick:n,children:[(0,b.jsx)(t,{className:`w-3.5 h-3.5`}),`Spawn`]})]})]}),e.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:e.map(e=>(0,b.jsx)(S,{orch:e},e.id))})]})}export{C as OrchestratorsView};
|
|
2
|
+
//# sourceMappingURL=orchestrators-CQcJb6VE.js.map
|