source dump of claude code
at main 683 lines 83 kB view raw
1import { getSdkAgentProgressSummariesEnabled } from '../../bootstrap/state.js'; 2import { OUTPUT_FILE_TAG, STATUS_TAG, SUMMARY_TAG, TASK_ID_TAG, TASK_NOTIFICATION_TAG, TOOL_USE_ID_TAG, WORKTREE_BRANCH_TAG, WORKTREE_PATH_TAG, WORKTREE_TAG } from '../../constants/xml.js'; 3import { abortSpeculation } from '../../services/PromptSuggestion/speculation.js'; 4import type { AppState } from '../../state/AppState.js'; 5import type { SetAppState, Task, TaskStateBase } from '../../Task.js'; 6import { createTaskStateBase } from '../../Task.js'; 7import type { Tools } from '../../Tool.js'; 8import { findToolByName } from '../../Tool.js'; 9import type { AgentToolResult } from '../../tools/AgentTool/agentToolUtils.js'; 10import type { AgentDefinition } from '../../tools/AgentTool/loadAgentsDir.js'; 11import { SYNTHETIC_OUTPUT_TOOL_NAME } from '../../tools/SyntheticOutputTool/SyntheticOutputTool.js'; 12import { asAgentId } from '../../types/ids.js'; 13import type { Message } from '../../types/message.js'; 14import { createAbortController, createChildAbortController } from '../../utils/abortController.js'; 15import { registerCleanup } from '../../utils/cleanupRegistry.js'; 16import { getToolSearchOrReadInfo } from '../../utils/collapseReadSearch.js'; 17import { enqueuePendingNotification } from '../../utils/messageQueueManager.js'; 18import { getAgentTranscriptPath } from '../../utils/sessionStorage.js'; 19import { evictTaskOutput, getTaskOutputPath, initTaskOutputAsSymlink } from '../../utils/task/diskOutput.js'; 20import { PANEL_GRACE_MS, registerTask, updateTaskState } from '../../utils/task/framework.js'; 21import { emitTaskProgress } from '../../utils/task/sdkProgress.js'; 22import type { TaskState } from '../types.js'; 23export type ToolActivity = { 24 toolName: string; 25 input: Record<string, unknown>; 26 /** Pre-computed activity description from the tool, e.g. "Reading src/foo.ts" */ 27 activityDescription?: string; 28 /** Pre-computed: true if this is a search operation (Grep, Glob, etc.) */ 29 isSearch?: boolean; 30 /** Pre-computed: true if this is a read operation (Read, cat, etc.) */ 31 isRead?: boolean; 32}; 33export type AgentProgress = { 34 toolUseCount: number; 35 tokenCount: number; 36 lastActivity?: ToolActivity; 37 recentActivities?: ToolActivity[]; 38 summary?: string; 39}; 40const MAX_RECENT_ACTIVITIES = 5; 41export type ProgressTracker = { 42 toolUseCount: number; 43 // Track input and output separately to avoid double-counting. 44 // input_tokens in Claude API is cumulative per turn (includes all previous context), 45 // so we keep the latest value. output_tokens is per-turn, so we sum those. 46 latestInputTokens: number; 47 cumulativeOutputTokens: number; 48 recentActivities: ToolActivity[]; 49}; 50export function createProgressTracker(): ProgressTracker { 51 return { 52 toolUseCount: 0, 53 latestInputTokens: 0, 54 cumulativeOutputTokens: 0, 55 recentActivities: [] 56 }; 57} 58export function getTokenCountFromTracker(tracker: ProgressTracker): number { 59 return tracker.latestInputTokens + tracker.cumulativeOutputTokens; 60} 61 62/** 63 * Resolver function that returns a human-readable activity description 64 * for a given tool name and input. Used to pre-compute descriptions 65 * from Tool.getActivityDescription() at recording time. 66 */ 67export type ActivityDescriptionResolver = (toolName: string, input: Record<string, unknown>) => string | undefined; 68export function updateProgressFromMessage(tracker: ProgressTracker, message: Message, resolveActivityDescription?: ActivityDescriptionResolver, tools?: Tools): void { 69 if (message.type !== 'assistant') { 70 return; 71 } 72 const usage = message.message.usage; 73 // Keep latest input (it's cumulative in the API), sum outputs 74 tracker.latestInputTokens = usage.input_tokens + (usage.cache_creation_input_tokens ?? 0) + (usage.cache_read_input_tokens ?? 0); 75 tracker.cumulativeOutputTokens += usage.output_tokens; 76 for (const content of message.message.content) { 77 if (content.type === 'tool_use') { 78 tracker.toolUseCount++; 79 // Omit StructuredOutput from preview - it's an internal tool 80 if (content.name !== SYNTHETIC_OUTPUT_TOOL_NAME) { 81 const input = content.input as Record<string, unknown>; 82 const classification = tools ? getToolSearchOrReadInfo(content.name, input, tools) : undefined; 83 tracker.recentActivities.push({ 84 toolName: content.name, 85 input, 86 activityDescription: resolveActivityDescription?.(content.name, input), 87 isSearch: classification?.isSearch, 88 isRead: classification?.isRead 89 }); 90 } 91 } 92 } 93 while (tracker.recentActivities.length > MAX_RECENT_ACTIVITIES) { 94 tracker.recentActivities.shift(); 95 } 96} 97export function getProgressUpdate(tracker: ProgressTracker): AgentProgress { 98 return { 99 toolUseCount: tracker.toolUseCount, 100 tokenCount: getTokenCountFromTracker(tracker), 101 lastActivity: tracker.recentActivities.length > 0 ? tracker.recentActivities[tracker.recentActivities.length - 1] : undefined, 102 recentActivities: [...tracker.recentActivities] 103 }; 104} 105 106/** 107 * Creates an ActivityDescriptionResolver from a tools list. 108 * Looks up the tool by name and calls getActivityDescription if available. 109 */ 110export function createActivityDescriptionResolver(tools: Tools): ActivityDescriptionResolver { 111 return (toolName, input) => { 112 const tool = findToolByName(tools, toolName); 113 return tool?.getActivityDescription?.(input) ?? undefined; 114 }; 115} 116export type LocalAgentTaskState = TaskStateBase & { 117 type: 'local_agent'; 118 agentId: string; 119 prompt: string; 120 selectedAgent?: AgentDefinition; 121 agentType: string; 122 model?: string; 123 abortController?: AbortController; 124 unregisterCleanup?: () => void; 125 error?: string; 126 result?: AgentToolResult; 127 progress?: AgentProgress; 128 retrieved: boolean; 129 messages?: Message[]; 130 // Track what we last reported for computing deltas 131 lastReportedToolCount: number; 132 lastReportedTokenCount: number; 133 // Whether the task has been backgrounded (false = foreground running, true = backgrounded) 134 isBackgrounded: boolean; 135 // Messages queued mid-turn via SendMessage, drained at tool-round boundaries 136 pendingMessages: string[]; 137 // UI is holding this task: blocks eviction, enables stream-append, triggers 138 // disk bootstrap. Set by enterTeammateView. Separate from viewingAgentTaskId 139 // (which is "what am I LOOKING at") — retain is "what am I HOLDING." 140 retain: boolean; 141 // Bootstrap has read the sidechain JSONL and UUID-merged into messages. 142 // One-shot per retain cycle; stream appends from there. 143 diskLoaded: boolean; 144 // Panel visibility deadline. undefined = no deadline (running or retained); 145 // timestamp = hide + GC-eligible after this time. Set at terminal transition 146 // and on unselect; cleared on retain. 147 evictAfter?: number; 148}; 149export function isLocalAgentTask(task: unknown): task is LocalAgentTaskState { 150 return typeof task === 'object' && task !== null && 'type' in task && task.type === 'local_agent'; 151} 152 153/** 154 * A local_agent task that the CoordinatorTaskPanel manages (not main-session). 155 * For ants, these render in the panel instead of the background-task pill. 156 * This is the ONE predicate that all pill/panel filters must agree on — if 157 * the gate changes, change it here. 158 */ 159export function isPanelAgentTask(t: unknown): t is LocalAgentTaskState { 160 return isLocalAgentTask(t) && t.agentType !== 'main-session'; 161} 162export function queuePendingMessage(taskId: string, msg: string, setAppState: (f: (prev: AppState) => AppState) => void): void { 163 updateTaskState<LocalAgentTaskState>(taskId, setAppState, task => ({ 164 ...task, 165 pendingMessages: [...task.pendingMessages, msg] 166 })); 167} 168 169/** 170 * Append a message to task.messages so it appears in the viewed transcript 171 * immediately. Caller constructs the Message (breaks the messages.ts cycle). 172 * queuePendingMessage and resumeAgentBackground route the prompt to the 173 * agent's API input but don't touch the display. 174 */ 175export function appendMessageToLocalAgent(taskId: string, message: Message, setAppState: (f: (prev: AppState) => AppState) => void): void { 176 updateTaskState<LocalAgentTaskState>(taskId, setAppState, task => ({ 177 ...task, 178 messages: [...(task.messages ?? []), message] 179 })); 180} 181export function drainPendingMessages(taskId: string, getAppState: () => AppState, setAppState: (f: (prev: AppState) => AppState) => void): string[] { 182 const task = getAppState().tasks[taskId]; 183 if (!isLocalAgentTask(task) || task.pendingMessages.length === 0) { 184 return []; 185 } 186 const drained = task.pendingMessages; 187 updateTaskState<LocalAgentTaskState>(taskId, setAppState, t => ({ 188 ...t, 189 pendingMessages: [] 190 })); 191 return drained; 192} 193 194/** 195 * Enqueue an agent notification to the message queue. 196 */ 197export function enqueueAgentNotification({ 198 taskId, 199 description, 200 status, 201 error, 202 setAppState, 203 finalMessage, 204 usage, 205 toolUseId, 206 worktreePath, 207 worktreeBranch 208}: { 209 taskId: string; 210 description: string; 211 status: 'completed' | 'failed' | 'killed'; 212 error?: string; 213 setAppState: SetAppState; 214 finalMessage?: string; 215 usage?: { 216 totalTokens: number; 217 toolUses: number; 218 durationMs: number; 219 }; 220 toolUseId?: string; 221 worktreePath?: string; 222 worktreeBranch?: string; 223}): void { 224 // Atomically check and set notified flag to prevent duplicate notifications. 225 // If the task was already marked as notified (e.g., by TaskStopTool), skip 226 // enqueueing to avoid sending redundant messages to the model. 227 let shouldEnqueue = false; 228 updateTaskState<LocalAgentTaskState>(taskId, setAppState, task => { 229 if (task.notified) { 230 return task; 231 } 232 shouldEnqueue = true; 233 return { 234 ...task, 235 notified: true 236 }; 237 }); 238 if (!shouldEnqueue) { 239 return; 240 } 241 242 // Abort any active speculation — background task state changed, so speculated 243 // results may reference stale task output. The prompt suggestion text is 244 // preserved; only the pre-computed response is discarded. 245 abortSpeculation(setAppState); 246 const summary = status === 'completed' ? `Agent "${description}" completed` : status === 'failed' ? `Agent "${description}" failed: ${error || 'Unknown error'}` : `Agent "${description}" was stopped`; 247 const outputPath = getTaskOutputPath(taskId); 248 const toolUseIdLine = toolUseId ? `\n<${TOOL_USE_ID_TAG}>${toolUseId}</${TOOL_USE_ID_TAG}>` : ''; 249 const resultSection = finalMessage ? `\n<result>${finalMessage}</result>` : ''; 250 const usageSection = usage ? `\n<usage><total_tokens>${usage.totalTokens}</total_tokens><tool_uses>${usage.toolUses}</tool_uses><duration_ms>${usage.durationMs}</duration_ms></usage>` : ''; 251 const worktreeSection = worktreePath ? `\n<${WORKTREE_TAG}><${WORKTREE_PATH_TAG}>${worktreePath}</${WORKTREE_PATH_TAG}>${worktreeBranch ? `<${WORKTREE_BRANCH_TAG}>${worktreeBranch}</${WORKTREE_BRANCH_TAG}>` : ''}</${WORKTREE_TAG}>` : ''; 252 const message = `<${TASK_NOTIFICATION_TAG}> 253<${TASK_ID_TAG}>${taskId}</${TASK_ID_TAG}>${toolUseIdLine} 254<${OUTPUT_FILE_TAG}>${outputPath}</${OUTPUT_FILE_TAG}> 255<${STATUS_TAG}>${status}</${STATUS_TAG}> 256<${SUMMARY_TAG}>${summary}</${SUMMARY_TAG}>${resultSection}${usageSection}${worktreeSection} 257</${TASK_NOTIFICATION_TAG}>`; 258 enqueuePendingNotification({ 259 value: message, 260 mode: 'task-notification' 261 }); 262} 263 264/** 265 * LocalAgentTask - Handles background agent execution. 266 * 267 * Replaces the AsyncAgent implementation from src/tools/AgentTool/asyncAgentUtils.ts 268 * with a unified Task interface. 269 */ 270export const LocalAgentTask: Task = { 271 name: 'LocalAgentTask', 272 type: 'local_agent', 273 async kill(taskId, setAppState) { 274 killAsyncAgent(taskId, setAppState); 275 } 276}; 277 278/** 279 * Kill an agent task. No-op if already killed/completed. 280 */ 281export function killAsyncAgent(taskId: string, setAppState: SetAppState): void { 282 let killed = false; 283 updateTaskState<LocalAgentTaskState>(taskId, setAppState, task => { 284 if (task.status !== 'running') { 285 return task; 286 } 287 killed = true; 288 task.abortController?.abort(); 289 task.unregisterCleanup?.(); 290 return { 291 ...task, 292 status: 'killed', 293 endTime: Date.now(), 294 evictAfter: task.retain ? undefined : Date.now() + PANEL_GRACE_MS, 295 abortController: undefined, 296 unregisterCleanup: undefined, 297 selectedAgent: undefined 298 }; 299 }); 300 if (killed) { 301 void evictTaskOutput(taskId); 302 } 303} 304 305/** 306 * Kill all running agent tasks. 307 * Used by ESC cancellation in coordinator mode to stop all subagents. 308 */ 309export function killAllRunningAgentTasks(tasks: Record<string, TaskState>, setAppState: SetAppState): void { 310 for (const [taskId, task] of Object.entries(tasks)) { 311 if (task.type === 'local_agent' && task.status === 'running') { 312 killAsyncAgent(taskId, setAppState); 313 } 314 } 315} 316 317/** 318 * Mark a task as notified without enqueueing a notification. 319 * Used by chat:killAgents bulk kill to suppress per-agent async notifications 320 * when a single aggregate message is sent instead. 321 */ 322export function markAgentsNotified(taskId: string, setAppState: SetAppState): void { 323 updateTaskState<LocalAgentTaskState>(taskId, setAppState, task => { 324 if (task.notified) { 325 return task; 326 } 327 return { 328 ...task, 329 notified: true 330 }; 331 }); 332} 333 334/** 335 * Update progress for an agent task. 336 * Preserves the existing summary field so that background summarization 337 * results are not clobbered by progress updates from assistant messages. 338 */ 339export function updateAgentProgress(taskId: string, progress: AgentProgress, setAppState: SetAppState): void { 340 updateTaskState<LocalAgentTaskState>(taskId, setAppState, task => { 341 if (task.status !== 'running') { 342 return task; 343 } 344 const existingSummary = task.progress?.summary; 345 return { 346 ...task, 347 progress: existingSummary ? { 348 ...progress, 349 summary: existingSummary 350 } : progress 351 }; 352 }); 353} 354 355/** 356 * Update the background summary for an agent task. 357 * Called by the periodic summarization service to store a 1-2 sentence progress summary. 358 */ 359export function updateAgentSummary(taskId: string, summary: string, setAppState: SetAppState): void { 360 let captured: { 361 tokenCount: number; 362 toolUseCount: number; 363 startTime: number; 364 toolUseId: string | undefined; 365 } | null = null; 366 updateTaskState<LocalAgentTaskState>(taskId, setAppState, task => { 367 if (task.status !== 'running') { 368 return task; 369 } 370 captured = { 371 tokenCount: task.progress?.tokenCount ?? 0, 372 toolUseCount: task.progress?.toolUseCount ?? 0, 373 startTime: task.startTime, 374 toolUseId: task.toolUseId 375 }; 376 return { 377 ...task, 378 progress: { 379 ...task.progress, 380 toolUseCount: task.progress?.toolUseCount ?? 0, 381 tokenCount: task.progress?.tokenCount ?? 0, 382 summary 383 } 384 }; 385 }); 386 387 // Emit summary to SDK consumers (e.g. VS Code subagent panel). No-op in TUI. 388 // Gate on the SDK option so coordinator-mode sessions without the flag don't 389 // leak summary events to consumers who didn't opt in. 390 if (captured && getSdkAgentProgressSummariesEnabled()) { 391 const { 392 tokenCount, 393 toolUseCount, 394 startTime, 395 toolUseId 396 } = captured; 397 emitTaskProgress({ 398 taskId, 399 toolUseId, 400 description: summary, 401 startTime, 402 totalTokens: tokenCount, 403 toolUses: toolUseCount, 404 summary 405 }); 406 } 407} 408 409/** 410 * Complete an agent task with result. 411 */ 412export function completeAgentTask(result: AgentToolResult, setAppState: SetAppState): void { 413 const taskId = result.agentId; 414 updateTaskState<LocalAgentTaskState>(taskId, setAppState, task => { 415 if (task.status !== 'running') { 416 return task; 417 } 418 task.unregisterCleanup?.(); 419 return { 420 ...task, 421 status: 'completed', 422 result, 423 endTime: Date.now(), 424 evictAfter: task.retain ? undefined : Date.now() + PANEL_GRACE_MS, 425 abortController: undefined, 426 unregisterCleanup: undefined, 427 selectedAgent: undefined 428 }; 429 }); 430 void evictTaskOutput(taskId); 431 // Note: Notification is sent by AgentTool via enqueueAgentNotification 432} 433 434/** 435 * Fail an agent task with error. 436 */ 437export function failAgentTask(taskId: string, error: string, setAppState: SetAppState): void { 438 updateTaskState<LocalAgentTaskState>(taskId, setAppState, task => { 439 if (task.status !== 'running') { 440 return task; 441 } 442 task.unregisterCleanup?.(); 443 return { 444 ...task, 445 status: 'failed', 446 error, 447 endTime: Date.now(), 448 evictAfter: task.retain ? undefined : Date.now() + PANEL_GRACE_MS, 449 abortController: undefined, 450 unregisterCleanup: undefined, 451 selectedAgent: undefined 452 }; 453 }); 454 void evictTaskOutput(taskId); 455 // Note: Notification is sent by AgentTool via enqueueAgentNotification 456} 457 458/** 459 * Register an agent task. 460 * Called by AgentTool to create a new background agent. 461 * 462 * @param parentAbortController - Optional parent abort controller. If provided, 463 * the agent's abort controller will be a child that auto-aborts when parent aborts. 464 * This ensures subagents are aborted when their parent (e.g., in-process teammate) aborts. 465 */ 466export function registerAsyncAgent({ 467 agentId, 468 description, 469 prompt, 470 selectedAgent, 471 setAppState, 472 parentAbortController, 473 toolUseId 474}: { 475 agentId: string; 476 description: string; 477 prompt: string; 478 selectedAgent: AgentDefinition; 479 setAppState: SetAppState; 480 parentAbortController?: AbortController; 481 toolUseId?: string; 482}): LocalAgentTaskState { 483 void initTaskOutputAsSymlink(agentId, getAgentTranscriptPath(asAgentId(agentId))); 484 485 // Create abort controller - if parent provided, create child that auto-aborts with parent 486 const abortController = parentAbortController ? createChildAbortController(parentAbortController) : createAbortController(); 487 const taskState: LocalAgentTaskState = { 488 ...createTaskStateBase(agentId, 'local_agent', description, toolUseId), 489 type: 'local_agent', 490 status: 'running', 491 agentId, 492 prompt, 493 selectedAgent, 494 agentType: selectedAgent.agentType ?? 'general-purpose', 495 abortController, 496 retrieved: false, 497 lastReportedToolCount: 0, 498 lastReportedTokenCount: 0, 499 isBackgrounded: true, 500 // registerAsyncAgent immediately backgrounds 501 pendingMessages: [], 502 retain: false, 503 diskLoaded: false 504 }; 505 506 // Register cleanup handler 507 const unregisterCleanup = registerCleanup(async () => { 508 killAsyncAgent(agentId, setAppState); 509 }); 510 taskState.unregisterCleanup = unregisterCleanup; 511 512 // Register task in AppState 513 registerTask(taskState, setAppState); 514 return taskState; 515} 516 517// Map of taskId -> resolve function for background signals 518// When backgroundAgentTask is called, it resolves the corresponding promise 519const backgroundSignalResolvers = new Map<string, () => void>(); 520 521/** 522 * Register a foreground agent task that could be backgrounded later. 523 * Called when an agent has been running long enough to show the BackgroundHint. 524 * @returns object with taskId and backgroundSignal promise 525 */ 526export function registerAgentForeground({ 527 agentId, 528 description, 529 prompt, 530 selectedAgent, 531 setAppState, 532 autoBackgroundMs, 533 toolUseId 534}: { 535 agentId: string; 536 description: string; 537 prompt: string; 538 selectedAgent: AgentDefinition; 539 setAppState: SetAppState; 540 autoBackgroundMs?: number; 541 toolUseId?: string; 542}): { 543 taskId: string; 544 backgroundSignal: Promise<void>; 545 cancelAutoBackground?: () => void; 546} { 547 void initTaskOutputAsSymlink(agentId, getAgentTranscriptPath(asAgentId(agentId))); 548 const abortController = createAbortController(); 549 const unregisterCleanup = registerCleanup(async () => { 550 killAsyncAgent(agentId, setAppState); 551 }); 552 const taskState: LocalAgentTaskState = { 553 ...createTaskStateBase(agentId, 'local_agent', description, toolUseId), 554 type: 'local_agent', 555 status: 'running', 556 agentId, 557 prompt, 558 selectedAgent, 559 agentType: selectedAgent.agentType ?? 'general-purpose', 560 abortController, 561 unregisterCleanup, 562 retrieved: false, 563 lastReportedToolCount: 0, 564 lastReportedTokenCount: 0, 565 isBackgrounded: false, 566 // Not yet backgrounded - running in foreground 567 pendingMessages: [], 568 retain: false, 569 diskLoaded: false 570 }; 571 572 // Create background signal promise 573 let resolveBackgroundSignal: () => void; 574 const backgroundSignal = new Promise<void>(resolve => { 575 resolveBackgroundSignal = resolve; 576 }); 577 backgroundSignalResolvers.set(agentId, resolveBackgroundSignal!); 578 registerTask(taskState, setAppState); 579 580 // Auto-background after timeout if configured 581 let cancelAutoBackground: (() => void) | undefined; 582 if (autoBackgroundMs !== undefined && autoBackgroundMs > 0) { 583 const timer = setTimeout((setAppState, agentId) => { 584 // Mark task as backgrounded and resolve the signal 585 setAppState(prev => { 586 const prevTask = prev.tasks[agentId]; 587 if (!isLocalAgentTask(prevTask) || prevTask.isBackgrounded) { 588 return prev; 589 } 590 return { 591 ...prev, 592 tasks: { 593 ...prev.tasks, 594 [agentId]: { 595 ...prevTask, 596 isBackgrounded: true 597 } 598 } 599 }; 600 }); 601 const resolver = backgroundSignalResolvers.get(agentId); 602 if (resolver) { 603 resolver(); 604 backgroundSignalResolvers.delete(agentId); 605 } 606 }, autoBackgroundMs, setAppState, agentId); 607 cancelAutoBackground = () => clearTimeout(timer); 608 } 609 return { 610 taskId: agentId, 611 backgroundSignal, 612 cancelAutoBackground 613 }; 614} 615 616/** 617 * Background a specific foreground agent task. 618 * @returns true if backgrounded successfully, false otherwise 619 */ 620export function backgroundAgentTask(taskId: string, getAppState: () => AppState, setAppState: SetAppState): boolean { 621 const state = getAppState(); 622 const task = state.tasks[taskId]; 623 if (!isLocalAgentTask(task) || task.isBackgrounded) { 624 return false; 625 } 626 627 // Update state to mark as backgrounded 628 setAppState(prev => { 629 const prevTask = prev.tasks[taskId]; 630 if (!isLocalAgentTask(prevTask)) { 631 return prev; 632 } 633 return { 634 ...prev, 635 tasks: { 636 ...prev.tasks, 637 [taskId]: { 638 ...prevTask, 639 isBackgrounded: true 640 } 641 } 642 }; 643 }); 644 645 // Resolve the background signal to interrupt the agent loop 646 const resolver = backgroundSignalResolvers.get(taskId); 647 if (resolver) { 648 resolver(); 649 backgroundSignalResolvers.delete(taskId); 650 } 651 return true; 652} 653 654/** 655 * Unregister a foreground agent task when the agent completes without being backgrounded. 656 */ 657export function unregisterAgentForeground(taskId: string, setAppState: SetAppState): void { 658 // Clean up the background signal resolver 659 backgroundSignalResolvers.delete(taskId); 660 let cleanupFn: (() => void) | undefined; 661 setAppState(prev => { 662 const task = prev.tasks[taskId]; 663 // Only remove if it's a foreground task (not backgrounded) 664 if (!isLocalAgentTask(task) || task.isBackgrounded) { 665 return prev; 666 } 667 668 // Capture cleanup function to call outside of updater 669 cleanupFn = task.unregisterCleanup; 670 const { 671 [taskId]: removed, 672 ...rest 673 } = prev.tasks; 674 return { 675 ...prev, 676 tasks: rest 677 }; 678 }); 679 680 // Call cleanup outside of the state updater (avoid side effects in updater) 681 cleanupFn?.(); 682} 683//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["getSdkAgentProgressSummariesEnabled","OUTPUT_FILE_TAG","STATUS_TAG","SUMMARY_TAG","TASK_ID_TAG","TASK_NOTIFICATION_TAG","TOOL_USE_ID_TAG","WORKTREE_BRANCH_TAG","WORKTREE_PATH_TAG","WORKTREE_TAG","abortSpeculation","AppState","SetAppState","Task","TaskStateBase","createTaskStateBase","Tools","findToolByName","AgentToolResult","AgentDefinition","SYNTHETIC_OUTPUT_TOOL_NAME","asAgentId","Message","createAbortController","createChildAbortController","registerCleanup","getToolSearchOrReadInfo","enqueuePendingNotification","getAgentTranscriptPath","evictTaskOutput","getTaskOutputPath","initTaskOutputAsSymlink","PANEL_GRACE_MS","registerTask","updateTaskState","emitTaskProgress","TaskState","ToolActivity","toolName","input","Record","activityDescription","isSearch","isRead","AgentProgress","toolUseCount","tokenCount","lastActivity","recentActivities","summary","MAX_RECENT_ACTIVITIES","ProgressTracker","latestInputTokens","cumulativeOutputTokens","createProgressTracker","getTokenCountFromTracker","tracker","ActivityDescriptionResolver","updateProgressFromMessage","message","resolveActivityDescription","tools","type","usage","input_tokens","cache_creation_input_tokens","cache_read_input_tokens","output_tokens","content","name","classification","undefined","push","length","shift","getProgressUpdate","createActivityDescriptionResolver","tool","getActivityDescription","LocalAgentTaskState","agentId","prompt","selectedAgent","agentType","model","abortController","AbortController","unregisterCleanup","error","result","progress","retrieved","messages","lastReportedToolCount","lastReportedTokenCount","isBackgrounded","pendingMessages","retain","diskLoaded","evictAfter","isLocalAgentTask","task","isPanelAgentTask","t","queuePendingMessage","taskId","msg","setAppState","f","prev","appendMessageToLocalAgent","drainPendingMessages","getAppState","tasks","drained","enqueueAgentNotification","description","status","finalMessage","toolUseId","worktreePath","worktreeBranch","totalTokens","toolUses","durationMs","shouldEnqueue","notified","outputPath","toolUseIdLine","resultSection","usageSection","worktreeSection","value","mode","LocalAgentTask","kill","killAsyncAgent","killed","abort","endTime","Date","now","killAllRunningAgentTasks","Object","entries","markAgentsNotified","updateAgentProgress","existingSummary","updateAgentSummary","captured","startTime","completeAgentTask","failAgentTask","registerAsyncAgent","parentAbortController","taskState","backgroundSignalResolvers","Map","registerAgentForeground","autoBackgroundMs","backgroundSignal","Promise","cancelAutoBackground","resolveBackgroundSignal","resolve","set","timer","setTimeout","prevTask","resolver","get","delete","clearTimeout","backgroundAgentTask","state","unregisterAgentForeground","cleanupFn","removed","rest"],"sources":["LocalAgentTask.tsx"],"sourcesContent":["import { getSdkAgentProgressSummariesEnabled } from '../../bootstrap/state.js'\nimport {\n  OUTPUT_FILE_TAG,\n  STATUS_TAG,\n  SUMMARY_TAG,\n  TASK_ID_TAG,\n  TASK_NOTIFICATION_TAG,\n  TOOL_USE_ID_TAG,\n  WORKTREE_BRANCH_TAG,\n  WORKTREE_PATH_TAG,\n  WORKTREE_TAG,\n} from '../../constants/xml.js'\nimport { abortSpeculation } from '../../services/PromptSuggestion/speculation.js'\nimport type { AppState } from '../../state/AppState.js'\nimport type { SetAppState, Task, TaskStateBase } from '../../Task.js'\nimport { createTaskStateBase } from '../../Task.js'\nimport type { Tools } from '../../Tool.js'\nimport { findToolByName } from '../../Tool.js'\nimport type { AgentToolResult } from '../../tools/AgentTool/agentToolUtils.js'\nimport type { AgentDefinition } from '../../tools/AgentTool/loadAgentsDir.js'\nimport { SYNTHETIC_OUTPUT_TOOL_NAME } from '../../tools/SyntheticOutputTool/SyntheticOutputTool.js'\nimport { asAgentId } from '../../types/ids.js'\nimport type { Message } from '../../types/message.js'\nimport {\n  createAbortController,\n  createChildAbortController,\n} from '../../utils/abortController.js'\nimport { registerCleanup } from '../../utils/cleanupRegistry.js'\nimport { getToolSearchOrReadInfo } from '../../utils/collapseReadSearch.js'\nimport { enqueuePendingNotification } from '../../utils/messageQueueManager.js'\nimport { getAgentTranscriptPath } from '../../utils/sessionStorage.js'\nimport {\n  evictTaskOutput,\n  getTaskOutputPath,\n  initTaskOutputAsSymlink,\n} from '../../utils/task/diskOutput.js'\nimport {\n  PANEL_GRACE_MS,\n  registerTask,\n  updateTaskState,\n} from '../../utils/task/framework.js'\nimport { emitTaskProgress } from '../../utils/task/sdkProgress.js'\nimport type { TaskState } from '../types.js'\n\nexport type ToolActivity = {\n  toolName: string\n  input: Record<string, unknown>\n  /** Pre-computed activity description from the tool, e.g. \"Reading src/foo.ts\" */\n  activityDescription?: string\n  /** Pre-computed: true if this is a search operation (Grep, Glob, etc.) */\n  isSearch?: boolean\n  /** Pre-computed: true if this is a read operation (Read, cat, etc.) */\n  isRead?: boolean\n}\n\nexport type AgentProgress = {\n  toolUseCount: number\n  tokenCount: number\n  lastActivity?: ToolActivity\n  recentActivities?: ToolActivity[]\n  summary?: string\n}\n\nconst MAX_RECENT_ACTIVITIES = 5\n\nexport type ProgressTracker = {\n  toolUseCount: number\n  // Track input and output separately to avoid double-counting.\n  // input_tokens in Claude API is cumulative per turn (includes all previous context),\n  // so we keep the latest value. output_tokens is per-turn, so we sum those.\n  latestInputTokens: number\n  cumulativeOutputTokens: number\n  recentActivities: ToolActivity[]\n}\n\nexport function createProgressTracker(): ProgressTracker {\n  return {\n    toolUseCount: 0,\n    latestInputTokens: 0,\n    cumulativeOutputTokens: 0,\n    recentActivities: [],\n  }\n}\n\nexport function getTokenCountFromTracker(tracker: ProgressTracker): number {\n  return tracker.latestInputTokens + tracker.cumulativeOutputTokens\n}\n\n/**\n * Resolver function that returns a human-readable activity description\n * for a given tool name and input. Used to pre-compute descriptions\n * from Tool.getActivityDescription() at recording time.\n */\nexport type ActivityDescriptionResolver = (\n  toolName: string,\n  input: Record<string, unknown>,\n) => string | undefined\n\nexport function updateProgressFromMessage(\n  tracker: ProgressTracker,\n  message: Message,\n  resolveActivityDescription?: ActivityDescriptionResolver,\n  tools?: Tools,\n): void {\n  if (message.type !== 'assistant') {\n    return\n  }\n  const usage = message.message.usage\n  // Keep latest input (it's cumulative in the API), sum outputs\n  tracker.latestInputTokens =\n    usage.input_tokens +\n    (usage.cache_creation_input_tokens ?? 0) +\n    (usage.cache_read_input_tokens ?? 0)\n  tracker.cumulativeOutputTokens += usage.output_tokens\n  for (const content of message.message.content) {\n    if (content.type === 'tool_use') {\n      tracker.toolUseCount++\n      // Omit StructuredOutput from preview - it's an internal tool\n      if (content.name !== SYNTHETIC_OUTPUT_TOOL_NAME) {\n        const input = content.input as Record<string, unknown>\n        const classification = tools\n          ? getToolSearchOrReadInfo(content.name, input, tools)\n          : undefined\n        tracker.recentActivities.push({\n          toolName: content.name,\n          input,\n          activityDescription: resolveActivityDescription?.(\n            content.name,\n            input,\n          ),\n          isSearch: classification?.isSearch,\n          isRead: classification?.isRead,\n        })\n      }\n    }\n  }\n  while (tracker.recentActivities.length > MAX_RECENT_ACTIVITIES) {\n    tracker.recentActivities.shift()\n  }\n}\n\nexport function getProgressUpdate(tracker: ProgressTracker): AgentProgress {\n  return {\n    toolUseCount: tracker.toolUseCount,\n    tokenCount: getTokenCountFromTracker(tracker),\n    lastActivity:\n      tracker.recentActivities.length > 0\n        ? tracker.recentActivities[tracker.recentActivities.length - 1]\n        : undefined,\n    recentActivities: [...tracker.recentActivities],\n  }\n}\n\n/**\n * Creates an ActivityDescriptionResolver from a tools list.\n * Looks up the tool by name and calls getActivityDescription if available.\n */\nexport function createActivityDescriptionResolver(\n  tools: Tools,\n): ActivityDescriptionResolver {\n  return (toolName, input) => {\n    const tool = findToolByName(tools, toolName)\n    return tool?.getActivityDescription?.(input) ?? undefined\n  }\n}\n\nexport type LocalAgentTaskState = TaskStateBase & {\n  type: 'local_agent'\n  agentId: string\n  prompt: string\n  selectedAgent?: AgentDefinition\n  agentType: string\n  model?: string\n  abortController?: AbortController\n  unregisterCleanup?: () => void\n  error?: string\n  result?: AgentToolResult\n  progress?: AgentProgress\n  retrieved: boolean\n  messages?: Message[]\n  // Track what we last reported for computing deltas\n  lastReportedToolCount: number\n  lastReportedTokenCount: number\n  // Whether the task has been backgrounded (false = foreground running, true = backgrounded)\n  isBackgrounded: boolean\n  // Messages queued mid-turn via SendMessage, drained at tool-round boundaries\n  pendingMessages: string[]\n  // UI is holding this task: blocks eviction, enables stream-append, triggers\n  // disk bootstrap. Set by enterTeammateView. Separate from viewingAgentTaskId\n  // (which is \"what am I LOOKING at\") — retain is \"what am I HOLDING.\"\n  retain: boolean\n  // Bootstrap has read the sidechain JSONL and UUID-merged into messages.\n  // One-shot per retain cycle; stream appends from there.\n  diskLoaded: boolean\n  // Panel visibility deadline. undefined = no deadline (running or retained);\n  // timestamp = hide + GC-eligible after this time. Set at terminal transition\n  // and on unselect; cleared on retain.\n  evictAfter?: number\n}\n\nexport function isLocalAgentTask(task: unknown): task is LocalAgentTaskState {\n  return (\n    typeof task === 'object' &&\n    task !== null &&\n    'type' in task &&\n    task.type === 'local_agent'\n  )\n}\n\n/**\n * A local_agent task that the CoordinatorTaskPanel manages (not main-session).\n * For ants, these render in the panel instead of the background-task pill.\n * This is the ONE predicate that all pill/panel filters must agree on — if\n * the gate changes, change it here.\n */\nexport function isPanelAgentTask(t: unknown): t is LocalAgentTaskState {\n  return isLocalAgentTask(t) && t.agentType !== 'main-session'\n}\n\nexport function queuePendingMessage(\n  taskId: string,\n  msg: string,\n  setAppState: (f: (prev: AppState) => AppState) => void,\n): void {\n  updateTaskState<LocalAgentTaskState>(taskId, setAppState, task => ({\n    ...task,\n    pendingMessages: [...task.pendingMessages, msg],\n  }))\n}\n\n/**\n * Append a message to task.messages so it appears in the viewed transcript\n * immediately. Caller constructs the Message (breaks the messages.ts cycle).\n * queuePendingMessage and resumeAgentBackground route the prompt to the\n * agent's API input but don't touch the display.\n */\nexport function appendMessageToLocalAgent(\n  taskId: string,\n  message: Message,\n  setAppState: (f: (prev: AppState) => AppState) => void,\n): void {\n  updateTaskState<LocalAgentTaskState>(taskId, setAppState, task => ({\n    ...task,\n    messages: [...(task.messages ?? []), message],\n  }))\n}\n\nexport function drainPendingMessages(\n  taskId: string,\n  getAppState: () => AppState,\n  setAppState: (f: (prev: AppState) => AppState) => void,\n): string[] {\n  const task = getAppState().tasks[taskId]\n  if (!isLocalAgentTask(task) || task.pendingMessages.length === 0) {\n    return []\n  }\n  const drained = task.pendingMessages\n  updateTaskState<LocalAgentTaskState>(taskId, setAppState, t => ({\n    ...t,\n    pendingMessages: [],\n  }))\n  return drained\n}\n\n/**\n * Enqueue an agent notification to the message queue.\n */\nexport function enqueueAgentNotification({\n  taskId,\n  description,\n  status,\n  error,\n  setAppState,\n  finalMessage,\n  usage,\n  toolUseId,\n  worktreePath,\n  worktreeBranch,\n}: {\n  taskId: string\n  description: string\n  status: 'completed' | 'failed' | 'killed'\n  error?: string\n  setAppState: SetAppState\n  finalMessage?: string\n  usage?: {\n    totalTokens: number\n    toolUses: number\n    durationMs: number\n  }\n  toolUseId?: string\n  worktreePath?: string\n  worktreeBranch?: string\n}): void {\n  // Atomically check and set notified flag to prevent duplicate notifications.\n  // If the task was already marked as notified (e.g., by TaskStopTool), skip\n  // enqueueing to avoid sending redundant messages to the model.\n  let shouldEnqueue = false\n  updateTaskState<LocalAgentTaskState>(taskId, setAppState, task => {\n    if (task.notified) {\n      return task\n    }\n    shouldEnqueue = true\n    return {\n      ...task,\n      notified: true,\n    }\n  })\n\n  if (!shouldEnqueue) {\n    return\n  }\n\n  // Abort any active speculation — background task state changed, so speculated\n  // results may reference stale task output. The prompt suggestion text is\n  // preserved; only the pre-computed response is discarded.\n  abortSpeculation(setAppState)\n\n  const summary =\n    status === 'completed'\n      ? `Agent \"${description}\" completed`\n      : status === 'failed'\n        ? `Agent \"${description}\" failed: ${error || 'Unknown error'}`\n        : `Agent \"${description}\" was stopped`\n\n  const outputPath = getTaskOutputPath(taskId)\n  const toolUseIdLine = toolUseId\n    ? `\\n<${TOOL_USE_ID_TAG}>${toolUseId}</${TOOL_USE_ID_TAG}>`\n    : ''\n  const resultSection = finalMessage ? `\\n<result>${finalMessage}</result>` : ''\n  const usageSection = usage\n    ? `\\n<usage><total_tokens>${usage.totalTokens}</total_tokens><tool_uses>${usage.toolUses}</tool_uses><duration_ms>${usage.durationMs}</duration_ms></usage>`\n    : ''\n  const worktreeSection = worktreePath\n    ? `\\n<${WORKTREE_TAG}><${WORKTREE_PATH_TAG}>${worktreePath}</${WORKTREE_PATH_TAG}>${worktreeBranch ? `<${WORKTREE_BRANCH_TAG}>${worktreeBranch}</${WORKTREE_BRANCH_TAG}>` : ''}</${WORKTREE_TAG}>`\n    : ''\n\n  const message = `<${TASK_NOTIFICATION_TAG}>\n<${TASK_ID_TAG}>${taskId}</${TASK_ID_TAG}>${toolUseIdLine}\n<${OUTPUT_FILE_TAG}>${outputPath}</${OUTPUT_FILE_TAG}>\n<${STATUS_TAG}>${status}</${STATUS_TAG}>\n<${SUMMARY_TAG}>${summary}</${SUMMARY_TAG}>${resultSection}${usageSection}${worktreeSection}\n</${TASK_NOTIFICATION_TAG}>`\n\n  enqueuePendingNotification({ value: message, mode: 'task-notification' })\n}\n\n/**\n * LocalAgentTask - Handles background agent execution.\n *\n * Replaces the AsyncAgent implementation from src/tools/AgentTool/asyncAgentUtils.ts\n * with a unified Task interface.\n */\nexport const LocalAgentTask: Task = {\n  name: 'LocalAgentTask',\n  type: 'local_agent',\n\n  async kill(taskId, setAppState) {\n    killAsyncAgent(taskId, setAppState)\n  },\n}\n\n/**\n * Kill an agent task. No-op if already killed/completed.\n */\nexport function killAsyncAgent(taskId: string, setAppState: SetAppState): void {\n  let killed = false\n  updateTaskState<LocalAgentTaskState>(taskId, setAppState, task => {\n    if (task.status !== 'running') {\n      return task\n    }\n    killed = true\n    task.abortController?.abort()\n    task.unregisterCleanup?.()\n    return {\n      ...task,\n      status: 'killed',\n      endTime: Date.now(),\n      evictAfter: task.retain ? undefined : Date.now() + PANEL_GRACE_MS,\n      abortController: undefined,\n      unregisterCleanup: undefined,\n      selectedAgent: undefined,\n    }\n  })\n  if (killed) {\n    void evictTaskOutput(taskId)\n  }\n}\n\n/**\n * Kill all running agent tasks.\n * Used by ESC cancellation in coordinator mode to stop all subagents.\n */\nexport function killAllRunningAgentTasks(\n  tasks: Record<string, TaskState>,\n  setAppState: SetAppState,\n): void {\n  for (const [taskId, task] of Object.entries(tasks)) {\n    if (task.type === 'local_agent' && task.status === 'running') {\n      killAsyncAgent(taskId, setAppState)\n    }\n  }\n}\n\n/**\n * Mark a task as notified without enqueueing a notification.\n * Used by chat:killAgents bulk kill to suppress per-agent async notifications\n * when a single aggregate message is sent instead.\n */\nexport function markAgentsNotified(\n  taskId: string,\n  setAppState: SetAppState,\n): void {\n  updateTaskState<LocalAgentTaskState>(taskId, setAppState, task => {\n    if (task.notified) {\n      return task\n    }\n    return {\n      ...task,\n      notified: true,\n    }\n  })\n}\n\n/**\n * Update progress for an agent task.\n * Preserves the existing summary field so that background summarization\n * results are not clobbered by progress updates from assistant messages.\n */\nexport function updateAgentProgress(\n  taskId: string,\n  progress: AgentProgress,\n  setAppState: SetAppState,\n): void {\n  updateTaskState<LocalAgentTaskState>(taskId, setAppState, task => {\n    if (task.status !== 'running') {\n      return task\n    }\n\n    const existingSummary = task.progress?.summary\n    return {\n      ...task,\n      progress: existingSummary\n        ? { ...progress, summary: existingSummary }\n        : progress,\n    }\n  })\n}\n\n/**\n * Update the background summary for an agent task.\n * Called by the periodic summarization service to store a 1-2 sentence progress summary.\n */\nexport function updateAgentSummary(\n  taskId: string,\n  summary: string,\n  setAppState: SetAppState,\n): void {\n  let captured: {\n    tokenCount: number\n    toolUseCount: number\n    startTime: number\n    toolUseId: string | undefined\n  } | null = null\n\n  updateTaskState<LocalAgentTaskState>(taskId, setAppState, task => {\n    if (task.status !== 'running') {\n      return task\n    }\n\n    captured = {\n      tokenCount: task.progress?.tokenCount ?? 0,\n      toolUseCount: task.progress?.toolUseCount ?? 0,\n      startTime: task.startTime,\n      toolUseId: task.toolUseId,\n    }\n\n    return {\n      ...task,\n      progress: {\n        ...task.progress,\n        toolUseCount: task.progress?.toolUseCount ?? 0,\n        tokenCount: task.progress?.tokenCount ?? 0,\n        summary,\n      },\n    }\n  })\n\n  // Emit summary to SDK consumers (e.g. VS Code subagent panel). No-op in TUI.\n  // Gate on the SDK option so coordinator-mode sessions without the flag don't\n  // leak summary events to consumers who didn't opt in.\n  if (captured && getSdkAgentProgressSummariesEnabled()) {\n    const { tokenCount, toolUseCount, startTime, toolUseId } = captured\n    emitTaskProgress({\n      taskId,\n      toolUseId,\n      description: summary,\n      startTime,\n      totalTokens: tokenCount,\n      toolUses: toolUseCount,\n      summary,\n    })\n  }\n}\n\n/**\n * Complete an agent task with result.\n */\nexport function completeAgentTask(\n  result: AgentToolResult,\n  setAppState: SetAppState,\n): void {\n  const taskId = result.agentId\n  updateTaskState<LocalAgentTaskState>(taskId, setAppState, task => {\n    if (task.status !== 'running') {\n      return task\n    }\n\n    task.unregisterCleanup?.()\n\n    return {\n      ...task,\n      status: 'completed',\n      result,\n      endTime: Date.now(),\n      evictAfter: task.retain ? undefined : Date.now() + PANEL_GRACE_MS,\n      abortController: undefined,\n      unregisterCleanup: undefined,\n      selectedAgent: undefined,\n    }\n  })\n  void evictTaskOutput(taskId)\n  // Note: Notification is sent by AgentTool via enqueueAgentNotification\n}\n\n/**\n * Fail an agent task with error.\n */\nexport function failAgentTask(\n  taskId: string,\n  error: string,\n  setAppState: SetAppState,\n): void {\n  updateTaskState<LocalAgentTaskState>(taskId, setAppState, task => {\n    if (task.status !== 'running') {\n      return task\n    }\n\n    task.unregisterCleanup?.()\n\n    return {\n      ...task,\n      status: 'failed',\n      error,\n      endTime: Date.now(),\n      evictAfter: task.retain ? undefined : Date.now() + PANEL_GRACE_MS,\n      abortController: undefined,\n      unregisterCleanup: undefined,\n      selectedAgent: undefined,\n    }\n  })\n  void evictTaskOutput(taskId)\n  // Note: Notification is sent by AgentTool via enqueueAgentNotification\n}\n\n/**\n * Register an agent task.\n * Called by AgentTool to create a new background agent.\n *\n * @param parentAbortController - Optional parent abort controller. If provided,\n *   the agent's abort controller will be a child that auto-aborts when parent aborts.\n *   This ensures subagents are aborted when their parent (e.g., in-process teammate) aborts.\n */\nexport function registerAsyncAgent({\n  agentId,\n  description,\n  prompt,\n  selectedAgent,\n  setAppState,\n  parentAbortController,\n  toolUseId,\n}: {\n  agentId: string\n  description: string\n  prompt: string\n  selectedAgent: AgentDefinition\n  setAppState: SetAppState\n  parentAbortController?: AbortController\n  toolUseId?: string\n}): LocalAgentTaskState {\n  void initTaskOutputAsSymlink(\n    agentId,\n    getAgentTranscriptPath(asAgentId(agentId)),\n  )\n\n  // Create abort controller - if parent provided, create child that auto-aborts with parent\n  const abortController = parentAbortController\n    ? createChildAbortController(parentAbortController)\n    : createAbortController()\n\n  const taskState: LocalAgentTaskState = {\n    ...createTaskStateBase(agentId, 'local_agent', description, toolUseId),\n    type: 'local_agent',\n    status: 'running',\n    agentId,\n    prompt,\n    selectedAgent,\n    agentType: selectedAgent.agentType ?? 'general-purpose',\n    abortController,\n    retrieved: false,\n    lastReportedToolCount: 0,\n    lastReportedTokenCount: 0,\n    isBackgrounded: true, // registerAsyncAgent immediately backgrounds\n    pendingMessages: [],\n    retain: false,\n    diskLoaded: false,\n  }\n\n  // Register cleanup handler\n  const unregisterCleanup = registerCleanup(async () => {\n    killAsyncAgent(agentId, setAppState)\n  })\n\n  taskState.unregisterCleanup = unregisterCleanup\n\n  // Register task in AppState\n  registerTask(taskState, setAppState)\n\n  return taskState\n}\n\n// Map of taskId -> resolve function for background signals\n// When backgroundAgentTask is called, it resolves the corresponding promise\nconst backgroundSignalResolvers = new Map<string, () => void>()\n\n/**\n * Register a foreground agent task that could be backgrounded later.\n * Called when an agent has been running long enough to show the BackgroundHint.\n * @returns object with taskId and backgroundSignal promise\n */\nexport function registerAgentForeground({\n  agentId,\n  description,\n  prompt,\n  selectedAgent,\n  setAppState,\n  autoBackgroundMs,\n  toolUseId,\n}: {\n  agentId: string\n  description: string\n  prompt: string\n  selectedAgent: AgentDefinition\n  setAppState: SetAppState\n  autoBackgroundMs?: number\n  toolUseId?: string\n}): {\n  taskId: string\n  backgroundSignal: Promise<void>\n  cancelAutoBackground?: () => void\n} {\n  void initTaskOutputAsSymlink(\n    agentId,\n    getAgentTranscriptPath(asAgentId(agentId)),\n  )\n\n  const abortController = createAbortController()\n\n  const unregisterCleanup = registerCleanup(async () => {\n    killAsyncAgent(agentId, setAppState)\n  })\n\n  const taskState: LocalAgentTaskState = {\n    ...createTaskStateBase(agentId, 'local_agent', description, toolUseId),\n    type: 'local_agent',\n    status: 'running',\n    agentId,\n    prompt,\n    selectedAgent,\n    agentType: selectedAgent.agentType ?? 'general-purpose',\n    abortController,\n    unregisterCleanup,\n    retrieved: false,\n    lastReportedToolCount: 0,\n    lastReportedTokenCount: 0,\n    isBackgrounded: false, // Not yet backgrounded - running in foreground\n    pendingMessages: [],\n    retain: false,\n    diskLoaded: false,\n  }\n\n  // Create background signal promise\n  let resolveBackgroundSignal: () => void\n  const backgroundSignal = new Promise<void>(resolve => {\n    resolveBackgroundSignal = resolve\n  })\n  backgroundSignalResolvers.set(agentId, resolveBackgroundSignal!)\n\n  registerTask(taskState, setAppState)\n\n  // Auto-background after timeout if configured\n  let cancelAutoBackground: (() => void) | undefined\n  if (autoBackgroundMs !== undefined && autoBackgroundMs > 0) {\n    const timer = setTimeout(\n      (setAppState, agentId) => {\n        // Mark task as backgrounded and resolve the signal\n        setAppState(prev => {\n          const prevTask = prev.tasks[agentId]\n          if (!isLocalAgentTask(prevTask) || prevTask.isBackgrounded) {\n            return prev\n          }\n          return {\n            ...prev,\n            tasks: {\n              ...prev.tasks,\n              [agentId]: { ...prevTask, isBackgrounded: true },\n            },\n          }\n        })\n        const resolver = backgroundSignalResolvers.get(agentId)\n        if (resolver) {\n          resolver()\n          backgroundSignalResolvers.delete(agentId)\n        }\n      },\n      autoBackgroundMs,\n      setAppState,\n      agentId,\n    )\n    cancelAutoBackground = () => clearTimeout(timer)\n  }\n\n  return { taskId: agentId, backgroundSignal, cancelAutoBackground }\n}\n\n/**\n * Background a specific foreground agent task.\n * @returns true if backgrounded successfully, false otherwise\n */\nexport function backgroundAgentTask(\n  taskId: string,\n  getAppState: () => AppState,\n  setAppState: SetAppState,\n): boolean {\n  const state = getAppState()\n  const task = state.tasks[taskId]\n  if (!isLocalAgentTask(task) || task.isBackgrounded) {\n    return false\n  }\n\n  // Update state to mark as backgrounded\n  setAppState(prev => {\n    const prevTask = prev.tasks[taskId]\n    if (!isLocalAgentTask(prevTask)) {\n      return prev\n    }\n    return {\n      ...prev,\n      tasks: {\n        ...prev.tasks,\n        [taskId]: { ...prevTask, isBackgrounded: true },\n      },\n    }\n  })\n\n  // Resolve the background signal to interrupt the agent loop\n  const resolver = backgroundSignalResolvers.get(taskId)\n  if (resolver) {\n    resolver()\n    backgroundSignalResolvers.delete(taskId)\n  }\n\n  return true\n}\n\n/**\n * Unregister a foreground agent task when the agent completes without being backgrounded.\n */\nexport function unregisterAgentForeground(\n  taskId: string,\n  setAppState: SetAppState,\n): void {\n  // Clean up the background signal resolver\n  backgroundSignalResolvers.delete(taskId)\n\n  let cleanupFn: (() => void) | undefined\n\n  setAppState(prev => {\n    const task = prev.tasks[taskId]\n    // Only remove if it's a foreground task (not backgrounded)\n    if (!isLocalAgentTask(task) || task.isBackgrounded) {\n      return prev\n    }\n\n    // Capture cleanup function to call outside of updater\n    cleanupFn = task.unregisterCleanup\n\n    const { [taskId]: removed, ...rest } = prev.tasks\n    return { ...prev, tasks: rest }\n  })\n\n  // Call cleanup outside of the state updater (avoid side effects in updater)\n  cleanupFn?.()\n}\n"],"mappings":"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","ignoreList":[]}