yzcode-cli 1.0.1 → 1.0.3

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.
Files changed (117) hide show
  1. package/assistant/sessionHistory.ts +87 -0
  2. package/bootstrap/state.ts +1769 -0
  3. package/bridge/bridgeApi.ts +539 -0
  4. package/bridge/bridgeConfig.ts +48 -0
  5. package/bridge/bridgeDebug.ts +135 -0
  6. package/bridge/bridgeEnabled.ts +202 -0
  7. package/bridge/bridgeMain.ts +2999 -0
  8. package/bridge/bridgeMessaging.ts +461 -0
  9. package/bridge/bridgePermissionCallbacks.ts +43 -0
  10. package/bridge/bridgePointer.ts +210 -0
  11. package/bridge/bridgeStatusUtil.ts +163 -0
  12. package/bridge/bridgeUI.ts +530 -0
  13. package/bridge/capacityWake.ts +56 -0
  14. package/bridge/codeSessionApi.ts +168 -0
  15. package/bridge/createSession.ts +384 -0
  16. package/bridge/debugUtils.ts +141 -0
  17. package/bridge/envLessBridgeConfig.ts +165 -0
  18. package/bridge/flushGate.ts +71 -0
  19. package/bridge/inboundAttachments.ts +175 -0
  20. package/bridge/inboundMessages.ts +80 -0
  21. package/bridge/initReplBridge.ts +569 -0
  22. package/bridge/jwtUtils.ts +256 -0
  23. package/bridge/pollConfig.ts +110 -0
  24. package/bridge/pollConfigDefaults.ts +82 -0
  25. package/bridge/remoteBridgeCore.ts +1008 -0
  26. package/bridge/replBridge.ts +2406 -0
  27. package/bridge/replBridgeHandle.ts +36 -0
  28. package/bridge/replBridgeTransport.ts +370 -0
  29. package/bridge/sessionIdCompat.ts +57 -0
  30. package/bridge/sessionRunner.ts +550 -0
  31. package/bridge/trustedDevice.ts +210 -0
  32. package/bridge/types.ts +262 -0
  33. package/bridge/workSecret.ts +127 -0
  34. package/buddy/CompanionSprite.tsx +371 -0
  35. package/buddy/companion.ts +133 -0
  36. package/buddy/prompt.ts +36 -0
  37. package/buddy/sprites.ts +514 -0
  38. package/buddy/types.ts +148 -0
  39. package/buddy/useBuddyNotification.tsx +98 -0
  40. package/coordinator/coordinatorMode.ts +369 -0
  41. package/memdir/findRelevantMemories.ts +141 -0
  42. package/memdir/memdir.ts +507 -0
  43. package/memdir/memoryAge.ts +53 -0
  44. package/memdir/memoryScan.ts +94 -0
  45. package/memdir/memoryTypes.ts +271 -0
  46. package/memdir/paths.ts +278 -0
  47. package/memdir/teamMemPaths.ts +292 -0
  48. package/memdir/teamMemPrompts.ts +100 -0
  49. package/migrations/migrateAutoUpdatesToSettings.ts +61 -0
  50. package/migrations/migrateBypassPermissionsAcceptedToSettings.ts +40 -0
  51. package/migrations/migrateEnableAllProjectMcpServersToSettings.ts +118 -0
  52. package/migrations/migrateFennecToOpus.ts +45 -0
  53. package/migrations/migrateLegacyOpusToCurrent.ts +57 -0
  54. package/migrations/migrateOpusToOpus1m.ts +43 -0
  55. package/migrations/migrateReplBridgeEnabledToRemoteControlAtStartup.ts +22 -0
  56. package/migrations/migrateSonnet1mToSonnet45.ts +48 -0
  57. package/migrations/migrateSonnet45ToSonnet46.ts +67 -0
  58. package/migrations/resetAutoModeOptInForDefaultOffer.ts +51 -0
  59. package/migrations/resetProToOpusDefault.ts +51 -0
  60. package/native-ts/color-diff/index.ts +999 -0
  61. package/native-ts/file-index/index.ts +370 -0
  62. package/native-ts/yoga-layout/enums.ts +134 -0
  63. package/native-ts/yoga-layout/index.ts +2578 -0
  64. package/outputStyles/loadOutputStylesDir.ts +98 -0
  65. package/package.json +22 -5
  66. package/plugins/builtinPlugins.ts +159 -0
  67. package/plugins/bundled/index.ts +23 -0
  68. package/schemas/hooks.ts +222 -0
  69. package/screens/Doctor.tsx +575 -0
  70. package/screens/REPL.tsx +5006 -0
  71. package/screens/ResumeConversation.tsx +399 -0
  72. package/server/createDirectConnectSession.ts +88 -0
  73. package/server/directConnectManager.ts +213 -0
  74. package/server/types.ts +57 -0
  75. package/skills/bundled/batch.ts +124 -0
  76. package/skills/bundled/claudeApi.ts +196 -0
  77. package/skills/bundled/claudeApiContent.ts +75 -0
  78. package/skills/bundled/claudeInChrome.ts +34 -0
  79. package/skills/bundled/debug.ts +103 -0
  80. package/skills/bundled/index.ts +79 -0
  81. package/skills/bundled/keybindings.ts +339 -0
  82. package/skills/bundled/loop.ts +92 -0
  83. package/skills/bundled/loremIpsum.ts +282 -0
  84. package/skills/bundled/remember.ts +82 -0
  85. package/skills/bundled/scheduleRemoteAgents.ts +447 -0
  86. package/skills/bundled/simplify.ts +69 -0
  87. package/skills/bundled/skillify.ts +197 -0
  88. package/skills/bundled/stuck.ts +79 -0
  89. package/skills/bundled/updateConfig.ts +475 -0
  90. package/skills/bundled/verify/SKILL.md +3 -0
  91. package/skills/bundled/verify/examples/cli.md +3 -0
  92. package/skills/bundled/verify/examples/server.md +3 -0
  93. package/skills/bundled/verify.ts +30 -0
  94. package/skills/bundled/verifyContent.ts +13 -0
  95. package/skills/bundledSkills.ts +220 -0
  96. package/skills/loadSkillsDir.ts +1086 -0
  97. package/skills/mcpSkillBuilders.ts +44 -0
  98. package/tasks/DreamTask/DreamTask.ts +157 -0
  99. package/tasks/InProcessTeammateTask/InProcessTeammateTask.tsx +126 -0
  100. package/tasks/InProcessTeammateTask/types.ts +121 -0
  101. package/tasks/LocalAgentTask/LocalAgentTask.tsx +683 -0
  102. package/tasks/LocalMainSessionTask.ts +479 -0
  103. package/tasks/LocalShellTask/LocalShellTask.tsx +523 -0
  104. package/tasks/LocalShellTask/guards.ts +41 -0
  105. package/tasks/LocalShellTask/killShellTasks.ts +76 -0
  106. package/tasks/RemoteAgentTask/RemoteAgentTask.tsx +856 -0
  107. package/tasks/pillLabel.ts +82 -0
  108. package/tasks/stopTask.ts +100 -0
  109. package/tasks/types.ts +46 -0
  110. package/upstreamproxy/relay.ts +455 -0
  111. package/upstreamproxy/upstreamproxy.ts +285 -0
  112. package/vim/motions.ts +82 -0
  113. package/vim/operators.ts +556 -0
  114. package/vim/textObjects.ts +186 -0
  115. package/vim/transitions.ts +490 -0
  116. package/vim/types.ts +199 -0
  117. package/voice/voiceModeEnabled.ts +54 -0
