@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,880 @@
|
|
|
1
|
+
import type {
|
|
2
|
+
Base64ImageSource,
|
|
3
|
+
ImageBlockParam,
|
|
4
|
+
} from '@anthropic-ai/sdk/resources/messages.mjs'
|
|
5
|
+
import {
|
|
6
|
+
API_IMAGE_MAX_BASE64_SIZE,
|
|
7
|
+
IMAGE_MAX_HEIGHT,
|
|
8
|
+
IMAGE_MAX_WIDTH,
|
|
9
|
+
IMAGE_TARGET_RAW_SIZE,
|
|
10
|
+
} from '../constants/apiLimits.js'
|
|
11
|
+
import { logEvent } from '../services/analytics/index.js'
|
|
12
|
+
import {
|
|
13
|
+
getImageProcessor,
|
|
14
|
+
type SharpFunction,
|
|
15
|
+
type SharpInstance,
|
|
16
|
+
} from '../tools/FileReadTool/imageProcessor.js'
|
|
17
|
+
import { logForDebugging } from './debug.js'
|
|
18
|
+
import { errorMessage } from './errors.js'
|
|
19
|
+
import { formatFileSize } from './format.js'
|
|
20
|
+
import { logError } from './log.js'
|
|
21
|
+
|
|
22
|
+
type ImageMediaType = 'image/png' | 'image/jpeg' | 'image/gif' | 'image/webp'
|
|
23
|
+
|
|
24
|
+
// Error type constants for analytics (numeric to comply with logEvent restrictions)
|
|
25
|
+
const ERROR_TYPE_MODULE_LOAD = 1
|
|
26
|
+
const ERROR_TYPE_PROCESSING = 2
|
|
27
|
+
const ERROR_TYPE_UNKNOWN = 3
|
|
28
|
+
const ERROR_TYPE_PIXEL_LIMIT = 4
|
|
29
|
+
const ERROR_TYPE_MEMORY = 5
|
|
30
|
+
const ERROR_TYPE_TIMEOUT = 6
|
|
31
|
+
const ERROR_TYPE_VIPS = 7
|
|
32
|
+
const ERROR_TYPE_PERMISSION = 8
|
|
33
|
+
|
|
34
|
+
/**
|
|
35
|
+
* Error thrown when image resizing fails and the image exceeds the API limit.
|
|
36
|
+
*/
|
|
37
|
+
export class ImageResizeError extends Error {
|
|
38
|
+
constructor(message: string) {
|
|
39
|
+
super(message)
|
|
40
|
+
this.name = 'ImageResizeError'
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
/**
|
|
45
|
+
* Classifies image processing errors for analytics.
|
|
46
|
+
*
|
|
47
|
+
* Uses error codes when available (Node.js module errors), falls back to
|
|
48
|
+
* message matching for libraries like sharp that don't expose error codes.
|
|
49
|
+
*/
|
|
50
|
+
function classifyImageError(error: unknown): number {
|
|
51
|
+
// Check for Node.js error codes first (more reliable than string matching)
|
|
52
|
+
if (error instanceof Error) {
|
|
53
|
+
const errorWithCode = error as Error & { code?: string }
|
|
54
|
+
if (
|
|
55
|
+
errorWithCode.code === 'MODULE_NOT_FOUND' ||
|
|
56
|
+
errorWithCode.code === 'ERR_MODULE_NOT_FOUND' ||
|
|
57
|
+
errorWithCode.code === 'ERR_DLOPEN_FAILED'
|
|
58
|
+
) {
|
|
59
|
+
return ERROR_TYPE_MODULE_LOAD
|
|
60
|
+
}
|
|
61
|
+
if (errorWithCode.code === 'EACCES' || errorWithCode.code === 'EPERM') {
|
|
62
|
+
return ERROR_TYPE_PERMISSION
|
|
63
|
+
}
|
|
64
|
+
if (errorWithCode.code === 'ENOMEM') {
|
|
65
|
+
return ERROR_TYPE_MEMORY
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
// Fall back to message matching for errors without codes
|
|
70
|
+
// Note: sharp doesn't expose error codes, so we must match on messages
|
|
71
|
+
const message = errorMessage(error)
|
|
72
|
+
|
|
73
|
+
// Module loading errors from our native wrapper
|
|
74
|
+
if (message.includes('Native image processor module not available')) {
|
|
75
|
+
return ERROR_TYPE_MODULE_LOAD
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
// Sharp/vips processing errors (format detection, corrupt data, etc.)
|
|
79
|
+
if (
|
|
80
|
+
message.includes('unsupported image format') ||
|
|
81
|
+
message.includes('Input buffer') ||
|
|
82
|
+
message.includes('Input file is missing') ||
|
|
83
|
+
message.includes('Input file has corrupt header') ||
|
|
84
|
+
message.includes('corrupt header') ||
|
|
85
|
+
message.includes('corrupt image') ||
|
|
86
|
+
message.includes('premature end') ||
|
|
87
|
+
message.includes('zlib: data error') ||
|
|
88
|
+
message.includes('zero width') ||
|
|
89
|
+
message.includes('zero height')
|
|
90
|
+
) {
|
|
91
|
+
return ERROR_TYPE_PROCESSING
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
// Pixel/dimension limit errors from sharp/vips
|
|
95
|
+
if (
|
|
96
|
+
message.includes('pixel limit') ||
|
|
97
|
+
message.includes('too many pixels') ||
|
|
98
|
+
message.includes('exceeds pixel') ||
|
|
99
|
+
message.includes('image dimensions')
|
|
100
|
+
) {
|
|
101
|
+
return ERROR_TYPE_PIXEL_LIMIT
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
// Memory allocation failures
|
|
105
|
+
if (
|
|
106
|
+
message.includes('out of memory') ||
|
|
107
|
+
message.includes('Cannot allocate') ||
|
|
108
|
+
message.includes('memory allocation')
|
|
109
|
+
) {
|
|
110
|
+
return ERROR_TYPE_MEMORY
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
// Timeout errors
|
|
114
|
+
if (message.includes('timeout') || message.includes('timed out')) {
|
|
115
|
+
return ERROR_TYPE_TIMEOUT
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
// Vips-specific errors (VipsJpeg, VipsPng, VipsWebp, etc.)
|
|
119
|
+
if (message.includes('Vips')) {
|
|
120
|
+
return ERROR_TYPE_VIPS
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
return ERROR_TYPE_UNKNOWN
|
|
124
|
+
}
|
|
125
|
+
|
|
126
|
+
/**
|
|
127
|
+
* Computes a simple numeric hash of a string for analytics grouping.
|
|
128
|
+
* Uses djb2 algorithm, returning a 32-bit unsigned integer.
|
|
129
|
+
*/
|
|
130
|
+
function hashString(str: string): number {
|
|
131
|
+
let hash = 5381
|
|
132
|
+
for (let i = 0; i < str.length; i++) {
|
|
133
|
+
hash = ((hash << 5) + hash + str.charCodeAt(i)) | 0
|
|
134
|
+
}
|
|
135
|
+
return hash >>> 0
|
|
136
|
+
}
|
|
137
|
+
|
|
138
|
+
export type ImageDimensions = {
|
|
139
|
+
originalWidth?: number
|
|
140
|
+
originalHeight?: number
|
|
141
|
+
displayWidth?: number
|
|
142
|
+
displayHeight?: number
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
export interface ResizeResult {
|
|
146
|
+
buffer: Buffer
|
|
147
|
+
mediaType: string
|
|
148
|
+
dimensions?: ImageDimensions
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
interface ImageCompressionContext {
|
|
152
|
+
imageBuffer: Buffer
|
|
153
|
+
metadata: { width?: number; height?: number; format?: string }
|
|
154
|
+
format: string
|
|
155
|
+
maxBytes: number
|
|
156
|
+
originalSize: number
|
|
157
|
+
}
|
|
158
|
+
|
|
159
|
+
interface CompressedImageResult {
|
|
160
|
+
base64: string
|
|
161
|
+
mediaType: Base64ImageSource['media_type']
|
|
162
|
+
originalSize: number
|
|
163
|
+
}
|
|
164
|
+
|
|
165
|
+
/**
|
|
166
|
+
* Extracted from FileReadTool's readImage function
|
|
167
|
+
* Resizes image buffer to meet size and dimension constraints
|
|
168
|
+
*/
|
|
169
|
+
export async function maybeResizeAndDownsampleImageBuffer(
|
|
170
|
+
imageBuffer: Buffer,
|
|
171
|
+
originalSize: number,
|
|
172
|
+
ext: string,
|
|
173
|
+
): Promise<ResizeResult> {
|
|
174
|
+
if (imageBuffer.length === 0) {
|
|
175
|
+
// Empty buffer would fall through the catch block below (sharp throws
|
|
176
|
+
// "Unable to determine image format"), and the fallback's size check
|
|
177
|
+
// `0 ≤ 5MB` would pass it through, yielding an empty base64 string
|
|
178
|
+
// that the API rejects with `image cannot be empty`.
|
|
179
|
+
throw new ImageResizeError('Image file is empty (0 bytes)')
|
|
180
|
+
}
|
|
181
|
+
try {
|
|
182
|
+
const sharp = await getImageProcessor()
|
|
183
|
+
const image = sharp(imageBuffer)
|
|
184
|
+
const metadata = await image.metadata()
|
|
185
|
+
|
|
186
|
+
const mediaType = metadata.format ?? ext
|
|
187
|
+
// Normalize "jpg" to "jpeg" for media type compatibility
|
|
188
|
+
const normalizedMediaType = mediaType === 'jpg' ? 'jpeg' : mediaType
|
|
189
|
+
|
|
190
|
+
// If dimensions aren't available from metadata
|
|
191
|
+
if (!metadata.width || !metadata.height) {
|
|
192
|
+
if (originalSize > IMAGE_TARGET_RAW_SIZE) {
|
|
193
|
+
// Create fresh sharp instance for compression
|
|
194
|
+
const compressedBuffer = await sharp(imageBuffer)
|
|
195
|
+
.jpeg({ quality: 80 })
|
|
196
|
+
.toBuffer()
|
|
197
|
+
return { buffer: compressedBuffer, mediaType: 'jpeg' }
|
|
198
|
+
}
|
|
199
|
+
// Return without dimensions if we can't determine them
|
|
200
|
+
return { buffer: imageBuffer, mediaType: normalizedMediaType }
|
|
201
|
+
}
|
|
202
|
+
|
|
203
|
+
// Store original dimensions (guaranteed to be defined here)
|
|
204
|
+
const originalWidth = metadata.width
|
|
205
|
+
const originalHeight = metadata.height
|
|
206
|
+
|
|
207
|
+
// Calculate dimensions while maintaining aspect ratio
|
|
208
|
+
let width = originalWidth
|
|
209
|
+
let height = originalHeight
|
|
210
|
+
|
|
211
|
+
// Check if the original file just works
|
|
212
|
+
if (
|
|
213
|
+
originalSize <= IMAGE_TARGET_RAW_SIZE &&
|
|
214
|
+
width <= IMAGE_MAX_WIDTH &&
|
|
215
|
+
height <= IMAGE_MAX_HEIGHT
|
|
216
|
+
) {
|
|
217
|
+
return {
|
|
218
|
+
buffer: imageBuffer,
|
|
219
|
+
mediaType: normalizedMediaType,
|
|
220
|
+
dimensions: {
|
|
221
|
+
originalWidth,
|
|
222
|
+
originalHeight,
|
|
223
|
+
displayWidth: width,
|
|
224
|
+
displayHeight: height,
|
|
225
|
+
},
|
|
226
|
+
}
|
|
227
|
+
}
|
|
228
|
+
|
|
229
|
+
const needsDimensionResize =
|
|
230
|
+
width > IMAGE_MAX_WIDTH || height > IMAGE_MAX_HEIGHT
|
|
231
|
+
const isPng = normalizedMediaType === 'png'
|
|
232
|
+
|
|
233
|
+
// If dimensions are within limits but file is too large, try compression first
|
|
234
|
+
// This preserves full resolution when possible
|
|
235
|
+
if (!needsDimensionResize && originalSize > IMAGE_TARGET_RAW_SIZE) {
|
|
236
|
+
// For PNGs, try PNG compression first to preserve transparency
|
|
237
|
+
if (isPng) {
|
|
238
|
+
// Create fresh sharp instance for each compression attempt
|
|
239
|
+
const pngCompressed = await sharp(imageBuffer)
|
|
240
|
+
.png({ compressionLevel: 9, palette: true })
|
|
241
|
+
.toBuffer()
|
|
242
|
+
if (pngCompressed.length <= IMAGE_TARGET_RAW_SIZE) {
|
|
243
|
+
return {
|
|
244
|
+
buffer: pngCompressed,
|
|
245
|
+
mediaType: 'png',
|
|
246
|
+
dimensions: {
|
|
247
|
+
originalWidth,
|
|
248
|
+
originalHeight,
|
|
249
|
+
displayWidth: width,
|
|
250
|
+
displayHeight: height,
|
|
251
|
+
},
|
|
252
|
+
}
|
|
253
|
+
}
|
|
254
|
+
}
|
|
255
|
+
// Try JPEG compression (lossy but much smaller)
|
|
256
|
+
for (const quality of [80, 60, 40, 20]) {
|
|
257
|
+
// Create fresh sharp instance for each attempt
|
|
258
|
+
const compressedBuffer = await sharp(imageBuffer)
|
|
259
|
+
.jpeg({ quality })
|
|
260
|
+
.toBuffer()
|
|
261
|
+
if (compressedBuffer.length <= IMAGE_TARGET_RAW_SIZE) {
|
|
262
|
+
return {
|
|
263
|
+
buffer: compressedBuffer,
|
|
264
|
+
mediaType: 'jpeg',
|
|
265
|
+
dimensions: {
|
|
266
|
+
originalWidth,
|
|
267
|
+
originalHeight,
|
|
268
|
+
displayWidth: width,
|
|
269
|
+
displayHeight: height,
|
|
270
|
+
},
|
|
271
|
+
}
|
|
272
|
+
}
|
|
273
|
+
}
|
|
274
|
+
// Quality reduction alone wasn't enough, fall through to resize
|
|
275
|
+
}
|
|
276
|
+
|
|
277
|
+
// Constrain dimensions if needed
|
|
278
|
+
if (width > IMAGE_MAX_WIDTH) {
|
|
279
|
+
height = Math.round((height * IMAGE_MAX_WIDTH) / width)
|
|
280
|
+
width = IMAGE_MAX_WIDTH
|
|
281
|
+
}
|
|
282
|
+
|
|
283
|
+
if (height > IMAGE_MAX_HEIGHT) {
|
|
284
|
+
width = Math.round((width * IMAGE_MAX_HEIGHT) / height)
|
|
285
|
+
height = IMAGE_MAX_HEIGHT
|
|
286
|
+
}
|
|
287
|
+
|
|
288
|
+
// IMPORTANT: Always create fresh sharp(imageBuffer) instances for each operation.
|
|
289
|
+
// The native image-processor-napi module doesn't properly apply format conversions
|
|
290
|
+
// when reusing a sharp instance after calling toBuffer(). This caused a bug where
|
|
291
|
+
// all compression attempts (PNG, JPEG at various qualities) returned identical sizes.
|
|
292
|
+
logForDebugging(`Resizing to ${width}x${height}`)
|
|
293
|
+
const resizedImageBuffer = await sharp(imageBuffer)
|
|
294
|
+
.resize(width, height, {
|
|
295
|
+
fit: 'inside',
|
|
296
|
+
withoutEnlargement: true,
|
|
297
|
+
})
|
|
298
|
+
.toBuffer()
|
|
299
|
+
|
|
300
|
+
// If still too large after resize, try compression
|
|
301
|
+
if (resizedImageBuffer.length > IMAGE_TARGET_RAW_SIZE) {
|
|
302
|
+
// For PNGs, try PNG compression first to preserve transparency
|
|
303
|
+
if (isPng) {
|
|
304
|
+
const pngCompressed = await sharp(imageBuffer)
|
|
305
|
+
.resize(width, height, {
|
|
306
|
+
fit: 'inside',
|
|
307
|
+
withoutEnlargement: true,
|
|
308
|
+
})
|
|
309
|
+
.png({ compressionLevel: 9, palette: true })
|
|
310
|
+
.toBuffer()
|
|
311
|
+
if (pngCompressed.length <= IMAGE_TARGET_RAW_SIZE) {
|
|
312
|
+
return {
|
|
313
|
+
buffer: pngCompressed,
|
|
314
|
+
mediaType: 'png',
|
|
315
|
+
dimensions: {
|
|
316
|
+
originalWidth,
|
|
317
|
+
originalHeight,
|
|
318
|
+
displayWidth: width,
|
|
319
|
+
displayHeight: height,
|
|
320
|
+
},
|
|
321
|
+
}
|
|
322
|
+
}
|
|
323
|
+
}
|
|
324
|
+
|
|
325
|
+
// Try JPEG with progressively lower quality
|
|
326
|
+
for (const quality of [80, 60, 40, 20]) {
|
|
327
|
+
const compressedBuffer = await sharp(imageBuffer)
|
|
328
|
+
.resize(width, height, {
|
|
329
|
+
fit: 'inside',
|
|
330
|
+
withoutEnlargement: true,
|
|
331
|
+
})
|
|
332
|
+
.jpeg({ quality })
|
|
333
|
+
.toBuffer()
|
|
334
|
+
if (compressedBuffer.length <= IMAGE_TARGET_RAW_SIZE) {
|
|
335
|
+
return {
|
|
336
|
+
buffer: compressedBuffer,
|
|
337
|
+
mediaType: 'jpeg',
|
|
338
|
+
dimensions: {
|
|
339
|
+
originalWidth,
|
|
340
|
+
originalHeight,
|
|
341
|
+
displayWidth: width,
|
|
342
|
+
displayHeight: height,
|
|
343
|
+
},
|
|
344
|
+
}
|
|
345
|
+
}
|
|
346
|
+
}
|
|
347
|
+
// If still too large, resize smaller and compress aggressively
|
|
348
|
+
const smallerWidth = Math.min(width, 1000)
|
|
349
|
+
const smallerHeight = Math.round(
|
|
350
|
+
(height * smallerWidth) / Math.max(width, 1),
|
|
351
|
+
)
|
|
352
|
+
logForDebugging('Still too large, compressing with JPEG')
|
|
353
|
+
const compressedBuffer = await sharp(imageBuffer)
|
|
354
|
+
.resize(smallerWidth, smallerHeight, {
|
|
355
|
+
fit: 'inside',
|
|
356
|
+
withoutEnlargement: true,
|
|
357
|
+
})
|
|
358
|
+
.jpeg({ quality: 20 })
|
|
359
|
+
.toBuffer()
|
|
360
|
+
logForDebugging(`JPEG compressed buffer size: ${compressedBuffer.length}`)
|
|
361
|
+
return {
|
|
362
|
+
buffer: compressedBuffer,
|
|
363
|
+
mediaType: 'jpeg',
|
|
364
|
+
dimensions: {
|
|
365
|
+
originalWidth,
|
|
366
|
+
originalHeight,
|
|
367
|
+
displayWidth: smallerWidth,
|
|
368
|
+
displayHeight: smallerHeight,
|
|
369
|
+
},
|
|
370
|
+
}
|
|
371
|
+
}
|
|
372
|
+
|
|
373
|
+
return {
|
|
374
|
+
buffer: resizedImageBuffer,
|
|
375
|
+
mediaType: normalizedMediaType,
|
|
376
|
+
dimensions: {
|
|
377
|
+
originalWidth,
|
|
378
|
+
originalHeight,
|
|
379
|
+
displayWidth: width,
|
|
380
|
+
displayHeight: height,
|
|
381
|
+
},
|
|
382
|
+
}
|
|
383
|
+
} catch (error) {
|
|
384
|
+
// Log the error and emit analytics event
|
|
385
|
+
logError(error as Error)
|
|
386
|
+
const errorType = classifyImageError(error)
|
|
387
|
+
const errorMsg = errorMessage(error)
|
|
388
|
+
logEvent('tengu_image_resize_failed', {
|
|
389
|
+
original_size_bytes: originalSize,
|
|
390
|
+
error_type: errorType,
|
|
391
|
+
error_message_hash: hashString(errorMsg),
|
|
392
|
+
})
|
|
393
|
+
|
|
394
|
+
// Detect actual format from magic bytes instead of trusting extension
|
|
395
|
+
const detected = detectImageFormatFromBuffer(imageBuffer)
|
|
396
|
+
const normalizedExt = detected.slice(6) // Remove 'image/' prefix
|
|
397
|
+
|
|
398
|
+
// Calculate the base64 size (API limit is on base64-encoded length)
|
|
399
|
+
const base64Size = Math.ceil((originalSize * 4) / 3)
|
|
400
|
+
|
|
401
|
+
// Size-under-5MB does not imply dimensions-under-cap. Don't return the
|
|
402
|
+
// raw buffer if the PNG header says it's oversized — fall through to
|
|
403
|
+
// ImageResizeError instead. PNG sig is 8 bytes, IHDR dims at 16-24.
|
|
404
|
+
const overDim =
|
|
405
|
+
imageBuffer.length >= 24 &&
|
|
406
|
+
imageBuffer[0] === 0x89 &&
|
|
407
|
+
imageBuffer[1] === 0x50 &&
|
|
408
|
+
imageBuffer[2] === 0x4e &&
|
|
409
|
+
imageBuffer[3] === 0x47 &&
|
|
410
|
+
(imageBuffer.readUInt32BE(16) > IMAGE_MAX_WIDTH ||
|
|
411
|
+
imageBuffer.readUInt32BE(20) > IMAGE_MAX_HEIGHT)
|
|
412
|
+
|
|
413
|
+
// If original image's base64 encoding is within API limit, allow it through uncompressed
|
|
414
|
+
if (base64Size <= API_IMAGE_MAX_BASE64_SIZE && !overDim) {
|
|
415
|
+
logEvent('tengu_image_resize_fallback', {
|
|
416
|
+
original_size_bytes: originalSize,
|
|
417
|
+
base64_size_bytes: base64Size,
|
|
418
|
+
error_type: errorType,
|
|
419
|
+
})
|
|
420
|
+
return { buffer: imageBuffer, mediaType: normalizedExt }
|
|
421
|
+
}
|
|
422
|
+
|
|
423
|
+
// Image is too large and we failed to compress it - fail with user-friendly error
|
|
424
|
+
throw new ImageResizeError(
|
|
425
|
+
overDim
|
|
426
|
+
? `Unable to resize image — dimensions exceed the ${IMAGE_MAX_WIDTH}x${IMAGE_MAX_HEIGHT}px limit and image processing failed. ` +
|
|
427
|
+
`Please resize the image to reduce its pixel dimensions.`
|
|
428
|
+
: `Unable to resize image (${formatFileSize(originalSize)} raw, ${formatFileSize(base64Size)} base64). ` +
|
|
429
|
+
`The image exceeds the 5MB API limit and compression failed. ` +
|
|
430
|
+
`Please resize the image manually or use a smaller image.`,
|
|
431
|
+
)
|
|
432
|
+
}
|
|
433
|
+
}
|
|
434
|
+
|
|
435
|
+
export interface ImageBlockWithDimensions {
|
|
436
|
+
block: ImageBlockParam
|
|
437
|
+
dimensions?: ImageDimensions
|
|
438
|
+
}
|
|
439
|
+
|
|
440
|
+
/**
|
|
441
|
+
* Resizes an image content block if needed
|
|
442
|
+
* Takes an image ImageBlockParam and returns a resized version if necessary
|
|
443
|
+
* Also returns dimension information for coordinate mapping
|
|
444
|
+
*/
|
|
445
|
+
export async function maybeResizeAndDownsampleImageBlock(
|
|
446
|
+
imageBlock: ImageBlockParam,
|
|
447
|
+
): Promise<ImageBlockWithDimensions> {
|
|
448
|
+
// Only process base64 images
|
|
449
|
+
if (imageBlock.source.type !== 'base64') {
|
|
450
|
+
return { block: imageBlock }
|
|
451
|
+
}
|
|
452
|
+
|
|
453
|
+
// Decode base64 to buffer
|
|
454
|
+
const imageBuffer = Buffer.from(imageBlock.source.data, 'base64')
|
|
455
|
+
const originalSize = imageBuffer.length
|
|
456
|
+
|
|
457
|
+
// Extract extension from media type
|
|
458
|
+
const mediaType = imageBlock.source.media_type
|
|
459
|
+
const ext = mediaType?.split('/')[1] || 'png'
|
|
460
|
+
|
|
461
|
+
// Resize if needed
|
|
462
|
+
const resized = await maybeResizeAndDownsampleImageBuffer(
|
|
463
|
+
imageBuffer,
|
|
464
|
+
originalSize,
|
|
465
|
+
ext,
|
|
466
|
+
)
|
|
467
|
+
|
|
468
|
+
// Return resized image block with dimension info
|
|
469
|
+
return {
|
|
470
|
+
block: {
|
|
471
|
+
type: 'image',
|
|
472
|
+
source: {
|
|
473
|
+
type: 'base64',
|
|
474
|
+
media_type:
|
|
475
|
+
`image/${resized.mediaType}` as Base64ImageSource['media_type'],
|
|
476
|
+
data: resized.buffer.toString('base64'),
|
|
477
|
+
},
|
|
478
|
+
},
|
|
479
|
+
dimensions: resized.dimensions,
|
|
480
|
+
}
|
|
481
|
+
}
|
|
482
|
+
|
|
483
|
+
/**
|
|
484
|
+
* Compresses an image buffer to fit within a maximum byte size.
|
|
485
|
+
*
|
|
486
|
+
* Uses a multi-strategy fallback approach because simple compression often fails for
|
|
487
|
+
* large screenshots, high-resolution photos, or images with complex gradients. Each
|
|
488
|
+
* strategy is progressively more aggressive to handle edge cases where earlier
|
|
489
|
+
* strategies produce files still exceeding the size limit.
|
|
490
|
+
*
|
|
491
|
+
* Strategy (from FileReadTool):
|
|
492
|
+
* 1. Try to preserve original format (PNG, JPEG, WebP) with progressive resizing
|
|
493
|
+
* 2. For PNG: Use palette optimization and color reduction if needed
|
|
494
|
+
* 3. Last resort: Convert to JPEG with aggressive compression
|
|
495
|
+
*
|
|
496
|
+
* This ensures images fit within context windows while maintaining format when possible.
|
|
497
|
+
*/
|
|
498
|
+
export async function compressImageBuffer(
|
|
499
|
+
imageBuffer: Buffer,
|
|
500
|
+
maxBytes: number = IMAGE_TARGET_RAW_SIZE,
|
|
501
|
+
originalMediaType?: string,
|
|
502
|
+
): Promise<CompressedImageResult> {
|
|
503
|
+
// Extract format from originalMediaType if provided (e.g., "image/png" -> "png")
|
|
504
|
+
const fallbackFormat = originalMediaType?.split('/')[1] || 'jpeg'
|
|
505
|
+
const normalizedFallback = fallbackFormat === 'jpg' ? 'jpeg' : fallbackFormat
|
|
506
|
+
|
|
507
|
+
try {
|
|
508
|
+
const sharp = await getImageProcessor()
|
|
509
|
+
const metadata = await sharp(imageBuffer).metadata()
|
|
510
|
+
const format = metadata.format || normalizedFallback
|
|
511
|
+
const originalSize = imageBuffer.length
|
|
512
|
+
|
|
513
|
+
const context: ImageCompressionContext = {
|
|
514
|
+
imageBuffer,
|
|
515
|
+
metadata,
|
|
516
|
+
format,
|
|
517
|
+
maxBytes,
|
|
518
|
+
originalSize,
|
|
519
|
+
}
|
|
520
|
+
|
|
521
|
+
// If image is already within size limit, return as-is without processing
|
|
522
|
+
if (originalSize <= maxBytes) {
|
|
523
|
+
return createCompressedImageResult(imageBuffer, format, originalSize)
|
|
524
|
+
}
|
|
525
|
+
|
|
526
|
+
// Try progressive resizing with format preservation
|
|
527
|
+
const resizedResult = await tryProgressiveResizing(context, sharp)
|
|
528
|
+
if (resizedResult) {
|
|
529
|
+
return resizedResult
|
|
530
|
+
}
|
|
531
|
+
|
|
532
|
+
// For PNG, try palette optimization
|
|
533
|
+
if (format === 'png') {
|
|
534
|
+
const palettizedResult = await tryPalettePNG(context, sharp)
|
|
535
|
+
if (palettizedResult) {
|
|
536
|
+
return palettizedResult
|
|
537
|
+
}
|
|
538
|
+
}
|
|
539
|
+
|
|
540
|
+
// Try JPEG conversion with moderate compression
|
|
541
|
+
const jpegResult = await tryJPEGConversion(context, 50, sharp)
|
|
542
|
+
if (jpegResult) {
|
|
543
|
+
return jpegResult
|
|
544
|
+
}
|
|
545
|
+
|
|
546
|
+
// Last resort: ultra-compressed JPEG
|
|
547
|
+
return await createUltraCompressedJPEG(context, sharp)
|
|
548
|
+
} catch (error) {
|
|
549
|
+
// Log the error and emit analytics event
|
|
550
|
+
logError(error as Error)
|
|
551
|
+
const errorType = classifyImageError(error)
|
|
552
|
+
const errorMsg = errorMessage(error)
|
|
553
|
+
logEvent('tengu_image_compress_failed', {
|
|
554
|
+
original_size_bytes: imageBuffer.length,
|
|
555
|
+
max_bytes: maxBytes,
|
|
556
|
+
error_type: errorType,
|
|
557
|
+
error_message_hash: hashString(errorMsg),
|
|
558
|
+
})
|
|
559
|
+
|
|
560
|
+
// If original image is within the requested limit, allow it through
|
|
561
|
+
if (imageBuffer.length <= maxBytes) {
|
|
562
|
+
// Detect actual format from magic bytes instead of trusting the provided media type
|
|
563
|
+
const detected = detectImageFormatFromBuffer(imageBuffer)
|
|
564
|
+
return {
|
|
565
|
+
base64: imageBuffer.toString('base64'),
|
|
566
|
+
mediaType: detected,
|
|
567
|
+
originalSize: imageBuffer.length,
|
|
568
|
+
}
|
|
569
|
+
}
|
|
570
|
+
|
|
571
|
+
// Image is too large and compression failed - throw error
|
|
572
|
+
throw new ImageResizeError(
|
|
573
|
+
`Unable to compress image (${formatFileSize(imageBuffer.length)}) to fit within ${formatFileSize(maxBytes)}. ` +
|
|
574
|
+
`Please use a smaller image.`,
|
|
575
|
+
)
|
|
576
|
+
}
|
|
577
|
+
}
|
|
578
|
+
|
|
579
|
+
/**
|
|
580
|
+
* Compresses an image buffer to fit within a token limit.
|
|
581
|
+
* Converts tokens to bytes using the formula: maxBytes = (maxTokens / 0.125) * 0.75
|
|
582
|
+
*/
|
|
583
|
+
export async function compressImageBufferWithTokenLimit(
|
|
584
|
+
imageBuffer: Buffer,
|
|
585
|
+
maxTokens: number,
|
|
586
|
+
originalMediaType?: string,
|
|
587
|
+
): Promise<CompressedImageResult> {
|
|
588
|
+
// Convert token limit to byte limit
|
|
589
|
+
// base64 uses about 4/3 the original size, so we reverse this
|
|
590
|
+
const maxBase64Chars = Math.floor(maxTokens / 0.125)
|
|
591
|
+
const maxBytes = Math.floor(maxBase64Chars * 0.75)
|
|
592
|
+
|
|
593
|
+
return compressImageBuffer(imageBuffer, maxBytes, originalMediaType)
|
|
594
|
+
}
|
|
595
|
+
|
|
596
|
+
/**
|
|
597
|
+
* Compresses an image block to fit within a maximum byte size.
|
|
598
|
+
* Wrapper around compressImageBuffer for ImageBlockParam.
|
|
599
|
+
*/
|
|
600
|
+
export async function compressImageBlock(
|
|
601
|
+
imageBlock: ImageBlockParam,
|
|
602
|
+
maxBytes: number = IMAGE_TARGET_RAW_SIZE,
|
|
603
|
+
): Promise<ImageBlockParam> {
|
|
604
|
+
// Only process base64 images
|
|
605
|
+
if (imageBlock.source.type !== 'base64') {
|
|
606
|
+
return imageBlock
|
|
607
|
+
}
|
|
608
|
+
|
|
609
|
+
// Decode base64 to buffer
|
|
610
|
+
const imageBuffer = Buffer.from(imageBlock.source.data, 'base64')
|
|
611
|
+
|
|
612
|
+
// Check if already within size limit
|
|
613
|
+
if (imageBuffer.length <= maxBytes) {
|
|
614
|
+
return imageBlock
|
|
615
|
+
}
|
|
616
|
+
|
|
617
|
+
// Compress the image
|
|
618
|
+
const compressed = await compressImageBuffer(imageBuffer, maxBytes)
|
|
619
|
+
|
|
620
|
+
return {
|
|
621
|
+
type: 'image',
|
|
622
|
+
source: {
|
|
623
|
+
type: 'base64',
|
|
624
|
+
media_type: compressed.mediaType,
|
|
625
|
+
data: compressed.base64,
|
|
626
|
+
},
|
|
627
|
+
}
|
|
628
|
+
}
|
|
629
|
+
|
|
630
|
+
// Helper functions for compression pipeline
|
|
631
|
+
|
|
632
|
+
function createCompressedImageResult(
|
|
633
|
+
buffer: Buffer,
|
|
634
|
+
mediaType: string,
|
|
635
|
+
originalSize: number,
|
|
636
|
+
): CompressedImageResult {
|
|
637
|
+
const normalizedMediaType = mediaType === 'jpg' ? 'jpeg' : mediaType
|
|
638
|
+
return {
|
|
639
|
+
base64: buffer.toString('base64'),
|
|
640
|
+
mediaType:
|
|
641
|
+
`image/${normalizedMediaType}` as Base64ImageSource['media_type'],
|
|
642
|
+
originalSize,
|
|
643
|
+
}
|
|
644
|
+
}
|
|
645
|
+
|
|
646
|
+
async function tryProgressiveResizing(
|
|
647
|
+
context: ImageCompressionContext,
|
|
648
|
+
sharp: SharpFunction,
|
|
649
|
+
): Promise<CompressedImageResult | null> {
|
|
650
|
+
const scalingFactors = [1.0, 0.75, 0.5, 0.25]
|
|
651
|
+
|
|
652
|
+
for (const scalingFactor of scalingFactors) {
|
|
653
|
+
const newWidth = Math.round(
|
|
654
|
+
(context.metadata.width || 2000) * scalingFactor,
|
|
655
|
+
)
|
|
656
|
+
const newHeight = Math.round(
|
|
657
|
+
(context.metadata.height || 2000) * scalingFactor,
|
|
658
|
+
)
|
|
659
|
+
|
|
660
|
+
let resizedImage = sharp(context.imageBuffer).resize(newWidth, newHeight, {
|
|
661
|
+
fit: 'inside',
|
|
662
|
+
withoutEnlargement: true,
|
|
663
|
+
})
|
|
664
|
+
|
|
665
|
+
// Apply format-specific optimizations
|
|
666
|
+
resizedImage = applyFormatOptimizations(resizedImage, context.format)
|
|
667
|
+
|
|
668
|
+
const resizedBuffer = await resizedImage.toBuffer()
|
|
669
|
+
|
|
670
|
+
if (resizedBuffer.length <= context.maxBytes) {
|
|
671
|
+
return createCompressedImageResult(
|
|
672
|
+
resizedBuffer,
|
|
673
|
+
context.format,
|
|
674
|
+
context.originalSize,
|
|
675
|
+
)
|
|
676
|
+
}
|
|
677
|
+
}
|
|
678
|
+
|
|
679
|
+
return null
|
|
680
|
+
}
|
|
681
|
+
|
|
682
|
+
function applyFormatOptimizations(
|
|
683
|
+
image: SharpInstance,
|
|
684
|
+
format: string,
|
|
685
|
+
): SharpInstance {
|
|
686
|
+
switch (format) {
|
|
687
|
+
case 'png':
|
|
688
|
+
return image.png({
|
|
689
|
+
compressionLevel: 9,
|
|
690
|
+
palette: true,
|
|
691
|
+
})
|
|
692
|
+
case 'jpeg':
|
|
693
|
+
case 'jpg':
|
|
694
|
+
return image.jpeg({ quality: 80 })
|
|
695
|
+
case 'webp':
|
|
696
|
+
return image.webp({ quality: 80 })
|
|
697
|
+
default:
|
|
698
|
+
return image
|
|
699
|
+
}
|
|
700
|
+
}
|
|
701
|
+
|
|
702
|
+
async function tryPalettePNG(
|
|
703
|
+
context: ImageCompressionContext,
|
|
704
|
+
sharp: SharpFunction,
|
|
705
|
+
): Promise<CompressedImageResult | null> {
|
|
706
|
+
const palettePng = await sharp(context.imageBuffer)
|
|
707
|
+
.resize(800, 800, {
|
|
708
|
+
fit: 'inside',
|
|
709
|
+
withoutEnlargement: true,
|
|
710
|
+
})
|
|
711
|
+
.png({
|
|
712
|
+
compressionLevel: 9,
|
|
713
|
+
palette: true,
|
|
714
|
+
colors: 64, // Reduce colors to 64 for better compression
|
|
715
|
+
})
|
|
716
|
+
.toBuffer()
|
|
717
|
+
|
|
718
|
+
if (palettePng.length <= context.maxBytes) {
|
|
719
|
+
return createCompressedImageResult(palettePng, 'png', context.originalSize)
|
|
720
|
+
}
|
|
721
|
+
|
|
722
|
+
return null
|
|
723
|
+
}
|
|
724
|
+
|
|
725
|
+
async function tryJPEGConversion(
|
|
726
|
+
context: ImageCompressionContext,
|
|
727
|
+
quality: number,
|
|
728
|
+
sharp: SharpFunction,
|
|
729
|
+
): Promise<CompressedImageResult | null> {
|
|
730
|
+
const jpegBuffer = await sharp(context.imageBuffer)
|
|
731
|
+
.resize(600, 600, {
|
|
732
|
+
fit: 'inside',
|
|
733
|
+
withoutEnlargement: true,
|
|
734
|
+
})
|
|
735
|
+
.jpeg({ quality })
|
|
736
|
+
.toBuffer()
|
|
737
|
+
|
|
738
|
+
if (jpegBuffer.length <= context.maxBytes) {
|
|
739
|
+
return createCompressedImageResult(jpegBuffer, 'jpeg', context.originalSize)
|
|
740
|
+
}
|
|
741
|
+
|
|
742
|
+
return null
|
|
743
|
+
}
|
|
744
|
+
|
|
745
|
+
async function createUltraCompressedJPEG(
|
|
746
|
+
context: ImageCompressionContext,
|
|
747
|
+
sharp: SharpFunction,
|
|
748
|
+
): Promise<CompressedImageResult> {
|
|
749
|
+
const ultraCompressedBuffer = await sharp(context.imageBuffer)
|
|
750
|
+
.resize(400, 400, {
|
|
751
|
+
fit: 'inside',
|
|
752
|
+
withoutEnlargement: true,
|
|
753
|
+
})
|
|
754
|
+
.jpeg({ quality: 20 })
|
|
755
|
+
.toBuffer()
|
|
756
|
+
|
|
757
|
+
return createCompressedImageResult(
|
|
758
|
+
ultraCompressedBuffer,
|
|
759
|
+
'jpeg',
|
|
760
|
+
context.originalSize,
|
|
761
|
+
)
|
|
762
|
+
}
|
|
763
|
+
|
|
764
|
+
/**
|
|
765
|
+
* Detect image format from a buffer using magic bytes
|
|
766
|
+
* @param buffer Buffer containing image data
|
|
767
|
+
* @returns Media type string (e.g., 'image/png', 'image/jpeg') or 'image/png' as default
|
|
768
|
+
*/
|
|
769
|
+
export function detectImageFormatFromBuffer(buffer: Buffer): ImageMediaType {
|
|
770
|
+
if (buffer.length < 4) return 'image/png' // default
|
|
771
|
+
|
|
772
|
+
// Check PNG signature
|
|
773
|
+
if (
|
|
774
|
+
buffer[0] === 0x89 &&
|
|
775
|
+
buffer[1] === 0x50 &&
|
|
776
|
+
buffer[2] === 0x4e &&
|
|
777
|
+
buffer[3] === 0x47
|
|
778
|
+
) {
|
|
779
|
+
return 'image/png'
|
|
780
|
+
}
|
|
781
|
+
|
|
782
|
+
// Check JPEG signature (FFD8FF)
|
|
783
|
+
if (buffer[0] === 0xff && buffer[1] === 0xd8 && buffer[2] === 0xff) {
|
|
784
|
+
return 'image/jpeg'
|
|
785
|
+
}
|
|
786
|
+
|
|
787
|
+
// Check GIF signature (GIF87a or GIF89a)
|
|
788
|
+
if (buffer[0] === 0x47 && buffer[1] === 0x49 && buffer[2] === 0x46) {
|
|
789
|
+
return 'image/gif'
|
|
790
|
+
}
|
|
791
|
+
|
|
792
|
+
// Check WebP signature (RIFF....WEBP)
|
|
793
|
+
if (
|
|
794
|
+
buffer[0] === 0x52 &&
|
|
795
|
+
buffer[1] === 0x49 &&
|
|
796
|
+
buffer[2] === 0x46 &&
|
|
797
|
+
buffer[3] === 0x46
|
|
798
|
+
) {
|
|
799
|
+
if (
|
|
800
|
+
buffer.length >= 12 &&
|
|
801
|
+
buffer[8] === 0x57 &&
|
|
802
|
+
buffer[9] === 0x45 &&
|
|
803
|
+
buffer[10] === 0x42 &&
|
|
804
|
+
buffer[11] === 0x50
|
|
805
|
+
) {
|
|
806
|
+
return 'image/webp'
|
|
807
|
+
}
|
|
808
|
+
}
|
|
809
|
+
|
|
810
|
+
// Default to PNG if unknown
|
|
811
|
+
return 'image/png'
|
|
812
|
+
}
|
|
813
|
+
|
|
814
|
+
/**
|
|
815
|
+
* Detect image format from base64 data using magic bytes
|
|
816
|
+
* @param base64Data Base64 encoded image data
|
|
817
|
+
* @returns Media type string (e.g., 'image/png', 'image/jpeg') or 'image/png' as default
|
|
818
|
+
*/
|
|
819
|
+
export function detectImageFormatFromBase64(
|
|
820
|
+
base64Data: string,
|
|
821
|
+
): ImageMediaType {
|
|
822
|
+
try {
|
|
823
|
+
const buffer = Buffer.from(base64Data, 'base64')
|
|
824
|
+
return detectImageFormatFromBuffer(buffer)
|
|
825
|
+
} catch {
|
|
826
|
+
// Default to PNG on any error
|
|
827
|
+
return 'image/png'
|
|
828
|
+
}
|
|
829
|
+
}
|
|
830
|
+
|
|
831
|
+
/**
|
|
832
|
+
* Creates a text description of image metadata including dimensions and source path.
|
|
833
|
+
* Returns null if no useful metadata is available.
|
|
834
|
+
*/
|
|
835
|
+
export function createImageMetadataText(
|
|
836
|
+
dims: ImageDimensions,
|
|
837
|
+
sourcePath?: string,
|
|
838
|
+
): string | null {
|
|
839
|
+
const { originalWidth, originalHeight, displayWidth, displayHeight } = dims
|
|
840
|
+
// Skip if dimensions are not available or invalid
|
|
841
|
+
// Note: checks for undefined/null and zero to prevent division by zero
|
|
842
|
+
if (
|
|
843
|
+
!originalWidth ||
|
|
844
|
+
!originalHeight ||
|
|
845
|
+
!displayWidth ||
|
|
846
|
+
!displayHeight ||
|
|
847
|
+
displayWidth <= 0 ||
|
|
848
|
+
displayHeight <= 0
|
|
849
|
+
) {
|
|
850
|
+
// If we have a source path but no valid dimensions, still return source info
|
|
851
|
+
if (sourcePath) {
|
|
852
|
+
return `[Image source: ${sourcePath}]`
|
|
853
|
+
}
|
|
854
|
+
return null
|
|
855
|
+
}
|
|
856
|
+
// Check if image was resized
|
|
857
|
+
const wasResized =
|
|
858
|
+
originalWidth !== displayWidth || originalHeight !== displayHeight
|
|
859
|
+
|
|
860
|
+
// Only include metadata if there's useful info (resized or has source path)
|
|
861
|
+
if (!wasResized && !sourcePath) {
|
|
862
|
+
return null
|
|
863
|
+
}
|
|
864
|
+
|
|
865
|
+
// Build metadata parts
|
|
866
|
+
const parts: string[] = []
|
|
867
|
+
|
|
868
|
+
if (sourcePath) {
|
|
869
|
+
parts.push(`source: ${sourcePath}`)
|
|
870
|
+
}
|
|
871
|
+
|
|
872
|
+
if (wasResized) {
|
|
873
|
+
const scaleFactor = originalWidth / displayWidth
|
|
874
|
+
parts.push(
|
|
875
|
+
`original ${originalWidth}x${originalHeight}, displayed at ${displayWidth}x${displayHeight}. Multiply coordinates by ${scaleFactor.toFixed(2)} to map to original image.`,
|
|
876
|
+
)
|
|
877
|
+
}
|
|
878
|
+
|
|
879
|
+
return `[Image: ${parts.join(', ')}]`
|
|
880
|
+
}
|