381 lines
48 KiB
JavaScript
381 lines
48 KiB
JavaScript
(() => {
|
|
const AGENT_META = {
|
|
rose: { emoji: "\u{1F339}", name: "Rose", tier: 1 },
|
|
lotus: { emoji: "\u{1FAB7}", name: "Lotus", tier: 2 },
|
|
"forget-me-not": { emoji: "\u{1F33C}", name: "Forget-me-not", tier: 2 },
|
|
sunflower: { emoji: "\u{1F33B}", name: "Sunflower", tier: 2 },
|
|
iris: { emoji: "\u269C\uFE0F", name: "Iris", tier: 2 },
|
|
ivy: { emoji: "\u{1F33F}", name: "Ivy", tier: 2 },
|
|
dandelion: { emoji: "\u{1F6E1}", name: "Dandelion", tier: 2 },
|
|
root: { emoji: "\u{1F333}", name: "Root", tier: 2 }
|
|
};
|
|
let _nodeCounter = 0;
|
|
function _nextNodeId(agentId) {
|
|
return `${agentId}-${++_nodeCounter}`;
|
|
}
|
|
function initActivityTree() {
|
|
const tree = {
|
|
version: 1,
|
|
rootId: "rose",
|
|
nodes: {
|
|
rose: {
|
|
id: "rose",
|
|
parentId: null,
|
|
agentId: "rose",
|
|
agentEmoji: "\u{1F339}",
|
|
agentName: "Rose",
|
|
tier: 1,
|
|
status: "running",
|
|
task: "Orchestrating",
|
|
toolCalls: [],
|
|
startedAt: Date.now(),
|
|
endedAt: null,
|
|
duration: null,
|
|
children: [],
|
|
collapsed: false,
|
|
metadata: {}
|
|
}
|
|
},
|
|
stats: _emptyStats()
|
|
};
|
|
S.activityTree = tree;
|
|
S.mcFilter = {};
|
|
S.mcSort = "runtime";
|
|
return tree;
|
|
}
|
|
function _emptyStats() {
|
|
return {
|
|
totalAgents: 0,
|
|
runningAgents: 0,
|
|
pendingAgents: 0,
|
|
doneAgents: 0,
|
|
errorAgents: 0,
|
|
totalTools: 0,
|
|
doneTools: 0,
|
|
runningTools: 0,
|
|
avgResponseTime: 0,
|
|
totalElapsed: 0
|
|
};
|
|
}
|
|
function atAddNode(opts) {
|
|
const tree = S.activityTree;
|
|
if (!tree) return null;
|
|
const parentId = opts.parentId || tree.rootId;
|
|
const parent = tree.nodes[parentId];
|
|
if (!parent) return null;
|
|
const meta = AGENT_META[opts.agentId];
|
|
const tier = opts.tier || (meta ? meta.tier : 3);
|
|
const id = _nextNodeId(opts.agentId);
|
|
const node = {
|
|
id,
|
|
parentId,
|
|
agentId: opts.agentId,
|
|
agentEmoji: meta?.emoji || "\u2699\uFE0F",
|
|
agentName: meta?.name || opts.agentId,
|
|
tier,
|
|
status: opts.status || "pending",
|
|
task: opts.task,
|
|
toolCalls: [],
|
|
startedAt: opts.status === "running" ? Date.now() : null,
|
|
endedAt: null,
|
|
duration: null,
|
|
children: [],
|
|
collapsed: false,
|
|
metadata: {}
|
|
};
|
|
tree.nodes[id] = node;
|
|
parent.children.push(id);
|
|
_recalcStats(tree);
|
|
return node;
|
|
}
|
|
function atUpdateNode(nodeId, updates) {
|
|
const tree = S.activityTree;
|
|
if (!tree) return;
|
|
const node = tree.nodes[nodeId];
|
|
if (!node) return;
|
|
Object.assign(node, updates);
|
|
_recalcStats(tree);
|
|
}
|
|
function atGetNode(nodeId) {
|
|
return S.activityTree?.nodes[nodeId] || null;
|
|
}
|
|
function atAddToolCall(nodeId, tc) {
|
|
const tree = S.activityTree;
|
|
if (!tree) return null;
|
|
const node = tree.nodes[nodeId];
|
|
if (!node) return null;
|
|
if (node.status === "pending") {
|
|
node.status = "running";
|
|
node.startedAt = node.startedAt || Date.now();
|
|
}
|
|
const toolCall = {
|
|
id: `tc-${++_nodeCounter}`,
|
|
name: tc.name,
|
|
status: tc.status || "running",
|
|
args: tc.args || {},
|
|
startedAt: Date.now(),
|
|
endedAt: null
|
|
};
|
|
node.toolCalls.push(toolCall);
|
|
_recalcStats(tree);
|
|
return toolCall;
|
|
}
|
|
function atUpdateToolCall(nodeId, toolId, updates) {
|
|
const tree = S.activityTree;
|
|
if (!tree) return;
|
|
const node = tree.nodes[nodeId];
|
|
if (!node) return;
|
|
const tc = node.toolCalls.find((t) => t.id === toolId);
|
|
if (!tc) return;
|
|
Object.assign(tc, updates);
|
|
if (updates.status === "done" || updates.status === "error") {
|
|
tc.endedAt = tc.endedAt || Date.now();
|
|
tc.duration = tc.startedAt ? (tc.endedAt - tc.startedAt) / 1e3 : null;
|
|
}
|
|
_recalcStats(tree);
|
|
}
|
|
function atFinalizeNode(nodeId, status) {
|
|
const tree = S.activityTree;
|
|
if (!tree) return;
|
|
const node = tree.nodes[nodeId];
|
|
if (!node) return;
|
|
node.status = status;
|
|
node.endedAt = Date.now();
|
|
node.duration = node.startedAt ? (node.endedAt - node.startedAt) / 1e3 : null;
|
|
for (const tc of node.toolCalls) {
|
|
if (tc.status === "running" || tc.status === "pending") {
|
|
tc.status = status === "error" ? "error" : "done";
|
|
tc.endedAt = node.endedAt;
|
|
tc.duration = tc.startedAt ? (tc.endedAt - tc.startedAt) / 1e3 : null;
|
|
}
|
|
}
|
|
for (const childId of node.children) {
|
|
const child = tree.nodes[childId];
|
|
if (child && (child.status === "running" || child.status === "thinking" || child.status === "pending")) {
|
|
atFinalizeNode(childId, status);
|
|
}
|
|
}
|
|
_recalcStats(tree);
|
|
}
|
|
function _recalcStats(tree) {
|
|
const nodes = Object.values(tree.nodes).filter((n) => n.id !== tree.rootId);
|
|
const tools = nodes.flatMap((n) => n.toolCalls);
|
|
const doneDurations = nodes.filter((n) => n.duration !== null).map((n) => n.duration);
|
|
const runningNodes = nodes.filter((n) => n.status === "running" || n.status === "thinking");
|
|
const maxElapsed = runningNodes.reduce((max, n) => {
|
|
const elapsed = n.startedAt ? (Date.now() - n.startedAt) / 1e3 : 0;
|
|
return Math.max(max, elapsed);
|
|
}, 0);
|
|
tree.stats = {
|
|
totalAgents: nodes.length,
|
|
runningAgents: runningNodes.length,
|
|
pendingAgents: nodes.filter((n) => n.status === "pending").length,
|
|
doneAgents: nodes.filter((n) => n.status === "done").length,
|
|
errorAgents: nodes.filter((n) => n.status === "error").length,
|
|
totalTools: tools.length,
|
|
doneTools: tools.filter((t) => t.status === "done").length,
|
|
runningTools: tools.filter((t) => t.status === "running").length,
|
|
avgResponseTime: doneDurations.length ? doneDurations.reduce((a, b) => a + b, 0) / doneDurations.length : 0,
|
|
totalElapsed: maxElapsed
|
|
};
|
|
}
|
|
function atGetStats() {
|
|
return S.activityTree?.stats || _emptyStats();
|
|
}
|
|
function atReset() {
|
|
_nodeCounter = 0;
|
|
initActivityTree();
|
|
}
|
|
function formatElapsed(ms) {
|
|
if (ms < 1e3) return `${Math.round(ms)}ms`;
|
|
const s = Math.floor(ms / 1e3);
|
|
if (s < 60) return `${s}s`;
|
|
const m = Math.floor(s / 60);
|
|
const rs = s % 60;
|
|
if (m < 60) return `${m}m ${rs}s`;
|
|
const h = Math.floor(m / 60);
|
|
const rm = m % 60;
|
|
return `${h}h ${rm}m`;
|
|
}
|
|
function formatDuration(seconds) {
|
|
if (seconds === null) return "\u2014";
|
|
if (seconds < 0.01) return "<0.01s";
|
|
if (seconds < 1) return `${seconds.toFixed(2)}s`;
|
|
if (seconds < 60) return `${seconds.toFixed(1)}s`;
|
|
const m = Math.floor(seconds / 60);
|
|
const rs = Math.round(seconds % 60);
|
|
return `${m}m ${rs}s`;
|
|
}
|
|
function createMockActivityTree() {
|
|
initActivityTree();
|
|
const tree = S.activityTree;
|
|
const now = Date.now();
|
|
const lotus = atAddNode({ agentId: "lotus", task: "Analysiere Gesundheitsdaten", status: "running" });
|
|
lotus.startedAt = now - 12e3;
|
|
atAddToolCall(lotus.id, { name: "search_files", args: { pattern: "health*", path: "/data" } });
|
|
atUpdateToolCall(lotus.id, lotus.toolCalls[0].id, { status: "done", result: "12 matches", duration: 0.3 });
|
|
lotus.toolCalls[0].startedAt = now - 11e3;
|
|
lotus.toolCalls[0].endedAt = now - 10997;
|
|
atAddToolCall(lotus.id, { name: "read_file", args: { path: "/data/health-log.md" } });
|
|
const sunflower = atAddNode({ agentId: "sunflower", task: "Portfolio-Analyse Q1 2026", status: "running" });
|
|
sunflower.startedAt = now - 8e3;
|
|
atAddToolCall(sunflower.id, { name: "browser_navigate", args: { url: "https://finance.example.com" } });
|
|
atUpdateToolCall(sunflower.id, sunflower.toolCalls[0].id, { status: "done", result: "Page loaded", duration: 2.1 });
|
|
sunflower.toolCalls[0].startedAt = now - 7500;
|
|
sunflower.toolCalls[0].endedAt = now - 5400;
|
|
atAddToolCall(sunflower.id, { name: "terminal", args: { command: "python3 analyse.py" } });
|
|
const dandelion = atAddNode({ agentId: "dandelion", task: "Triaging unread messages", status: "pending" });
|
|
const researcher = atAddNode({ parentId: lotus.id, agentId: "researcher", task: "Looking up nutrition data", status: "running", tier: 3 });
|
|
researcher.agentEmoji = "\u{1F50D}";
|
|
researcher.agentName = "Researcher";
|
|
researcher.startedAt = now - 3e3;
|
|
atAddToolCall(researcher.id, { name: "web_search", args: { query: "nutrition database API" } });
|
|
_recalcStats(tree);
|
|
return tree;
|
|
}
|
|
window.initActivityTree = initActivityTree;
|
|
window.createMockActivityTree = createMockActivityTree;
|
|
window.formatElapsed = formatElapsed;
|
|
window.formatDuration = formatDuration;
|
|
window.atAddNode = atAddNode;
|
|
window.atUpdateNode = atUpdateNode;
|
|
window.atGetNode = atGetNode;
|
|
window.atAddToolCall = atAddToolCall;
|
|
window.atUpdateToolCall = atUpdateToolCall;
|
|
window.atFinalizeNode = atFinalizeNode;
|
|
window.atGetStats = atGetStats;
|
|
window.atReset = atReset;
|
|
window._atTrackTool = _atTrackTool;
|
|
window._atTrackToolComplete = _atTrackToolComplete;
|
|
window._atTrackDone = _atTrackDone;
|
|
window._atTrackSubagent = _atTrackSubagent;
|
|
window._at = {
|
|
init: initActivityTree,
|
|
addNode: atAddNode,
|
|
updateNode: atUpdateNode,
|
|
getNode: atGetNode,
|
|
addToolCall: atAddToolCall,
|
|
updateToolCall: atUpdateToolCall,
|
|
finalize: atFinalizeNode,
|
|
stats: atGetStats,
|
|
reset: atReset,
|
|
mock: createMockActivityTree,
|
|
formatElapsed,
|
|
formatDuration,
|
|
AGENT_META,
|
|
trackSubagent: _atTrackSubagent,
|
|
};
|
|
let _activeAgentNodeId = null;
|
|
let _lastToolId = null;
|
|
function _atTrackTool(d) {
|
|
if (!S.activityTree) initActivityTree();
|
|
const tree = S.activityTree;
|
|
if (d.name === "delegate_task") {
|
|
const agentId = d.args?.agentId || d.args?.agent || d.args?.name || "unknown";
|
|
const task = d.args?.goal || d.args?.prompt || d.args?.task || d.preview || "Delegated task";
|
|
const node = atAddNode({ agentId, task, status: "running" });
|
|
if (node) {
|
|
_activeAgentNodeId = node.id;
|
|
}
|
|
return;
|
|
}
|
|
let targetNodeId = _activeAgentNodeId || tree.rootId;
|
|
const inflight = window.INFLIGHT?.[S.session?.session_id];
|
|
if (inflight?.thisTurnAgent) {
|
|
const agentKey = inflight.thisTurnAgent;
|
|
const existingNode = Object.values(tree.nodes).find(
|
|
(n) => n.agentId === agentKey && (n.status === "running" || n.status === "thinking")
|
|
);
|
|
if (existingNode) {
|
|
targetNodeId = existingNode.id;
|
|
_activeAgentNodeId = existingNode.id;
|
|
} else if (AGENT_META[agentKey]) {
|
|
const task = inflight.thisTurnModel ? `${inflight.thisTurnModel}` : "Working...";
|
|
const node = atAddNode({ agentId: agentKey, task, status: "running" });
|
|
if (node) {
|
|
_activeAgentNodeId = node.id;
|
|
targetNodeId = node.id;
|
|
}
|
|
}
|
|
}
|
|
const tc = atAddToolCall(targetNodeId, {
|
|
name: d.name,
|
|
args: d.args || {},
|
|
status: "running"
|
|
});
|
|
if (tc) {
|
|
_lastToolId = tc.id;
|
|
}
|
|
}
|
|
function _atTrackToolComplete(d) {
|
|
if (!S.activityTree) return;
|
|
const tree = S.activityTree;
|
|
const activeNodes = Object.values(tree.nodes).filter(
|
|
(n) => n.status === "running" || n.status === "thinking"
|
|
);
|
|
for (const node of activeNodes) {
|
|
for (const tc of node.toolCalls) {
|
|
if (tc.status === "running" && tc.name === d.name) {
|
|
atUpdateToolCall(node.id, tc.id, {
|
|
status: d.is_error ? "error" : "done",
|
|
result: d.preview || d.result,
|
|
duration: d.duration
|
|
});
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
function _atTrackSubagent(d) {
|
|
if (!S.activityTree) initActivityTree();
|
|
const tree = S.activityTree;
|
|
|
|
if (d.event_type === 'subagent.start') {
|
|
// Derive agentId from subagent_id if available, otherwise from goal hint.
|
|
// subagent_id format: "lotus-1" → agentId = "lotus"
|
|
let agentId = 'unknown';
|
|
if (d.subagent_id) {
|
|
// Strip numeric suffix e.g. "lotus-1" → "lotus"
|
|
agentId = d.subagent_id.replace(/-\d+$/, '');
|
|
}
|
|
const task = d.goal || d.preview || `Subagent ${d.task_index ?? ''}`.trim();
|
|
const node = atAddNode({ agentId, task, status: 'running' });
|
|
if (node) {
|
|
_activeAgentNodeId = node.id;
|
|
}
|
|
return;
|
|
}
|
|
|
|
if (d.event_type === 'subagent.complete' || d.event_type === 'subagent.done') {
|
|
// Find the running node for this subagent and finalize it.
|
|
// Use subagent_id to locate the right node.
|
|
let targetNodeId = null;
|
|
if (d.subagent_id) {
|
|
const agentId = d.subagent_id.replace(/-\d+$/, '');
|
|
const found = Object.values(tree.nodes).find(
|
|
(n) => n.agentId === agentId && n.status === 'running'
|
|
);
|
|
if (found) targetNodeId = found.id;
|
|
}
|
|
// Fall back to _activeAgentNodeId if no subagent_id match
|
|
if (!targetNodeId) targetNodeId = _activeAgentNodeId;
|
|
if (targetNodeId && tree.nodes[targetNodeId]) {
|
|
const status = d.status === 'timeout' || d.status === 'error' ? 'error' : 'done';
|
|
atFinalizeNode(targetNodeId, status);
|
|
}
|
|
}
|
|
}
|
|
function _atTrackDone() {
|
|
if (!S.activityTree) return;
|
|
if (_activeAgentNodeId) {
|
|
atFinalizeNode(_activeAgentNodeId, "done");
|
|
_activeAgentNodeId = null;
|
|
}
|
|
const root = S.activityTree.nodes[S.activityTree.rootId];
|
|
if (root) {
|
|
root.status = "running";
|
|
}
|
|
}
|
|
})();
|
|
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["activity-tree.ts"],
  "sourcesContent": ["// \u2500\u2500\u2500 Agent Activity Tree (Mission Control) \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n// Full-depth parallel agent activity tracking for the 3-Tier system.\n// Provides: initActivityTree, addNode, addToolCall, updateToolCall,\n//           finalizeNode, getStats, createMockActivityTree, formatElapsed\n\n// Agent metadata \u2014 maps agent IDs to display info\nconst AGENT_META: Record<string, { emoji: string; name: string; tier: 1|2|3 }> = {\n  rose:           { emoji: '\uD83C\uDF39', name: 'Rose',           tier: 1 },\n  lotus:          { emoji: '\uD83E\uDEB7', name: 'Lotus',          tier: 2 },\n  'forget-me-not':{ emoji: '\uD83C\uDF3C', name: 'Forget-me-not',  tier: 2 },\n  sunflower:      { emoji: '\uD83C\uDF3B', name: 'Sunflower',      tier: 2 },\n  iris:           { emoji: '\u269C\uFE0F', name: 'Iris',           tier: 2 },\n  ivy:            { emoji: '\uD83C\uDF3F', name: 'Ivy',            tier: 2 },\n  dandelion:      { emoji: '\uD83D\uDEE1', name: 'Dandelion',      tier: 2 },\n  root:           { emoji: '\uD83C\uDF33', name: 'Root',           tier: 2 },\n};\n\n// Tier-3 sub-agent detection \u2014 tool names that indicate sub-agent work\nconst TIER3_TOOL_NAMES = new Set([\n  'search_files', 'read_file', 'write_file', 'terminal', 'browser_navigate',\n  'browser_snapshot', 'browser_click', 'browser_type', 'browser_press',\n  'web_search', 'delegate_task', 'execute_code', 'patch',\n]);\n\nlet _nodeCounter = 0;\nfunction _nextNodeId(agentId: string): string {\n  return `${agentId}-${++_nodeCounter}`;\n}\n\n// \u2500\u2500\u2500 Core Functions \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\nfunction initActivityTree(): ActivityTree {\n  const tree: ActivityTree = {\n    version: 1,\n    rootId: 'rose',\n    nodes: {\n      rose: {\n        id: 'rose',\n        parentId: null,\n        agentId: 'rose',\n        agentEmoji: '\uD83C\uDF39',\n        agentName: 'Rose',\n        tier: 1,\n        status: 'running',\n        task: 'Orchestrating',\n        toolCalls: [],\n        startedAt: Date.now(),\n        endedAt: null,\n        duration: null,\n        children: [],\n        collapsed: false,\n        metadata: {},\n      }\n    },\n    stats: _emptyStats(),\n  };\n  S.activityTree = tree;\n  S.mcFilter = {};\n  S.mcSort = 'runtime';\n  return tree;\n}\n\nfunction _emptyStats(): MCStats {\n  return {\n    totalAgents: 0,\n    runningAgents: 0,\n    pendingAgents: 0,\n    doneAgents: 0,\n    errorAgents: 0,\n    totalTools: 0,\n    doneTools: 0,\n    runningTools: 0,\n    avgResponseTime: 0,\n    totalElapsed: 0,\n  };\n}\n\n// \u2500\u2500\u2500 Node Operations \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\nfunction atAddNode(opts: {\n  parentId?: string;\n  agentId: string;\n  task: string;\n  status?: ActivityNode['status'];\n  tier?: 1|2|3;\n}): ActivityNode | null {\n  const tree = S.activityTree;\n  if (!tree) return null;\n\n  const parentId = opts.parentId || tree.rootId;\n  const parent = tree.nodes[parentId];\n  if (!parent) return null;\n\n  const meta = AGENT_META[opts.agentId];\n  const tier = opts.tier || (meta ? meta.tier : 3);\n  const id = _nextNodeId(opts.agentId);\n\n  const node: ActivityNode = {\n    id,\n    parentId,\n    agentId: opts.agentId,\n    agentEmoji: meta?.emoji || '\u2699\uFE0F',\n    agentName: meta?.name || opts.agentId,\n    tier,\n    status: opts.status || 'pending',\n    task: opts.task,\n    toolCalls: [],\n    startedAt: opts.status === 'running' ? Date.now() : null,\n    endedAt: null,\n    duration: null,\n    children: [],\n    collapsed: false,\n    metadata: {},\n  };\n\n  tree.nodes[id] = node;\n  parent.children.push(id);\n  _recalcStats(tree);\n  return node;\n}\n\nfunction atUpdateNode(nodeId: string, updates: Partial<ActivityNode>): void {\n  const tree = S.activityTree;\n  if (!tree) return;\n  const node = tree.nodes[nodeId];\n  if (!node) return;\n  Object.assign(node, updates);\n  _recalcStats(tree);\n}\n\nfunction atGetNode(nodeId: string): ActivityNode | null {\n  return S.activityTree?.nodes[nodeId] || null;\n}\n\nfunction atGetChildren(nodeId: string): ActivityNode[] {\n  const tree = S.activityTree;\n  if (!tree) return [];\n  const node = tree.nodes[nodeId];\n  if (!node) return [];\n  return node.children.map(id => tree.nodes[id]).filter(Boolean);\n}\n\nfunction atGetRunningNodes(): ActivityNode[] {\n  const tree = S.activityTree;\n  if (!tree) return [];\n  return Object.values(tree.nodes).filter(\n    n => n.status === 'running' || n.status === 'thinking'\n  );\n}\n\n// \u2500\u2500\u2500 Tool Call Operations \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\nfunction atAddToolCall(nodeId: string, tc: {\n  name: string;\n  args?: Record<string, any>;\n  status?: ActivityToolCall['status'];\n}): ActivityToolCall | null {\n  const tree = S.activityTree;\n  if (!tree) return null;\n  const node = tree.nodes[nodeId];\n  if (!node) return null;\n\n  // If node was pending, it's now running\n  if (node.status === 'pending') {\n    node.status = 'running';\n    node.startedAt = node.startedAt || Date.now();\n  }\n\n  const toolCall: ActivityToolCall = {\n    id: `tc-${++_nodeCounter}`,\n    name: tc.name,\n    status: tc.status || 'running',\n    args: tc.args || {},\n    startedAt: Date.now(),\n    endedAt: null,\n  };\n\n  node.toolCalls.push(toolCall);\n  _recalcStats(tree);\n  return toolCall;\n}\n\nfunction atUpdateToolCall(nodeId: string, toolId: string, updates: Partial<ActivityToolCall>): void {\n  const tree = S.activityTree;\n  if (!tree) return;\n  const node = tree.nodes[nodeId];\n  if (!node) return;\n\n  const tc = node.toolCalls.find(t => t.id === toolId);\n  if (!tc) return;\n\n  Object.assign(tc, updates);\n  if (updates.status === 'done' || updates.status === 'error') {\n    tc.endedAt = tc.endedAt || Date.now();\n    tc.duration = tc.startedAt ? (tc.endedAt - tc.startedAt) / 1000 : null;\n  }\n  _recalcStats(tree);\n}\n\n// \u2500\u2500\u2500 Finalization \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\nfunction atFinalizeNode(nodeId: string, status: ActivityNode['status']): void {\n  const tree = S.activityTree;\n  if (!tree) return;\n  const node = tree.nodes[nodeId];\n  if (!node) return;\n\n  node.status = status;\n  node.endedAt = Date.now();\n  node.duration = node.startedAt ? (node.endedAt - node.startedAt) / 1000 : null;\n\n  // Finalize any still-running tool calls\n  for (const tc of node.toolCalls) {\n    if (tc.status === 'running' || tc.status === 'pending') {\n      tc.status = status === 'error' ? 'error' : 'done';\n      tc.endedAt = node.endedAt;\n      tc.duration = tc.startedAt ? (tc.endedAt - tc.startedAt) / 1000 : null;\n    }\n  }\n\n  // Recursively finalize children that are still active\n  for (const childId of node.children) {\n    const child = tree.nodes[childId];\n    if (child && (child.status === 'running' || child.status === 'thinking' || child.status === 'pending')) {\n      atFinalizeNode(childId, status);\n    }\n  }\n\n  _recalcStats(tree);\n}\n\n// \u2500\u2500\u2500 Stats \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\nfunction _recalcStats(tree: ActivityTree): void {\n  const nodes = Object.values(tree.nodes).filter(n => n.id !== tree.rootId);\n  const tools = nodes.flatMap(n => n.toolCalls);\n\n  const doneDurations = nodes\n    .filter(n => n.duration !== null)\n    .map(n => n.duration!);\n\n  const runningNodes = nodes.filter(n => n.status === 'running' || n.status === 'thinking');\n  const maxElapsed = runningNodes.reduce((max, n) => {\n    const elapsed = n.startedAt ? (Date.now() - n.startedAt) / 1000 : 0;\n    return Math.max(max, elapsed);\n  }, 0);\n\n  tree.stats = {\n    totalAgents: nodes.length,\n    runningAgents: runningNodes.length,\n    pendingAgents: nodes.filter(n => n.status === 'pending').length,\n    doneAgents: nodes.filter(n => n.status === 'done').length,\n    errorAgents: nodes.filter(n => n.status === 'error').length,\n    totalTools: tools.length,\n    doneTools: tools.filter(t => t.status === 'done').length,\n    runningTools: tools.filter(t => t.status === 'running').length,\n    avgResponseTime: doneDurations.length\n      ? doneDurations.reduce((a, b) => a + b, 0) / doneDurations.length\n      : 0,\n    totalElapsed: maxElapsed,\n  };\n}\n\nfunction atGetStats(): MCStats {\n  return S.activityTree?.stats || _emptyStats();\n}\n\n// \u2500\u2500\u2500 Reset \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\nfunction atReset(): void {\n  _nodeCounter = 0;\n  initActivityTree();\n}\n\n// \u2500\u2500\u2500 Formatting \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\nfunction formatElapsed(ms: number): string {\n  if (ms < 1000) return `${Math.round(ms)}ms`;\n  const s = Math.floor(ms / 1000);\n  if (s < 60) return `${s}s`;\n  const m = Math.floor(s / 60);\n  const rs = s % 60;\n  if (m < 60) return `${m}m ${rs}s`;\n  const h = Math.floor(m / 60);\n  const rm = m % 60;\n  return `${h}h ${rm}m`;\n}\n\nfunction formatDuration(seconds: number | null): string {\n  if (seconds === null) return '\u2014';\n  if (seconds < 0.01) return '<0.01s';\n  if (seconds < 1) return `${seconds.toFixed(2)}s`;\n  if (seconds < 60) return `${seconds.toFixed(1)}s`;\n  const m = Math.floor(seconds / 60);\n  const rs = Math.round(seconds % 60);\n  return `${m}m ${rs}s`;\n}\n\n// \u2500\u2500\u2500 Mock Data \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\nfunction createMockActivityTree(): ActivityTree {\n  initActivityTree();\n  const tree = S.activityTree!;\n  const now = Date.now();\n\n  // Lotus \u2014 running with 2 tool calls\n  const lotus = atAddNode({ agentId: 'lotus', task: 'Analysiere Gesundheitsdaten', status: 'running' })!;\n  lotus.startedAt = now - 12000;\n  atAddToolCall(lotus.id, { name: 'search_files', args: { pattern: 'health*', path: '/data' } });\n  atUpdateToolCall(lotus.id, lotus.toolCalls[0].id, { status: 'done', result: '12 matches', duration: 0.3 });\n  lotus.toolCalls[0].startedAt = now - 11000;\n  lotus.toolCalls[0].endedAt = now - 10997;\n\n  atAddToolCall(lotus.id, { name: 'read_file', args: { path: '/data/health-log.md' } });\n\n  // Sunflower \u2014 running with 1 done, 1 running tool\n  const sunflower = atAddNode({ agentId: 'sunflower', task: 'Portfolio-Analyse Q1 2026', status: 'running' })!;\n  sunflower.startedAt = now - 8000;\n  atAddToolCall(sunflower.id, { name: 'browser_navigate', args: { url: 'https://finance.example.com' } });\n  atUpdateToolCall(sunflower.id, sunflower.toolCalls[0].id, { status: 'done', result: 'Page loaded', duration: 2.1 });\n  sunflower.toolCalls[0].startedAt = now - 7500;\n  sunflower.toolCalls[0].endedAt = now - 5400;\n\n  atAddToolCall(sunflower.id, { name: 'terminal', args: { command: 'python3 analyse.py' } });\n\n  // Dandelion \u2014 pending\n  const dandelion = atAddNode({ agentId: 'dandelion', task: 'Triaging unread messages', status: 'pending' })!;\n\n  // Add a tier-3 sub-agent under Lotus\n  const researcher = atAddNode({ parentId: lotus.id, agentId: 'researcher', task: 'Looking up nutrition data', status: 'running', tier: 3 })!;\n  researcher.agentEmoji = '\uD83D\uDD0D';\n  researcher.agentName = 'Researcher';\n  researcher.startedAt = now - 3000;\n\n  atAddToolCall(researcher.id, { name: 'web_search', args: { query: 'nutrition database API' } });\n\n  _recalcStats(tree);\n  return tree;\n}\n\n// \u2500\u2500\u2500 Make core functions global (IIFE-safe) \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\nwindow.initActivityTree = initActivityTree;\nwindow.createMockActivityTree = createMockActivityTree;\nwindow.formatElapsed = formatElapsed;\nwindow.formatDuration = formatDuration;\nwindow.atAddNode = atAddNode;\nwindow.atUpdateNode = atUpdateNode;\nwindow.atGetNode = atGetNode;\nwindow.atAddToolCall = atAddToolCall;\nwindow.atUpdateToolCall = atUpdateToolCall;\nwindow.atFinalizeNode = atFinalizeNode;\nwindow.atGetStats = atGetStats;\nwindow.atReset = atReset;\nwindow._atTrackTool = _atTrackTool;\nwindow._atTrackToolComplete = _atTrackToolComplete;\nwindow._atTrackDone = _atTrackDone;\n\n// \u2500\u2500\u2500 Export for console testing \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n(window as any)._at = {\n  init: initActivityTree,\n  addNode: atAddNode,\n  updateNode: atUpdateNode,\n  getNode: atGetNode,\n  addToolCall: atAddToolCall,\n  updateToolCall: atUpdateToolCall,\n  finalize: atFinalizeNode,\n  stats: atGetStats,\n  reset: atReset,\n  mock: createMockActivityTree,\n  formatElapsed,\n  formatDuration,\n  AGENT_META,\n};\n\n// \u2500\u2500\u2500 SSE Event Bridge (called from messages.ts) \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n// Tracks which agent is currently active based on tool calls.\n// When delegate_task is detected, creates a new agent node.\n// Otherwise, attaches tool calls to the current active agent node.\n\nlet _activeAgentNodeId: string | null = null;  // Current agent node receiving tool calls\nlet _lastToolId: string | null = null;          // Last tool call ID (for complete matching)\n\nfunction _atTrackTool(d: any): void {\n  if (!S.activityTree) initActivityTree();\n  const tree = S.activityTree!;\n\n  // Detect delegation \u2014 when tool name is 'delegate_task'\n  if (d.name === 'delegate_task') {\n    const agentId = d.args?.agentId || d.args?.agent || d.args?.name || 'unknown';\n    const task = d.args?.goal || d.args?.prompt || d.args?.task || d.preview || 'Delegated task';\n    const node = atAddNode({ agentId, task, status: 'running' });\n    if (node) {\n      _activeAgentNodeId = node.id;\n    }\n    return;\n  }\n\n  // Determine which node to attach the tool call to\n  let targetNodeId = _activeAgentNodeId || tree.rootId;\n\n  // Check if this tool is from a specific agent (from inflight metadata)\n  const inflight = (window as any).INFLIGHT?.[S.session?.session_id];\n  if (inflight?.thisTurnAgent) {\n    const agentKey = inflight.thisTurnAgent;\n    // Find or create node for this agent\n    const existingNode = Object.values(tree.nodes).find(\n      (n: ActivityNode) => n.agentId === agentKey && (n.status === 'running' || n.status === 'thinking')\n    );\n    if (existingNode) {\n      targetNodeId = existingNode.id;\n      _activeAgentNodeId = existingNode.id;\n    } else if (AGENT_META[agentKey]) {\n      // New agent appeared \u2014 create node\n      const task = inflight.thisTurnModel ? `${inflight.thisTurnModel}` : 'Working...';\n      const node = atAddNode({ agentId: agentKey, task, status: 'running' });\n      if (node) {\n        _activeAgentNodeId = node.id;\n        targetNodeId = node.id;\n      }\n    }\n  }\n\n  const tc = atAddToolCall(targetNodeId, {\n    name: d.name,\n    args: d.args || {},\n    status: 'running',\n  });\n  if (tc) {\n    _lastToolId = tc.id;\n  }\n}\n\nfunction _atTrackToolComplete(d: any): void {\n  if (!S.activityTree) return;\n  const tree = S.activityTree!;\n\n  // Try to find the matching running tool call across all active nodes\n  const activeNodes = Object.values(tree.nodes).filter(\n    (n: ActivityNode) => n.status === 'running' || n.status === 'thinking'\n  );\n\n  for (const node of activeNodes) {\n    for (const tc of node.toolCalls) {\n      if (tc.status === 'running' && tc.name === d.name) {\n        atUpdateToolCall(node.id, tc.id, {\n          status: d.is_error ? 'error' : 'done',\n          result: d.preview || d.result,\n          duration: d.duration,\n        });\n        return;\n      }\n    }\n  }\n}\n\nfunction _atTrackDone(): void {\n  if (!S.activityTree) return;\n\n  // Finalize the active agent node\n  if (_activeAgentNodeId) {\n    atFinalizeNode(_activeAgentNodeId, 'done');\n    _activeAgentNodeId = null;\n  }\n\n  // Reset root status\n  const root = S.activityTree.nodes[S.activityTree.rootId];\n  if (root) {\n    root.status = 'running';\n  }\n}\n\nfunction _atTrackNewSession(): void {\n  _activeAgentNodeId = null;\n  _lastToolId = null;\n  if (S.activityTree) {\n    atReset();\n  }\n}\n"],
  "mappings": "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",
  "names": []
}

|