@@ -0,0 +1,523 @@
1
+ import { feature } from 'bun:bundle';
2
+ import { stat } from 'fs/promises';
3
+ import { OUTPUT_FILE_TAG, STATUS_TAG, SUMMARY_TAG, TASK_ID_TAG, TASK_NOTIFICATION_TAG, TOOL_USE_ID_TAG } from '../../constants/xml.js';
4
+ import { abortSpeculation } from '../../services/PromptSuggestion/speculation.js';
5
+ import type { AppState } from '../../state/AppState.js';
6
+ import type { LocalShellSpawnInput, SetAppState, Task, TaskContext, TaskHandle } from '../../Task.js';
7
+ import { createTaskStateBase } from '../../Task.js';
8
+ import type { AgentId } from '../../types/ids.js';
9
+ import { registerCleanup } from '../../utils/cleanupRegistry.js';
10
+ import { tailFile } from '../../utils/fsOperations.js';
11
+ import { logError } from '../../utils/log.js';
12
+ import { enqueuePendingNotification } from '../../utils/messageQueueManager.js';
13
+ import type { ShellCommand } from '../../utils/ShellCommand.js';
14
+ import { evictTaskOutput, getTaskOutputPath } from '../../utils/task/diskOutput.js';
15
+ import { registerTask, updateTaskState } from '../../utils/task/framework.js';
16
+ import { escapeXml } from '../../utils/xml.js';
17
+ import { backgroundAgentTask, isLocalAgentTask } from '../LocalAgentTask/LocalAgentTask.js';
18
+ import { isMainSessionTask } from '../LocalMainSessionTask.js';
19
+ import { type BashTaskKind, isLocalShellTask, type LocalShellTaskState } from './guards.js';
20
+ import { killTask } from './killShellTasks.js';
21
+
22
+ /** Prefix that identifies a LocalShellTask summary to the UI collapse transform. */
23
+ export const BACKGROUND_BASH_SUMMARY_PREFIX = 'Background command ';
24
+ const STALL_CHECK_INTERVAL_MS = 5_000;
25
+ const STALL_THRESHOLD_MS = 45_000;
26
+ const STALL_TAIL_BYTES = 1024;
27
+
28
+ // Last-line patterns that suggest a command is blocked waiting for keyboard
29
+ // input. Used to gate the stall notification — we stay silent on commands that
30
+ // are merely slow (git log -S, long builds) and only notify when the tail
31
+ // looks like an interactive prompt the model can act on. See CC-1175.
32
+ const PROMPT_PATTERNS = [/\(y\/n\)/i,
33
+ // (Y/n), (y/N)
34
+ /\[y\/n\]/i,
35
+ // [Y/n], [y/N]
36
+ /\(yes\/no\)/i, /\b(?:Do you|Would you|Shall I|Are you sure|Ready to)\b.*\? *$/i,
37
+ // directed questions
38
+ /Press (any key|Enter)/i, /Continue\?/i, /Overwrite\?/i];
39
+ export function looksLikePrompt(tail: string): boolean {
40
+ const lastLine = tail.trimEnd().split('\n').pop() ?? '';
41
+ return PROMPT_PATTERNS.some(p => p.test(lastLine));
42
+ }
43
+
44
+ // Output-side analog of peekForStdinData (utils/process.ts): fire a one-shot
45
+ // notification if output stops growing and the tail looks like a prompt.
46
+ function startStallWatchdog(taskId: string, description: string, kind: BashTaskKind | undefined, toolUseId?: string, agentId?: AgentId): () => void {
47
+ if (kind === 'monitor') return () => {};
48
+ const outputPath = getTaskOutputPath(taskId);
49
+ let lastSize = 0;
50
+ let lastGrowth = Date.now();
51
+ let cancelled = false;
52
+ const timer = setInterval(() => {
53
+ void stat(outputPath).then(s => {
54
+ if (s.size > lastSize) {
55
+ lastSize = s.size;
56
+ lastGrowth = Date.now();
57
+ return;
58
+ }
59
+ if (Date.now() - lastGrowth < STALL_THRESHOLD_MS) return;
60
+ void tailFile(outputPath, STALL_TAIL_BYTES).then(({
61
+ content
62
+ }) => {
63
+ if (cancelled) return;
64
+ if (!looksLikePrompt(content)) {
65
+ // Not a prompt — keep watching. Reset so the next check is
66
+ // 45s out instead of re-reading the tail on every tick.
67
+ lastGrowth = Date.now();
68
+ return;
69
+ }
70
+ // Latch before the async-boundary-visible side effects so an
71
+ // overlapping tick's callback sees cancelled=true and bails.
72
+ cancelled = true;
73
+ clearInterval(timer);
74
+ const toolUseIdLine = toolUseId ? `\n<${TOOL_USE_ID_TAG}>${toolUseId}</${TOOL_USE_ID_TAG}>` : '';
75
+ const summary = `${BACKGROUND_BASH_SUMMARY_PREFIX}"${description}" appears to be waiting for interactive input`;
76
+ // No <status> tag — print.ts treats <status> as a terminal
77
+ // signal and an unknown value falls through to 'completed',
78
+ // falsely closing the task for SDK consumers. Statusless
79
+ // notifications are skipped by the SDK emitter (progress ping).
80
+ const message = `<${TASK_NOTIFICATION_TAG}>
81
+ <${TASK_ID_TAG}>${taskId}</${TASK_ID_TAG}>${toolUseIdLine}
82
+ <${OUTPUT_FILE_TAG}>${outputPath}</${OUTPUT_FILE_TAG}>
83
+ <${SUMMARY_TAG}>${escapeXml(summary)}</${SUMMARY_TAG}>
84
+ </${TASK_NOTIFICATION_TAG}>
85
+ Last output:
86
+ ${content.trimEnd()}
87
+
88
+ The command is likely blocked on an interactive prompt. Kill this task and re-run with piped input (e.g., \`echo y | command\`) or a non-interactive flag if one exists.`;
89
+ enqueuePendingNotification({
90
+ value: message,
91
+ mode: 'task-notification',
92
+ priority: 'next',
93
+ agentId
94
+ });
95
+ }, () => {});
96
+ }, () => {} // File may not exist yet
97
+ );
98
+ }, STALL_CHECK_INTERVAL_MS);
99
+ timer.unref();
100
+ return () => {
101
+ cancelled = true;
102
+ clearInterval(timer);
103
+ };
104
+ }
105
+ function enqueueShellNotification(taskId: string, description: string, status: 'completed' | 'failed' | 'killed', exitCode: number | undefined, setAppState: SetAppState, toolUseId?: string, kind: BashTaskKind = 'bash', agentId?: AgentId): void {
106
+ // Atomically check and set notified flag to prevent duplicate notifications.
107
+ // If the task was already marked as notified (e.g., by TaskStopTool), skip
108
+ // enqueueing to avoid sending redundant messages to the model.
109
+ let shouldEnqueue = false;
110
+ updateTaskState(taskId, setAppState, task => {
111
+ if (task.notified) {
112
+ return task;
113
+ }
114
+ shouldEnqueue = true;
115
+ return {
116
+ ...task,
117
+ notified: true
118
+ };
119
+ });
120
+ if (!shouldEnqueue) {
121
+ return;
122
+ }
123
+
124
+ // Abort any active speculation — background task state changed, so speculated
125
+ // results may reference stale task output. The prompt suggestion text is
126
+ // preserved; only the pre-computed response is discarded.
127
+ abortSpeculation(setAppState);
128
+ let summary: string;
129
+ if (feature('MONITOR_TOOL') && kind === 'monitor') {
130
+ // Monitor is streaming-only (post-#22764) — the script exiting means
131
+ // the stream ended, not "condition met". Distinct from the bash prefix
132
+ // so Monitor completions don't fold into the "N background commands
133
+ // completed" collapse.
134
+ switch (status) {
135
+ case 'completed':
136
+ summary = `Monitor "${description}" stream ended`;
137
+ break;
138
+ case 'failed':
139
+ summary = `Monitor "${description}" script failed${exitCode !== undefined ? ` (exit ${exitCode})` : ''}`;
140
+ break;
141
+ case 'killed':
142
+ summary = `Monitor "${description}" stopped`;
143
+ break;
144
+ }
145
+ } else {
146
+ switch (status) {
147
+ case 'completed':
148
+ summary = `${BACKGROUND_BASH_SUMMARY_PREFIX}"${description}" completed${exitCode !== undefined ? ` (exit code ${exitCode})` : ''}`;
149
+ break;
150
+ case 'failed':
151
+ summary = `${BACKGROUND_BASH_SUMMARY_PREFIX}"${description}" failed${exitCode !== undefined ? ` with exit code ${exitCode}` : ''}`;
152
+ break;
153
+ case 'killed':
154
+ summary = `${BACKGROUND_BASH_SUMMARY_PREFIX}"${description}" was stopped`;
155
+ break;
156
+ }
157
+ }
158
+ const outputPath = getTaskOutputPath(taskId);
159
+ const toolUseIdLine = toolUseId ? `\n<${TOOL_USE_ID_TAG}>${toolUseId}</${TOOL_USE_ID_TAG}>` : '';
160
+ const message = `<${TASK_NOTIFICATION_TAG}>
161
+ <${TASK_ID_TAG}>${taskId}</${TASK_ID_TAG}>${toolUseIdLine}
162
+ <${OUTPUT_FILE_TAG}>${outputPath}</${OUTPUT_FILE_TAG}>
163
+ <${STATUS_TAG}>${status}</${STATUS_TAG}>
164
+ <${SUMMARY_TAG}>${escapeXml(summary)}</${SUMMARY_TAG}>
165
+ </${TASK_NOTIFICATION_TAG}>`;
166
+ enqueuePendingNotification({
167
+ value: message,
168
+ mode: 'task-notification',
169
+ priority: feature('MONITOR_TOOL') ? 'next' : 'later',
170
+ agentId
171
+ });
172
+ }
173
+ export const LocalShellTask: Task = {
174
+ name: 'LocalShellTask',
175
+ type: 'local_bash',
176
+ async kill(taskId, setAppState) {
177
+ killTask(taskId, setAppState);
178
+ }
179
+ };
180
+ export async function spawnShellTask(input: LocalShellSpawnInput & {
181
+ shellCommand: ShellCommand;
182
+ }, context: TaskContext): Promise<TaskHandle> {
183
+ const {
184
+ command,
185
+ description,
186
+ shellCommand,
187
+ toolUseId,
188
+ agentId,
189
+ kind
190
+ } = input;
191
+ const {
192
+ setAppState
193
+ } = context;
194
+
195
+ // TaskOutput owns the data — use its taskId so disk writes are consistent
196
+ const {
197
+ taskOutput
198
+ } = shellCommand;
199
+ const taskId = taskOutput.taskId;
200
+ const unregisterCleanup = registerCleanup(async () => {
201
+ killTask(taskId, setAppState);
202
+ });
203
+ const taskState: LocalShellTaskState = {
204
+ ...createTaskStateBase(taskId, 'local_bash', description, toolUseId),
205
+ type: 'local_bash',
206
+ status: 'running',
207
+ command,
208
+ completionStatusSentInAttachment: false,
209
+ shellCommand,
210
+ unregisterCleanup,
211
+ lastReportedTotalLines: 0,
212
+ isBackgrounded: true,
213
+ agentId,
214
+ kind
215
+ };
216
+ registerTask(taskState, setAppState);
217
+
218
+ // Data flows through TaskOutput automatically — no stream listeners needed.
219
+ // Just transition to backgrounded state so the process keeps running.
220
+ shellCommand.background(taskId);
221
+ const cancelStallWatchdog = startStallWatchdog(taskId, description, kind, toolUseId, agentId);
222
+ void shellCommand.result.then(async result => {
223
+ cancelStallWatchdog();
224
+ await flushAndCleanup(shellCommand);
225
+ let wasKilled = false;
226
+ updateTaskState<LocalShellTaskState>(taskId, setAppState, task => {
227
+ if (task.status === 'killed') {
228
+ wasKilled = true;
229
+ return task;
230
+ }
231
+ return {
232
+ ...task,
233
+ status: result.code === 0 ? 'completed' : 'failed',
234
+ result: {
235
+ code: result.code,
236
+ interrupted: result.interrupted
237
+ },
238
+ shellCommand: null,
239
+ unregisterCleanup: undefined,
240
+ endTime: Date.now()
241
+ };
242
+ });
243
+ enqueueShellNotification(taskId, description, wasKilled ? 'killed' : result.code === 0 ? 'completed' : 'failed', result.code, setAppState, toolUseId, kind, agentId);
244
+ void evictTaskOutput(taskId);
245
+ });
246
+ return {
247
+ taskId,
248
+ cleanup: () => {
249
+ unregisterCleanup();
250
+ }
251
+ };
252
+ }
253
+
254
+ /**
255
+ * Register a foreground task that could be backgrounded later.
256
+ * Called when a bash command has been running long enough to show the BackgroundHint.
257
+ * @returns taskId for the registered task
258
+ */
259
+ export function registerForeground(input: LocalShellSpawnInput & {
260
+ shellCommand: ShellCommand;
261
+ }, setAppState: SetAppState, toolUseId?: string): string {
262
+ const {
263
+ command,
264
+ description,
265
+ shellCommand,
266
+ agentId
267
+ } = input;
268
+ const taskId = shellCommand.taskOutput.taskId;
269
+ const unregisterCleanup = registerCleanup(async () => {
270
+ killTask(taskId, setAppState);
271
+ });
272
+ const taskState: LocalShellTaskState = {
273
+ ...createTaskStateBase(taskId, 'local_bash', description, toolUseId),
274
+ type: 'local_bash',
275
+ status: 'running',
276
+ command,
277
+ completionStatusSentInAttachment: false,
278
+ shellCommand,
279
+ unregisterCleanup,
280
+ lastReportedTotalLines: 0,
281
+ isBackgrounded: false,
282
+ // Not yet backgrounded - running in foreground
283
+ agentId
284
+ };
285
+ registerTask(taskState, setAppState);
286
+ return taskId;
287
+ }
288
+
289
+ /**
290
+ * Background a specific foreground task.
291
+ * @returns true if backgrounded successfully, false otherwise
292
+ */
293
+ function backgroundTask(taskId: string, getAppState: () => AppState, setAppState: SetAppState): boolean {
294
+ // Step 1: Get the task and shell command from current state
295
+ const state = getAppState();
296
+ const task = state.tasks[taskId];
297
+ if (!isLocalShellTask(task) || task.isBackgrounded || !task.shellCommand) {
298
+ return false;
299
+ }
300
+ const shellCommand = task.shellCommand;
301
+ const description = task.description;
302
+ const {
303
+ toolUseId,
304
+ kind,
305
+ agentId
306
+ } = task;
307
+
308
+ // Transition to backgrounded — TaskOutput continues receiving data automatically
309
+ if (!shellCommand.background(taskId)) {
310
+ return false;
311
+ }
312
+ setAppState(prev => {
313
+ const prevTask = prev.tasks[taskId];
314
+ if (!isLocalShellTask(prevTask) || prevTask.isBackgrounded) {
315
+ return prev;
316
+ }
317
+ return {
318
+ ...prev,
319
+ tasks: {
320
+ ...prev.tasks,
321
+ [taskId]: {
322
+ ...prevTask,
323
+ isBackgrounded: true
324
+ }
325
+ }
326
+ };
327
+ });
328
+ const cancelStallWatchdog = startStallWatchdog(taskId, description, kind, toolUseId, agentId);
329
+
330
+ // Set up result handler
331
+ void shellCommand.result.then(async result => {
332
+ cancelStallWatchdog();
333
+ await flushAndCleanup(shellCommand);
334
+ let wasKilled = false;
335
+ let cleanupFn: (() => void) | undefined;
336
+ updateTaskState<LocalShellTaskState>(taskId, setAppState, t => {
337
+ if (t.status === 'killed') {
338
+ wasKilled = true;
339
+ return t;
340
+ }
341
+
342
+ // Capture cleanup function to call outside of updater
343
+ cleanupFn = t.unregisterCleanup;
344
+ return {
345
+ ...t,
346
+ status: result.code === 0 ? 'completed' : 'failed',
347
+ result: {
348
+ code: result.code,
349
+ interrupted: result.interrupted
350
+ },
351
+ shellCommand: null,
352
+ unregisterCleanup: undefined,
353
+ endTime: Date.now()
354
+ };
355
+ });
356
+
357
+ // Call cleanup outside of the state updater (avoid side effects in updater)
358
+ cleanupFn?.();
359
+ if (wasKilled) {
360
+ enqueueShellNotification(taskId, description, 'killed', result.code, setAppState, toolUseId, kind, agentId);
361
+ } else {
362
+ const finalStatus = result.code === 0 ? 'completed' : 'failed';
363
+ enqueueShellNotification(taskId, description, finalStatus, result.code, setAppState, toolUseId, kind, agentId);
364
+ }
365
+ void evictTaskOutput(taskId);
366
+ });
367
+ return true;
368
+ }
369
+
370
+ /**
371
+ * Background ALL foreground tasks (bash commands and agents).
372
+ * Called when user presses Ctrl+B to background all running tasks.
373
+ */
374
+ /**
375
+ * Check if there are any foreground tasks (bash or agent) that can be backgrounded.
376
+ * Used to determine whether Ctrl+B should background existing tasks vs. background the session.
377
+ */
378
+ export function hasForegroundTasks(state: AppState): boolean {
379
+ return Object.values(state.tasks).some(task => {
380
+ if (isLocalShellTask(task) && !task.isBackgrounded && task.shellCommand) {
381
+ return true;
382
+ }
383
+ // Exclude main session tasks - they display in the main view, not as foreground tasks
384
+ if (isLocalAgentTask(task) && !task.isBackgrounded && !isMainSessionTask(task)) {
385
+ return true;
386
+ }
387
+ return false;
388
+ });
389
+ }
390
+ export function backgroundAll(getAppState: () => AppState, setAppState: SetAppState): void {
391
+ const state = getAppState();
392
+
393
+ // Background all foreground bash tasks
394
+ const foregroundBashTaskIds = Object.keys(state.tasks).filter(id => {
395
+ const task = state.tasks[id];
396
+ return isLocalShellTask(task) && !task.isBackgrounded && task.shellCommand;
397
+ });
398
+ for (const taskId of foregroundBashTaskIds) {
399
+ backgroundTask(taskId, getAppState, setAppState);
400
+ }
401
+
402
+ // Background all foreground agent tasks
403
+ const foregroundAgentTaskIds = Object.keys(state.tasks).filter(id => {
404
+ const task = state.tasks[id];
405
+ return isLocalAgentTask(task) && !task.isBackgrounded;
406
+ });
407
+ for (const taskId of foregroundAgentTaskIds) {
408
+ backgroundAgentTask(taskId, getAppState, setAppState);
409
+ }
410
+ }
411
+
412
+ /**
413
+ * Background an already-registered foreground task in-place.
414
+ * Unlike spawn(), this does NOT re-register the task — it flips isBackgrounded
415
+ * on the existing registration and sets up a completion handler.
416
+ * Used when the auto-background timer fires after registerForeground() has
417
+ * already registered the task (avoiding duplicate task_started SDK events
418
+ * and leaked cleanup callbacks).
419
+ */
420
+ export function backgroundExistingForegroundTask(taskId: string, shellCommand: ShellCommand, description: string, setAppState: SetAppState, toolUseId?: string): boolean {
421
+ if (!shellCommand.background(taskId)) {
422
+ return false;
423
+ }
424
+ let agentId: AgentId | undefined;
425
+ setAppState(prev => {
426
+ const prevTask = prev.tasks[taskId];
427
+ if (!isLocalShellTask(prevTask) || prevTask.isBackgrounded) {
428
+ return prev;
429
+ }
430
+ agentId = prevTask.agentId;
431
+ return {
432
+ ...prev,
433
+ tasks: {
434
+ ...prev.tasks,
435
+ [taskId]: {
436
+ ...prevTask,
437
+ isBackgrounded: true
438
+ }
439
+ }
440
+ };
441
+ });
442
+ const cancelStallWatchdog = startStallWatchdog(taskId, description, undefined, toolUseId, agentId);
443
+
444
+ // Set up result handler (mirrors backgroundTask's handler)
445
+ void shellCommand.result.then(async result => {
446
+ cancelStallWatchdog();
447
+ await flushAndCleanup(shellCommand);
448
+ let wasKilled = false;
449
+ let cleanupFn: (() => void) | undefined;
450
+ updateTaskState<LocalShellTaskState>(taskId, setAppState, t => {
451
+ if (t.status === 'killed') {
452
+ wasKilled = true;
453
+ return t;
454
+ }
455
+ cleanupFn = t.unregisterCleanup;
456
+ return {
457
+ ...t,
458
+ status: result.code === 0 ? 'completed' : 'failed',
459
+ result: {
460
+ code: result.code,
461
+ interrupted: result.interrupted
462
+ },
463
+ shellCommand: null,
464
+ unregisterCleanup: undefined,
465
+ endTime: Date.now()
466
+ };
467
+ });
468
+ cleanupFn?.();
469
+ const finalStatus = wasKilled ? 'killed' : result.code === 0 ? 'completed' : 'failed';
470
+ enqueueShellNotification(taskId, description, finalStatus, result.code, setAppState, toolUseId, undefined, agentId);
471
+ void evictTaskOutput(taskId);
472
+ });
473
+ return true;
474
+ }
475
+
476
+ /**
477
+ * Mark a task as notified to suppress a pending enqueueShellNotification.
478
+ * Used when backgrounding raced with completion — the tool result already
479
+ * carries the full output, so the <task_notification> would be redundant.
480
+ */
481
+ export function markTaskNotified(taskId: string, setAppState: SetAppState): void {
482
+ updateTaskState(taskId, setAppState, t => t.notified ? t : {
483
+ ...t,
484
+ notified: true
485
+ });
486
+ }
487
+
488
+ /**
489
+ * Unregister a foreground task when the command completes without being backgrounded.
490
+ */
491
+ export function unregisterForeground(taskId: string, setAppState: SetAppState): void {
492
+ let cleanupFn: (() => void) | undefined;
493
+ setAppState(prev => {
494
+ const task = prev.tasks[taskId];
495
+ // Only remove if it's a foreground task (not backgrounded)
496
+ if (!isLocalShellTask(task) || task.isBackgrounded) {
497
+ return prev;
498
+ }
499
+
500
+ // Capture cleanup function to call outside of updater
501
+ cleanupFn = task.unregisterCleanup;
502
+ const {
503
+ [taskId]: removed,
504
+ ...rest
505
+ } = prev.tasks;
506
+ return {
507
+ ...prev,
508
+ tasks: rest
509
+ };
510
+ });
511
+
512
+ // Call cleanup outside of the state updater (avoid side effects in updater)
513
+ cleanupFn?.();
514
+ }
515
+ async function flushAndCleanup(shellCommand: ShellCommand): Promise<void> {
516
+ try {
517
+ await shellCommand.taskOutput.flush();
518
+ shellCommand.cleanup();
519
+ } catch (error) {
520
+ logError(error);
521
+ }
522
+ }
523
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJmZWF0dXJlIiwic3RhdCIsIk9VVFBVVF9GSUxFX1RBRyIsIlNUQVRVU19UQUciLCJTVU1NQVJZX1RBRyIsIlRBU0tfSURfVEFHIiwiVEFTS19OT1RJRklDQVRJT05fVEFHIiwiVE9PTF9VU0VfSURfVEFHIiwiYWJvcnRTcGVjdWxhdGlvbiIsIkFwcFN0YXRlIiwiTG9jYWxTaGVsbFNwYXduSW5wdXQiLCJTZXRBcHBTdGF0ZSIsIlRhc2siLCJUYXNrQ29udGV4dCIsIlRhc2tIYW5kbGUiLCJjcmVhdGVUYXNrU3RhdGVCYXNlIiwiQWdlbnRJZCIsInJlZ2lzdGVyQ2xlYW51cCIsInRhaWxGaWxlIiwibG9nRXJyb3IiLCJlbnF1ZXVlUGVuZGluZ05vdGlmaWNhdGlvbiIsIlNoZWxsQ29tbWFuZCIsImV2aWN0VGFza091dHB1dCIsImdldFRhc2tPdXRwdXRQYXRoIiwicmVnaXN0ZXJUYXNrIiwidXBkYXRlVGFza1N0YXRlIiwiZXNjYXBlWG1sIiwiYmFja2dyb3VuZEFnZW50VGFzayIsImlzTG9jYWxBZ2VudFRhc2siLCJpc01haW5TZXNzaW9uVGFzayIsIkJhc2hUYXNrS2luZCIsImlzTG9jYWxTaGVsbFRhc2siLCJMb2NhbFNoZWxsVGFza1N0YXRlIiwia2lsbFRhc2siLCJCQUNLR1JPVU5EX0JBU0hfU1VNTUFSWV9QUkVGSVgiLCJTVEFMTF9DSEVDS19JTlRFUlZBTF9NUyIsIlNUQUxMX1RIUkVTSE9MRF9NUyIsIlNUQUxMX1RBSUxfQllURVMiLCJQUk9NUFRfUEFUVEVSTlMiLCJsb29rc0xpa2VQcm9tcHQiLCJ0YWlsIiwibGFzdExpbmUiLCJ0cmltRW5kIiwic3BsaXQiLCJwb3AiLCJzb21lIiwicCIsInRlc3QiLCJzdGFydFN0YWxsV2F0Y2hkb2ciLCJ0YXNrSWQiLCJkZXNjcmlwdGlvbiIsImtpbmQiLCJ0b29sVXNlSWQiLCJhZ2VudElkIiwib3V0cHV0UGF0aCIsImxhc3RTaXplIiwibGFzdEdyb3d0aCIsIkRhdGUiLCJub3ciLCJjYW5jZWxsZWQiLCJ0aW1lciIsInNldEludGVydmFsIiwidGhlbiIsInMiLCJzaXplIiwiY29udGVudCIsImNsZWFySW50ZXJ2YWwiLCJ0b29sVXNlSWRMaW5lIiwic3VtbWFyeSIsIm1lc3NhZ2UiLCJ2YWx1ZSIsIm1vZGUiLCJwcmlvcml0eSIsInVucmVmIiwiZW5xdWV1ZVNoZWxsTm90aWZpY2F0aW9uIiwic3RhdHVzIiwiZXhpdENvZGUiLCJzZXRBcHBTdGF0ZSIsInNob3VsZEVucXVldWUiLCJ0YXNrIiwibm90aWZpZWQiLCJ1bmRlZmluZWQiLCJMb2NhbFNoZWxsVGFzayIsIm5hbWUiLCJ0eXBlIiwia2lsbCIsInNwYXduU2hlbGxUYXNrIiwiaW5wdXQiLCJzaGVsbENvbW1hbmQiLCJjb250ZXh0IiwiUHJvbWlzZSIsImNvbW1hbmQiLCJ0YXNrT3V0cHV0IiwidW5yZWdpc3RlckNsZWFudXAiLCJ0YXNrU3RhdGUiLCJjb21wbGV0aW9uU3RhdHVzU2VudEluQXR0YWNobWVudCIsImxhc3RSZXBvcnRlZFRvdGFsTGluZXMiLCJpc0JhY2tncm91bmRlZCIsImJhY2tncm91bmQiLCJjYW5jZWxTdGFsbFdhdGNoZG9nIiwicmVzdWx0IiwiZmx1c2hBbmRDbGVhbnVwIiwid2FzS2lsbGVkIiwiY29kZSIsImludGVycnVwdGVkIiwiZW5kVGltZSIsImNsZWFudXAiLCJyZWdpc3RlckZvcmVncm91bmQiLCJiYWNrZ3JvdW5kVGFzayIsImdldEFwcFN0YXRlIiwic3RhdGUiLCJ0YXNrcyIsInByZXYiLCJwcmV2VGFzayIsImNsZWFudXBGbiIsInQiLCJmaW5hbFN0YXR1cyIsImhhc0ZvcmVncm91bmRUYXNrcyIsIk9iamVjdCIsInZhbHVlcyIsImJhY2tncm91bmRBbGwiLCJmb3JlZ3JvdW5kQmFzaFRhc2tJZHMiLCJrZXlzIiwiZmlsdGVyIiwiaWQiLCJmb3JlZ3JvdW5kQWdlbnRUYXNrSWRzIiwiYmFja2dyb3VuZEV4aXN0aW5nRm9yZWdyb3VuZFRhc2siLCJtYXJrVGFza05vdGlmaWVkIiwidW5yZWdpc3RlckZvcmVncm91bmQiLCJyZW1vdmVkIiwicmVzdCIsImZsdXNoIiwiZXJyb3IiXSwic291cmNlcyI6WyJMb2NhbFNoZWxsVGFzay50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgZmVhdHVyZSB9IGZyb20gJ2J1bjpidW5kbGUnXG5pbXBvcnQgeyBzdGF0IH0gZnJvbSAnZnMvcHJvbWlzZXMnXG5pbXBvcnQge1xuICBPVVRQVVRfRklMRV9UQUcsXG4gIFNUQVRVU19UQUcsXG4gIFNVTU1BUllfVEFHLFxuICBUQVNLX0lEX1RBRyxcbiAgVEFTS19OT1RJRklDQVRJT05fVEFHLFxuICBUT09MX1VTRV9JRF9UQUcsXG59IGZyb20gJy4uLy4uL2NvbnN0YW50cy94bWwuanMnXG5pbXBvcnQgeyBhYm9ydFNwZWN1bGF0aW9uIH0gZnJvbSAnLi4vLi4vc2VydmljZXMvUHJvbXB0U3VnZ2VzdGlvbi9zcGVjdWxhdGlvbi5qcydcbmltcG9ydCB0eXBlIHsgQXBwU3RhdGUgfSBmcm9tICcuLi8uLi9zdGF0ZS9BcHBTdGF0ZS5qcydcbmltcG9ydCB0eXBlIHtcbiAgTG9jYWxTaGVsbFNwYXduSW5wdXQsXG4gIFNldEFwcFN0YXRlLFxuICBUYXNrLFxuICBUYXNrQ29udGV4dCxcbiAgVGFza0hhbmRsZSxcbn0gZnJvbSAnLi4vLi4vVGFzay5qcydcbmltcG9ydCB7IGNyZWF0ZVRhc2tTdGF0ZUJhc2UgfSBmcm9tICcuLi8uLi9UYXNrLmpzJ1xuaW1wb3J0IHR5cGUgeyBBZ2VudElkIH0gZnJvbSAnLi4vLi4vdHlwZXMvaWRzLmpzJ1xuaW1wb3J0IHsgcmVnaXN0ZXJDbGVhbnVwIH0gZnJvbSAnLi4vLi4vdXRpbHMvY2xlYW51cFJlZ2lzdHJ5LmpzJ1xuaW1wb3J0IHsgdGFpbEZpbGUgfSBmcm9tICcuLi8uLi91dGlscy9mc09wZXJhdGlvbnMuanMnXG5pbXBvcnQgeyBsb2dFcnJvciB9IGZyb20gJy4uLy4uL3V0aWxzL2xvZy5qcydcbmltcG9ydCB7IGVucXVldWVQZW5kaW5nTm90aWZpY2F0aW9uIH0gZnJvbSAnLi4vLi4vdXRpbHMvbWVzc2FnZVF1ZXVlTWFuYWdlci5qcydcbmltcG9ydCB0eXBlIHsgU2hlbGxDb21tYW5kIH0gZnJvbSAnLi4vLi4vdXRpbHMvU2hlbGxDb21tYW5kLmpzJ1xuaW1wb3J0IHtcbiAgZXZpY3RUYXNrT3V0cHV0LFxuICBnZXRUYXNrT3V0cHV0UGF0aCxcbn0gZnJvbSAnLi4vLi4vdXRpbHMvdGFzay9kaXNrT3V0cHV0LmpzJ1xuaW1wb3J0IHsgcmVnaXN0ZXJUYXNrLCB1cGRhdGVUYXNrU3RhdGUgfSBmcm9tICcuLi8uLi91dGlscy90YXNrL2ZyYW1ld29yay5qcydcbmltcG9ydCB7IGVzY2FwZVhtbCB9IGZyb20gJy4uLy4uL3V0aWxzL3htbC5qcydcbmltcG9ydCB7XG4gIGJhY2tncm91bmRBZ2VudFRhc2ssXG4gIGlzTG9jYWxBZ2VudFRhc2ssXG59IGZyb20gJy4uL0xvY2FsQWdlbnRUYXNrL0xvY2FsQWdlbnRUYXNrLmpzJ1xuaW1wb3J0IHsgaXNNYWluU2Vzc2lvblRhc2sgfSBmcm9tICcuLi9Mb2NhbE1haW5TZXNzaW9uVGFzay5qcydcbmltcG9ydCB7XG4gIHR5cGUgQmFzaFRhc2tLaW5kLFxuICBpc0xvY2FsU2hlbGxUYXNrLFxuICB0eXBlIExvY2FsU2hlbGxUYXNrU3RhdGUsXG59IGZyb20gJy4vZ3VhcmRzLmpzJ1xuaW1wb3J0IHsga2lsbFRhc2sgfSBmcm9tICcuL2tpbGxTaGVsbFRhc2tzLmpzJ1xuXG4vKiogUHJlZml4IHRoYXQgaWRlbnRpZmllcyBhIExvY2FsU2hlbGxUYXNrIHN1bW1hcnkgdG8gdGhlIFVJIGNvbGxhcHNlIHRyYW5zZm9ybS4gKi9cbmV4cG9ydCBjb25zdCBCQUNLR1JPVU5EX0JBU0hfU1VNTUFSWV9QUkVGSVggPSAnQmFja2dyb3VuZCBjb21tYW5kICdcblxuY29uc3QgU1RBTExfQ0hFQ0tfSU5URVJWQUxfTVMgPSA1XzAwMFxuY29uc3QgU1RBTExfVEhSRVNIT0xEX01TID0gNDVfMDAwXG5jb25zdCBTVEFMTF9UQUlMX0JZVEVTID0gMTAyNFxuXG4vLyBMYXN0LWxpbmUgcGF0dGVybnMgdGhhdCBzdWdnZXN0IGEgY29tbWFuZCBpcyBibG9ja2VkIHdhaXRpbmcgZm9yIGtleWJvYXJkXG4vLyBpbnB1dC4gVXNlZCB0byBnYXRlIHRoZSBzdGFsbCBub3RpZmljYXRpb24g4oCUIHdlIHN0YXkgc2lsZW50IG9uIGNvbW1hbmRzIHRoYXRcbi8vIGFyZSBtZXJlbHkgc2xvdyAoZ2l0IGxvZyAtUywgbG9uZyBidWlsZHMpIGFuZCBvbmx5IG5vdGlmeSB3aGVuIHRoZSB0YWlsXG4vLyBsb29rcyBsaWtlIGFuIGludGVyYWN0aXZlIHByb21wdCB0aGUgbW9kZWwgY2FuIGFjdCBvbi4gU2VlIENDLTExNzUuXG5jb25zdCBQUk9NUFRfUEFUVEVSTlMgPSBbXG4gIC9cXCh5XFwvblxcKS9pLCAvLyAoWS9uKSwgKHkvTilcbiAgL1xcW3lcXC9uXFxdL2ksIC8vIFtZL25dLCBbeS9OXVxuICAvXFwoeWVzXFwvbm9cXCkvaSxcbiAgL1xcYig/OkRvIHlvdXxXb3VsZCB5b3V8U2hhbGwgSXxBcmUgeW91IHN1cmV8UmVhZHkgdG8pXFxiLipcXD8gKiQvaSwgLy8gZGlyZWN0ZWQgcXVlc3Rpb25zXG4gIC9QcmVzcyAoYW55IGtleXxFbnRlcikvaSxcbiAgL0NvbnRpbnVlXFw/L2ksXG4gIC9PdmVyd3JpdGVcXD8vaSxcbl1cblxuZXhwb3J0IGZ1bmN0aW9uIGxvb2tzTGlrZVByb21wdCh0YWlsOiBzdHJpbmcpOiBib29sZWFuIHtcbiAgY29uc3QgbGFzdExpbmUgPSB0YWlsLnRyaW1FbmQoKS5zcGxpdCgnXFxuJykucG9wKCkgPz8gJydcbiAgcmV0dXJuIFBST01QVF9QQVRURVJOUy5zb21lKHAgPT4gcC50ZXN0KGxhc3RMaW5lKSlcbn1cblxuLy8gT3V0cHV0LXNpZGUgYW5hbG9nIG9mIHBlZWtGb3JTdGRpbkRhdGEgKHV0aWxzL3Byb2Nlc3MudHMpOiBmaXJlIGEgb25lLXNob3Rcbi8vIG5vdGlmaWNhdGlvbiBpZiBvdXRwdXQgc3RvcHMgZ3Jvd2luZyBhbmQgdGhlIHRhaWwgbG9va3MgbGlrZSBhIHByb21wdC5cbmZ1bmN0aW9uIHN0YXJ0U3RhbGxXYXRjaGRvZyhcbiAgdGFza0lkOiBzdHJpbmcsXG4gIGRlc2NyaXB0aW9uOiBzdHJpbmcsXG4gIGtpbmQ6IEJhc2hUYXNrS2luZCB8IHVuZGVmaW5lZCxcbiAgdG9vbFVzZUlkPzogc3RyaW5nLFxuICBhZ2VudElkPzogQWdlbnRJZCxcbik6ICgpID0+IHZvaWQge1xuICBpZiAoa2luZCA9PT0gJ21vbml0b3InKSByZXR1cm4gKCkgPT4ge31cbiAgY29uc3Qgb3V0cHV0UGF0aCA9IGdldFRhc2tPdXRwdXRQYXRoKHRhc2tJZClcbiAgbGV0IGxhc3RTaXplID0gMFxuICBsZXQgbGFzdEdyb3d0aCA9IERhdGUubm93KClcbiAgbGV0IGNhbmNlbGxlZCA9IGZhbHNlXG5cbiAgY29uc3QgdGltZXIgPSBzZXRJbnRlcnZhbCgoKSA9PiB7XG4gICAgdm9pZCBzdGF0KG91dHB1dFBhdGgpLnRoZW4oXG4gICAgICBzID0+IHtcbiAgICAgICAgaWYgKHMuc2l6ZSA+IGxhc3RTaXplKSB7XG4gICAgICAgICAgbGFzdFNpemUgPSBzLnNpemVcbiAgICAgICAgICBsYXN0R3Jvd3RoID0gRGF0ZS5ub3coKVxuICAgICAgICAgIHJldHVyblxuICAgICAgICB9XG4gICAgICAgIGlmIChEYXRlLm5vdygpIC0gbGFzdEdyb3d0aCA8IFNUQUxMX1RIUkVTSE9MRF9NUykgcmV0dXJuXG4gICAgICAgIHZvaWQgdGFpbEZpbGUob3V0cHV0UGF0aCwgU1RBTExfVEFJTF9CWVRFUykudGhlbihcbiAgICAgICAgICAoeyBjb250ZW50IH0pID0+IHtcbiAgICAgICAgICAgIGlmIChjYW5jZWxsZWQpIHJldHVyblxuICAgICAgICAgICAgaWYgKCFsb29rc0xpa2VQcm9tcHQoY29udGVudCkpIHtcbiAgICAgICAgICAgICAgLy8gTm90IGEgcHJvbXB0IOKAlCBrZWVwIHdhdGNoaW5nLiBSZXNldCBzbyB0aGUgbmV4dCBjaGVjayBpc1xuICAgICAgICAgICAgICAvLyA0NXMgb3V0IGluc3RlYWQgb2YgcmUtcmVhZGluZyB0aGUgdGFpbCBvbiBldmVyeSB0aWNrLlxuICAgICAgICAgICAgICBsYXN0R3Jvd3RoID0gRGF0ZS5ub3coKVxuICAgICAgICAgICAgICByZXR1cm5cbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIC8vIExhdGNoIGJlZm9yZSB0aGUgYXN5bmMtYm91bmRhcnktdmlzaWJsZSBzaWRlIGVmZmVjdHMgc28gYW5cbiAgICAgICAgICAgIC8vIG92ZXJsYXBwaW5nIHRpY2sncyBjYWxsYmFjayBzZWVzIGNhbmNlbGxlZD10cnVlIGFuZCBiYWlscy5cbiAgICAgICAgICAgIGNhbmNlbGxlZCA9IHRydWVcbiAgICAgICAgICAgIGNsZWFySW50ZXJ2YWwodGltZXIpXG4gICAgICAgICAgICBjb25zdCB0b29sVXNlSWRMaW5lID0gdG9vbFVzZUlkXG4gICAgICAgICAgICAgID8gYFxcbjwke1RPT0xfVVNFX0lEX1RBR30+JHt0b29sVXNlSWR9PC8ke1RPT0xfVVNFX0lEX1RBR30+YFxuICAgICAgICAgICAgICA6ICcnXG4gICAgICAgICAgICBjb25zdCBzdW1tYXJ5ID0gYCR7QkFDS0dST1VORF9CQVNIX1NVTU1BUllfUFJFRklYfVwiJHtkZXNjcmlwdGlvbn1cIiBhcHBlYXJzIHRvIGJlIHdhaXRpbmcgZm9yIGludGVyYWN0aXZlIGlucHV0YFxuICAgICAgICAgICAgLy8gTm8gPHN0YXR1cz4gdGFnIOKAlCBwcmludC50cyB0cmVhdHMgPHN0YXR1cz4gYXMgYSB0ZXJtaW5hbFxuICAgICAgICAgICAgLy8gc2lnbmFsIGFuZCBhbiB1bmtub3duIHZhbHVlIGZhbGxzIHRocm91Z2ggdG8gJ2NvbXBsZXRlZCcsXG4gICAgICAgICAgICAvLyBmYWxzZWx5IGNsb3NpbmcgdGhlIHRhc2sgZm9yIFNESyBjb25zdW1lcnMuIFN0YXR1c2xlc3NcbiAgICAgICAgICAgIC8vIG5vdGlmaWNhdGlvbnMgYXJlIHNraXBwZWQgYnkgdGhlIFNESyBlbWl0dGVyIChwcm9ncmVzcyBwaW5nKS5cbiAgICAgICAgICAgIGNvbnN0IG1lc3NhZ2UgPSBgPCR7VEFTS19OT1RJRklDQVRJT05fVEFHfT5cbjwke1RBU0tfSURfVEFHfT4ke3Rhc2tJZH08LyR7VEFTS19JRF9UQUd9PiR7dG9vbFVzZUlkTGluZX1cbjwke09VVFBVVF9GSUxFX1RBR30+JHtvdXRwdXRQYXRofTwvJHtPVVRQVVRfRklMRV9UQUd9PlxuPCR7U1VNTUFSWV9UQUd9PiR7ZXNjYXBlWG1sKHN1bW1hcnkpfTwvJHtTVU1NQVJZX1RBR30+XG48LyR7VEFTS19OT1RJRklDQVRJT05fVEFHfT5cbkxhc3Qgb3V0cHV0OlxuJHtjb250ZW50LnRyaW1FbmQoKX1cblxuVGhlIGNvbW1hbmQgaXMgbGlrZWx5IGJsb2NrZWQgb24gYW4gaW50ZXJhY3RpdmUgcHJvbXB0LiBLaWxsIHRoaXMgdGFzayBhbmQgcmUtcnVuIHdpdGggcGlwZWQgaW5wdXQgKGUuZy4sIFxcYGVjaG8geSB8IGNvbW1hbmRcXGApIG9yIGEgbm9uLWludGVyYWN0aXZlIGZsYWcgaWYgb25lIGV4aXN0cy5gXG4gICAgICAgICAgICBlbnF1ZXVlUGVuZGluZ05vdGlmaWNhdGlvbih7XG4gICAgICAgICAgICAgIHZhbHVlOiBtZXNzYWdlLFxuICAgICAgICAgICAgICBtb2RlOiAndGFzay1ub3RpZmljYXRpb24nLFxuICAgICAgICAgICAgICBwcmlvcml0eTogJ25leHQnLFxuICAgICAgICAgICAgICBhZ2VudElkLFxuICAgICAgICAgICAgfSlcbiAgICAgICAgICB9LFxuICAgICAgICAgICgpID0+IHt9LFxuICAgICAgICApXG4gICAgICB9LFxuICAgICAgKCkgPT4ge30sIC8vIEZpbGUgbWF5IG5vdCBleGlzdCB5ZXRcbiAgICApXG4gIH0sIFNUQUxMX0NIRUNLX0lOVEVSVkFMX01TKVxuICB0aW1lci51bnJlZigpXG5cbiAgcmV0dXJuICgpID0+IHtcbiAgICBjYW5jZWxsZWQgPSB0cnVlXG4gICAgY2xlYXJJbnRlcnZhbCh0aW1lcilcbiAgfVxufVxuXG5mdW5jdGlvbiBlbnF1ZXVlU2hlbGxOb3RpZmljYXRpb24oXG4gIHRhc2tJZDogc3RyaW5nLFxuICBkZXNjcmlwdGlvbjogc3RyaW5nLFxuICBzdGF0dXM6ICdjb21wbGV0ZWQnIHwgJ2ZhaWxlZCcgfCAna2lsbGVkJyxcbiAgZXhpdENvZGU6IG51bWJlciB8IHVuZGVmaW5lZCxcbiAgc2V0QXBwU3RhdGU6IFNldEFwcFN0YXRlLFxuICB0b29sVXNlSWQ/OiBzdHJpbmcsXG4gIGtpbmQ6IEJhc2hUYXNrS2luZCA9ICdiYXNoJyxcbiAgYWdlbnRJZD86IEFnZW50SWQsXG4pOiB2b2lkIHtcbiAgLy8gQXRvbWljYWxseSBjaGVjayBhbmQgc2V0IG5vdGlmaWVkIGZsYWcgdG8gcHJldmVudCBkdXBsaWNhdGUgbm90aWZpY2F0aW9ucy5cbiAgLy8gSWYgdGhlIHRhc2sgd2FzIGFscmVhZHkgbWFya2VkIGFzIG5vdGlmaWVkIChlLmcuLCBieSBUYXNrU3RvcFRvb2wpLCBza2lwXG4gIC8vIGVucXVldWVpbmcgdG8gYXZvaWQgc2VuZGluZyByZWR1bmRhbnQgbWVzc2FnZXMgdG8gdGhlIG1vZGVsLlxuICBsZXQgc2hvdWxkRW5xdWV1ZSA9IGZhbHNlXG4gIHVwZGF0ZVRhc2tTdGF0ZSh0YXNrSWQsIHNldEFwcFN0YXRlLCB0YXNrID0+IHtcbiAgICBpZiAodGFzay5ub3RpZmllZCkge1xuICAgICAgcmV0dXJuIHRhc2tcbiAgICB9XG4gICAgc2hvdWxkRW5xdWV1ZSA9IHRydWVcbiAgICByZXR1cm4geyAuLi50YXNrLCBub3RpZmllZDogdHJ1ZSB9XG4gIH0pXG5cbiAgaWYgKCFzaG91bGRFbnF1ZXVlKSB7XG4gICAgcmV0dXJuXG4gIH1cblxuICAvLyBBYm9ydCBhbnkgYWN0aXZlIHNwZWN1bGF0aW9uIOKAlCBiYWNrZ3JvdW5kIHRhc2sgc3RhdGUgY2hhbmdlZCwgc28gc3BlY3VsYXRlZFxuICAvLyByZXN1bHRzIG1heSByZWZlcmVuY2Ugc3RhbGUgdGFzayBvdXRwdXQuIFRoZSBwcm9tcHQgc3VnZ2VzdGlvbiB0ZXh0IGlzXG4gIC8vIHByZXNlcnZlZDsgb25seSB0aGUgcHJlLWNvbXB1dGVkIHJlc3BvbnNlIGlzIGRpc2NhcmRlZC5cbiAgYWJvcnRTcGVjdWxhdGlvbihzZXRBcHBTdGF0ZSlcblxuICBsZXQgc3VtbWFyeTogc3RyaW5nXG4gIGlmIChmZWF0dXJlKCdNT05JVE9SX1RPT0wnKSAmJiBraW5kID09PSAnbW9uaXRvcicpIHtcbiAgICAvLyBNb25pdG9yIGlzIHN0cmVhbWluZy1vbmx5IChwb3N0LSMyMjc2NCkg4oCUIHRoZSBzY3JpcHQgZXhpdGluZyBtZWFuc1xuICAgIC8vIHRoZSBzdHJlYW0gZW5kZWQsIG5vdCBcImNvbmRpdGlvbiBtZXRcIi4gRGlzdGluY3QgZnJvbSB0aGUgYmFzaCBwcmVmaXhcbiAgICAvLyBzbyBNb25pdG9yIGNvbXBsZXRpb25zIGRvbid0IGZvbGQgaW50byB0aGUgXCJOIGJhY2tncm91bmQgY29tbWFuZHNcbiAgICAvLyBjb21wbGV0ZWRcIiBjb2xsYXBzZS5cbiAgICBzd2l0Y2ggKHN0YXR1cykge1xuICAgICAgY2FzZSAnY29tcGxldGVkJzpcbiAgICAgICAgc3VtbWFyeSA9IGBNb25pdG9yIFwiJHtkZXNjcmlwdGlvbn1cIiBzdHJlYW0gZW5kZWRgXG4gICAgICAgIGJyZWFrXG4gICAgICBjYXNlICdmYWlsZWQnOlxuICAgICAgICBzdW1tYXJ5ID0gYE1vbml0b3IgXCIke2Rlc2NyaXB0aW9ufVwiIHNjcmlwdCBmYWlsZWQke2V4aXRDb2RlICE9PSB1bmRlZmluZWQgPyBgIChleGl0ICR7ZXhpdENvZGV9KWAgOiAnJ31gXG4gICAgICAgIGJyZWFrXG4gICAgICBjYXNlICdraWxsZWQnOlxuICAgICAgICBzdW1tYXJ5ID0gYE1vbml0b3IgXCIke2Rlc2NyaXB0aW9ufVwiIHN0b3BwZWRgXG4gICAgICAgIGJyZWFrXG4gICAgfVxuICB9IGVsc2Uge1xuICAgIHN3aXRjaCAoc3RhdHVzKSB7XG4gICAgICBjYXNlICdjb21wbGV0ZWQnOlxuICAgICAgICBzdW1tYXJ5ID0gYCR7QkFDS0dST1VORF9CQVNIX1NVTU1BUllfUFJFRklYfVwiJHtkZXNjcmlwdGlvbn1cIiBjb21wbGV0ZWQke2V4aXRDb2RlICE9PSB1bmRlZmluZWQgPyBgIChleGl0IGNvZGUgJHtleGl0Q29kZX0pYCA6ICcnfWBcbiAgICAgICAgYnJlYWtcbiAgICAgIGNhc2UgJ2ZhaWxlZCc6XG4gICAgICAgIHN1bW1hcnkgPSBgJHtCQUNLR1JPVU5EX0JBU0hfU1VNTUFSWV9QUkVGSVh9XCIke2Rlc2NyaXB0aW9ufVwiIGZhaWxlZCR7ZXhpdENvZGUgIT09IHVuZGVmaW5lZCA/IGAgd2l0aCBleGl0IGNvZGUgJHtleGl0Q29kZX1gIDogJyd9YFxuICAgICAgICBicmVha1xuICAgICAgY2FzZSAna2lsbGVkJzpcbiAgICAgICAgc3VtbWFyeSA9IGAke0JBQ0tHUk9VTkRfQkFTSF9TVU1NQVJZX1BSRUZJWH1cIiR7ZGVzY3JpcHRpb259XCIgd2FzIHN0b3BwZWRgXG4gICAgICAgIGJyZWFrXG4gICAgfVxuICB9XG5cbiAgY29uc3Qgb3V0cHV0UGF0aCA9IGdldFRhc2tPdXRwdXRQYXRoKHRhc2tJZClcbiAgY29uc3QgdG9vbFVzZUlkTGluZSA9IHRvb2xVc2VJZFxuICAgID8gYFxcbjwke1RPT0xfVVNFX0lEX1RBR30+JHt0b29sVXNlSWR9PC8ke1RPT0xfVVNFX0lEX1RBR30+YFxuICAgIDogJydcbiAgY29uc3QgbWVzc2FnZSA9IGA8JHtUQVNLX05PVElGSUNBVElPTl9UQUd9PlxuPCR7VEFTS19JRF9UQUd9PiR7dGFza0lkfTwvJHtUQVNLX0lEX1RBR30+JHt0b29sVXNlSWRMaW5lfVxuPCR7T1VUUFVUX0ZJTEVfVEFHfT4ke291dHB1dFBhdGh9PC8ke09VVFBVVF9GSUxFX1RBR30+XG48JHtTVEFUVVNfVEFHfT4ke3N0YXR1c308LyR7U1RBVFVTX1RBR30+XG48JHtTVU1NQVJZX1RBR30+JHtlc2NhcGVYbWwoc3VtbWFyeSl9PC8ke1NVTU1BUllfVEFHfT5cbjwvJHtUQVNLX05PVElGSUNBVElPTl9UQUd9PmBcblxuICBlbnF1ZXVlUGVuZGluZ05vdGlmaWNhdGlvbih7XG4gICAgdmFsdWU6IG1lc3NhZ2UsXG4gICAgbW9kZTogJ3Rhc2stbm90aWZpY2F0aW9uJyxcbiAgICBwcmlvcml0eTogZmVhdHVyZSgnTU9OSVRPUl9UT09MJykgPyAnbmV4dCcgOiAnbGF0ZXInLFxuICAgIGFnZW50SWQsXG4gIH0pXG59XG5cbmV4cG9ydCBjb25zdCBMb2NhbFNoZWxsVGFzazogVGFzayA9IHtcbiAgbmFtZTogJ0xvY2FsU2hlbGxUYXNrJyxcbiAgdHlwZTogJ2xvY2FsX2Jhc2gnLFxuICBhc3luYyBraWxsKHRhc2tJZCwgc2V0QXBwU3RhdGUpIHtcbiAgICBraWxsVGFzayh0YXNrSWQsIHNldEFwcFN0YXRlKVxuICB9LFxufVxuXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gc3Bhd25TaGVsbFRhc2soXG4gIGlucHV0OiBMb2NhbFNoZWxsU3Bhd25JbnB1dCAmIHsgc2hlbGxDb21tYW5kOiBTaGVsbENvbW1hbmQgfSxcbiAgY29udGV4dDogVGFza0NvbnRleHQsXG4pOiBQcm9taXNlPFRhc2tIYW5kbGU+IHtcbiAgY29uc3QgeyBjb21tYW5kLCBkZXNjcmlwdGlvbiwgc2hlbGxDb21tYW5kLCB0b29sVXNlSWQsIGFnZW50SWQsIGtpbmQgfSA9IGlucHV0XG4gIGNvbnN0IHsgc2V0QXBwU3RhdGUgfSA9IGNvbnRleHRcblxuICAvLyBUYXNrT3V0cHV0IG93bnMgdGhlIGRhdGEg4oCUIHVzZSBpdHMgdGFza0lkIHNvIGRpc2sgd3JpdGVzIGFyZSBjb25zaXN0ZW50XG4gIGNvbnN0IHsgdGFza091dHB1dCB9ID0gc2hlbGxDb21tYW5kXG4gIGNvbnN0IHRhc2tJZCA9IHRhc2tPdXRwdXQudGFza0lkXG5cbiAgY29uc3QgdW5yZWdpc3RlckNsZWFudXAgPSByZWdpc3RlckNsZWFudXAoYXN5bmMgKCkgPT4ge1xuICAgIGtpbGxUYXNrKHRhc2tJZCwgc2V0QXBwU3RhdGUpXG4gIH0pXG5cbiAgY29uc3QgdGFza1N0YXRlOiBMb2NhbFNoZWxsVGFza1N0YXRlID0ge1xuICAgIC4uLmNyZWF0ZVRhc2tTdGF0ZUJhc2UodGFza0lkLCAnbG9jYWxfYmFzaCcsIGRlc2NyaXB0aW9uLCB0b29sVXNlSWQpLFxuICAgIHR5cGU6ICdsb2NhbF9iYXNoJyxcbiAgICBzdGF0dXM6ICdydW5uaW5nJyxcbiAgICBjb21tYW5kLFxuICAgIGNvbXBsZXRpb25TdGF0dXNTZW50SW5BdHRhY2htZW50OiBmYWxzZSxcbiAgICBzaGVsbENvbW1hbmQsXG4gICAgdW5yZWdpc3RlckNsZWFudXAsXG4gICAgbGFzdFJlcG9ydGVkVG90YWxMaW5lczogMCxcbiAgICBpc0JhY2tncm91bmRlZDogdHJ1ZSxcbiAgICBhZ2VudElkLFxuICAgIGtpbmQsXG4gIH1cblxuICByZWdpc3RlclRhc2sodGFza1N0YXRlLCBzZXRBcHBTdGF0ZSlcblxuICAvLyBEYXRhIGZsb3dzIHRocm91Z2ggVGFza091dHB1dCBhdXRvbWF0aWNhbGx5IOKAlCBubyBzdHJlYW0gbGlzdGVuZXJzIG5lZWRlZC5cbiAgLy8gSnVzdCB0cmFuc2l0aW9uIHRvIGJhY2tncm91bmRlZCBzdGF0ZSBzbyB0aGUgcHJvY2VzcyBrZWVwcyBydW5uaW5nLlxuICBzaGVsbENvbW1hbmQuYmFja2dyb3VuZCh0YXNrSWQpXG5cbiAgY29uc3QgY2FuY2VsU3RhbGxXYXRjaGRvZyA9IHN0YXJ0U3RhbGxXYXRjaGRvZyhcbiAgICB0YXNrSWQsXG4gICAgZGVzY3JpcHRpb24sXG4gICAga2luZCxcbiAgICB0b29sVXNlSWQsXG4gICAgYWdlbnRJZCxcbiAgKVxuXG4gIHZvaWQgc2hlbGxDb21tYW5kLnJlc3VsdC50aGVuKGFzeW5jIHJlc3VsdCA9PiB7XG4gICAgY2FuY2VsU3RhbGxXYXRjaGRvZygpXG4gICAgYXdhaXQgZmx1c2hBbmRDbGVhbnVwKHNoZWxsQ29tbWFuZClcbiAgICBsZXQgd2FzS2lsbGVkID0gZmFsc2VcblxuICAgIHVwZGF0ZVRhc2tTdGF0ZTxMb2NhbFNoZWxsVGFza1N0YXRlPih0YXNrSWQsIHNldEFwcFN0YXRlLCB0YXNrID0+IHtcbiAgICAgIGlmICh0YXNrLnN0YXR1cyA9PT0gJ2tpbGxlZCcpIHtcbiAgICAgICAgd2FzS2lsbGVkID0gdHJ1ZVxuICAgICAgICByZXR1cm4gdGFza1xuICAgICAgfVxuXG4gICAgICByZXR1cm4ge1xuICAgICAgICAuLi50YXNrLFxuICAgICAgICBzdGF0dXM6IHJlc3VsdC5jb2RlID09PSAwID8gJ2NvbXBsZXRlZCcgOiAnZmFpbGVkJyxcbiAgICAgICAgcmVzdWx0OiB7IGNvZGU6IHJlc3VsdC5jb2RlLCBpbnRlcnJ1cHRlZDogcmVzdWx0LmludGVycnVwdGVkIH0sXG4gICAgICAgIHNoZWxsQ29tbWFuZDogbnVsbCxcbiAgICAgICAgdW5yZWdpc3RlckNsZWFudXA6IHVuZGVmaW5lZCxcbiAgICAgICAgZW5kVGltZTogRGF0ZS5ub3coKSxcbiAgICAgIH1cbiAgICB9KVxuXG4gICAgZW5xdWV1ZVNoZWxsTm90aWZpY2F0aW9uKFxuICAgICAgdGFza0lkLFxuICAgICAgZGVzY3JpcHRpb24sXG4gICAgICB3YXNLaWxsZWQgPyAna2lsbGVkJyA6IHJlc3VsdC5jb2RlID09PSAwID8gJ2NvbXBsZXRlZCcgOiAnZmFpbGVkJyxcbiAgICAgIHJlc3VsdC5jb2RlLFxuICAgICAgc2V0QXBwU3RhdGUsXG4gICAgICB0b29sVXNlSWQsXG4gICAgICBraW5kLFxuICAgICAgYWdlbnRJZCxcbiAgICApXG5cbiAgICB2b2lkIGV2aWN0VGFza091dHB1dCh0YXNrSWQpXG4gIH0pXG5cbiAgcmV0dXJuIHtcbiAgICB0YXNrSWQsXG4gICAgY2xlYW51cDogKCkgPT4ge1xuICAgICAgdW5yZWdpc3RlckNsZWFudXAoKVxuICAgIH0sXG4gIH1cbn1cblxuLyoqXG4gKiBSZWdpc3RlciBhIGZvcmVncm91bmQgdGFzayB0aGF0IGNvdWxkIGJlIGJhY2tncm91bmRlZCBsYXRlci5cbiAqIENhbGxlZCB3aGVuIGEgYmFzaCBjb21tYW5kIGhhcyBiZWVuIHJ1bm5pbmcgbG9uZyBlbm91Z2ggdG8gc2hvdyB0aGUgQmFja2dyb3VuZEhpbnQuXG4gKiBAcmV0dXJucyB0YXNrSWQgZm9yIHRoZSByZWdpc3RlcmVkIHRhc2tcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIHJlZ2lzdGVyRm9yZWdyb3VuZChcbiAgaW5wdXQ6IExvY2FsU2hlbGxTcGF3bklucHV0ICYgeyBzaGVsbENvbW1hbmQ6IFNoZWxsQ29tbWFuZCB9LFxuICBzZXRBcHBTdGF0ZTogU2V0QXBwU3RhdGUsXG4gIHRvb2xVc2VJZD86IHN0cmluZyxcbik6IHN0cmluZyB7XG4gIGNvbnN0IHsgY29tbWFuZCwgZGVzY3JpcHRpb24sIHNoZWxsQ29tbWFuZCwgYWdlbnRJZCB9ID0gaW5wdXRcblxuICBjb25zdCB0YXNrSWQgPSBzaGVsbENvbW1hbmQudGFza091dHB1dC50YXNrSWRcblxuICBjb25zdCB1bnJlZ2lzdGVyQ2xlYW51cCA9IHJlZ2lzdGVyQ2xlYW51cChhc3luYyAoKSA9PiB7XG4gICAga2lsbFRhc2sodGFza0lkLCBzZXRBcHBTdGF0ZSlcbiAgfSlcblxuICBjb25zdCB0YXNrU3RhdGU6IExvY2FsU2hlbGxUYXNrU3RhdGUgPSB7XG4gICAgLi4uY3JlYXRlVGFza1N0YXRlQmFzZSh0YXNrSWQsICdsb2NhbF9iYXNoJywgZGVzY3JpcHRpb24sIHRvb2xVc2VJZCksXG4gICAgdHlwZTogJ2xvY2FsX2Jhc2gnLFxuICAgIHN0YXR1czogJ3J1bm5pbmcnLFxuICAgIGNvbW1hbmQsXG4gICAgY29tcGxldGlvblN0YXR1c1NlbnRJbkF0dGFjaG1lbnQ6IGZhbHNlLFxuICAgIHNoZWxsQ29tbWFuZCxcbiAgICB1bnJlZ2lzdGVyQ2xlYW51cCxcbiAgICBsYXN0UmVwb3J0ZWRUb3RhbExpbmVzOiAwLFxuICAgIGlzQmFja2dyb3VuZGVkOiBmYWxzZSwgLy8gTm90IHlldCBiYWNrZ3JvdW5kZWQgLSBydW5uaW5nIGluIGZvcmVncm91bmRcbiAgICBhZ2VudElkLFxuICB9XG5cbiAgcmVnaXN0ZXJUYXNrKHRhc2tTdGF0ZSwgc2V0QXBwU3RhdGUpXG4gIHJldHVybiB0YXNrSWRcbn1cblxuLyoqXG4gKiBCYWNrZ3JvdW5kIGEgc3BlY2lmaWMgZm9yZWdyb3VuZCB0YXNrLlxuICogQHJldHVybnMgdHJ1ZSBpZiBiYWNrZ3JvdW5kZWQgc3VjY2Vzc2Z1bGx5LCBmYWxzZSBvdGhlcndpc2VcbiAqL1xuZnVuY3Rpb24gYmFja2dyb3VuZFRhc2soXG4gIHRhc2tJZDogc3RyaW5nLFxuICBnZXRBcHBTdGF0ZTogKCkgPT4gQXBwU3RhdGUsXG4gIHNldEFwcFN0YXRlOiBTZXRBcHBTdGF0ZSxcbik6IGJvb2xlYW4ge1xuICAvLyBTdGVwIDE6IEdldCB0aGUgdGFzayBhbmQgc2hlbGwgY29tbWFuZCBmcm9tIGN1cnJlbnQgc3RhdGVcbiAgY29uc3Qgc3RhdGUgPSBnZXRBcHBTdGF0ZSgpXG4gIGNvbnN0IHRhc2sgPSBzdGF0ZS50YXNrc1t0YXNrSWRdXG4gIGlmICghaXNMb2NhbFNoZWxsVGFzayh0YXNrKSB8fCB0YXNrLmlzQmFja2dyb3VuZGVkIHx8ICF0YXNrLnNoZWxsQ29tbWFuZCkge1xuICAgIHJldHVybiBmYWxzZVxuICB9XG5cbiAgY29uc3Qgc2hlbGxDb21tYW5kID0gdGFzay5zaGVsbENvbW1hbmRcbiAgY29uc3QgZGVzY3JpcHRpb24gPSB0YXNrLmRlc2NyaXB0aW9uXG4gIGNvbnN0IHsgdG9vbFVzZUlkLCBraW5kLCBhZ2VudElkIH0gPSB0YXNrXG5cbiAgLy8gVHJhbnNpdGlvbiB0byBiYWNrZ3JvdW5kZWQg4oCUIFRhc2tPdXRwdXQgY29udGludWVzIHJlY2VpdmluZyBkYXRhIGF1dG9tYXRpY2FsbHlcbiAgaWYgKCFzaGVsbENvbW1hbmQuYmFja2dyb3VuZCh0YXNrSWQpKSB7XG4gICAgcmV0dXJuIGZhbHNlXG4gIH1cblxuICBzZXRBcHBTdGF0ZShwcmV2ID0+IHtcbiAgICBjb25zdCBwcmV2VGFzayA9IHByZXYudGFza3NbdGFza0lkXVxuICAgIGlmICghaXNMb2NhbFNoZWxsVGFzayhwcmV2VGFzaykgfHwgcHJldlRhc2suaXNCYWNrZ3JvdW5kZWQpIHtcbiAgICAgIHJldHVybiBwcmV2XG4gICAgfVxuICAgIHJldHVybiB7XG4gICAgICAuLi5wcmV2LFxuICAgICAgdGFza3M6IHtcbiAgICAgICAgLi4ucHJldi50YXNrcyxcbiAgICAgICAgW3Rhc2tJZF06IHsgLi4ucHJldlRhc2ssIGlzQmFja2dyb3VuZGVkOiB0cnVlIH0sXG4gICAgICB9LFxuICAgIH1cbiAgfSlcblxuICBjb25zdCBjYW5jZWxTdGFsbFdhdGNoZG9nID0gc3RhcnRTdGFsbFdhdGNoZG9nKFxuICAgIHRhc2tJZCxcbiAgICBkZXNjcmlwdGlvbixcbiAgICBraW5kLFxuICAgIHRvb2xVc2VJZCxcbiAgICBhZ2VudElkLFxuICApXG5cbiAgLy8gU2V0IHVwIHJlc3VsdCBoYW5kbGVyXG4gIHZvaWQgc2hlbGxDb21tYW5kLnJlc3VsdC50aGVuKGFzeW5jIHJlc3VsdCA9PiB7XG4gICAgY2FuY2VsU3RhbGxXYXRjaGRvZygpXG4gICAgYXdhaXQgZmx1c2hBbmRDbGVhbnVwKHNoZWxsQ29tbWFuZClcbiAgICBsZXQgd2FzS2lsbGVkID0gZmFsc2VcbiAgICBsZXQgY2xlYW51cEZuOiAoKCkgPT4gdm9pZCkgfCB1bmRlZmluZWRcblxuICAgIHVwZGF0ZVRhc2tTdGF0ZTxMb2NhbFNoZWxsVGFza1N0YXRlPih0YXNrSWQsIHNldEFwcFN0YXRlLCB0ID0+IHtcbiAgICAgIGlmICh0LnN0YXR1cyA9PT0gJ2tpbGxlZCcpIHtcbiAgICAgICAgd2FzS2lsbGVkID0gdHJ1ZVxuICAgICAgICByZXR1cm4gdFxuICAgICAgfVxuXG4gICAgICAvLyBDYXB0dXJlIGNsZWFudXAgZnVuY3Rpb24gdG8gY2FsbCBvdXRzaWRlIG9mIHVwZGF0ZXJcbiAgICAgIGNsZWFudXBGbiA9IHQudW5yZWdpc3RlckNsZWFudXBcblxuICAgICAgcmV0dXJuIHtcbiAgICAgICAgLi4udCxcbiAgICAgICAgc3RhdHVzOiByZXN1bHQuY29kZSA9PT0gMCA/ICdjb21wbGV0ZWQnIDogJ2ZhaWxlZCcsXG4gICAgICAgIHJlc3VsdDogeyBjb2RlOiByZXN1bHQuY29kZSwgaW50ZXJydXB0ZWQ6IHJlc3VsdC5pbnRlcnJ1cHRlZCB9LFxuICAgICAgICBzaGVsbENvbW1hbmQ6IG51bGwsXG4gICAgICAgIHVucmVnaXN0ZXJDbGVhbnVwOiB1bmRlZmluZWQsXG4gICAgICAgIGVuZFRpbWU6IERhdGUubm93KCksXG4gICAgICB9XG4gICAgfSlcblxuICAgIC8vIENhbGwgY2xlYW51cCBvdXRzaWRlIG9mIHRoZSBzdGF0ZSB1cGRhdGVyIChhdm9pZCBzaWRlIGVmZmVjdHMgaW4gdXBkYXRlcilcbiAgICBjbGVhbnVwRm4/LigpXG5cbiAgICBpZiAod2FzS2lsbGVkKSB7XG4gICAgICBlbnF1ZXVlU2hlbGxOb3RpZmljYXRpb24oXG4gICAgICAgIHRhc2tJZCxcbiAgICAgICAgZGVzY3JpcHRpb24sXG4gICAgICAgICdraWxsZWQnLFxuICAgICAgICByZXN1bHQuY29kZSxcbiAgICAgICAgc2V0QXBwU3RhdGUsXG4gICAgICAgIHRvb2xVc2VJZCxcbiAgICAgICAga2luZCxcbiAgICAgICAgYWdlbnRJZCxcbiAgICAgIClcbiAgICB9IGVsc2Uge1xuICAgICAgY29uc3QgZmluYWxTdGF0dXMgPSByZXN1bHQuY29kZSA9PT0gMCA/ICdjb21wbGV0ZWQnIDogJ2ZhaWxlZCdcbiAgICAgIGVucXVldWVTaGVsbE5vdGlmaWNhdGlvbihcbiAgICAgICAgdGFza0lkLFxuICAgICAgICBkZXNjcmlwdGlvbixcbiAgICAgICAgZmluYWxTdGF0dXMsXG4gICAgICAgIHJlc3VsdC5jb2RlLFxuICAgICAgICBzZXRBcHBTdGF0ZSxcbiAgICAgICAgdG9vbFVzZUlkLFxuICAgICAgICBraW5kLFxuICAgICAgICBhZ2VudElkLFxuICAgICAgKVxuICAgIH1cblxuICAgIHZvaWQgZXZpY3RUYXNrT3V0cHV0KHRhc2tJZClcbiAgfSlcblxuICByZXR1cm4gdHJ1ZVxufVxuXG4vKipcbiAqIEJhY2tncm91bmQgQUxMIGZvcmVncm91bmQgdGFza3MgKGJhc2ggY29tbWFuZHMgYW5kIGFnZW50cykuXG4gKiBDYWxsZWQgd2hlbiB1c2VyIHByZXNzZXMgQ3RybCtCIHRvIGJhY2tncm91bmQgYWxsIHJ1bm5pbmcgdGFza3MuXG4gKi9cbi8qKlxuICogQ2hlY2sgaWYgdGhlcmUgYXJlIGFueSBmb3JlZ3JvdW5kIHRhc2tzIChiYXNoIG9yIGFnZW50KSB0aGF0IGNhbiBiZSBiYWNrZ3JvdW5kZWQuXG4gKiBVc2VkIHRvIGRldGVybWluZSB3aGV0aGVyIEN0cmwrQiBzaG91bGQgYmFja2dyb3VuZCBleGlzdGluZyB0YXNrcyB2cy4gYmFja2dyb3VuZCB0aGUgc2Vzc2lvbi5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGhhc0ZvcmVncm91bmRUYXNrcyhzdGF0ZTogQXBwU3RhdGUpOiBib29sZWFuIHtcbiAgcmV0dXJuIE9iamVjdC52YWx1ZXMoc3RhdGUudGFza3MpLnNvbWUodGFzayA9PiB7XG4gICAgaWYgKGlzTG9jYWxTaGVsbFRhc2sodGFzaykgJiYgIXRhc2suaXNCYWNrZ3JvdW5kZWQgJiYgdGFzay5zaGVsbENvbW1hbmQpIHtcbiAgICAgIHJldHVybiB0cnVlXG4gICAgfVxuICAgIC8vIEV4Y2x1ZGUgbWFpbiBzZXNzaW9uIHRhc2tzIC0gdGhleSBkaXNwbGF5IGluIHRoZSBtYWluIHZpZXcsIG5vdCBhcyBmb3JlZ3JvdW5kIHRhc2tzXG4gICAgaWYgKFxuICAgICAgaXNMb2NhbEFnZW50VGFzayh0YXNrKSAmJlxuICAgICAgIXRhc2suaXNCYWNrZ3JvdW5kZWQgJiZcbiAgICAgICFpc01haW5TZXNzaW9uVGFzayh0YXNrKVxuICAgICkge1xuICAgICAgcmV0dXJuIHRydWVcbiAgICB9XG4gICAgcmV0dXJuIGZhbHNlXG4gIH0pXG59XG5cbmV4cG9ydCBmdW5jdGlvbiBiYWNrZ3JvdW5kQWxsKFxuICBnZXRBcHBTdGF0ZTogKCkgPT4gQXBwU3RhdGUsXG4gIHNldEFwcFN0YXRlOiBTZXRBcHBTdGF0ZSxcbik6IHZvaWQge1xuICBjb25zdCBzdGF0ZSA9IGdldEFwcFN0YXRlKClcblxuICAvLyBCYWNrZ3JvdW5kIGFsbCBmb3JlZ3JvdW5kIGJhc2ggdGFza3NcbiAgY29uc3QgZm9yZWdyb3VuZEJhc2hUYXNrSWRzID0gT2JqZWN0LmtleXMoc3RhdGUudGFza3MpLmZpbHRlcihpZCA9PiB7XG4gICAgY29uc3QgdGFzayA9IHN0YXRlLnRhc2tzW2lkXVxuICAgIHJldHVybiBpc0xvY2FsU2hlbGxUYXNrKHRhc2spICYmICF0YXNrLmlzQmFja2dyb3VuZGVkICYmIHRhc2suc2hlbGxDb21tYW5kXG4gIH0pXG4gIGZvciAoY29uc3QgdGFza0lkIG9mIGZvcmVncm91bmRCYXNoVGFza0lkcykge1xuICAgIGJhY2tncm91bmRUYXNrKHRhc2tJZCwgZ2V0QXBwU3RhdGUsIHNldEFwcFN0YXRlKVxuICB9XG5cbiAgLy8gQmFja2dyb3VuZCBhbGwgZm9yZWdyb3VuZCBhZ2VudCB0YXNrc1xuICBjb25zdCBmb3JlZ3JvdW5kQWdlbnRUYXNrSWRzID0gT2JqZWN0LmtleXMoc3RhdGUudGFza3MpLmZpbHRlcihpZCA9PiB7XG4gICAgY29uc3QgdGFzayA9IHN0YXRlLnRhc2tzW2lkXVxuICAgIHJldHVybiBpc0xvY2FsQWdlbnRUYXNrKHRhc2spICYmICF0YXNrLmlzQmFja2dyb3VuZGVkXG4gIH0pXG4gIGZvciAoY29uc3QgdGFza0lkIG9mIGZvcmVncm91bmRBZ2VudFRhc2tJZHMpIHtcbiAgICBiYWNrZ3JvdW5kQWdlbnRUYXNrKHRhc2tJZCwgZ2V0QXBwU3RhdGUsIHNldEFwcFN0YXRlKVxuICB9XG59XG5cbi8qKlxuICogQmFja2dyb3VuZCBhbiBhbHJlYWR5LXJlZ2lzdGVyZWQgZm9yZWdyb3VuZCB0YXNrIGluLXBsYWNlLlxuICogVW5saWtlIHNwYXduKCksIHRoaXMgZG9lcyBOT1QgcmUtcmVnaXN0ZXIgdGhlIHRhc2sg4oCUIGl0IGZsaXBzIGlzQmFja2dyb3VuZGVkXG4gKiBvbiB0aGUgZXhpc3RpbmcgcmVnaXN0cmF0aW9uIGFuZCBzZXRzIHVwIGEgY29tcGxldGlvbiBoYW5kbGVyLlxuICogVXNlZCB3aGVuIHRoZSBhdXRvLWJhY2tncm91bmQgdGltZXIgZmlyZXMgYWZ0ZXIgcmVnaXN0ZXJGb3JlZ3JvdW5kKCkgaGFzXG4gKiBhbHJlYWR5IHJlZ2lzdGVyZWQgdGhlIHRhc2sgKGF2b2lkaW5nIGR1cGxpY2F0ZSB0YXNrX3N0YXJ0ZWQgU0RLIGV2ZW50c1xuICogYW5kIGxlYWtlZCBjbGVhbnVwIGNhbGxiYWNrcykuXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBiYWNrZ3JvdW5kRXhpc3RpbmdGb3JlZ3JvdW5kVGFzayhcbiAgdGFza0lkOiBzdHJpbmcsXG4gIHNoZWxsQ29tbWFuZDogU2hlbGxDb21tYW5kLFxuICBkZXNjcmlwdGlvbjogc3RyaW5nLFxuICBzZXRBcHBTdGF0ZTogU2V0QXBwU3RhdGUsXG4gIHRvb2xVc2VJZD86IHN0cmluZyxcbik6IGJvb2xlYW4ge1xuICBpZiAoIXNoZWxsQ29tbWFuZC5iYWNrZ3JvdW5kKHRhc2tJZCkpIHtcbiAgICByZXR1cm4gZmFsc2VcbiAgfVxuXG4gIGxldCBhZ2VudElkOiBBZ2VudElkIHwgdW5kZWZpbmVkXG4gIHNldEFwcFN0YXRlKHByZXYgPT4ge1xuICAgIGNvbnN0IHByZXZUYXNrID0gcHJldi50YXNrc1t0YXNrSWRdXG4gICAgaWYgKCFpc0xvY2FsU2hlbGxUYXNrKHByZXZUYXNrKSB8fCBwcmV2VGFzay5pc0JhY2tncm91bmRlZCkge1xuICAgICAgcmV0dXJuIHByZXZcbiAgICB9XG4gICAgYWdlbnRJZCA9IHByZXZUYXNrLmFnZW50SWRcbiAgICByZXR1cm4ge1xuICAgICAgLi4ucHJldixcbiAgICAgIHRhc2tzOiB7XG4gICAgICAgIC4uLnByZXYudGFza3MsXG4gICAgICAgIFt0YXNrSWRdOiB7IC4uLnByZXZUYXNrLCBpc0JhY2tncm91bmRlZDogdHJ1ZSB9LFxuICAgICAgfSxcbiAgICB9XG4gIH0pXG5cbiAgY29uc3QgY2FuY2VsU3RhbGxXYXRjaGRvZyA9IHN0YXJ0U3RhbGxXYXRjaGRvZyhcbiAgICB0YXNrSWQsXG4gICAgZGVzY3JpcHRpb24sXG4gICAgdW5kZWZpbmVkLFxuICAgIHRvb2xVc2VJZCxcbiAgICBhZ2VudElkLFxuICApXG5cbiAgLy8gU2V0IHVwIHJlc3VsdCBoYW5kbGVyIChtaXJyb3JzIGJhY2tncm91bmRUYXNrJ3MgaGFuZGxlcilcbiAgdm9pZCBzaGVsbENvbW1hbmQucmVzdWx0LnRoZW4oYXN5bmMgcmVzdWx0ID0+IHtcbiAgICBjYW5jZWxTdGFsbFdhdGNoZG9nKClcbiAgICBhd2FpdCBmbHVzaEFuZENsZWFudXAoc2hlbGxDb21tYW5kKVxuICAgIGxldCB3YXNLaWxsZWQgPSBmYWxzZVxuICAgIGxldCBjbGVhbnVwRm46ICgoKSA9PiB2b2lkKSB8IHVuZGVmaW5lZFxuXG4gICAgdXBkYXRlVGFza1N0YXRlPExvY2FsU2hlbGxUYXNrU3RhdGU+KHRhc2tJZCwgc2V0QXBwU3RhdGUsIHQgPT4ge1xuICAgICAgaWYgKHQuc3RhdHVzID09PSAna2lsbGVkJykge1xuICAgICAgICB3YXNLaWxsZWQgPSB0cnVlXG4gICAgICAgIHJldHVybiB0XG4gICAgICB9XG4gICAgICBjbGVhbnVwRm4gPSB0LnVucmVnaXN0ZXJDbGVhbnVwXG4gICAgICByZXR1cm4ge1xuICAgICAgICAuLi50LFxuICAgICAgICBzdGF0dXM6IHJlc3VsdC5jb2RlID09PSAwID8gJ2NvbXBsZXRlZCcgOiAnZmFpbGVkJyxcbiAgICAgICAgcmVzdWx0OiB7IGNvZGU6IHJlc3VsdC5jb2RlLCBpbnRlcnJ1cHRlZDogcmVzdWx0LmludGVycnVwdGVkIH0sXG4gICAgICAgIHNoZWxsQ29tbWFuZDogbnVsbCxcbiAgICAgICAgdW5yZWdpc3RlckNsZWFudXA6IHVuZGVmaW5lZCxcbiAgICAgICAgZW5kVGltZTogRGF0ZS5ub3coKSxcbiAgICAgIH1cbiAgICB9KVxuXG4gICAgY2xlYW51cEZuPy4oKVxuXG4gICAgY29uc3QgZmluYWxTdGF0dXMgPSB3YXNLaWxsZWRcbiAgICAgID8gJ2tpbGxlZCdcbiAgICAgIDogcmVzdWx0LmNvZGUgPT09IDBcbiAgICAgICAgPyAnY29tcGxldGVkJ1xuICAgICAgICA6ICdmYWlsZWQnXG4gICAgZW5xdWV1ZVNoZWxsTm90aWZpY2F0aW9uKFxuICAgICAgdGFza0lkLFxuICAgICAgZGVzY3JpcHRpb24sXG4gICAgICBmaW5hbFN0YXR1cyxcbiAgICAgIHJlc3VsdC5jb2RlLFxuICAgICAgc2V0QXBwU3RhdGUsXG4gICAgICB0b29sVXNlSWQsXG4gICAgICB1bmRlZmluZWQsXG4gICAgICBhZ2VudElkLFxuICAgIClcblxuICAgIHZvaWQgZXZpY3RUYXNrT3V0cHV0KHRhc2tJZClcbiAgfSlcblxuICByZXR1cm4gdHJ1ZVxufVxuXG4vKipcbiAqIE1hcmsgYSB0YXNrIGFzIG5vdGlmaWVkIHRvIHN1cHByZXNzIGEgcGVuZGluZyBlbnF1ZXVlU2hlbGxOb3RpZmljYXRpb24uXG4gKiBVc2VkIHdoZW4gYmFja2dyb3VuZGluZyByYWNlZCB3aXRoIGNvbXBsZXRpb24g4oCUIHRoZSB0b29sIHJlc3VsdCBhbHJlYWR5XG4gKiBjYXJyaWVzIHRoZSBmdWxsIG91dHB1dCwgc28gdGhlIDx0YXNrX25vdGlmaWNhdGlvbj4gd291bGQgYmUgcmVkdW5kYW50LlxuICovXG5leHBvcnQgZnVuY3Rpb24gbWFya1Rhc2tOb3RpZmllZChcbiAgdGFza0lkOiBzdHJpbmcsXG4gIHNldEFwcFN0YXRlOiBTZXRBcHBTdGF0ZSxcbik6IHZvaWQge1xuICB1cGRhdGVUYXNrU3RhdGUodGFza0lkLCBzZXRBcHBTdGF0ZSwgdCA9PlxuICAgIHQubm90aWZpZWQgPyB0IDogeyAuLi50LCBub3RpZmllZDogdHJ1ZSB9LFxuICApXG59XG5cbi8qKlxuICogVW5yZWdpc3RlciBhIGZvcmVncm91bmQgdGFzayB3aGVuIHRoZSBjb21tYW5kIGNvbXBsZXRlcyB3aXRob3V0IGJlaW5nIGJhY2tncm91bmRlZC5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIHVucmVnaXN0ZXJGb3JlZ3JvdW5kKFxuICB0YXNrSWQ6IHN0cmluZyxcbiAgc2V0QXBwU3RhdGU6IFNldEFwcFN0YXRlLFxuKTogdm9pZCB7XG4gIGxldCBjbGVhbnVwRm46ICgoKSA9PiB2b2lkKSB8IHVuZGVmaW5lZFxuXG4gIHNldEFwcFN0YXRlKHByZXYgPT4ge1xuICAgIGNvbnN0IHRhc2sgPSBwcmV2LnRhc2tzW3Rhc2tJZF1cbiAgICAvLyBPbmx5IHJlbW92ZSBpZiBpdCdzIGEgZm9yZWdyb3VuZCB0YXNrIChub3QgYmFja2dyb3VuZGVkKVxuICAgIGlmICghaXNMb2NhbFNoZWxsVGFzayh0YXNrKSB8fCB0YXNrLmlzQmFja2dyb3VuZGVkKSB7XG4gICAgICByZXR1cm4gcHJldlxuICAgIH1cblxuICAgIC8vIENhcHR1cmUgY2xlYW51cCBmdW5jdGlvbiB0byBjYWxsIG91dHNpZGUgb2YgdXBkYXRlclxuICAgIGNsZWFudXBGbiA9IHRhc2sudW5yZWdpc3RlckNsZWFudXBcblxuICAgIGNvbnN0IHsgW3Rhc2tJZF06IHJlbW92ZWQsIC4uLnJlc3QgfSA9IHByZXYudGFza3NcbiAgICByZXR1cm4geyAuLi5wcmV2LCB0YXNrczogcmVzdCB9XG4gIH0pXG5cbiAgLy8gQ2FsbCBjbGVhbnVwIG91dHNpZGUgb2YgdGhlIHN0YXRlIHVwZGF0ZXIgKGF2b2lkIHNpZGUgZWZmZWN0cyBpbiB1cGRhdGVyKVxuICBjbGVhbnVwRm4/LigpXG59XG5cbmFzeW5jIGZ1bmN0aW9uIGZsdXNoQW5kQ2xlYW51cChzaGVsbENvbW1hbmQ6IFNoZWxsQ29tbWFuZCk6IFByb21pc2U8dm9pZD4ge1xuICB0cnkge1xuICAgIGF3YWl0IHNoZWxsQ29tbWFuZC50YXNrT3V0cHV0LmZsdXNoKClcbiAgICBzaGVsbENvbW1hbmQuY2xlYW51cCgpXG4gIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgbG9nRXJyb3IoZXJyb3IpXG4gIH1cbn1cbiJdLCJtYXBwaW5ncyI6IkFBQUEsU0FBU0EsT0FBTyxRQUFRLFlBQVk7QUFDcEMsU0FBU0Ms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
@@ -0,0 +1,41 @@
1
+ // Pure type + type guard for LocalShellTask state.
2
+ // Extracted from LocalShellTask.tsx so non-React consumers (stopTask.ts via
3
+ // print.ts) don't pull React/ink into the module graph.
4
+
5
+ import type { TaskStateBase } from '../../Task.js'
6
+ import type { AgentId } from '../../types/ids.js'
7
+ import type { ShellCommand } from '../../utils/ShellCommand.js'
8
+
9
+ export type BashTaskKind = 'bash' | 'monitor'
10
+
11
+ export type LocalShellTaskState = TaskStateBase & {
12
+ type: 'local_bash' // Keep as 'local_bash' for backward compatibility with persisted session state
13
+ command: string
14
+ result?: {
15
+ code: number
16
+ interrupted: boolean
17
+ }
18
+ completionStatusSentInAttachment: boolean
19
+ shellCommand: ShellCommand | null
20
+ unregisterCleanup?: () => void
21
+ cleanupTimeoutId?: NodeJS.Timeout
22
+ // Track what we last reported for computing deltas (total lines from TaskOutput)
23
+ lastReportedTotalLines: number
24
+ // Whether the task has been backgrounded (false = foreground running, true = backgrounded)
25
+ isBackgrounded: boolean
26
+ // Agent that spawned this task. Used to kill orphaned bash tasks when the
27
+ // agent exits (see killShellTasksForAgent). Undefined = main thread.
28
+ agentId?: AgentId
29
+ // UI display variant. 'monitor' → shows description instead of command,
30
+ // 'Monitor details' dialog title, distinct status bar pill.
31
+ kind?: BashTaskKind
32
+ }
33
+
34
+ export function isLocalShellTask(task: unknown): task is LocalShellTaskState {
35
+ return (
36
+ typeof task === 'object' &&
37
+ task !== null &&
38
+ 'type' in task &&
39
+ task.type === 'local_bash'
40
+ )
41
+ }