@xortex/xcode 3.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +21 -0
- package/README.md +171 -0
- package/bin/xcode +127 -0
- package/bin/xcode-test +84 -0
- package/bin/xcode.cmd +31 -0
- package/constants/apiLimits.ts +94 -0
- package/constants/betas.ts +52 -0
- package/constants/common.ts +33 -0
- package/constants/cyberRiskInstruction.ts +24 -0
- package/constants/errorIds.ts +15 -0
- package/constants/figures.ts +45 -0
- package/constants/files.ts +156 -0
- package/constants/github-app.ts +144 -0
- package/constants/keys.ts +11 -0
- package/constants/messages.ts +1 -0
- package/constants/oauth.ts +234 -0
- package/constants/outputStyles.ts +216 -0
- package/constants/product.ts +76 -0
- package/constants/prompts.ts +939 -0
- package/constants/spinnerVerbs.ts +204 -0
- package/constants/system.ts +95 -0
- package/constants/systemPromptSections.ts +68 -0
- package/constants/toolLimits.ts +56 -0
- package/constants/tools.ts +112 -0
- package/constants/turnCompletionVerbs.ts +12 -0
- package/constants/xml.ts +86 -0
- package/entrypoints/agentSdkTypes.ts +443 -0
- package/entrypoints/cli.tsx +307 -0
- package/entrypoints/init.ts +340 -0
- package/entrypoints/mcp.ts +196 -0
- package/entrypoints/sandboxTypes.ts +156 -0
- package/entrypoints/sdk/controlSchemas.ts +663 -0
- package/entrypoints/sdk/coreSchemas.ts +1889 -0
- package/entrypoints/sdk/coreTypes.generated.ts +2 -0
- package/entrypoints/sdk/coreTypes.ts +62 -0
- package/entrypoints/sdk/runtimeTypes.ts +140 -0
- package/entrypoints/sdk/sdkUtilityTypes.ts +3 -0
- package/entrypoints/sdk/toolTypes.ts +90 -0
- package/main.tsx +4686 -0
- package/package.json +120 -0
- package/services/AgentSummary/agentSummary.ts +179 -0
- package/services/MagicDocs/magicDocs.ts +254 -0
- package/services/MagicDocs/prompts.ts +127 -0
- package/services/PromptSuggestion/promptSuggestion.ts +523 -0
- package/services/PromptSuggestion/speculation.ts +991 -0
- package/services/SessionMemory/prompts.ts +324 -0
- package/services/SessionMemory/sessionMemory.ts +495 -0
- package/services/SessionMemory/sessionMemoryUtils.ts +207 -0
- package/services/analytics/config.ts +38 -0
- package/services/analytics/datadog.ts +307 -0
- package/services/analytics/firstPartyEventLogger.ts +449 -0
- package/services/analytics/firstPartyEventLoggingExporter.ts +806 -0
- package/services/analytics/growthbook.ts +1155 -0
- package/services/analytics/index.ts +173 -0
- package/services/analytics/metadata.ts +973 -0
- package/services/analytics/sink.ts +114 -0
- package/services/analytics/sinkKillswitch.ts +25 -0
- package/services/api/adminRequests.ts +119 -0
- package/services/api/bootstrap.ts +141 -0
- package/services/api/claude.ts +3422 -0
- package/services/api/client.ts +406 -0
- package/services/api/dumpPrompts.ts +226 -0
- package/services/api/emptyUsage.ts +22 -0
- package/services/api/errorUtils.ts +260 -0
- package/services/api/errors.ts +1207 -0
- package/services/api/filesApi.ts +748 -0
- package/services/api/firstTokenDate.ts +60 -0
- package/services/api/gemini.ts +359 -0
- package/services/api/geminiAdapter.ts +123 -0
- package/services/api/geminiClient.ts +291 -0
- package/services/api/grove.ts +357 -0
- package/services/api/logging.ts +788 -0
- package/services/api/metricsOptOut.ts +159 -0
- package/services/api/openRouterClient.ts +453 -0
- package/services/api/overageCreditGrant.ts +137 -0
- package/services/api/promptCacheBreakDetection.ts +727 -0
- package/services/api/referral.ts +281 -0
- package/services/api/sessionIngress.ts +514 -0
- package/services/api/ultrareviewQuota.ts +38 -0
- package/services/api/usage.ts +63 -0
- package/services/api/withRetry.ts +822 -0
- package/services/autoDream/autoDream.ts +324 -0
- package/services/autoDream/config.ts +21 -0
- package/services/autoDream/consolidationLock.ts +140 -0
- package/services/autoDream/consolidationPrompt.ts +65 -0
- package/services/awaySummary.ts +74 -0
- package/services/claudeAiLimits.ts +515 -0
- package/services/claudeAiLimitsHook.ts +23 -0
- package/services/compact/apiMicrocompact.ts +153 -0
- package/services/compact/autoCompact.ts +351 -0
- package/services/compact/compact.ts +1705 -0
- package/services/compact/compactWarningHook.ts +16 -0
- package/services/compact/compactWarningState.ts +18 -0
- package/services/compact/grouping.ts +63 -0
- package/services/compact/microCompact.ts +530 -0
- package/services/compact/postCompactCleanup.ts +77 -0
- package/services/compact/prompt.ts +374 -0
- package/services/compact/sessionMemoryCompact.ts +630 -0
- package/services/compact/timeBasedMCConfig.ts +43 -0
- package/services/diagnosticTracking.ts +397 -0
- package/services/extractMemories/extractMemories.ts +517 -0
- package/services/extractMemories/prompts.ts +154 -0
- package/services/internalLogging.ts +90 -0
- package/services/lsp/LSPClient.ts +447 -0
- package/services/lsp/LSPDiagnosticRegistry.ts +386 -0
- package/services/lsp/LSPServerInstance.ts +511 -0
- package/services/lsp/LSPServerManager.ts +420 -0
- package/services/lsp/config.ts +79 -0
- package/services/lsp/manager.ts +289 -0
- package/services/lsp/passiveFeedback.ts +328 -0
- package/services/mcp/InProcessTransport.ts +63 -0
- package/services/mcp/MCPConnectionManager.tsx +73 -0
- package/services/mcp/SdkControlTransport.ts +136 -0
- package/services/mcp/auth.ts +2465 -0
- package/services/mcp/channelAllowlist.ts +76 -0
- package/services/mcp/channelNotification.ts +316 -0
- package/services/mcp/channelPermissions.ts +240 -0
- package/services/mcp/claudeai.ts +164 -0
- package/services/mcp/client.ts +3348 -0
- package/services/mcp/config.ts +1578 -0
- package/services/mcp/elicitationHandler.ts +313 -0
- package/services/mcp/envExpansion.ts +38 -0
- package/services/mcp/headersHelper.ts +138 -0
- package/services/mcp/mcpStringUtils.ts +106 -0
- package/services/mcp/normalization.ts +23 -0
- package/services/mcp/oauthPort.ts +78 -0
- package/services/mcp/officialRegistry.ts +72 -0
- package/services/mcp/types.ts +258 -0
- package/services/mcp/useManageMCPConnections.ts +1141 -0
- package/services/mcp/utils.ts +575 -0
- package/services/mcp/vscodeSdkMcp.ts +112 -0
- package/services/mcp/xaa.ts +511 -0
- package/services/mcp/xaaIdpLogin.ts +487 -0
- package/services/mcpServerApproval.tsx +41 -0
- package/services/mockRateLimits.ts +882 -0
- package/services/notifier.ts +156 -0
- package/services/oauth/auth-code-listener.ts +211 -0
- package/services/oauth/client.ts +566 -0
- package/services/oauth/crypto.ts +23 -0
- package/services/oauth/getOauthProfile.ts +53 -0
- package/services/oauth/index.ts +198 -0
- package/services/plugins/PluginInstallationManager.ts +184 -0
- package/services/plugins/pluginCliCommands.ts +344 -0
- package/services/plugins/pluginOperations.ts +1088 -0
- package/services/policyLimits/index.ts +663 -0
- package/services/policyLimits/types.ts +27 -0
- package/services/preventSleep.ts +165 -0
- package/services/rateLimitMessages.ts +344 -0
- package/services/rateLimitMocking.ts +144 -0
- package/services/remoteManagedSettings/index.ts +638 -0
- package/services/remoteManagedSettings/securityCheck.tsx +74 -0
- package/services/remoteManagedSettings/syncCache.ts +112 -0
- package/services/remoteManagedSettings/syncCacheState.ts +96 -0
- package/services/remoteManagedSettings/types.ts +31 -0
- package/services/settingsSync/index.ts +581 -0
- package/services/settingsSync/types.ts +67 -0
- package/services/teamMemorySync/index.ts +1256 -0
- package/services/teamMemorySync/secretScanner.ts +324 -0
- package/services/teamMemorySync/teamMemSecretGuard.ts +44 -0
- package/services/teamMemorySync/types.ts +156 -0
- package/services/teamMemorySync/watcher.ts +387 -0
- package/services/tips/tipHistory.ts +17 -0
- package/services/tips/tipRegistry.ts +686 -0
- package/services/tips/tipScheduler.ts +58 -0
- package/services/tokenEstimation.ts +495 -0
- package/services/toolUseSummary/toolUseSummaryGenerator.ts +112 -0
- package/services/tools/StreamingToolExecutor.ts +530 -0
- package/services/tools/toolExecution.ts +1745 -0
- package/services/tools/toolHooks.ts +650 -0
- package/services/tools/toolOrchestration.ts +188 -0
- package/services/vcr.ts +406 -0
- package/services/voice.ts +525 -0
- package/services/voiceKeyterms.ts +106 -0
- package/services/voiceStreamSTT.ts +544 -0
- package/tools/AgentTool/AgentTool.tsx +1398 -0
- package/tools/AgentTool/UI.tsx +872 -0
- package/tools/AgentTool/agentColorManager.ts +66 -0
- package/tools/AgentTool/agentDisplay.ts +104 -0
- package/tools/AgentTool/agentMemory.ts +177 -0
- package/tools/AgentTool/agentMemorySnapshot.ts +197 -0
- package/tools/AgentTool/agentToolUtils.ts +686 -0
- package/tools/AgentTool/built-in/claudeCodeGuideAgent.ts +205 -0
- package/tools/AgentTool/built-in/exploreAgent.ts +83 -0
- package/tools/AgentTool/built-in/generalPurposeAgent.ts +34 -0
- package/tools/AgentTool/built-in/planAgent.ts +92 -0
- package/tools/AgentTool/built-in/statuslineSetup.ts +144 -0
- package/tools/AgentTool/built-in/verificationAgent.ts +152 -0
- package/tools/AgentTool/builtInAgents.ts +72 -0
- package/tools/AgentTool/constants.ts +12 -0
- package/tools/AgentTool/forkSubagent.ts +210 -0
- package/tools/AgentTool/loadAgentsDir.ts +755 -0
- package/tools/AgentTool/prompt.ts +287 -0
- package/tools/AgentTool/resumeAgent.ts +265 -0
- package/tools/AgentTool/runAgent.ts +973 -0
- package/tools/AskUserQuestionTool/AskUserQuestionTool.tsx +266 -0
- package/tools/AskUserQuestionTool/prompt.ts +44 -0
- package/tools/BashTool/BashTool.tsx +1144 -0
- package/tools/BashTool/BashToolResultMessage.tsx +191 -0
- package/tools/BashTool/UI.tsx +185 -0
- package/tools/BashTool/bashCommandHelpers.ts +265 -0
- package/tools/BashTool/bashPermissions.ts +2621 -0
- package/tools/BashTool/bashSecurity.ts +2592 -0
- package/tools/BashTool/commandSemantics.ts +140 -0
- package/tools/BashTool/commentLabel.ts +13 -0
- package/tools/BashTool/destructiveCommandWarning.ts +102 -0
- package/tools/BashTool/modeValidation.ts +115 -0
- package/tools/BashTool/pathValidation.ts +1303 -0
- package/tools/BashTool/prompt.ts +369 -0
- package/tools/BashTool/readOnlyValidation.ts +1990 -0
- package/tools/BashTool/sedEditParser.ts +322 -0
- package/tools/BashTool/sedValidation.ts +684 -0
- package/tools/BashTool/shouldUseSandbox.ts +153 -0
- package/tools/BashTool/toolName.ts +2 -0
- package/tools/BashTool/utils.ts +223 -0
- package/tools/BriefTool/BriefTool.ts +204 -0
- package/tools/BriefTool/UI.tsx +101 -0
- package/tools/BriefTool/attachments.ts +110 -0
- package/tools/BriefTool/prompt.ts +22 -0
- package/tools/BriefTool/upload.ts +174 -0
- package/tools/ConfigTool/ConfigTool.ts +467 -0
- package/tools/ConfigTool/UI.tsx +38 -0
- package/tools/ConfigTool/constants.ts +1 -0
- package/tools/ConfigTool/prompt.ts +93 -0
- package/tools/ConfigTool/supportedSettings.ts +211 -0
- package/tools/EnterPlanModeTool/EnterPlanModeTool.ts +126 -0
- package/tools/EnterPlanModeTool/UI.tsx +33 -0
- package/tools/EnterPlanModeTool/constants.ts +1 -0
- package/tools/EnterPlanModeTool/prompt.ts +170 -0
- package/tools/EnterWorktreeTool/EnterWorktreeTool.ts +127 -0
- package/tools/EnterWorktreeTool/UI.tsx +20 -0
- package/tools/EnterWorktreeTool/constants.ts +1 -0
- package/tools/EnterWorktreeTool/prompt.ts +30 -0
- package/tools/ExitPlanModeTool/ExitPlanModeV2Tool.ts +493 -0
- package/tools/ExitPlanModeTool/UI.tsx +82 -0
- package/tools/ExitPlanModeTool/constants.ts +2 -0
- package/tools/ExitPlanModeTool/prompt.ts +29 -0
- package/tools/ExitWorktreeTool/ExitWorktreeTool.ts +329 -0
- package/tools/ExitWorktreeTool/UI.tsx +25 -0
- package/tools/ExitWorktreeTool/constants.ts +1 -0
- package/tools/ExitWorktreeTool/prompt.ts +32 -0
- package/tools/FileEditTool/FileEditTool.ts +625 -0
- package/tools/FileEditTool/UI.tsx +289 -0
- package/tools/FileEditTool/constants.ts +11 -0
- package/tools/FileEditTool/prompt.ts +28 -0
- package/tools/FileEditTool/types.ts +85 -0
- package/tools/FileEditTool/utils.ts +775 -0
- package/tools/FileReadTool/FileReadTool.ts +1183 -0
- package/tools/FileReadTool/UI.tsx +185 -0
- package/tools/FileReadTool/imageProcessor.ts +94 -0
- package/tools/FileReadTool/limits.ts +92 -0
- package/tools/FileReadTool/prompt.ts +49 -0
- package/tools/FileWriteTool/FileWriteTool.ts +434 -0
- package/tools/FileWriteTool/UI.tsx +405 -0
- package/tools/FileWriteTool/prompt.ts +18 -0
- package/tools/GlobTool/GlobTool.ts +198 -0
- package/tools/GlobTool/UI.tsx +63 -0
- package/tools/GlobTool/prompt.ts +7 -0
- package/tools/GrepTool/GrepTool.ts +577 -0
- package/tools/GrepTool/UI.tsx +201 -0
- package/tools/GrepTool/prompt.ts +18 -0
- package/tools/LSPTool/LSPTool.ts +860 -0
- package/tools/LSPTool/UI.tsx +228 -0
- package/tools/LSPTool/formatters.ts +592 -0
- package/tools/LSPTool/prompt.ts +21 -0
- package/tools/LSPTool/schemas.ts +215 -0
- package/tools/LSPTool/symbolContext.ts +90 -0
- package/tools/ListMcpResourcesTool/ListMcpResourcesTool.ts +123 -0
- package/tools/ListMcpResourcesTool/UI.tsx +29 -0
- package/tools/ListMcpResourcesTool/prompt.ts +20 -0
- package/tools/MCPTool/MCPTool.ts +77 -0
- package/tools/MCPTool/UI.tsx +403 -0
- package/tools/MCPTool/classifyForCollapse.ts +604 -0
- package/tools/MCPTool/prompt.ts +3 -0
- package/tools/McpAuthTool/McpAuthTool.ts +215 -0
- package/tools/NotebookEditTool/NotebookEditTool.ts +490 -0
- package/tools/NotebookEditTool/UI.tsx +93 -0
- package/tools/NotebookEditTool/constants.ts +2 -0
- package/tools/NotebookEditTool/prompt.ts +3 -0
- package/tools/PowerShellTool/PowerShellTool.tsx +1001 -0
- package/tools/PowerShellTool/UI.tsx +131 -0
- package/tools/PowerShellTool/clmTypes.ts +211 -0
- package/tools/PowerShellTool/commandSemantics.ts +142 -0
- package/tools/PowerShellTool/commonParameters.ts +30 -0
- package/tools/PowerShellTool/destructiveCommandWarning.ts +109 -0
- package/tools/PowerShellTool/gitSafety.ts +176 -0
- package/tools/PowerShellTool/modeValidation.ts +404 -0
- package/tools/PowerShellTool/pathValidation.ts +2049 -0
- package/tools/PowerShellTool/powershellPermissions.ts +1648 -0
- package/tools/PowerShellTool/powershellSecurity.ts +1090 -0
- package/tools/PowerShellTool/prompt.ts +145 -0
- package/tools/PowerShellTool/readOnlyValidation.ts +1823 -0
- package/tools/PowerShellTool/toolName.ts +2 -0
- package/tools/REPLTool/constants.ts +46 -0
- package/tools/REPLTool/primitiveTools.ts +39 -0
- package/tools/ReadMcpResourceTool/ReadMcpResourceTool.ts +158 -0
- package/tools/ReadMcpResourceTool/UI.tsx +37 -0
- package/tools/ReadMcpResourceTool/prompt.ts +16 -0
- package/tools/RemoteTriggerTool/RemoteTriggerTool.ts +161 -0
- package/tools/RemoteTriggerTool/UI.tsx +17 -0
- package/tools/RemoteTriggerTool/prompt.ts +15 -0
- package/tools/ScheduleCronTool/CronCreateTool.ts +157 -0
- package/tools/ScheduleCronTool/CronDeleteTool.ts +95 -0
- package/tools/ScheduleCronTool/CronListTool.ts +97 -0
- package/tools/ScheduleCronTool/UI.tsx +60 -0
- package/tools/ScheduleCronTool/prompt.ts +135 -0
- package/tools/SendMessageTool/SendMessageTool.ts +917 -0
- package/tools/SendMessageTool/UI.tsx +31 -0
- package/tools/SendMessageTool/constants.ts +1 -0
- package/tools/SendMessageTool/prompt.ts +49 -0
- package/tools/SkillTool/SkillTool.ts +1108 -0
- package/tools/SkillTool/UI.tsx +128 -0
- package/tools/SkillTool/constants.ts +1 -0
- package/tools/SkillTool/prompt.ts +241 -0
- package/tools/SleepTool/prompt.ts +17 -0
- package/tools/SyntheticOutputTool/SyntheticOutputTool.ts +163 -0
- package/tools/TaskCreateTool/TaskCreateTool.ts +138 -0
- package/tools/TaskCreateTool/constants.ts +1 -0
- package/tools/TaskCreateTool/prompt.ts +56 -0
- package/tools/TaskGetTool/TaskGetTool.ts +128 -0
- package/tools/TaskGetTool/constants.ts +1 -0
- package/tools/TaskGetTool/prompt.ts +24 -0
- package/tools/TaskListTool/TaskListTool.ts +116 -0
- package/tools/TaskListTool/constants.ts +1 -0
- package/tools/TaskListTool/prompt.ts +49 -0
- package/tools/TaskOutputTool/TaskOutputTool.tsx +584 -0
- package/tools/TaskOutputTool/constants.ts +1 -0
- package/tools/TaskStopTool/TaskStopTool.ts +131 -0
- package/tools/TaskStopTool/UI.tsx +41 -0
- package/tools/TaskStopTool/prompt.ts +8 -0
- package/tools/TaskUpdateTool/TaskUpdateTool.ts +406 -0
- package/tools/TaskUpdateTool/constants.ts +1 -0
- package/tools/TaskUpdateTool/prompt.ts +77 -0
- package/tools/TeamCreateTool/TeamCreateTool.ts +240 -0
- package/tools/TeamCreateTool/UI.tsx +6 -0
- package/tools/TeamCreateTool/constants.ts +1 -0
- package/tools/TeamCreateTool/prompt.ts +113 -0
- package/tools/TeamDeleteTool/TeamDeleteTool.ts +139 -0
- package/tools/TeamDeleteTool/UI.tsx +20 -0
- package/tools/TeamDeleteTool/constants.ts +1 -0
- package/tools/TeamDeleteTool/prompt.ts +16 -0
- package/tools/TodoWriteTool/TodoWriteTool.ts +115 -0
- package/tools/TodoWriteTool/constants.ts +1 -0
- package/tools/TodoWriteTool/prompt.ts +184 -0
- package/tools/ToolSearchTool/ToolSearchTool.ts +471 -0
- package/tools/ToolSearchTool/constants.ts +1 -0
- package/tools/ToolSearchTool/prompt.ts +121 -0
- package/tools/TungstenTool/TungstenTool.ts +4 -0
- package/tools/WebFetchTool/UI.tsx +72 -0
- package/tools/WebFetchTool/WebFetchTool.ts +318 -0
- package/tools/WebFetchTool/preapproved.ts +166 -0
- package/tools/WebFetchTool/prompt.ts +46 -0
- package/tools/WebFetchTool/utils.ts +530 -0
- package/tools/WebSearchTool/UI.tsx +101 -0
- package/tools/WebSearchTool/WebSearchTool.ts +435 -0
- package/tools/WebSearchTool/prompt.ts +34 -0
- package/tools/WorkflowTool/constants.ts +2 -0
- package/tools/XMemIngestTool/XMemIngestTool.ts +140 -0
- package/tools/XMemIngestTool/prompt.ts +13 -0
- package/tools/XMemRetrieveTool/XMemRetrieveTool.ts +177 -0
- package/tools/XMemRetrieveTool/prompt.ts +16 -0
- package/tools/XMemSearchTool/XMemSearchTool.ts +172 -0
- package/tools/XMemSearchTool/prompt.ts +11 -0
- package/tools/shared/gitOperationTracking.ts +277 -0
- package/tools/shared/spawnMultiAgent.ts +1093 -0
- package/tools/testing/TestingPermissionTool.tsx +74 -0
- package/tools/utils.ts +40 -0
- package/utils/CircularBuffer.ts +84 -0
- package/utils/Cursor.ts +1530 -0
- package/utils/QueryGuard.ts +121 -0
- package/utils/Shell.ts +474 -0
- package/utils/ShellCommand.ts +465 -0
- package/utils/abortController.ts +99 -0
- package/utils/activityManager.ts +164 -0
- package/utils/advisor.ts +145 -0
- package/utils/agentContext.ts +178 -0
- package/utils/agentId.ts +99 -0
- package/utils/agentSwarmsEnabled.ts +44 -0
- package/utils/agenticSessionSearch.ts +307 -0
- package/utils/analyzeContext.ts +1382 -0
- package/utils/ansiToPng.ts +334 -0
- package/utils/ansiToSvg.ts +272 -0
- package/utils/api.ts +718 -0
- package/utils/apiPreconnect.ts +71 -0
- package/utils/appleTerminalBackup.ts +124 -0
- package/utils/argumentSubstitution.ts +145 -0
- package/utils/array.ts +13 -0
- package/utils/asciicast.ts +239 -0
- package/utils/attachments.ts +4091 -0
- package/utils/attribution.ts +393 -0
- package/utils/auth.ts +2002 -0
- package/utils/authFileDescriptor.ts +196 -0
- package/utils/authPortable.ts +19 -0
- package/utils/autoModeDenials.ts +26 -0
- package/utils/autoRunIssue.tsx +122 -0
- package/utils/autoUpdater.ts +561 -0
- package/utils/aws.ts +74 -0
- package/utils/awsAuthStatusManager.ts +81 -0
- package/utils/axios.ts +8 -0
- package/utils/background/remote/preconditions.ts +235 -0
- package/utils/background/remote/remoteSession.ts +98 -0
- package/utils/backgroundHousekeeping.ts +94 -0
- package/utils/bash/ParsedCommand.ts +318 -0
- package/utils/bash/ShellSnapshot.ts +582 -0
- package/utils/bash/ast.ts +2679 -0
- package/utils/bash/bashParser.ts +4436 -0
- package/utils/bash/bashPipeCommand.ts +294 -0
- package/utils/bash/commands.ts +1339 -0
- package/utils/bash/heredoc.ts +733 -0
- package/utils/bash/parser.ts +230 -0
- package/utils/bash/prefix.ts +204 -0
- package/utils/bash/registry.ts +53 -0
- package/utils/bash/shellCompletion.ts +259 -0
- package/utils/bash/shellPrefix.ts +28 -0
- package/utils/bash/shellQuote.ts +304 -0
- package/utils/bash/shellQuoting.ts +128 -0
- package/utils/bash/specs/alias.ts +14 -0
- package/utils/bash/specs/index.ts +18 -0
- package/utils/bash/specs/nohup.ts +13 -0
- package/utils/bash/specs/pyright.ts +91 -0
- package/utils/bash/specs/sleep.ts +13 -0
- package/utils/bash/specs/srun.ts +31 -0
- package/utils/bash/specs/time.ts +13 -0
- package/utils/bash/specs/timeout.ts +20 -0
- package/utils/bash/treeSitterAnalysis.ts +506 -0
- package/utils/betas.ts +434 -0
- package/utils/billing.ts +78 -0
- package/utils/binaryCheck.ts +53 -0
- package/utils/browser.ts +68 -0
- package/utils/bufferedWriter.ts +100 -0
- package/utils/bundledMode.ts +22 -0
- package/utils/caCerts.ts +115 -0
- package/utils/caCertsConfig.ts +88 -0
- package/utils/cachePaths.ts +38 -0
- package/utils/classifierApprovals.ts +88 -0
- package/utils/classifierApprovalsHook.ts +17 -0
- package/utils/claudeCodeHints.ts +193 -0
- package/utils/claudeDesktop.ts +152 -0
- package/utils/claudeInChrome/chromeNativeHost.ts +527 -0
- package/utils/claudeInChrome/common.ts +540 -0
- package/utils/claudeInChrome/mcpServer.ts +292 -0
- package/utils/claudeInChrome/prompt.ts +83 -0
- package/utils/claudeInChrome/setup.ts +400 -0
- package/utils/claudeInChrome/setupPortable.ts +233 -0
- package/utils/claudeInChrome/toolRendering.tsx +262 -0
- package/utils/claudemd.ts +1479 -0
- package/utils/cleanup.ts +602 -0
- package/utils/cleanupRegistry.ts +25 -0
- package/utils/cliArgs.ts +60 -0
- package/utils/cliHighlight.ts +54 -0
- package/utils/codeIndexing.ts +206 -0
- package/utils/collapseBackgroundBashNotifications.ts +84 -0
- package/utils/collapseHookSummaries.ts +59 -0
- package/utils/collapseReadSearch.ts +1109 -0
- package/utils/collapseTeammateShutdowns.ts +55 -0
- package/utils/color-diff-mock.ts +27 -0
- package/utils/combinedAbortSignal.ts +47 -0
- package/utils/commandLifecycle.ts +21 -0
- package/utils/commitAttribution.ts +961 -0
- package/utils/completionCache.ts +166 -0
- package/utils/computerUse/appNames.ts +196 -0
- package/utils/computerUse/cleanup.ts +86 -0
- package/utils/computerUse/common.ts +61 -0
- package/utils/computerUse/computerUseLock.ts +215 -0
- package/utils/computerUse/drainRunLoop.ts +79 -0
- package/utils/computerUse/escHotkey.ts +54 -0
- package/utils/computerUse/executor.ts +658 -0
- package/utils/computerUse/gates.ts +72 -0
- package/utils/computerUse/hostAdapter.ts +69 -0
- package/utils/computerUse/inputLoader.ts +30 -0
- package/utils/computerUse/mcpServer.ts +106 -0
- package/utils/computerUse/setup.ts +53 -0
- package/utils/computerUse/swiftLoader.ts +23 -0
- package/utils/computerUse/toolRendering.tsx +125 -0
- package/utils/computerUse/wrapper.tsx +336 -0
- package/utils/concurrentSessions.ts +204 -0
- package/utils/config.ts +1817 -0
- package/utils/configConstants.ts +21 -0
- package/utils/contentArray.ts +51 -0
- package/utils/context.ts +221 -0
- package/utils/contextAnalysis.ts +272 -0
- package/utils/contextSuggestions.ts +235 -0
- package/utils/controlMessageCompat.ts +32 -0
- package/utils/conversationRecovery.ts +597 -0
- package/utils/cron.ts +308 -0
- package/utils/cronJitterConfig.ts +75 -0
- package/utils/cronScheduler.ts +565 -0
- package/utils/cronTasks.ts +458 -0
- package/utils/cronTasksLock.ts +195 -0
- package/utils/crossProjectResume.ts +75 -0
- package/utils/crypto.ts +13 -0
- package/utils/cwd.ts +32 -0
- package/utils/debug.ts +268 -0
- package/utils/debugFilter.ts +157 -0
- package/utils/deepLink/banner.ts +123 -0
- package/utils/deepLink/parseDeepLink.ts +170 -0
- package/utils/deepLink/protocolHandler.ts +136 -0
- package/utils/deepLink/registerProtocol.ts +348 -0
- package/utils/deepLink/terminalLauncher.ts +557 -0
- package/utils/deepLink/terminalPreference.ts +54 -0
- package/utils/desktopDeepLink.ts +236 -0
- package/utils/detectRepository.ts +178 -0
- package/utils/diagLogs.ts +94 -0
- package/utils/diff.ts +177 -0
- package/utils/directMemberMessage.ts +69 -0
- package/utils/displayTags.ts +51 -0
- package/utils/doctorContextWarnings.ts +265 -0
- package/utils/doctorDiagnostic.ts +625 -0
- package/utils/dxt/helpers.ts +88 -0
- package/utils/dxt/zip.ts +226 -0
- package/utils/earlyInput.ts +191 -0
- package/utils/editor.ts +183 -0
- package/utils/effort.ts +329 -0
- package/utils/embeddedTools.ts +29 -0
- package/utils/env.ts +347 -0
- package/utils/envDynamic.ts +151 -0
- package/utils/envUtils.ts +183 -0
- package/utils/envValidation.ts +38 -0
- package/utils/errorLogSink.ts +235 -0
- package/utils/errors.ts +238 -0
- package/utils/exampleCommands.ts +184 -0
- package/utils/execFileNoThrow.ts +150 -0
- package/utils/execFileNoThrowPortable.ts +89 -0
- package/utils/execSyncWrapper.ts +38 -0
- package/utils/exportRenderer.tsx +98 -0
- package/utils/extraUsage.ts +23 -0
- package/utils/fastMode.ts +532 -0
- package/utils/file.ts +584 -0
- package/utils/fileHistory.ts +1115 -0
- package/utils/fileOperationAnalytics.ts +71 -0
- package/utils/filePersistence/filePersistence.ts +287 -0
- package/utils/filePersistence/outputsScanner.ts +126 -0
- package/utils/fileRead.ts +102 -0
- package/utils/fileReadCache.ts +96 -0
- package/utils/fileStateCache.ts +142 -0
- package/utils/findExecutable.ts +17 -0
- package/utils/fingerprint.ts +76 -0
- package/utils/forkedAgent.ts +689 -0
- package/utils/format.ts +308 -0
- package/utils/formatBriefTimestamp.ts +81 -0
- package/utils/fpsTracker.ts +47 -0
- package/utils/frontmatterParser.ts +370 -0
- package/utils/fsOperations.ts +770 -0
- package/utils/fullscreen.ts +202 -0
- package/utils/generatedFiles.ts +136 -0
- package/utils/generators.ts +88 -0
- package/utils/genericProcessUtils.ts +184 -0
- package/utils/getWorktreePaths.ts +70 -0
- package/utils/getWorktreePathsPortable.ts +27 -0
- package/utils/ghPrStatus.ts +106 -0
- package/utils/git/gitConfigParser.ts +277 -0
- package/utils/git/gitFilesystem.ts +699 -0
- package/utils/git/gitignore.ts +99 -0
- package/utils/git.ts +926 -0
- package/utils/gitDiff.ts +532 -0
- package/utils/gitSettings.ts +18 -0
- package/utils/github/ghAuthStatus.ts +29 -0
- package/utils/githubRepoPathMapping.ts +162 -0
- package/utils/glob.ts +130 -0
- package/utils/gracefulShutdown.ts +529 -0
- package/utils/groupToolUses.ts +182 -0
- package/utils/handlePromptSubmit.ts +610 -0
- package/utils/hash.ts +46 -0
- package/utils/headlessProfiler.ts +178 -0
- package/utils/heapDumpService.ts +303 -0
- package/utils/heatmap.ts +198 -0
- package/utils/highlightMatch.tsx +28 -0
- package/utils/hooks/AsyncHookRegistry.ts +309 -0
- package/utils/hooks/apiQueryHookHelper.ts +141 -0
- package/utils/hooks/execAgentHook.ts +339 -0
- package/utils/hooks/execHttpHook.ts +242 -0
- package/utils/hooks/execPromptHook.ts +211 -0
- package/utils/hooks/fileChangedWatcher.ts +191 -0
- package/utils/hooks/hookEvents.ts +192 -0
- package/utils/hooks/hookHelpers.ts +83 -0
- package/utils/hooks/hooksConfigManager.ts +400 -0
- package/utils/hooks/hooksConfigSnapshot.ts +133 -0
- package/utils/hooks/hooksSettings.ts +271 -0
- package/utils/hooks/postSamplingHooks.ts +70 -0
- package/utils/hooks/registerFrontmatterHooks.ts +67 -0
- package/utils/hooks/registerSkillHooks.ts +64 -0
- package/utils/hooks/sessionHooks.ts +447 -0
- package/utils/hooks/skillImprovement.ts +267 -0
- package/utils/hooks/ssrfGuard.ts +294 -0
- package/utils/hooks.ts +5022 -0
- package/utils/horizontalScroll.ts +137 -0
- package/utils/http.ts +136 -0
- package/utils/hyperlink.ts +39 -0
- package/utils/iTermBackup.ts +73 -0
- package/utils/ide.ts +1494 -0
- package/utils/idePathConversion.ts +90 -0
- package/utils/idleTimeout.ts +53 -0
- package/utils/imagePaste.ts +416 -0
- package/utils/imageResizer.ts +880 -0
- package/utils/imageStore.ts +167 -0
- package/utils/imageValidation.ts +104 -0
- package/utils/immediateCommand.ts +15 -0
- package/utils/inProcessTeammateHelpers.ts +102 -0
- package/utils/ink.ts +26 -0
- package/utils/intl.ts +94 -0
- package/utils/jetbrains.ts +191 -0
- package/utils/json.ts +277 -0
- package/utils/jsonRead.ts +16 -0
- package/utils/keyboardShortcuts.ts +14 -0
- package/utils/lazySchema.ts +8 -0
- package/utils/listSessionsImpl.ts +454 -0
- package/utils/localInstaller.ts +162 -0
- package/utils/lockfile.ts +43 -0
- package/utils/log.ts +362 -0
- package/utils/logoV2Utils.ts +347 -0
- package/utils/mailbox.ts +73 -0
- package/utils/managedEnv.ts +199 -0
- package/utils/managedEnvConstants.ts +191 -0
- package/utils/markdown.ts +381 -0
- package/utils/markdownConfigLoader.ts +600 -0
- package/utils/mcp/dateTimeParser.ts +121 -0
- package/utils/mcp/elicitationValidation.ts +336 -0
- package/utils/mcpInstructionsDelta.ts +130 -0
- package/utils/mcpOutputStorage.ts +189 -0
- package/utils/mcpValidation.ts +208 -0
- package/utils/mcpWebSocketTransport.ts +200 -0
- package/utils/memoize.ts +269 -0
- package/utils/memory/types.ts +12 -0
- package/utils/memory/versions.ts +8 -0
- package/utils/memoryFileDetection.ts +289 -0
- package/utils/messagePredicates.ts +8 -0
- package/utils/messageQueueManager.ts +547 -0
- package/utils/messages/mappers.ts +290 -0
- package/utils/messages/systemInit.ts +96 -0
- package/utils/messages.ts +5520 -0
- package/utils/model/agent.ts +157 -0
- package/utils/model/aliases.ts +35 -0
- package/utils/model/antModels.ts +64 -0
- package/utils/model/bedrock.ts +265 -0
- package/utils/model/check1mAccess.ts +72 -0
- package/utils/model/configs.ts +158 -0
- package/utils/model/contextWindowUpgradeCheck.ts +47 -0
- package/utils/model/deprecation.ts +101 -0
- package/utils/model/model.ts +654 -0
- package/utils/model/modelAllowlist.ts +170 -0
- package/utils/model/modelCapabilities.ts +118 -0
- package/utils/model/modelOptions.ts +589 -0
- package/utils/model/modelStrings.ts +170 -0
- package/utils/model/modelSupportOverrides.ts +50 -0
- package/utils/model/providers.ts +42 -0
- package/utils/model/validateModel.ts +159 -0
- package/utils/modelCost.ts +231 -0
- package/utils/modifiers.ts +36 -0
- package/utils/mtls.ts +179 -0
- package/utils/nativeInstaller/download.ts +523 -0
- package/utils/nativeInstaller/index.ts +18 -0
- package/utils/nativeInstaller/installer.ts +1708 -0
- package/utils/nativeInstaller/packageManagers.ts +336 -0
- package/utils/nativeInstaller/pidLock.ts +433 -0
- package/utils/notebook.ts +224 -0
- package/utils/objectGroupBy.ts +18 -0
- package/utils/pasteStore.ts +104 -0
- package/utils/path.ts +155 -0
- package/utils/pdf.ts +300 -0
- package/utils/pdfUtils.ts +70 -0
- package/utils/peerAddress.ts +21 -0
- package/utils/permissions/PermissionMode.ts +141 -0
- package/utils/permissions/PermissionPromptToolResultSchema.ts +127 -0
- package/utils/permissions/PermissionResult.ts +35 -0
- package/utils/permissions/PermissionRule.ts +40 -0
- package/utils/permissions/PermissionUpdate.ts +389 -0
- package/utils/permissions/PermissionUpdateSchema.ts +78 -0
- package/utils/permissions/autoModeState.ts +39 -0
- package/utils/permissions/bashClassifier.ts +61 -0
- package/utils/permissions/bypassPermissionsKillswitch.ts +155 -0
- package/utils/permissions/classifierDecision.ts +98 -0
- package/utils/permissions/classifierShared.ts +39 -0
- package/utils/permissions/dangerousPatterns.ts +80 -0
- package/utils/permissions/denialTracking.ts +45 -0
- package/utils/permissions/filesystem.ts +1777 -0
- package/utils/permissions/getNextPermissionMode.ts +101 -0
- package/utils/permissions/pathValidation.ts +485 -0
- package/utils/permissions/permissionExplainer.ts +250 -0
- package/utils/permissions/permissionRuleParser.ts +198 -0
- package/utils/permissions/permissionSetup.ts +1532 -0
- package/utils/permissions/permissions.ts +1486 -0
- package/utils/permissions/permissionsLoader.ts +296 -0
- package/utils/permissions/shadowedRuleDetection.ts +234 -0
- package/utils/permissions/shellRuleMatching.ts +228 -0
- package/utils/permissions/yoloClassifier.ts +1495 -0
- package/utils/planModeV2.ts +95 -0
- package/utils/plans.ts +397 -0
- package/utils/platform.ts +150 -0
- package/utils/plugins/addDirPluginSettings.ts +71 -0
- package/utils/plugins/cacheUtils.ts +196 -0
- package/utils/plugins/dependencyResolver.ts +305 -0
- package/utils/plugins/fetchTelemetry.ts +135 -0
- package/utils/plugins/gitAvailability.ts +69 -0
- package/utils/plugins/headlessPluginInstall.ts +174 -0
- package/utils/plugins/hintRecommendation.ts +164 -0
- package/utils/plugins/installCounts.ts +292 -0
- package/utils/plugins/installedPluginsManager.ts +1268 -0
- package/utils/plugins/loadPluginAgents.ts +348 -0
- package/utils/plugins/loadPluginCommands.ts +946 -0
- package/utils/plugins/loadPluginHooks.ts +287 -0
- package/utils/plugins/loadPluginOutputStyles.ts +178 -0
- package/utils/plugins/lspPluginIntegration.ts +387 -0
- package/utils/plugins/lspRecommendation.ts +374 -0
- package/utils/plugins/managedPlugins.ts +27 -0
- package/utils/plugins/marketplaceHelpers.ts +592 -0
- package/utils/plugins/marketplaceManager.ts +2643 -0
- package/utils/plugins/mcpPluginIntegration.ts +634 -0
- package/utils/plugins/mcpbHandler.ts +968 -0
- package/utils/plugins/officialMarketplace.ts +25 -0
- package/utils/plugins/officialMarketplaceGcs.ts +216 -0
- package/utils/plugins/officialMarketplaceStartupCheck.ts +439 -0
- package/utils/plugins/orphanedPluginFilter.ts +114 -0
- package/utils/plugins/parseMarketplaceInput.ts +162 -0
- package/utils/plugins/performStartupChecks.tsx +70 -0
- package/utils/plugins/pluginAutoupdate.ts +284 -0
- package/utils/plugins/pluginBlocklist.ts +127 -0
- package/utils/plugins/pluginDirectories.ts +178 -0
- package/utils/plugins/pluginFlagging.ts +208 -0
- package/utils/plugins/pluginIdentifier.ts +123 -0
- package/utils/plugins/pluginInstallationHelpers.ts +595 -0
- package/utils/plugins/pluginLoader.ts +3302 -0
- package/utils/plugins/pluginOptionsStorage.ts +400 -0
- package/utils/plugins/pluginPolicy.ts +20 -0
- package/utils/plugins/pluginStartupCheck.ts +341 -0
- package/utils/plugins/pluginVersioning.ts +157 -0
- package/utils/plugins/reconciler.ts +265 -0
- package/utils/plugins/refresh.ts +215 -0
- package/utils/plugins/schemas.ts +1681 -0
- package/utils/plugins/validatePlugin.ts +903 -0
- package/utils/plugins/walkPluginMarkdown.ts +69 -0
- package/utils/plugins/zipCache.ts +406 -0
- package/utils/plugins/zipCacheAdapters.ts +164 -0
- package/utils/powershell/dangerousCmdlets.ts +185 -0
- package/utils/powershell/parser.ts +1804 -0
- package/utils/powershell/staticPrefix.ts +316 -0
- package/utils/preflightChecks.tsx +151 -0
- package/utils/privacyLevel.ts +55 -0
- package/utils/process.ts +68 -0
- package/utils/processUserInput/processBashCommand.tsx +140 -0
- package/utils/processUserInput/processSlashCommand.tsx +922 -0
- package/utils/processUserInput/processTextPrompt.ts +100 -0
- package/utils/processUserInput/processUserInput.ts +605 -0
- package/utils/profilerBase.ts +46 -0
- package/utils/promptCategory.ts +49 -0
- package/utils/promptEditor.ts +188 -0
- package/utils/promptShellExecution.ts +183 -0
- package/utils/proxy.ts +426 -0
- package/utils/queryContext.ts +179 -0
- package/utils/queryHelpers.ts +552 -0
- package/utils/queryProfiler.ts +301 -0
- package/utils/queueProcessor.ts +95 -0
- package/utils/readEditContext.ts +227 -0
- package/utils/readFileInRange.ts +383 -0
- package/utils/releaseNotes.ts +360 -0
- package/utils/renderOptions.ts +113 -0
- package/utils/ripgrep.ts +679 -0
- package/utils/sandbox/sandbox-adapter.ts +985 -0
- package/utils/sandbox/sandbox-ui-utils.ts +12 -0
- package/utils/sanitization.ts +91 -0
- package/utils/screenshotClipboard.ts +121 -0
- package/utils/sdkEventQueue.ts +134 -0
- package/utils/secureStorage/fallbackStorage.ts +70 -0
- package/utils/secureStorage/index.ts +17 -0
- package/utils/secureStorage/keychainPrefetch.ts +116 -0
- package/utils/secureStorage/macOsKeychainHelpers.ts +111 -0
- package/utils/secureStorage/macOsKeychainStorage.ts +231 -0
- package/utils/secureStorage/plainTextStorage.ts +84 -0
- package/utils/semanticBoolean.ts +29 -0
- package/utils/semanticNumber.ts +36 -0
- package/utils/semver.ts +59 -0
- package/utils/sequential.ts +56 -0
- package/utils/sessionActivity.ts +133 -0
- package/utils/sessionEnvVars.ts +22 -0
- package/utils/sessionEnvironment.ts +166 -0
- package/utils/sessionFileAccessHooks.ts +250 -0
- package/utils/sessionIngressAuth.ts +140 -0
- package/utils/sessionRestore.ts +551 -0
- package/utils/sessionStart.ts +232 -0
- package/utils/sessionState.ts +150 -0
- package/utils/sessionStorage.ts +5105 -0
- package/utils/sessionStoragePortable.ts +793 -0
- package/utils/sessionTitle.ts +129 -0
- package/utils/sessionUrl.ts +64 -0
- package/utils/set.ts +53 -0
- package/utils/settings/allErrors.ts +32 -0
- package/utils/settings/applySettingsChange.ts +92 -0
- package/utils/settings/changeDetector.ts +488 -0
- package/utils/settings/constants.ts +202 -0
- package/utils/settings/internalWrites.ts +37 -0
- package/utils/settings/managedPath.ts +34 -0
- package/utils/settings/mdm/constants.ts +81 -0
- package/utils/settings/mdm/rawRead.ts +130 -0
- package/utils/settings/mdm/settings.ts +316 -0
- package/utils/settings/permissionValidation.ts +262 -0
- package/utils/settings/pluginOnlyPolicy.ts +60 -0
- package/utils/settings/schemaOutput.ts +8 -0
- package/utils/settings/settings.ts +1015 -0
- package/utils/settings/settingsCache.ts +80 -0
- package/utils/settings/toolValidationConfig.ts +103 -0
- package/utils/settings/types.ts +1149 -0
- package/utils/settings/validateEditTool.ts +45 -0
- package/utils/settings/validation.ts +265 -0
- package/utils/settings/validationTips.ts +164 -0
- package/utils/shell/bashProvider.ts +255 -0
- package/utils/shell/outputLimits.ts +14 -0
- package/utils/shell/powershellDetection.ts +107 -0
- package/utils/shell/powershellProvider.ts +123 -0
- package/utils/shell/prefix.ts +367 -0
- package/utils/shell/readOnlyCommandValidation.ts +1893 -0
- package/utils/shell/resolveDefaultShell.ts +14 -0
- package/utils/shell/shellProvider.ts +33 -0
- package/utils/shell/shellToolUtils.ts +22 -0
- package/utils/shell/specPrefix.ts +241 -0
- package/utils/shellConfig.ts +167 -0
- package/utils/sideQuery.ts +222 -0
- package/utils/sideQuestion.ts +155 -0
- package/utils/signal.ts +43 -0
- package/utils/sinks.ts +16 -0
- package/utils/skills/skillChangeDetector.ts +311 -0
- package/utils/slashCommandParsing.ts +60 -0
- package/utils/sleep.ts +84 -0
- package/utils/sliceAnsi.ts +91 -0
- package/utils/slowOperations.ts +286 -0
- package/utils/standaloneAgent.ts +23 -0
- package/utils/startupProfiler.ts +194 -0
- package/utils/staticRender.tsx +116 -0
- package/utils/stats.ts +1061 -0
- package/utils/statsCache.ts +434 -0
- package/utils/status.tsx +362 -0
- package/utils/statusNoticeDefinitions.tsx +198 -0
- package/utils/statusNoticeHelpers.ts +20 -0
- package/utils/stream.ts +76 -0
- package/utils/streamJsonStdoutGuard.ts +123 -0
- package/utils/streamlinedTransform.ts +201 -0
- package/utils/stringUtils.ts +235 -0
- package/utils/subprocessEnv.ts +99 -0
- package/utils/suggestions/commandSuggestions.ts +567 -0
- package/utils/suggestions/directoryCompletion.ts +263 -0
- package/utils/suggestions/shellHistoryCompletion.ts +119 -0
- package/utils/suggestions/skillUsageTracking.ts +55 -0
- package/utils/suggestions/slackChannelSuggestions.ts +209 -0
- package/utils/swarm/It2SetupPrompt.tsx +380 -0
- package/utils/swarm/backends/ITermBackend.ts +370 -0
- package/utils/swarm/backends/InProcessBackend.ts +339 -0
- package/utils/swarm/backends/PaneBackendExecutor.ts +354 -0
- package/utils/swarm/backends/TmuxBackend.ts +764 -0
- package/utils/swarm/backends/detection.ts +128 -0
- package/utils/swarm/backends/it2Setup.ts +245 -0
- package/utils/swarm/backends/registry.ts +464 -0
- package/utils/swarm/backends/teammateModeSnapshot.ts +87 -0
- package/utils/swarm/backends/types.ts +311 -0
- package/utils/swarm/constants.ts +33 -0
- package/utils/swarm/inProcessRunner.ts +1552 -0
- package/utils/swarm/leaderPermissionBridge.ts +54 -0
- package/utils/swarm/permissionSync.ts +928 -0
- package/utils/swarm/reconnection.ts +119 -0
- package/utils/swarm/spawnInProcess.ts +328 -0
- package/utils/swarm/spawnUtils.ts +146 -0
- package/utils/swarm/teamHelpers.ts +683 -0
- package/utils/swarm/teammateInit.ts +129 -0
- package/utils/swarm/teammateLayoutManager.ts +107 -0
- package/utils/swarm/teammateModel.ts +10 -0
- package/utils/swarm/teammatePromptAddendum.ts +18 -0
- package/utils/systemDirectories.ts +74 -0
- package/utils/systemPrompt.ts +123 -0
- package/utils/systemPromptType.ts +14 -0
- package/utils/systemTheme.ts +119 -0
- package/utils/taggedId.ts +54 -0
- package/utils/task/TaskOutput.ts +390 -0
- package/utils/task/diskOutput.ts +451 -0
- package/utils/task/framework.ts +308 -0
- package/utils/task/outputFormatting.ts +38 -0
- package/utils/task/sdkProgress.ts +36 -0
- package/utils/tasks.ts +862 -0
- package/utils/teamDiscovery.ts +81 -0
- package/utils/teamMemoryOps.ts +88 -0
- package/utils/teammate.ts +292 -0
- package/utils/teammateContext.ts +96 -0
- package/utils/teammateMailbox.ts +1183 -0
- package/utils/telemetry/betaSessionTracing.ts +491 -0
- package/utils/telemetry/bigqueryExporter.ts +252 -0
- package/utils/telemetry/events.ts +75 -0
- package/utils/telemetry/instrumentation.ts +825 -0
- package/utils/telemetry/logger.ts +26 -0
- package/utils/telemetry/perfettoTracing.ts +1120 -0
- package/utils/telemetry/pluginTelemetry.ts +289 -0
- package/utils/telemetry/sessionTracing.ts +927 -0
- package/utils/telemetry/skillLoadedEvent.ts +39 -0
- package/utils/telemetryAttributes.ts +71 -0
- package/utils/teleport/api.ts +466 -0
- package/utils/teleport/environmentSelection.ts +77 -0
- package/utils/teleport/environments.ts +120 -0
- package/utils/teleport/gitBundle.ts +292 -0
- package/utils/teleport.tsx +1226 -0
- package/utils/tempfile.ts +31 -0
- package/utils/terminal.ts +131 -0
- package/utils/terminalPanel.ts +191 -0
- package/utils/textHighlighting.ts +166 -0
- package/utils/theme.ts +639 -0
- package/utils/thinking.ts +162 -0
- package/utils/timeouts.ts +39 -0
- package/utils/tmuxSocket.ts +427 -0
- package/utils/todo/types.ts +18 -0
- package/utils/tokenBudget.ts +73 -0
- package/utils/tokens.ts +261 -0
- package/utils/toolErrors.ts +132 -0
- package/utils/toolPool.ts +79 -0
- package/utils/toolResultStorage.ts +1040 -0
- package/utils/toolSchemaCache.ts +26 -0
- package/utils/toolSearch.ts +756 -0
- package/utils/transcriptSearch.ts +202 -0
- package/utils/treeify.ts +170 -0
- package/utils/truncate.ts +179 -0
- package/utils/ultraplan/ccrSession.ts +349 -0
- package/utils/ultraplan/keyword.ts +127 -0
- package/utils/ultraplan/prompt.txt +1 -0
- package/utils/unaryLogging.ts +39 -0
- package/utils/undercover.ts +89 -0
- package/utils/user.ts +194 -0
- package/utils/userAgent.ts +10 -0
- package/utils/userPromptKeywords.ts +27 -0
- package/utils/uuid.ts +27 -0
- package/utils/warningHandler.ts +121 -0
- package/utils/which.ts +82 -0
- package/utils/windowsPaths.ts +173 -0
- package/utils/withResolvers.ts +13 -0
- package/utils/words.ts +800 -0
- package/utils/workloadContext.ts +57 -0
- package/utils/worktree.ts +1519 -0
- package/utils/worktreeModeEnabled.ts +11 -0
- package/utils/xdg.ts +65 -0
- package/utils/xmem.ts +6 -0
- package/utils/xml.ts +16 -0
- package/utils/yaml.ts +15 -0
- package/utils/zodToJsonSchema.ts +23 -0
|
@@ -0,0 +1,1093 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Shared spawn module for teammate creation.
|
|
3
|
+
* Extracted from TeammateTool to allow reuse by AgentTool.
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
import React from 'react'
|
|
7
|
+
import {
|
|
8
|
+
getChromeFlagOverride,
|
|
9
|
+
getFlagSettingsPath,
|
|
10
|
+
getInlinePlugins,
|
|
11
|
+
getMainLoopModelOverride,
|
|
12
|
+
getSessionBypassPermissionsMode,
|
|
13
|
+
getSessionId,
|
|
14
|
+
} from '../../bootstrap/state.js'
|
|
15
|
+
import type { AppState } from '../../state/AppState.js'
|
|
16
|
+
import { createTaskStateBase, generateTaskId } from '../../Task.js'
|
|
17
|
+
import type { ToolUseContext } from '../../Tool.js'
|
|
18
|
+
import type { InProcessTeammateTaskState } from '../../tasks/InProcessTeammateTask/types.js'
|
|
19
|
+
import { formatAgentId } from '../../utils/agentId.js'
|
|
20
|
+
import { quote } from '../../utils/bash/shellQuote.js'
|
|
21
|
+
import { isInBundledMode } from '../../utils/bundledMode.js'
|
|
22
|
+
import { getGlobalConfig } from '../../utils/config.js'
|
|
23
|
+
import { getCwd } from '../../utils/cwd.js'
|
|
24
|
+
import { logForDebugging } from '../../utils/debug.js'
|
|
25
|
+
import { errorMessage } from '../../utils/errors.js'
|
|
26
|
+
import { execFileNoThrow } from '../../utils/execFileNoThrow.js'
|
|
27
|
+
import { parseUserSpecifiedModel } from '../../utils/model/model.js'
|
|
28
|
+
import type { PermissionMode } from '../../utils/permissions/PermissionMode.js'
|
|
29
|
+
import { isTmuxAvailable } from '../../utils/swarm/backends/detection.js'
|
|
30
|
+
import {
|
|
31
|
+
detectAndGetBackend,
|
|
32
|
+
getBackendByType,
|
|
33
|
+
isInProcessEnabled,
|
|
34
|
+
markInProcessFallback,
|
|
35
|
+
resetBackendDetection,
|
|
36
|
+
} from '../../utils/swarm/backends/registry.js'
|
|
37
|
+
import { getTeammateModeFromSnapshot } from '../../utils/swarm/backends/teammateModeSnapshot.js'
|
|
38
|
+
import type { BackendType } from '../../utils/swarm/backends/types.js'
|
|
39
|
+
import { isPaneBackend } from '../../utils/swarm/backends/types.js'
|
|
40
|
+
import {
|
|
41
|
+
SWARM_SESSION_NAME,
|
|
42
|
+
TEAM_LEAD_NAME,
|
|
43
|
+
TEAMMATE_COMMAND_ENV_VAR,
|
|
44
|
+
TMUX_COMMAND,
|
|
45
|
+
} from '../../utils/swarm/constants.js'
|
|
46
|
+
import { It2SetupPrompt } from '../../utils/swarm/It2SetupPrompt.js'
|
|
47
|
+
import { startInProcessTeammate } from '../../utils/swarm/inProcessRunner.js'
|
|
48
|
+
import {
|
|
49
|
+
type InProcessSpawnConfig,
|
|
50
|
+
spawnInProcessTeammate,
|
|
51
|
+
} from '../../utils/swarm/spawnInProcess.js'
|
|
52
|
+
import { buildInheritedEnvVars } from '../../utils/swarm/spawnUtils.js'
|
|
53
|
+
import {
|
|
54
|
+
readTeamFileAsync,
|
|
55
|
+
sanitizeAgentName,
|
|
56
|
+
sanitizeName,
|
|
57
|
+
writeTeamFileAsync,
|
|
58
|
+
} from '../../utils/swarm/teamHelpers.js'
|
|
59
|
+
import {
|
|
60
|
+
assignTeammateColor,
|
|
61
|
+
createTeammatePaneInSwarmView,
|
|
62
|
+
enablePaneBorderStatus,
|
|
63
|
+
isInsideTmux,
|
|
64
|
+
sendCommandToPane,
|
|
65
|
+
} from '../../utils/swarm/teammateLayoutManager.js'
|
|
66
|
+
import { getHardcodedTeammateModelFallback } from '../../utils/swarm/teammateModel.js'
|
|
67
|
+
import { registerTask } from '../../utils/task/framework.js'
|
|
68
|
+
import { writeToMailbox } from '../../utils/teammateMailbox.js'
|
|
69
|
+
import type { CustomAgentDefinition } from '../AgentTool/loadAgentsDir.js'
|
|
70
|
+
import { isCustomAgent } from '../AgentTool/loadAgentsDir.js'
|
|
71
|
+
|
|
72
|
+
function getDefaultTeammateModel(leaderModel: string | null): string {
|
|
73
|
+
const configured = getGlobalConfig().teammateDefaultModel
|
|
74
|
+
if (configured === null) {
|
|
75
|
+
// User picked "Default" in the /config picker — follow the leader.
|
|
76
|
+
return leaderModel ?? getHardcodedTeammateModelFallback()
|
|
77
|
+
}
|
|
78
|
+
if (configured !== undefined) {
|
|
79
|
+
return parseUserSpecifiedModel(configured)
|
|
80
|
+
}
|
|
81
|
+
return getHardcodedTeammateModelFallback()
|
|
82
|
+
}
|
|
83
|
+
|
|
84
|
+
/**
|
|
85
|
+
* Resolve a teammate model value. Handles the 'inherit' alias (from agent
|
|
86
|
+
* frontmatter) by substituting the leader's model. gh-31069: 'inherit' was
|
|
87
|
+
* passed literally to --model, producing "It may not exist or you may not
|
|
88
|
+
* have access". If leader model is null (not yet set), falls through to the
|
|
89
|
+
* default.
|
|
90
|
+
*
|
|
91
|
+
* Exported for testing.
|
|
92
|
+
*/
|
|
93
|
+
export function resolveTeammateModel(
|
|
94
|
+
inputModel: string | undefined,
|
|
95
|
+
leaderModel: string | null,
|
|
96
|
+
): string {
|
|
97
|
+
if (inputModel === 'inherit') {
|
|
98
|
+
return leaderModel ?? getDefaultTeammateModel(leaderModel)
|
|
99
|
+
}
|
|
100
|
+
return inputModel ?? getDefaultTeammateModel(leaderModel)
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
// ============================================================================
|
|
104
|
+
// Types
|
|
105
|
+
// ============================================================================
|
|
106
|
+
|
|
107
|
+
export type SpawnOutput = {
|
|
108
|
+
teammate_id: string
|
|
109
|
+
agent_id: string
|
|
110
|
+
agent_type?: string
|
|
111
|
+
model?: string
|
|
112
|
+
name: string
|
|
113
|
+
color?: string
|
|
114
|
+
tmux_session_name: string
|
|
115
|
+
tmux_window_name: string
|
|
116
|
+
tmux_pane_id: string
|
|
117
|
+
team_name?: string
|
|
118
|
+
is_splitpane?: boolean
|
|
119
|
+
plan_mode_required?: boolean
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
export type SpawnTeammateConfig = {
|
|
123
|
+
name: string
|
|
124
|
+
prompt: string
|
|
125
|
+
team_name?: string
|
|
126
|
+
cwd?: string
|
|
127
|
+
use_splitpane?: boolean
|
|
128
|
+
plan_mode_required?: boolean
|
|
129
|
+
model?: string
|
|
130
|
+
agent_type?: string
|
|
131
|
+
description?: string
|
|
132
|
+
/** request_id of the API call whose response contained the tool_use that
|
|
133
|
+
* spawned this teammate. Threaded through to TeammateAgentContext for
|
|
134
|
+
* lineage tracing on tengu_api_* events. */
|
|
135
|
+
invokingRequestId?: string
|
|
136
|
+
}
|
|
137
|
+
|
|
138
|
+
// Internal input type matching TeammateTool's spawn parameters
|
|
139
|
+
type SpawnInput = {
|
|
140
|
+
name: string
|
|
141
|
+
prompt: string
|
|
142
|
+
team_name?: string
|
|
143
|
+
cwd?: string
|
|
144
|
+
use_splitpane?: boolean
|
|
145
|
+
plan_mode_required?: boolean
|
|
146
|
+
model?: string
|
|
147
|
+
agent_type?: string
|
|
148
|
+
description?: string
|
|
149
|
+
invokingRequestId?: string
|
|
150
|
+
}
|
|
151
|
+
|
|
152
|
+
// ============================================================================
|
|
153
|
+
// Helper Functions
|
|
154
|
+
// ============================================================================
|
|
155
|
+
|
|
156
|
+
/**
|
|
157
|
+
* Checks if a tmux session exists
|
|
158
|
+
*/
|
|
159
|
+
async function hasSession(sessionName: string): Promise<boolean> {
|
|
160
|
+
const result = await execFileNoThrow(TMUX_COMMAND, [
|
|
161
|
+
'has-session',
|
|
162
|
+
'-t',
|
|
163
|
+
sessionName,
|
|
164
|
+
])
|
|
165
|
+
return result.code === 0
|
|
166
|
+
}
|
|
167
|
+
|
|
168
|
+
/**
|
|
169
|
+
* Creates a new tmux session if it doesn't exist
|
|
170
|
+
*/
|
|
171
|
+
async function ensureSession(sessionName: string): Promise<void> {
|
|
172
|
+
const exists = await hasSession(sessionName)
|
|
173
|
+
if (!exists) {
|
|
174
|
+
const result = await execFileNoThrow(TMUX_COMMAND, [
|
|
175
|
+
'new-session',
|
|
176
|
+
'-d',
|
|
177
|
+
'-s',
|
|
178
|
+
sessionName,
|
|
179
|
+
])
|
|
180
|
+
if (result.code !== 0) {
|
|
181
|
+
throw new Error(
|
|
182
|
+
`Failed to create tmux session '${sessionName}': ${result.stderr || 'Unknown error'}`,
|
|
183
|
+
)
|
|
184
|
+
}
|
|
185
|
+
}
|
|
186
|
+
}
|
|
187
|
+
|
|
188
|
+
/**
|
|
189
|
+
* Gets the command to spawn a teammate.
|
|
190
|
+
* For native builds (compiled binaries), use process.execPath.
|
|
191
|
+
* For non-native (node/bun running a script), use process.argv[1].
|
|
192
|
+
*/
|
|
193
|
+
function getTeammateCommand(): string {
|
|
194
|
+
if (process.env[TEAMMATE_COMMAND_ENV_VAR]) {
|
|
195
|
+
return process.env[TEAMMATE_COMMAND_ENV_VAR]
|
|
196
|
+
}
|
|
197
|
+
return isInBundledMode() ? process.execPath : process.argv[1]!
|
|
198
|
+
}
|
|
199
|
+
|
|
200
|
+
/**
|
|
201
|
+
* Builds CLI flags to propagate from the current session to spawned teammates.
|
|
202
|
+
* This ensures teammates inherit important settings like permission mode,
|
|
203
|
+
* model selection, and plugin configuration from their parent.
|
|
204
|
+
*
|
|
205
|
+
* @param options.planModeRequired - If true, don't inherit bypass permissions (plan mode takes precedence)
|
|
206
|
+
* @param options.permissionMode - Permission mode to propagate
|
|
207
|
+
*/
|
|
208
|
+
function buildInheritedCliFlags(options?: {
|
|
209
|
+
planModeRequired?: boolean
|
|
210
|
+
permissionMode?: PermissionMode
|
|
211
|
+
}): string {
|
|
212
|
+
const flags: string[] = []
|
|
213
|
+
const { planModeRequired, permissionMode } = options || {}
|
|
214
|
+
|
|
215
|
+
// Propagate permission mode to teammates, but NOT if plan mode is required
|
|
216
|
+
// Plan mode takes precedence over bypass permissions for safety
|
|
217
|
+
if (planModeRequired) {
|
|
218
|
+
// Don't inherit bypass permissions when plan mode is required
|
|
219
|
+
} else if (
|
|
220
|
+
permissionMode === 'bypassPermissions' ||
|
|
221
|
+
getSessionBypassPermissionsMode()
|
|
222
|
+
) {
|
|
223
|
+
flags.push('--dangerously-skip-permissions')
|
|
224
|
+
} else if (permissionMode === 'acceptEdits') {
|
|
225
|
+
flags.push('--permission-mode acceptEdits')
|
|
226
|
+
} else if (permissionMode === 'auto') {
|
|
227
|
+
// Teammates inherit auto mode so the classifier auto-approves their tool
|
|
228
|
+
// calls too. The teammate's own startup (permissionSetup.ts) handles
|
|
229
|
+
// GrowthBook gate checks and setAutoModeActive(true) independently.
|
|
230
|
+
flags.push('--permission-mode auto')
|
|
231
|
+
}
|
|
232
|
+
|
|
233
|
+
// Propagate --model if explicitly set via CLI
|
|
234
|
+
const modelOverride = getMainLoopModelOverride()
|
|
235
|
+
if (modelOverride) {
|
|
236
|
+
flags.push(`--model ${quote([modelOverride])}`)
|
|
237
|
+
}
|
|
238
|
+
|
|
239
|
+
// Propagate --settings if set via CLI
|
|
240
|
+
const settingsPath = getFlagSettingsPath()
|
|
241
|
+
if (settingsPath) {
|
|
242
|
+
flags.push(`--settings ${quote([settingsPath])}`)
|
|
243
|
+
}
|
|
244
|
+
|
|
245
|
+
// Propagate --plugin-dir for each inline plugin
|
|
246
|
+
const inlinePlugins = getInlinePlugins()
|
|
247
|
+
for (const pluginDir of inlinePlugins) {
|
|
248
|
+
flags.push(`--plugin-dir ${quote([pluginDir])}`)
|
|
249
|
+
}
|
|
250
|
+
|
|
251
|
+
// Propagate --chrome / --no-chrome if explicitly set on the CLI
|
|
252
|
+
const chromeFlagOverride = getChromeFlagOverride()
|
|
253
|
+
if (chromeFlagOverride === true) {
|
|
254
|
+
flags.push('--chrome')
|
|
255
|
+
} else if (chromeFlagOverride === false) {
|
|
256
|
+
flags.push('--no-chrome')
|
|
257
|
+
}
|
|
258
|
+
|
|
259
|
+
return flags.join(' ')
|
|
260
|
+
}
|
|
261
|
+
|
|
262
|
+
/**
|
|
263
|
+
* Generates a unique teammate name by checking existing team members.
|
|
264
|
+
* If the name already exists, appends a numeric suffix (e.g., tester-2, tester-3).
|
|
265
|
+
* @internal Exported for testing
|
|
266
|
+
*/
|
|
267
|
+
export async function generateUniqueTeammateName(
|
|
268
|
+
baseName: string,
|
|
269
|
+
teamName: string | undefined,
|
|
270
|
+
): Promise<string> {
|
|
271
|
+
if (!teamName) {
|
|
272
|
+
return baseName
|
|
273
|
+
}
|
|
274
|
+
|
|
275
|
+
const teamFile = await readTeamFileAsync(teamName)
|
|
276
|
+
if (!teamFile) {
|
|
277
|
+
return baseName
|
|
278
|
+
}
|
|
279
|
+
|
|
280
|
+
const existingNames = new Set(teamFile.members.map(m => m.name.toLowerCase()))
|
|
281
|
+
|
|
282
|
+
// If the base name doesn't exist, use it as-is
|
|
283
|
+
if (!existingNames.has(baseName.toLowerCase())) {
|
|
284
|
+
return baseName
|
|
285
|
+
}
|
|
286
|
+
|
|
287
|
+
// Find the next available suffix
|
|
288
|
+
let suffix = 2
|
|
289
|
+
while (existingNames.has(`${baseName}-${suffix}`.toLowerCase())) {
|
|
290
|
+
suffix++
|
|
291
|
+
}
|
|
292
|
+
|
|
293
|
+
return `${baseName}-${suffix}`
|
|
294
|
+
}
|
|
295
|
+
|
|
296
|
+
// ============================================================================
|
|
297
|
+
// Spawn Handlers
|
|
298
|
+
// ============================================================================
|
|
299
|
+
|
|
300
|
+
/**
|
|
301
|
+
* Handle spawn operation using split-pane view (default).
|
|
302
|
+
* When inside tmux: Creates teammates in a shared window with leader on left, teammates on right.
|
|
303
|
+
* When outside tmux: Creates a claude-swarm session with all teammates in a tiled layout.
|
|
304
|
+
*/
|
|
305
|
+
async function handleSpawnSplitPane(
|
|
306
|
+
input: SpawnInput,
|
|
307
|
+
context: ToolUseContext,
|
|
308
|
+
): Promise<{ data: SpawnOutput }> {
|
|
309
|
+
const { setAppState, getAppState } = context
|
|
310
|
+
const { name, prompt, agent_type, cwd, plan_mode_required } = input
|
|
311
|
+
|
|
312
|
+
// Resolve model: 'inherit' → leader's model; undefined → default Opus
|
|
313
|
+
const model = resolveTeammateModel(input.model, getAppState().mainLoopModel)
|
|
314
|
+
|
|
315
|
+
if (!name || !prompt) {
|
|
316
|
+
throw new Error('name and prompt are required for spawn operation')
|
|
317
|
+
}
|
|
318
|
+
|
|
319
|
+
// Get team name from input or inherit from leader's team context
|
|
320
|
+
const appState = getAppState()
|
|
321
|
+
const teamName = input.team_name || appState.teamContext?.teamName
|
|
322
|
+
|
|
323
|
+
if (!teamName) {
|
|
324
|
+
throw new Error(
|
|
325
|
+
'team_name is required for spawn operation. Either provide team_name in input or call spawnTeam first to establish team context.',
|
|
326
|
+
)
|
|
327
|
+
}
|
|
328
|
+
|
|
329
|
+
// Generate unique name if duplicate exists in team
|
|
330
|
+
const uniqueName = await generateUniqueTeammateName(name, teamName)
|
|
331
|
+
|
|
332
|
+
// Sanitize the name to prevent @ in agent IDs (would break agentName@teamName format)
|
|
333
|
+
const sanitizedName = sanitizeAgentName(uniqueName)
|
|
334
|
+
|
|
335
|
+
// Generate deterministic agent ID from name and team
|
|
336
|
+
const teammateId = formatAgentId(sanitizedName, teamName)
|
|
337
|
+
const workingDir = cwd || getCwd()
|
|
338
|
+
|
|
339
|
+
// Detect the appropriate backend and check if setup is needed
|
|
340
|
+
let detectionResult = await detectAndGetBackend()
|
|
341
|
+
|
|
342
|
+
// If in iTerm2 but it2 isn't set up, prompt the user
|
|
343
|
+
if (detectionResult.needsIt2Setup && context.setToolJSX) {
|
|
344
|
+
const tmuxAvailable = await isTmuxAvailable()
|
|
345
|
+
|
|
346
|
+
// Show the setup prompt and wait for user decision
|
|
347
|
+
const setupResult = await new Promise<
|
|
348
|
+
'installed' | 'use-tmux' | 'cancelled'
|
|
349
|
+
>(resolve => {
|
|
350
|
+
context.setToolJSX!({
|
|
351
|
+
jsx: React.createElement(It2SetupPrompt, {
|
|
352
|
+
onDone: resolve,
|
|
353
|
+
tmuxAvailable,
|
|
354
|
+
}),
|
|
355
|
+
shouldHidePromptInput: true,
|
|
356
|
+
})
|
|
357
|
+
})
|
|
358
|
+
|
|
359
|
+
// Clear the JSX
|
|
360
|
+
context.setToolJSX(null)
|
|
361
|
+
|
|
362
|
+
if (setupResult === 'cancelled') {
|
|
363
|
+
throw new Error('Teammate spawn cancelled - iTerm2 setup required')
|
|
364
|
+
}
|
|
365
|
+
|
|
366
|
+
// If they installed it2 or chose tmux, clear cached detection and re-fetch
|
|
367
|
+
// so the local detectionResult matches the backend that will actually
|
|
368
|
+
// spawn the pane.
|
|
369
|
+
// - 'installed': re-detect to pick up the ITermBackend (it2 is now available)
|
|
370
|
+
// - 'use-tmux': re-detect so needsIt2Setup is false (preferTmux is now saved)
|
|
371
|
+
// and subsequent spawns skip this prompt
|
|
372
|
+
if (setupResult === 'installed' || setupResult === 'use-tmux') {
|
|
373
|
+
resetBackendDetection()
|
|
374
|
+
detectionResult = await detectAndGetBackend()
|
|
375
|
+
}
|
|
376
|
+
}
|
|
377
|
+
|
|
378
|
+
// Check if we're inside tmux to determine session naming
|
|
379
|
+
const insideTmux = await isInsideTmux()
|
|
380
|
+
|
|
381
|
+
// Assign a unique color to this teammate
|
|
382
|
+
const teammateColor = assignTeammateColor(teammateId)
|
|
383
|
+
|
|
384
|
+
// Create a pane in the swarm view
|
|
385
|
+
// - Inside tmux: splits current window (leader on left, teammates on right)
|
|
386
|
+
// - In iTerm2 with it2: uses native iTerm2 split panes
|
|
387
|
+
// - Outside both: creates claude-swarm session with tiled teammates
|
|
388
|
+
const { paneId, isFirstTeammate } = await createTeammatePaneInSwarmView(
|
|
389
|
+
sanitizedName,
|
|
390
|
+
teammateColor,
|
|
391
|
+
)
|
|
392
|
+
|
|
393
|
+
// Enable pane border status on first teammate when inside tmux
|
|
394
|
+
// (outside tmux, this is handled in createTeammatePaneInSwarmView)
|
|
395
|
+
if (isFirstTeammate && insideTmux) {
|
|
396
|
+
await enablePaneBorderStatus()
|
|
397
|
+
}
|
|
398
|
+
|
|
399
|
+
// Build the command to spawn Claude Code with teammate identity
|
|
400
|
+
// Note: We spawn without a prompt - initial instructions are sent via mailbox
|
|
401
|
+
const binaryPath = getTeammateCommand()
|
|
402
|
+
|
|
403
|
+
// Build teammate identity CLI args (replaces CLAUDE_CODE_* env vars)
|
|
404
|
+
const teammateArgs = [
|
|
405
|
+
`--agent-id ${quote([teammateId])}`,
|
|
406
|
+
`--agent-name ${quote([sanitizedName])}`,
|
|
407
|
+
`--team-name ${quote([teamName])}`,
|
|
408
|
+
`--agent-color ${quote([teammateColor])}`,
|
|
409
|
+
`--parent-session-id ${quote([getSessionId()])}`,
|
|
410
|
+
plan_mode_required ? '--plan-mode-required' : '',
|
|
411
|
+
agent_type ? `--agent-type ${quote([agent_type])}` : '',
|
|
412
|
+
]
|
|
413
|
+
.filter(Boolean)
|
|
414
|
+
.join(' ')
|
|
415
|
+
|
|
416
|
+
// Build CLI flags to propagate to teammate
|
|
417
|
+
// Pass plan_mode_required to prevent inheriting bypass permissions
|
|
418
|
+
let inheritedFlags = buildInheritedCliFlags({
|
|
419
|
+
planModeRequired: plan_mode_required,
|
|
420
|
+
permissionMode: appState.toolPermissionContext.mode,
|
|
421
|
+
})
|
|
422
|
+
|
|
423
|
+
// If teammate has a custom model, add --model flag (or replace inherited one)
|
|
424
|
+
if (model) {
|
|
425
|
+
// Remove any inherited --model flag first
|
|
426
|
+
inheritedFlags = inheritedFlags
|
|
427
|
+
.split(' ')
|
|
428
|
+
.filter((flag, i, arr) => flag !== '--model' && arr[i - 1] !== '--model')
|
|
429
|
+
.join(' ')
|
|
430
|
+
// Add the teammate's model
|
|
431
|
+
inheritedFlags = inheritedFlags
|
|
432
|
+
? `${inheritedFlags} --model ${quote([model])}`
|
|
433
|
+
: `--model ${quote([model])}`
|
|
434
|
+
}
|
|
435
|
+
|
|
436
|
+
const flagsStr = inheritedFlags ? ` ${inheritedFlags}` : ''
|
|
437
|
+
// Propagate env vars that teammates need but may not inherit from tmux split-window shells.
|
|
438
|
+
// Includes CLAUDECODE, CLAUDE_CODE_EXPERIMENTAL_AGENT_TEAMS, and API provider vars.
|
|
439
|
+
const envStr = buildInheritedEnvVars()
|
|
440
|
+
const spawnCommand = `cd ${quote([workingDir])} && env ${envStr} ${quote([binaryPath])} ${teammateArgs}${flagsStr}`
|
|
441
|
+
|
|
442
|
+
// Send the command to the new pane
|
|
443
|
+
// Use swarm socket when running outside tmux (external swarm session)
|
|
444
|
+
await sendCommandToPane(paneId, spawnCommand, !insideTmux)
|
|
445
|
+
|
|
446
|
+
// Determine session/window names for output
|
|
447
|
+
const sessionName = insideTmux ? 'current' : SWARM_SESSION_NAME
|
|
448
|
+
const windowName = insideTmux ? 'current' : 'swarm-view'
|
|
449
|
+
|
|
450
|
+
// Track the teammate in AppState's teamContext with color
|
|
451
|
+
// If spawning without spawnTeam, set up the leader as team lead
|
|
452
|
+
setAppState(prev => ({
|
|
453
|
+
...prev,
|
|
454
|
+
teamContext: {
|
|
455
|
+
...prev.teamContext,
|
|
456
|
+
teamName: teamName ?? prev.teamContext?.teamName ?? 'default',
|
|
457
|
+
teamFilePath: prev.teamContext?.teamFilePath ?? '',
|
|
458
|
+
leadAgentId: prev.teamContext?.leadAgentId ?? '',
|
|
459
|
+
teammates: {
|
|
460
|
+
...(prev.teamContext?.teammates || {}),
|
|
461
|
+
[teammateId]: {
|
|
462
|
+
name: sanitizedName,
|
|
463
|
+
agentType: agent_type,
|
|
464
|
+
color: teammateColor,
|
|
465
|
+
tmuxSessionName: sessionName,
|
|
466
|
+
tmuxPaneId: paneId,
|
|
467
|
+
cwd: workingDir,
|
|
468
|
+
spawnedAt: Date.now(),
|
|
469
|
+
},
|
|
470
|
+
},
|
|
471
|
+
},
|
|
472
|
+
}))
|
|
473
|
+
|
|
474
|
+
// Register background task so teammates appear in the tasks pill/dialog
|
|
475
|
+
registerOutOfProcessTeammateTask(setAppState, {
|
|
476
|
+
teammateId,
|
|
477
|
+
sanitizedName,
|
|
478
|
+
teamName,
|
|
479
|
+
teammateColor,
|
|
480
|
+
prompt,
|
|
481
|
+
plan_mode_required,
|
|
482
|
+
paneId,
|
|
483
|
+
insideTmux,
|
|
484
|
+
backendType: detectionResult.backend.type,
|
|
485
|
+
toolUseId: context.toolUseId,
|
|
486
|
+
})
|
|
487
|
+
|
|
488
|
+
// Register agent in the team file
|
|
489
|
+
const teamFile = await readTeamFileAsync(teamName)
|
|
490
|
+
if (!teamFile) {
|
|
491
|
+
throw new Error(
|
|
492
|
+
`Team "${teamName}" does not exist. Call spawnTeam first to create the team.`,
|
|
493
|
+
)
|
|
494
|
+
}
|
|
495
|
+
teamFile.members.push({
|
|
496
|
+
agentId: teammateId,
|
|
497
|
+
name: sanitizedName,
|
|
498
|
+
agentType: agent_type,
|
|
499
|
+
model,
|
|
500
|
+
prompt,
|
|
501
|
+
color: teammateColor,
|
|
502
|
+
planModeRequired: plan_mode_required,
|
|
503
|
+
joinedAt: Date.now(),
|
|
504
|
+
tmuxPaneId: paneId,
|
|
505
|
+
cwd: workingDir,
|
|
506
|
+
subscriptions: [],
|
|
507
|
+
backendType: detectionResult.backend.type,
|
|
508
|
+
})
|
|
509
|
+
await writeTeamFileAsync(teamName, teamFile)
|
|
510
|
+
|
|
511
|
+
// Send initial instructions to teammate via mailbox
|
|
512
|
+
// The teammate's inbox poller will pick this up and submit it as their first turn
|
|
513
|
+
await writeToMailbox(
|
|
514
|
+
sanitizedName,
|
|
515
|
+
{
|
|
516
|
+
from: TEAM_LEAD_NAME,
|
|
517
|
+
text: prompt,
|
|
518
|
+
timestamp: new Date().toISOString(),
|
|
519
|
+
},
|
|
520
|
+
teamName,
|
|
521
|
+
)
|
|
522
|
+
|
|
523
|
+
return {
|
|
524
|
+
data: {
|
|
525
|
+
teammate_id: teammateId,
|
|
526
|
+
agent_id: teammateId,
|
|
527
|
+
agent_type,
|
|
528
|
+
model,
|
|
529
|
+
name: sanitizedName,
|
|
530
|
+
color: teammateColor,
|
|
531
|
+
tmux_session_name: sessionName,
|
|
532
|
+
tmux_window_name: windowName,
|
|
533
|
+
tmux_pane_id: paneId,
|
|
534
|
+
team_name: teamName,
|
|
535
|
+
is_splitpane: true,
|
|
536
|
+
plan_mode_required,
|
|
537
|
+
},
|
|
538
|
+
}
|
|
539
|
+
}
|
|
540
|
+
|
|
541
|
+
/**
|
|
542
|
+
* Handle spawn operation using separate windows (legacy behavior).
|
|
543
|
+
* Creates each teammate in its own tmux window.
|
|
544
|
+
*/
|
|
545
|
+
async function handleSpawnSeparateWindow(
|
|
546
|
+
input: SpawnInput,
|
|
547
|
+
context: ToolUseContext,
|
|
548
|
+
): Promise<{ data: SpawnOutput }> {
|
|
549
|
+
const { setAppState, getAppState } = context
|
|
550
|
+
const { name, prompt, agent_type, cwd, plan_mode_required } = input
|
|
551
|
+
|
|
552
|
+
// Resolve model: 'inherit' → leader's model; undefined → default Opus
|
|
553
|
+
const model = resolveTeammateModel(input.model, getAppState().mainLoopModel)
|
|
554
|
+
|
|
555
|
+
if (!name || !prompt) {
|
|
556
|
+
throw new Error('name and prompt are required for spawn operation')
|
|
557
|
+
}
|
|
558
|
+
|
|
559
|
+
// Get team name from input or inherit from leader's team context
|
|
560
|
+
const appState = getAppState()
|
|
561
|
+
const teamName = input.team_name || appState.teamContext?.teamName
|
|
562
|
+
|
|
563
|
+
if (!teamName) {
|
|
564
|
+
throw new Error(
|
|
565
|
+
'team_name is required for spawn operation. Either provide team_name in input or call spawnTeam first to establish team context.',
|
|
566
|
+
)
|
|
567
|
+
}
|
|
568
|
+
|
|
569
|
+
// Generate unique name if duplicate exists in team
|
|
570
|
+
const uniqueName = await generateUniqueTeammateName(name, teamName)
|
|
571
|
+
|
|
572
|
+
// Sanitize the name to prevent @ in agent IDs (would break agentName@teamName format)
|
|
573
|
+
const sanitizedName = sanitizeAgentName(uniqueName)
|
|
574
|
+
|
|
575
|
+
// Generate deterministic agent ID from name and team
|
|
576
|
+
const teammateId = formatAgentId(sanitizedName, teamName)
|
|
577
|
+
const windowName = `teammate-${sanitizeName(sanitizedName)}`
|
|
578
|
+
const workingDir = cwd || getCwd()
|
|
579
|
+
|
|
580
|
+
// Ensure the swarm session exists
|
|
581
|
+
await ensureSession(SWARM_SESSION_NAME)
|
|
582
|
+
|
|
583
|
+
// Assign a unique color to this teammate
|
|
584
|
+
const teammateColor = assignTeammateColor(teammateId)
|
|
585
|
+
|
|
586
|
+
// Create a new window for this teammate
|
|
587
|
+
const createWindowResult = await execFileNoThrow(TMUX_COMMAND, [
|
|
588
|
+
'new-window',
|
|
589
|
+
'-t',
|
|
590
|
+
SWARM_SESSION_NAME,
|
|
591
|
+
'-n',
|
|
592
|
+
windowName,
|
|
593
|
+
'-P',
|
|
594
|
+
'-F',
|
|
595
|
+
'#{pane_id}',
|
|
596
|
+
])
|
|
597
|
+
|
|
598
|
+
if (createWindowResult.code !== 0) {
|
|
599
|
+
throw new Error(
|
|
600
|
+
`Failed to create tmux window: ${createWindowResult.stderr}`,
|
|
601
|
+
)
|
|
602
|
+
}
|
|
603
|
+
|
|
604
|
+
const paneId = createWindowResult.stdout.trim()
|
|
605
|
+
|
|
606
|
+
// Build the command to spawn Claude Code with teammate identity
|
|
607
|
+
// Note: We spawn without a prompt - initial instructions are sent via mailbox
|
|
608
|
+
const binaryPath = getTeammateCommand()
|
|
609
|
+
|
|
610
|
+
// Build teammate identity CLI args (replaces CLAUDE_CODE_* env vars)
|
|
611
|
+
const teammateArgs = [
|
|
612
|
+
`--agent-id ${quote([teammateId])}`,
|
|
613
|
+
`--agent-name ${quote([sanitizedName])}`,
|
|
614
|
+
`--team-name ${quote([teamName])}`,
|
|
615
|
+
`--agent-color ${quote([teammateColor])}`,
|
|
616
|
+
`--parent-session-id ${quote([getSessionId()])}`,
|
|
617
|
+
plan_mode_required ? '--plan-mode-required' : '',
|
|
618
|
+
agent_type ? `--agent-type ${quote([agent_type])}` : '',
|
|
619
|
+
]
|
|
620
|
+
.filter(Boolean)
|
|
621
|
+
.join(' ')
|
|
622
|
+
|
|
623
|
+
// Build CLI flags to propagate to teammate
|
|
624
|
+
// Pass plan_mode_required to prevent inheriting bypass permissions
|
|
625
|
+
let inheritedFlags = buildInheritedCliFlags({
|
|
626
|
+
planModeRequired: plan_mode_required,
|
|
627
|
+
permissionMode: appState.toolPermissionContext.mode,
|
|
628
|
+
})
|
|
629
|
+
|
|
630
|
+
// If teammate has a custom model, add --model flag (or replace inherited one)
|
|
631
|
+
if (model) {
|
|
632
|
+
// Remove any inherited --model flag first
|
|
633
|
+
inheritedFlags = inheritedFlags
|
|
634
|
+
.split(' ')
|
|
635
|
+
.filter((flag, i, arr) => flag !== '--model' && arr[i - 1] !== '--model')
|
|
636
|
+
.join(' ')
|
|
637
|
+
// Add the teammate's model
|
|
638
|
+
inheritedFlags = inheritedFlags
|
|
639
|
+
? `${inheritedFlags} --model ${quote([model])}`
|
|
640
|
+
: `--model ${quote([model])}`
|
|
641
|
+
}
|
|
642
|
+
|
|
643
|
+
const flagsStr = inheritedFlags ? ` ${inheritedFlags}` : ''
|
|
644
|
+
// Propagate env vars that teammates need but may not inherit from tmux split-window shells.
|
|
645
|
+
// Includes CLAUDECODE, CLAUDE_CODE_EXPERIMENTAL_AGENT_TEAMS, and API provider vars.
|
|
646
|
+
const envStr = buildInheritedEnvVars()
|
|
647
|
+
const spawnCommand = `cd ${quote([workingDir])} && env ${envStr} ${quote([binaryPath])} ${teammateArgs}${flagsStr}`
|
|
648
|
+
|
|
649
|
+
// Send the command to the new window
|
|
650
|
+
const sendKeysResult = await execFileNoThrow(TMUX_COMMAND, [
|
|
651
|
+
'send-keys',
|
|
652
|
+
'-t',
|
|
653
|
+
`${SWARM_SESSION_NAME}:${windowName}`,
|
|
654
|
+
spawnCommand,
|
|
655
|
+
'Enter',
|
|
656
|
+
])
|
|
657
|
+
|
|
658
|
+
if (sendKeysResult.code !== 0) {
|
|
659
|
+
throw new Error(
|
|
660
|
+
`Failed to send command to tmux window: ${sendKeysResult.stderr}`,
|
|
661
|
+
)
|
|
662
|
+
}
|
|
663
|
+
|
|
664
|
+
// Track the teammate in AppState's teamContext
|
|
665
|
+
setAppState(prev => ({
|
|
666
|
+
...prev,
|
|
667
|
+
teamContext: {
|
|
668
|
+
...prev.teamContext,
|
|
669
|
+
teamName: teamName ?? prev.teamContext?.teamName ?? 'default',
|
|
670
|
+
teamFilePath: prev.teamContext?.teamFilePath ?? '',
|
|
671
|
+
leadAgentId: prev.teamContext?.leadAgentId ?? '',
|
|
672
|
+
teammates: {
|
|
673
|
+
...(prev.teamContext?.teammates || {}),
|
|
674
|
+
[teammateId]: {
|
|
675
|
+
name: sanitizedName,
|
|
676
|
+
agentType: agent_type,
|
|
677
|
+
color: teammateColor,
|
|
678
|
+
tmuxSessionName: SWARM_SESSION_NAME,
|
|
679
|
+
tmuxPaneId: paneId,
|
|
680
|
+
cwd: workingDir,
|
|
681
|
+
spawnedAt: Date.now(),
|
|
682
|
+
},
|
|
683
|
+
},
|
|
684
|
+
},
|
|
685
|
+
}))
|
|
686
|
+
|
|
687
|
+
// Register background task so tmux teammates appear in the tasks pill/dialog
|
|
688
|
+
// Separate window spawns are always outside tmux (external swarm session)
|
|
689
|
+
registerOutOfProcessTeammateTask(setAppState, {
|
|
690
|
+
teammateId,
|
|
691
|
+
sanitizedName,
|
|
692
|
+
teamName,
|
|
693
|
+
teammateColor,
|
|
694
|
+
prompt,
|
|
695
|
+
plan_mode_required,
|
|
696
|
+
paneId,
|
|
697
|
+
insideTmux: false,
|
|
698
|
+
backendType: 'tmux',
|
|
699
|
+
toolUseId: context.toolUseId,
|
|
700
|
+
})
|
|
701
|
+
|
|
702
|
+
// Register agent in the team file
|
|
703
|
+
const teamFile = await readTeamFileAsync(teamName)
|
|
704
|
+
if (!teamFile) {
|
|
705
|
+
throw new Error(
|
|
706
|
+
`Team "${teamName}" does not exist. Call spawnTeam first to create the team.`,
|
|
707
|
+
)
|
|
708
|
+
}
|
|
709
|
+
teamFile.members.push({
|
|
710
|
+
agentId: teammateId,
|
|
711
|
+
name: sanitizedName,
|
|
712
|
+
agentType: agent_type,
|
|
713
|
+
model,
|
|
714
|
+
prompt,
|
|
715
|
+
color: teammateColor,
|
|
716
|
+
planModeRequired: plan_mode_required,
|
|
717
|
+
joinedAt: Date.now(),
|
|
718
|
+
tmuxPaneId: paneId,
|
|
719
|
+
cwd: workingDir,
|
|
720
|
+
subscriptions: [],
|
|
721
|
+
backendType: 'tmux', // This handler always uses tmux directly
|
|
722
|
+
})
|
|
723
|
+
await writeTeamFileAsync(teamName, teamFile)
|
|
724
|
+
|
|
725
|
+
// Send initial instructions to teammate via mailbox
|
|
726
|
+
// The teammate's inbox poller will pick this up and submit it as their first turn
|
|
727
|
+
await writeToMailbox(
|
|
728
|
+
sanitizedName,
|
|
729
|
+
{
|
|
730
|
+
from: TEAM_LEAD_NAME,
|
|
731
|
+
text: prompt,
|
|
732
|
+
timestamp: new Date().toISOString(),
|
|
733
|
+
},
|
|
734
|
+
teamName,
|
|
735
|
+
)
|
|
736
|
+
|
|
737
|
+
return {
|
|
738
|
+
data: {
|
|
739
|
+
teammate_id: teammateId,
|
|
740
|
+
agent_id: teammateId,
|
|
741
|
+
agent_type,
|
|
742
|
+
model,
|
|
743
|
+
name: sanitizedName,
|
|
744
|
+
color: teammateColor,
|
|
745
|
+
tmux_session_name: SWARM_SESSION_NAME,
|
|
746
|
+
tmux_window_name: windowName,
|
|
747
|
+
tmux_pane_id: paneId,
|
|
748
|
+
team_name: teamName,
|
|
749
|
+
is_splitpane: false,
|
|
750
|
+
plan_mode_required,
|
|
751
|
+
},
|
|
752
|
+
}
|
|
753
|
+
}
|
|
754
|
+
|
|
755
|
+
/**
|
|
756
|
+
* Register a background task entry for an out-of-process (tmux/iTerm2) teammate.
|
|
757
|
+
* This makes tmux teammates visible in the background tasks pill and dialog,
|
|
758
|
+
* matching how in-process teammates are tracked.
|
|
759
|
+
*/
|
|
760
|
+
function registerOutOfProcessTeammateTask(
|
|
761
|
+
setAppState: (updater: (prev: AppState) => AppState) => void,
|
|
762
|
+
{
|
|
763
|
+
teammateId,
|
|
764
|
+
sanitizedName,
|
|
765
|
+
teamName,
|
|
766
|
+
teammateColor,
|
|
767
|
+
prompt,
|
|
768
|
+
plan_mode_required,
|
|
769
|
+
paneId,
|
|
770
|
+
insideTmux,
|
|
771
|
+
backendType,
|
|
772
|
+
toolUseId,
|
|
773
|
+
}: {
|
|
774
|
+
teammateId: string
|
|
775
|
+
sanitizedName: string
|
|
776
|
+
teamName: string
|
|
777
|
+
teammateColor: string
|
|
778
|
+
prompt: string
|
|
779
|
+
plan_mode_required?: boolean
|
|
780
|
+
paneId: string
|
|
781
|
+
insideTmux: boolean
|
|
782
|
+
backendType: BackendType
|
|
783
|
+
toolUseId?: string
|
|
784
|
+
},
|
|
785
|
+
): void {
|
|
786
|
+
const taskId = generateTaskId('in_process_teammate')
|
|
787
|
+
const description = `${sanitizedName}: ${prompt.substring(0, 50)}${prompt.length > 50 ? '...' : ''}`
|
|
788
|
+
|
|
789
|
+
const abortController = new AbortController()
|
|
790
|
+
|
|
791
|
+
const taskState: InProcessTeammateTaskState = {
|
|
792
|
+
...createTaskStateBase(
|
|
793
|
+
taskId,
|
|
794
|
+
'in_process_teammate',
|
|
795
|
+
description,
|
|
796
|
+
toolUseId,
|
|
797
|
+
),
|
|
798
|
+
type: 'in_process_teammate',
|
|
799
|
+
status: 'running',
|
|
800
|
+
identity: {
|
|
801
|
+
agentId: teammateId,
|
|
802
|
+
agentName: sanitizedName,
|
|
803
|
+
teamName,
|
|
804
|
+
color: teammateColor,
|
|
805
|
+
planModeRequired: plan_mode_required ?? false,
|
|
806
|
+
parentSessionId: getSessionId(),
|
|
807
|
+
},
|
|
808
|
+
prompt,
|
|
809
|
+
abortController,
|
|
810
|
+
awaitingPlanApproval: false,
|
|
811
|
+
permissionMode: plan_mode_required ? 'plan' : 'default',
|
|
812
|
+
isIdle: false,
|
|
813
|
+
shutdownRequested: false,
|
|
814
|
+
lastReportedToolCount: 0,
|
|
815
|
+
lastReportedTokenCount: 0,
|
|
816
|
+
pendingUserMessages: [],
|
|
817
|
+
}
|
|
818
|
+
|
|
819
|
+
registerTask(taskState, setAppState)
|
|
820
|
+
|
|
821
|
+
// When abort is signaled, kill the pane using the backend that created it
|
|
822
|
+
// (tmux kill-pane for tmux panes, it2 session close for iTerm2 native panes).
|
|
823
|
+
// SDK task_notification bookend is emitted by killInProcessTeammate (the
|
|
824
|
+
// sole abort trigger for this controller).
|
|
825
|
+
abortController.signal.addEventListener(
|
|
826
|
+
'abort',
|
|
827
|
+
() => {
|
|
828
|
+
if (isPaneBackend(backendType)) {
|
|
829
|
+
void getBackendByType(backendType).killPane(paneId, !insideTmux)
|
|
830
|
+
}
|
|
831
|
+
},
|
|
832
|
+
{ once: true },
|
|
833
|
+
)
|
|
834
|
+
}
|
|
835
|
+
|
|
836
|
+
/**
|
|
837
|
+
* Handle spawn operation for in-process teammates.
|
|
838
|
+
* In-process teammates run in the same Node.js process using AsyncLocalStorage.
|
|
839
|
+
*/
|
|
840
|
+
async function handleSpawnInProcess(
|
|
841
|
+
input: SpawnInput,
|
|
842
|
+
context: ToolUseContext,
|
|
843
|
+
): Promise<{ data: SpawnOutput }> {
|
|
844
|
+
const { setAppState, getAppState } = context
|
|
845
|
+
const { name, prompt, agent_type, plan_mode_required } = input
|
|
846
|
+
|
|
847
|
+
// Resolve model: 'inherit' → leader's model; undefined → default Opus
|
|
848
|
+
const model = resolveTeammateModel(input.model, getAppState().mainLoopModel)
|
|
849
|
+
|
|
850
|
+
if (!name || !prompt) {
|
|
851
|
+
throw new Error('name and prompt are required for spawn operation')
|
|
852
|
+
}
|
|
853
|
+
|
|
854
|
+
// Get team name from input or inherit from leader's team context
|
|
855
|
+
const appState = getAppState()
|
|
856
|
+
const teamName = input.team_name || appState.teamContext?.teamName
|
|
857
|
+
|
|
858
|
+
if (!teamName) {
|
|
859
|
+
throw new Error(
|
|
860
|
+
'team_name is required for spawn operation. Either provide team_name in input or call spawnTeam first to establish team context.',
|
|
861
|
+
)
|
|
862
|
+
}
|
|
863
|
+
|
|
864
|
+
// Generate unique name if duplicate exists in team
|
|
865
|
+
const uniqueName = await generateUniqueTeammateName(name, teamName)
|
|
866
|
+
|
|
867
|
+
// Sanitize the name to prevent @ in agent IDs
|
|
868
|
+
const sanitizedName = sanitizeAgentName(uniqueName)
|
|
869
|
+
|
|
870
|
+
// Generate deterministic agent ID from name and team
|
|
871
|
+
const teammateId = formatAgentId(sanitizedName, teamName)
|
|
872
|
+
|
|
873
|
+
// Assign a unique color to this teammate
|
|
874
|
+
const teammateColor = assignTeammateColor(teammateId)
|
|
875
|
+
|
|
876
|
+
// Look up custom agent definition if agent_type is provided
|
|
877
|
+
let agentDefinition: CustomAgentDefinition | undefined
|
|
878
|
+
if (agent_type) {
|
|
879
|
+
const allAgents = context.options.agentDefinitions.activeAgents
|
|
880
|
+
const foundAgent = allAgents.find(a => a.agentType === agent_type)
|
|
881
|
+
if (foundAgent && isCustomAgent(foundAgent)) {
|
|
882
|
+
agentDefinition = foundAgent
|
|
883
|
+
}
|
|
884
|
+
logForDebugging(
|
|
885
|
+
`[handleSpawnInProcess] agent_type=${agent_type}, found=${!!agentDefinition}`,
|
|
886
|
+
)
|
|
887
|
+
}
|
|
888
|
+
|
|
889
|
+
// Spawn in-process teammate
|
|
890
|
+
const config: InProcessSpawnConfig = {
|
|
891
|
+
name: sanitizedName,
|
|
892
|
+
teamName,
|
|
893
|
+
prompt,
|
|
894
|
+
color: teammateColor,
|
|
895
|
+
planModeRequired: plan_mode_required ?? false,
|
|
896
|
+
model,
|
|
897
|
+
}
|
|
898
|
+
|
|
899
|
+
const result = await spawnInProcessTeammate(config, context)
|
|
900
|
+
|
|
901
|
+
if (!result.success) {
|
|
902
|
+
throw new Error(result.error ?? 'Failed to spawn in-process teammate')
|
|
903
|
+
}
|
|
904
|
+
|
|
905
|
+
// Debug: log what spawn returned
|
|
906
|
+
logForDebugging(
|
|
907
|
+
`[handleSpawnInProcess] spawn result: taskId=${result.taskId}, hasContext=${!!result.teammateContext}, hasAbort=${!!result.abortController}`,
|
|
908
|
+
)
|
|
909
|
+
|
|
910
|
+
// Start the agent execution loop (fire-and-forget)
|
|
911
|
+
if (result.taskId && result.teammateContext && result.abortController) {
|
|
912
|
+
startInProcessTeammate({
|
|
913
|
+
identity: {
|
|
914
|
+
agentId: teammateId,
|
|
915
|
+
agentName: sanitizedName,
|
|
916
|
+
teamName,
|
|
917
|
+
color: teammateColor,
|
|
918
|
+
planModeRequired: plan_mode_required ?? false,
|
|
919
|
+
parentSessionId: result.teammateContext.parentSessionId,
|
|
920
|
+
},
|
|
921
|
+
taskId: result.taskId,
|
|
922
|
+
prompt,
|
|
923
|
+
description: input.description,
|
|
924
|
+
model,
|
|
925
|
+
agentDefinition,
|
|
926
|
+
teammateContext: result.teammateContext,
|
|
927
|
+
// Strip messages: the teammate never reads toolUseContext.messages
|
|
928
|
+
// (it builds its own history via allMessages in inProcessRunner).
|
|
929
|
+
// Passing the parent's full conversation here would pin it for the
|
|
930
|
+
// teammate's lifetime, surviving /clear and auto-compact.
|
|
931
|
+
toolUseContext: { ...context, messages: [] },
|
|
932
|
+
abortController: result.abortController,
|
|
933
|
+
invokingRequestId: input.invokingRequestId,
|
|
934
|
+
})
|
|
935
|
+
logForDebugging(
|
|
936
|
+
`[handleSpawnInProcess] Started agent execution for ${teammateId}`,
|
|
937
|
+
)
|
|
938
|
+
}
|
|
939
|
+
|
|
940
|
+
// Track the teammate in AppState's teamContext
|
|
941
|
+
// Auto-register leader if spawning without prior spawnTeam call
|
|
942
|
+
setAppState(prev => {
|
|
943
|
+
const needsLeaderSetup = !prev.teamContext?.leadAgentId
|
|
944
|
+
const leadAgentId = needsLeaderSetup
|
|
945
|
+
? formatAgentId(TEAM_LEAD_NAME, teamName)
|
|
946
|
+
: prev.teamContext!.leadAgentId
|
|
947
|
+
|
|
948
|
+
// Build teammates map, including leader if needed for inbox polling
|
|
949
|
+
const existingTeammates = prev.teamContext?.teammates || {}
|
|
950
|
+
const leadEntry = needsLeaderSetup
|
|
951
|
+
? {
|
|
952
|
+
[leadAgentId]: {
|
|
953
|
+
name: TEAM_LEAD_NAME,
|
|
954
|
+
agentType: TEAM_LEAD_NAME,
|
|
955
|
+
color: assignTeammateColor(leadAgentId),
|
|
956
|
+
tmuxSessionName: 'in-process',
|
|
957
|
+
tmuxPaneId: 'leader',
|
|
958
|
+
cwd: getCwd(),
|
|
959
|
+
spawnedAt: Date.now(),
|
|
960
|
+
},
|
|
961
|
+
}
|
|
962
|
+
: {}
|
|
963
|
+
|
|
964
|
+
return {
|
|
965
|
+
...prev,
|
|
966
|
+
teamContext: {
|
|
967
|
+
...prev.teamContext,
|
|
968
|
+
teamName: teamName ?? prev.teamContext?.teamName ?? 'default',
|
|
969
|
+
teamFilePath: prev.teamContext?.teamFilePath ?? '',
|
|
970
|
+
leadAgentId,
|
|
971
|
+
teammates: {
|
|
972
|
+
...existingTeammates,
|
|
973
|
+
...leadEntry,
|
|
974
|
+
[teammateId]: {
|
|
975
|
+
name: sanitizedName,
|
|
976
|
+
agentType: agent_type,
|
|
977
|
+
color: teammateColor,
|
|
978
|
+
tmuxSessionName: 'in-process',
|
|
979
|
+
tmuxPaneId: 'in-process',
|
|
980
|
+
cwd: getCwd(),
|
|
981
|
+
spawnedAt: Date.now(),
|
|
982
|
+
},
|
|
983
|
+
},
|
|
984
|
+
},
|
|
985
|
+
}
|
|
986
|
+
})
|
|
987
|
+
|
|
988
|
+
// Register agent in the team file
|
|
989
|
+
const teamFile = await readTeamFileAsync(teamName)
|
|
990
|
+
if (!teamFile) {
|
|
991
|
+
throw new Error(
|
|
992
|
+
`Team "${teamName}" does not exist. Call spawnTeam first to create the team.`,
|
|
993
|
+
)
|
|
994
|
+
}
|
|
995
|
+
teamFile.members.push({
|
|
996
|
+
agentId: teammateId,
|
|
997
|
+
name: sanitizedName,
|
|
998
|
+
agentType: agent_type,
|
|
999
|
+
model,
|
|
1000
|
+
prompt,
|
|
1001
|
+
color: teammateColor,
|
|
1002
|
+
planModeRequired: plan_mode_required,
|
|
1003
|
+
joinedAt: Date.now(),
|
|
1004
|
+
tmuxPaneId: 'in-process',
|
|
1005
|
+
cwd: getCwd(),
|
|
1006
|
+
subscriptions: [],
|
|
1007
|
+
backendType: 'in-process',
|
|
1008
|
+
})
|
|
1009
|
+
await writeTeamFileAsync(teamName, teamFile)
|
|
1010
|
+
|
|
1011
|
+
// Note: Do NOT send the prompt via mailbox for in-process teammates.
|
|
1012
|
+
// In-process teammates receive the prompt directly via startInProcessTeammate().
|
|
1013
|
+
// The mailbox is only needed for tmux-based teammates which poll for their initial message.
|
|
1014
|
+
// Sending via both paths would cause duplicate welcome messages.
|
|
1015
|
+
|
|
1016
|
+
return {
|
|
1017
|
+
data: {
|
|
1018
|
+
teammate_id: teammateId,
|
|
1019
|
+
agent_id: teammateId,
|
|
1020
|
+
agent_type,
|
|
1021
|
+
model,
|
|
1022
|
+
name: sanitizedName,
|
|
1023
|
+
color: teammateColor,
|
|
1024
|
+
tmux_session_name: 'in-process',
|
|
1025
|
+
tmux_window_name: 'in-process',
|
|
1026
|
+
tmux_pane_id: 'in-process',
|
|
1027
|
+
team_name: teamName,
|
|
1028
|
+
is_splitpane: false,
|
|
1029
|
+
plan_mode_required,
|
|
1030
|
+
},
|
|
1031
|
+
}
|
|
1032
|
+
}
|
|
1033
|
+
|
|
1034
|
+
/**
|
|
1035
|
+
* Handle spawn operation - creates a new Claude Code instance.
|
|
1036
|
+
* Uses in-process mode when enabled, otherwise uses tmux/iTerm2 split-pane view.
|
|
1037
|
+
* Falls back to in-process if pane backend detection fails (e.g., iTerm2 without
|
|
1038
|
+
* it2 CLI or tmux installed).
|
|
1039
|
+
*/
|
|
1040
|
+
async function handleSpawn(
|
|
1041
|
+
input: SpawnInput,
|
|
1042
|
+
context: ToolUseContext,
|
|
1043
|
+
): Promise<{ data: SpawnOutput }> {
|
|
1044
|
+
// Check if in-process mode is enabled via feature flag
|
|
1045
|
+
if (isInProcessEnabled()) {
|
|
1046
|
+
return handleSpawnInProcess(input, context)
|
|
1047
|
+
}
|
|
1048
|
+
|
|
1049
|
+
// Pre-flight: ensure a pane backend is available before attempting pane-based spawn.
|
|
1050
|
+
// This handles auto-mode cases like iTerm2 without it2 or tmux installed, where
|
|
1051
|
+
// isInProcessEnabled() returns false but detectAndGetBackend() has no viable backend.
|
|
1052
|
+
// Narrowly scoped so user cancellation and other spawn errors propagate normally.
|
|
1053
|
+
try {
|
|
1054
|
+
await detectAndGetBackend()
|
|
1055
|
+
} catch (error) {
|
|
1056
|
+
// Only fall back silently in auto mode. If the user explicitly configured
|
|
1057
|
+
// teammateMode: 'tmux', let the error propagate so they see the actionable
|
|
1058
|
+
// install instructions from getTmuxInstallInstructions().
|
|
1059
|
+
if (getTeammateModeFromSnapshot() !== 'auto') {
|
|
1060
|
+
throw error
|
|
1061
|
+
}
|
|
1062
|
+
logForDebugging(
|
|
1063
|
+
`[handleSpawn] No pane backend available, falling back to in-process: ${errorMessage(error)}`,
|
|
1064
|
+
)
|
|
1065
|
+
// Record the fallback so isInProcessEnabled() reflects the actual mode
|
|
1066
|
+
// (fixes banner and other UI that would otherwise show tmux attach commands).
|
|
1067
|
+
markInProcessFallback()
|
|
1068
|
+
return handleSpawnInProcess(input, context)
|
|
1069
|
+
}
|
|
1070
|
+
|
|
1071
|
+
// Backend is available (and now cached) - proceed with pane spawning.
|
|
1072
|
+
// Any errors here (user cancellation, validation, etc.) propagate to the caller.
|
|
1073
|
+
const useSplitPane = input.use_splitpane !== false
|
|
1074
|
+
if (useSplitPane) {
|
|
1075
|
+
return handleSpawnSplitPane(input, context)
|
|
1076
|
+
}
|
|
1077
|
+
return handleSpawnSeparateWindow(input, context)
|
|
1078
|
+
}
|
|
1079
|
+
|
|
1080
|
+
// ============================================================================
|
|
1081
|
+
// Main Export
|
|
1082
|
+
// ============================================================================
|
|
1083
|
+
|
|
1084
|
+
/**
|
|
1085
|
+
* Spawns a new teammate with the given configuration.
|
|
1086
|
+
* This is the main entry point for teammate spawning, used by both TeammateTool and AgentTool.
|
|
1087
|
+
*/
|
|
1088
|
+
export async function spawnTeammate(
|
|
1089
|
+
config: SpawnTeammateConfig,
|
|
1090
|
+
context: ToolUseContext,
|
|
1091
|
+
): Promise<{ data: SpawnOutput }> {
|
|
1092
|
+
return handleSpawn(config, context)
|
|
1093
|
+
}
|