@mseep/claudian 2.0.25

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (687) hide show
  1. package/.env.local.example +2 -0
  2. package/.github/workflows/ci.yml +59 -0
  3. package/.github/workflows/claude-code-review.yml +57 -0
  4. package/.github/workflows/claude.yml +50 -0
  5. package/.github/workflows/duplicate-issues.yml +22 -0
  6. package/.github/workflows/release.yml +73 -0
  7. package/.github/workflows/stale.yml +21 -0
  8. package/.node-version +1 -0
  9. package/AGENTS.md +3 -0
  10. package/CLAUDE.md +80 -0
  11. package/LICENSE +21 -0
  12. package/README.md +190 -0
  13. package/assets/Preview.png +0 -0
  14. package/assets/sponsors/MOMA.png +0 -0
  15. package/bun.lock +1618 -0
  16. package/esbuild.config.mjs +195 -0
  17. package/eslint.config.mjs +143 -0
  18. package/jest.config.js +41 -0
  19. package/main.js +104915 -0
  20. package/manifest.json +10 -0
  21. package/package.json +65 -0
  22. package/scripts/build-css.mjs +119 -0
  23. package/scripts/build.mjs +19 -0
  24. package/scripts/postinstall.mjs +25 -0
  25. package/scripts/rendererSafeUnref.js +205 -0
  26. package/scripts/run-jest.js +19 -0
  27. package/scripts/sync-version.js +16 -0
  28. package/src/app/settings/ClaudianSettingsStorage.ts +435 -0
  29. package/src/app/settings/defaultSettings.ts +54 -0
  30. package/src/app/storage/SharedStorageService.ts +106 -0
  31. package/src/core/CLAUDE.md +84 -0
  32. package/src/core/auxiliary/AuxQueryRunner.ts +11 -0
  33. package/src/core/auxiliary/QueryBackedInlineEditService.ts +73 -0
  34. package/src/core/auxiliary/QueryBackedInstructionRefineService.ts +78 -0
  35. package/src/core/auxiliary/QueryBackedTitleGenerationService.ts +90 -0
  36. package/src/core/bootstrap/SessionStorage.ts +179 -0
  37. package/src/core/bootstrap/StoragePaths.ts +7 -0
  38. package/src/core/bootstrap/storage.ts +20 -0
  39. package/src/core/commands/builtInCommands.ts +141 -0
  40. package/src/core/mcp/McpConfigParser.ts +102 -0
  41. package/src/core/mcp/McpServerManager.ts +119 -0
  42. package/src/core/mcp/McpTester.ts +310 -0
  43. package/src/core/prompt/inlineEdit.ts +252 -0
  44. package/src/core/prompt/instructionRefine.ts +72 -0
  45. package/src/core/prompt/mainAgent.ts +213 -0
  46. package/src/core/prompt/titleGeneration.ts +44 -0
  47. package/src/core/providers/ProviderRegistry.ts +253 -0
  48. package/src/core/providers/ProviderSettingsCoordinator.ts +434 -0
  49. package/src/core/providers/ProviderWorkspaceRegistry.ts +119 -0
  50. package/src/core/providers/commands/ProviderCommandCatalog.ts +21 -0
  51. package/src/core/providers/commands/ProviderCommandEntry.ts +33 -0
  52. package/src/core/providers/commands/hiddenCommands.ts +74 -0
  53. package/src/core/providers/modelRouting.ts +17 -0
  54. package/src/core/providers/modelSelection.ts +72 -0
  55. package/src/core/providers/providerConfig.ts +34 -0
  56. package/src/core/providers/providerEnvironment.ts +364 -0
  57. package/src/core/providers/types.ts +544 -0
  58. package/src/core/runtime/ChatRuntime.ts +66 -0
  59. package/src/core/runtime/QueuedTurn.ts +97 -0
  60. package/src/core/runtime/types.ts +118 -0
  61. package/src/core/security/ApprovalManager.ts +142 -0
  62. package/src/core/storage/HomeFileAdapter.ts +75 -0
  63. package/src/core/storage/VaultFileAdapter.ts +132 -0
  64. package/src/core/tools/todo.ts +65 -0
  65. package/src/core/tools/toolIcons.ts +80 -0
  66. package/src/core/tools/toolInput.ts +119 -0
  67. package/src/core/tools/toolNames.ts +149 -0
  68. package/src/core/tools/toolResultContent.ts +26 -0
  69. package/src/core/types/agent.ts +28 -0
  70. package/src/core/types/chat.ts +177 -0
  71. package/src/core/types/diff.ts +31 -0
  72. package/src/core/types/index.ts +78 -0
  73. package/src/core/types/mcp.ts +97 -0
  74. package/src/core/types/plugins.ts +9 -0
  75. package/src/core/types/provider.ts +1 -0
  76. package/src/core/types/settings.ts +152 -0
  77. package/src/core/types/tools.ts +80 -0
  78. package/src/features/chat/CLAUDE.md +136 -0
  79. package/src/features/chat/ClaudianView.ts +762 -0
  80. package/src/features/chat/constants.ts +114 -0
  81. package/src/features/chat/controllers/BrowserSelectionController.ts +295 -0
  82. package/src/features/chat/controllers/CanvasSelectionController.ts +142 -0
  83. package/src/features/chat/controllers/ConversationController.ts +1103 -0
  84. package/src/features/chat/controllers/InputController.ts +1707 -0
  85. package/src/features/chat/controllers/NavigationController.ts +209 -0
  86. package/src/features/chat/controllers/SelectionController.ts +430 -0
  87. package/src/features/chat/controllers/StreamController.ts +1560 -0
  88. package/src/features/chat/controllers/contextRowVisibility.ts +18 -0
  89. package/src/features/chat/rendering/DiffRenderer.ts +134 -0
  90. package/src/features/chat/rendering/InlineAskUserQuestion.ts +702 -0
  91. package/src/features/chat/rendering/InlineExitPlanMode.ts +263 -0
  92. package/src/features/chat/rendering/InlinePlanApproval.ts +183 -0
  93. package/src/features/chat/rendering/MessageRenderer.ts +907 -0
  94. package/src/features/chat/rendering/SubagentRenderer.ts +678 -0
  95. package/src/features/chat/rendering/ThinkingBlockRenderer.ts +126 -0
  96. package/src/features/chat/rendering/TodoListRenderer.ts +5 -0
  97. package/src/features/chat/rendering/ToolCallRenderer.ts +1161 -0
  98. package/src/features/chat/rendering/WriteEditRenderer.ts +232 -0
  99. package/src/features/chat/rendering/collapsible.ts +101 -0
  100. package/src/features/chat/rendering/subagentLifecycleResolution.ts +25 -0
  101. package/src/features/chat/rendering/todoUtils.ts +29 -0
  102. package/src/features/chat/rewind.ts +31 -0
  103. package/src/features/chat/services/BangBashService.ts +56 -0
  104. package/src/features/chat/services/SubagentManager.ts +1107 -0
  105. package/src/features/chat/state/ChatState.ts +436 -0
  106. package/src/features/chat/state/types.ts +138 -0
  107. package/src/features/chat/tabs/Tab.ts +1886 -0
  108. package/src/features/chat/tabs/TabBar.ts +179 -0
  109. package/src/features/chat/tabs/TabManager.ts +1021 -0
  110. package/src/features/chat/tabs/providerResolution.ts +34 -0
  111. package/src/features/chat/tabs/types.ts +287 -0
  112. package/src/features/chat/ui/BangBashModeManager.ts +121 -0
  113. package/src/features/chat/ui/FileContext.ts +385 -0
  114. package/src/features/chat/ui/ImageContext.ts +366 -0
  115. package/src/features/chat/ui/InputToolbar.ts +1244 -0
  116. package/src/features/chat/ui/InstructionModeManager.ts +158 -0
  117. package/src/features/chat/ui/NavigationSidebar.ts +126 -0
  118. package/src/features/chat/ui/StatusPanel.ts +589 -0
  119. package/src/features/chat/ui/file-context/state/FileContextState.ts +83 -0
  120. package/src/features/chat/ui/file-context/view/FileChipsView.ts +70 -0
  121. package/src/features/chat/ui/textareaResize.ts +47 -0
  122. package/src/features/chat/utils/usageInfo.ts +26 -0
  123. package/src/features/inline-edit/ui/InlineEditModal.ts +895 -0
  124. package/src/features/inline-edit/ui/inlineEditMarkdownPreview.ts +55 -0
  125. package/src/features/settings/ClaudianSettings.ts +672 -0
  126. package/src/features/settings/keyboardNavigation.ts +60 -0
  127. package/src/features/settings/ui/EnvSnippetManager.ts +430 -0
  128. package/src/features/settings/ui/EnvironmentSettingsSection.ts +85 -0
  129. package/src/features/settings/ui/McpServerModal.ts +335 -0
  130. package/src/features/settings/ui/McpSettingsManager.ts +400 -0
  131. package/src/features/settings/ui/McpTestModal.ts +346 -0
  132. package/src/i18n/constants.ts +58 -0
  133. package/src/i18n/i18n.ts +140 -0
  134. package/src/i18n/locales/de.json +322 -0
  135. package/src/i18n/locales/en.json +322 -0
  136. package/src/i18n/locales/es.json +322 -0
  137. package/src/i18n/locales/fr.json +322 -0
  138. package/src/i18n/locales/ja.json +322 -0
  139. package/src/i18n/locales/ko.json +322 -0
  140. package/src/i18n/locales/pt.json +322 -0
  141. package/src/i18n/locales/ru.json +322 -0
  142. package/src/i18n/locales/zh-CN.json +322 -0
  143. package/src/i18n/locales/zh-TW.json +322 -0
  144. package/src/i18n/types.ts +248 -0
  145. package/src/main.ts +772 -0
  146. package/src/providers/acp/AcpClientConnection.ts +361 -0
  147. package/src/providers/acp/AcpJsonRpcTransport.ts +427 -0
  148. package/src/providers/acp/AcpSessionConfig.ts +139 -0
  149. package/src/providers/acp/AcpSessionUpdateNormalizer.ts +371 -0
  150. package/src/providers/acp/AcpSubprocess.ts +155 -0
  151. package/src/providers/acp/AcpToolStreamAdapter.ts +132 -0
  152. package/src/providers/acp/buildAcpUsageInfo.ts +41 -0
  153. package/src/providers/acp/index.ts +9 -0
  154. package/src/providers/acp/methodNames.ts +50 -0
  155. package/src/providers/acp/types.ts +566 -0
  156. package/src/providers/claude/CLAUDE.md +78 -0
  157. package/src/providers/claude/agents/AgentManager.ts +225 -0
  158. package/src/providers/claude/agents/AgentStorage.ts +101 -0
  159. package/src/providers/claude/app/ClaudeWorkspaceServices.ts +90 -0
  160. package/src/providers/claude/auxiliary/ClaudeInlineEditService.ts +122 -0
  161. package/src/providers/claude/auxiliary/ClaudeInstructionRefineService.ts +90 -0
  162. package/src/providers/claude/auxiliary/ClaudeTitleGenerationService.ts +129 -0
  163. package/src/providers/claude/auxiliary/extractAssistantText.ts +28 -0
  164. package/src/providers/claude/capabilities.ts +17 -0
  165. package/src/providers/claude/cli/findClaudeCLIPath.ts +261 -0
  166. package/src/providers/claude/commands/ClaudeCommandCatalog.ts +149 -0
  167. package/src/providers/claude/commands/probeRuntimeCommands.ts +86 -0
  168. package/src/providers/claude/env/ClaudeSettingsReconciler.ts +90 -0
  169. package/src/providers/claude/env/claudeModelEnv.ts +86 -0
  170. package/src/providers/claude/history/ClaudeConversationHistoryService.ts +446 -0
  171. package/src/providers/claude/history/ClaudeHistoryStore.ts +170 -0
  172. package/src/providers/claude/history/sdkAsyncSubagent.ts +92 -0
  173. package/src/providers/claude/history/sdkBranchFilter.ts +271 -0
  174. package/src/providers/claude/history/sdkHistoryTypes.ts +61 -0
  175. package/src/providers/claude/history/sdkMessageParsing.ts +413 -0
  176. package/src/providers/claude/history/sdkSessionPaths.ts +98 -0
  177. package/src/providers/claude/history/sdkSubagentSidecar.ts +261 -0
  178. package/src/providers/claude/hooks/SubagentHooks.ts +31 -0
  179. package/src/providers/claude/modelLabels.ts +77 -0
  180. package/src/providers/claude/modelOptions.ts +113 -0
  181. package/src/providers/claude/modelSelection.ts +17 -0
  182. package/src/providers/claude/plugins/PluginManager.ts +194 -0
  183. package/src/providers/claude/prompt/ClaudeTurnEncoder.ts +46 -0
  184. package/src/providers/claude/registration.ts +39 -0
  185. package/src/providers/claude/runtime/ClaudeApprovalHandler.ts +153 -0
  186. package/src/providers/claude/runtime/ClaudeChatRuntime.ts +1812 -0
  187. package/src/providers/claude/runtime/ClaudeCliResolver.ts +94 -0
  188. package/src/providers/claude/runtime/ClaudeDynamicUpdates.ts +164 -0
  189. package/src/providers/claude/runtime/ClaudeMessageChannel.ts +209 -0
  190. package/src/providers/claude/runtime/ClaudeQueryOptionsBuilder.ts +315 -0
  191. package/src/providers/claude/runtime/ClaudeRewindService.ts +220 -0
  192. package/src/providers/claude/runtime/ClaudeSessionManager.ts +92 -0
  193. package/src/providers/claude/runtime/ClaudeTaskResultInterpreter.ts +172 -0
  194. package/src/providers/claude/runtime/ClaudeUserMessageFactory.ts +83 -0
  195. package/src/providers/claude/runtime/claudeColdStartQuery.ts +152 -0
  196. package/src/providers/claude/runtime/customSpawn.ts +87 -0
  197. package/src/providers/claude/runtime/types.ts +134 -0
  198. package/src/providers/claude/sdk/messages.ts +17 -0
  199. package/src/providers/claude/sdk/toolResultContent.ts +4 -0
  200. package/src/providers/claude/sdk/typeGuards.ts +14 -0
  201. package/src/providers/claude/sdk/types.ts +15 -0
  202. package/src/providers/claude/security/ClaudePermissionUpdates.ts +44 -0
  203. package/src/providers/claude/settings.ts +138 -0
  204. package/src/providers/claude/storage/AgentVaultStorage.ts +101 -0
  205. package/src/providers/claude/storage/CCSettingsStorage.ts +153 -0
  206. package/src/providers/claude/storage/ClaudianSettingsStorage.ts +6 -0
  207. package/src/providers/claude/storage/McpStorage.ts +139 -0
  208. package/src/providers/claude/storage/SessionStorage.ts +5 -0
  209. package/src/providers/claude/storage/SkillStorage.ts +61 -0
  210. package/src/providers/claude/storage/SlashCommandStorage.ts +96 -0
  211. package/src/providers/claude/storage/StorageService.ts +185 -0
  212. package/src/providers/claude/stream/toolInputStreamState.ts +318 -0
  213. package/src/providers/claude/stream/transformClaudeMessage.ts +586 -0
  214. package/src/providers/claude/types/agent.ts +2 -0
  215. package/src/providers/claude/types/models.ts +168 -0
  216. package/src/providers/claude/types/plugins.ts +14 -0
  217. package/src/providers/claude/types/providerState.ts +16 -0
  218. package/src/providers/claude/types/settings.ts +98 -0
  219. package/src/providers/claude/ui/AgentSettings.ts +389 -0
  220. package/src/providers/claude/ui/ClaudeChatUIConfig.ts +113 -0
  221. package/src/providers/claude/ui/ClaudeSettingsTab.ts +407 -0
  222. package/src/providers/claude/ui/PluginSettingsManager.ts +149 -0
  223. package/src/providers/claude/ui/SlashCommandSettings.ts +527 -0
  224. package/src/providers/codex/CLAUDE.md +64 -0
  225. package/src/providers/codex/agents/CodexAgentMentionProvider.ts +33 -0
  226. package/src/providers/codex/app/CodexWorkspaceServices.ts +76 -0
  227. package/src/providers/codex/auxiliary/CodexInlineEditService.ts +9 -0
  228. package/src/providers/codex/auxiliary/CodexInstructionRefineService.ts +9 -0
  229. package/src/providers/codex/auxiliary/CodexTaskResultInterpreter.ts +29 -0
  230. package/src/providers/codex/auxiliary/CodexTitleGenerationService.ts +22 -0
  231. package/src/providers/codex/capabilities.ts +16 -0
  232. package/src/providers/codex/commands/CodexSkillCatalog.ts +180 -0
  233. package/src/providers/codex/env/CodexSettingsReconciler.ts +68 -0
  234. package/src/providers/codex/history/CodexConversationHistoryService.ts +212 -0
  235. package/src/providers/codex/history/CodexHistoryStore.ts +1672 -0
  236. package/src/providers/codex/modelOptions.ts +99 -0
  237. package/src/providers/codex/modelSelection.ts +17 -0
  238. package/src/providers/codex/normalization/codexSubagentNormalization.ts +227 -0
  239. package/src/providers/codex/normalization/codexToolNormalization.ts +390 -0
  240. package/src/providers/codex/prompt/encodeCodexTurn.ts +57 -0
  241. package/src/providers/codex/registration.ts +29 -0
  242. package/src/providers/codex/runtime/CodexAppServerProcess.ts +105 -0
  243. package/src/providers/codex/runtime/CodexAuxQueryRunner.ts +180 -0
  244. package/src/providers/codex/runtime/CodexBinaryLocator.ts +49 -0
  245. package/src/providers/codex/runtime/CodexChatRuntime.ts +1296 -0
  246. package/src/providers/codex/runtime/CodexCliResolver.ts +65 -0
  247. package/src/providers/codex/runtime/CodexExecutionTargetResolver.ts +104 -0
  248. package/src/providers/codex/runtime/CodexLaunchSpecBuilder.ts +85 -0
  249. package/src/providers/codex/runtime/CodexNotificationRouter.ts +1033 -0
  250. package/src/providers/codex/runtime/CodexPathMapper.ts +155 -0
  251. package/src/providers/codex/runtime/CodexRpcTransport.ts +171 -0
  252. package/src/providers/codex/runtime/CodexRuntimeContext.ts +109 -0
  253. package/src/providers/codex/runtime/CodexServerRequestRouter.ts +331 -0
  254. package/src/providers/codex/runtime/CodexSessionFileTail.ts +792 -0
  255. package/src/providers/codex/runtime/CodexSessionManager.ts +39 -0
  256. package/src/providers/codex/runtime/codexAppServerSupport.ts +58 -0
  257. package/src/providers/codex/runtime/codexAppServerTypes.ts +705 -0
  258. package/src/providers/codex/runtime/codexLaunchTypes.ts +30 -0
  259. package/src/providers/codex/settings.ts +236 -0
  260. package/src/providers/codex/skills/CodexSkillListingService.ts +173 -0
  261. package/src/providers/codex/storage/CodexSkillStorage.ts +250 -0
  262. package/src/providers/codex/storage/CodexSubagentStorage.ts +212 -0
  263. package/src/providers/codex/types/index.ts +16 -0
  264. package/src/providers/codex/types/models.ts +46 -0
  265. package/src/providers/codex/types/subagent.ts +23 -0
  266. package/src/providers/codex/ui/CodexChatUIConfig.ts +128 -0
  267. package/src/providers/codex/ui/CodexSettingsTab.ts +432 -0
  268. package/src/providers/codex/ui/CodexSkillSettings.ts +275 -0
  269. package/src/providers/codex/ui/CodexSubagentSettings.ts +400 -0
  270. package/src/providers/defaultProviderConfigs.ts +14 -0
  271. package/src/providers/index.ts +30 -0
  272. package/src/providers/opencode/agents/OpencodeAgentMentionProvider.ts +42 -0
  273. package/src/providers/opencode/app/OpencodeRuntimeCommandLoader.ts +67 -0
  274. package/src/providers/opencode/app/OpencodeWorkspaceServices.ts +55 -0
  275. package/src/providers/opencode/auxiliary/OpencodeInlineEditService.ts +13 -0
  276. package/src/providers/opencode/auxiliary/OpencodeInstructionRefineService.ts +12 -0
  277. package/src/providers/opencode/auxiliary/OpencodeTaskResultInterpreter.ts +29 -0
  278. package/src/providers/opencode/auxiliary/OpencodeTitleGenerationService.ts +27 -0
  279. package/src/providers/opencode/capabilities.ts +16 -0
  280. package/src/providers/opencode/commands/OpencodeCommandCatalog.ts +92 -0
  281. package/src/providers/opencode/discoveryState.ts +135 -0
  282. package/src/providers/opencode/env/OpencodeSettingsReconciler.ts +178 -0
  283. package/src/providers/opencode/history/OpencodeConversationHistoryService.ts +84 -0
  284. package/src/providers/opencode/history/OpencodeHistoryStore.ts +472 -0
  285. package/src/providers/opencode/history/OpencodeSqliteReader.ts +284 -0
  286. package/src/providers/opencode/internal/compareCollections.ts +72 -0
  287. package/src/providers/opencode/internal/providerProjection.ts +15 -0
  288. package/src/providers/opencode/models.ts +378 -0
  289. package/src/providers/opencode/modes.ts +150 -0
  290. package/src/providers/opencode/normalization/opencodeToolNormalization.ts +406 -0
  291. package/src/providers/opencode/registration.ts +27 -0
  292. package/src/providers/opencode/runtime/OpencodeAuxQueryRunner.ts +436 -0
  293. package/src/providers/opencode/runtime/OpencodeChatRuntime.ts +1603 -0
  294. package/src/providers/opencode/runtime/OpencodeCliResolver.ts +57 -0
  295. package/src/providers/opencode/runtime/OpencodeLaunchArtifacts.ts +231 -0
  296. package/src/providers/opencode/runtime/OpencodePaths.ts +113 -0
  297. package/src/providers/opencode/runtime/OpencodeRuntimeEnvironment.ts +18 -0
  298. package/src/providers/opencode/runtime/buildOpencodePrompt.ts +66 -0
  299. package/src/providers/opencode/settings.ts +427 -0
  300. package/src/providers/opencode/storage/OpencodeAgentStorage.ts +346 -0
  301. package/src/providers/opencode/types/agent.ts +37 -0
  302. package/src/providers/opencode/types/index.ts +9 -0
  303. package/src/providers/opencode/ui/OpencodeAgentSettings.ts +579 -0
  304. package/src/providers/opencode/ui/OpencodeChatUIConfig.ts +316 -0
  305. package/src/providers/opencode/ui/OpencodeSettingsTab.ts +674 -0
  306. package/src/providers/pi/app/PiRuntimeCommandLoader.ts +57 -0
  307. package/src/providers/pi/app/PiWorkspaceServices.ts +39 -0
  308. package/src/providers/pi/auxiliary/PiInlineEditService.ts +9 -0
  309. package/src/providers/pi/auxiliary/PiInstructionRefineService.ts +9 -0
  310. package/src/providers/pi/auxiliary/PiTaskResultInterpreter.ts +29 -0
  311. package/src/providers/pi/auxiliary/PiTitleGenerationService.ts +19 -0
  312. package/src/providers/pi/capabilities.ts +16 -0
  313. package/src/providers/pi/commands/PiCommandCatalog.ts +92 -0
  314. package/src/providers/pi/env/PiSettingsReconciler.ts +180 -0
  315. package/src/providers/pi/history/PiConversationHistoryService.ts +123 -0
  316. package/src/providers/pi/history/PiHistoryStore.ts +664 -0
  317. package/src/providers/pi/internal/compareCollections.ts +4 -0
  318. package/src/providers/pi/internal/providerProjection.ts +18 -0
  319. package/src/providers/pi/models.ts +302 -0
  320. package/src/providers/pi/normalizations/piEventNormalization.ts +211 -0
  321. package/src/providers/pi/normalizations/piToolNormalization.ts +97 -0
  322. package/src/providers/pi/registration.ts +30 -0
  323. package/src/providers/pi/runtime/PiAuxQueryRunner.ts +216 -0
  324. package/src/providers/pi/runtime/PiChatRuntime.ts +1064 -0
  325. package/src/providers/pi/runtime/PiCliResolver.ts +53 -0
  326. package/src/providers/pi/runtime/PiExtensionUiBridge.ts +161 -0
  327. package/src/providers/pi/runtime/PiJsonl.ts +71 -0
  328. package/src/providers/pi/runtime/PiLaunchSpec.ts +70 -0
  329. package/src/providers/pi/runtime/PiModelDiscoveryService.ts +92 -0
  330. package/src/providers/pi/runtime/PiRpcPayloads.ts +18 -0
  331. package/src/providers/pi/runtime/PiRpcTransport.ts +243 -0
  332. package/src/providers/pi/runtime/PiSubprocess.ts +159 -0
  333. package/src/providers/pi/runtime/buildPiPrompt.ts +62 -0
  334. package/src/providers/pi/runtime/buildPiUsageInfo.ts +69 -0
  335. package/src/providers/pi/settings.ts +468 -0
  336. package/src/providers/pi/types.ts +64 -0
  337. package/src/providers/pi/ui/ObsidianPiExtensionUiRenderer.ts +251 -0
  338. package/src/providers/pi/ui/PiChatUIConfig.ts +265 -0
  339. package/src/providers/pi/ui/PiExtensionUiRenderer.ts +12 -0
  340. package/src/providers/pi/ui/PiSettingsTab.ts +642 -0
  341. package/src/shared/components/ResumeSessionDropdown.ts +185 -0
  342. package/src/shared/components/SelectableDropdown.ts +140 -0
  343. package/src/shared/components/SelectionHighlight.ts +77 -0
  344. package/src/shared/components/SlashCommandDropdown.ts +421 -0
  345. package/src/shared/icons.ts +180 -0
  346. package/src/shared/mention/MentionDropdownController.ts +627 -0
  347. package/src/shared/mention/VaultMentionCache.ts +106 -0
  348. package/src/shared/mention/VaultMentionDataProvider.ts +51 -0
  349. package/src/shared/mention/types.ts +67 -0
  350. package/src/shared/modals/ConfirmModal.ts +60 -0
  351. package/src/shared/modals/ForkTargetModal.ts +47 -0
  352. package/src/shared/modals/InstructionConfirmModal.ts +281 -0
  353. package/src/style/CLAUDE.md +49 -0
  354. package/src/style/accessibility.css +40 -0
  355. package/src/style/base/animations.css +44 -0
  356. package/src/style/base/container.css +20 -0
  357. package/src/style/base/variables.css +46 -0
  358. package/src/style/base/visibility.css +15 -0
  359. package/src/style/components/code.css +97 -0
  360. package/src/style/components/context-footer.css +76 -0
  361. package/src/style/components/header.css +27 -0
  362. package/src/style/components/history.css +221 -0
  363. package/src/style/components/input.css +312 -0
  364. package/src/style/components/messages.css +262 -0
  365. package/src/style/components/nav-sidebar.css +58 -0
  366. package/src/style/components/status-panel.css +202 -0
  367. package/src/style/components/subagent.css +248 -0
  368. package/src/style/components/tabs.css +112 -0
  369. package/src/style/components/thinking.css +88 -0
  370. package/src/style/components/toolcalls.css +278 -0
  371. package/src/style/features/ask-user-question.css +315 -0
  372. package/src/style/features/diff.css +197 -0
  373. package/src/style/features/file-context.css +188 -0
  374. package/src/style/features/file-link.css +22 -0
  375. package/src/style/features/image-context.css +179 -0
  376. package/src/style/features/image-embed.css +40 -0
  377. package/src/style/features/image-modal.css +52 -0
  378. package/src/style/features/inline-edit.css +278 -0
  379. package/src/style/features/plan-mode.css +103 -0
  380. package/src/style/features/resume-session.css +119 -0
  381. package/src/style/features/slash-commands.css +91 -0
  382. package/src/style/index.css +63 -0
  383. package/src/style/modals/fork-target.css +21 -0
  384. package/src/style/modals/instruction.css +161 -0
  385. package/src/style/modals/mcp-modal.css +241 -0
  386. package/src/style/settings/agent-settings.css +2 -0
  387. package/src/style/settings/base.css +300 -0
  388. package/src/style/settings/env-snippets.css +366 -0
  389. package/src/style/settings/mcp-settings.css +211 -0
  390. package/src/style/settings/plugin-settings.css +164 -0
  391. package/src/style/settings/provider-model-picker.css +367 -0
  392. package/src/style/settings/slash-settings.css +16 -0
  393. package/src/style/toolbar/external-context.css +177 -0
  394. package/src/style/toolbar/mcp-selector.css +176 -0
  395. package/src/style/toolbar/mode-selector.css +19 -0
  396. package/src/style/toolbar/model-selector.css +99 -0
  397. package/src/style/toolbar/permission-toggle.css +56 -0
  398. package/src/style/toolbar/service-tier-toggle.css +39 -0
  399. package/src/style/toolbar/thinking-selector.css +83 -0
  400. package/src/types/smol-toml.d.ts +4 -0
  401. package/src/utils/agent.ts +50 -0
  402. package/src/utils/animationFrame.ts +46 -0
  403. package/src/utils/browser.ts +46 -0
  404. package/src/utils/canvas.ts +14 -0
  405. package/src/utils/cliBinaryLocator.ts +97 -0
  406. package/src/utils/context.ts +117 -0
  407. package/src/utils/contextMentionResolver.ts +154 -0
  408. package/src/utils/date.ts +31 -0
  409. package/src/utils/diff.ts +384 -0
  410. package/src/utils/editor.ts +104 -0
  411. package/src/utils/electronCompat.ts +53 -0
  412. package/src/utils/env.ts +465 -0
  413. package/src/utils/externalContext.ts +143 -0
  414. package/src/utils/externalContextScanner.ts +135 -0
  415. package/src/utils/fileLink.ts +263 -0
  416. package/src/utils/frontmatter.ts +194 -0
  417. package/src/utils/imageEmbed.ts +139 -0
  418. package/src/utils/inlineEdit.ts +22 -0
  419. package/src/utils/interrupt.ts +23 -0
  420. package/src/utils/markdown.ts +25 -0
  421. package/src/utils/markdownMath.ts +130 -0
  422. package/src/utils/mcp.ts +96 -0
  423. package/src/utils/obsidianCompat.ts +23 -0
  424. package/src/utils/path.ts +342 -0
  425. package/src/utils/session.ts +240 -0
  426. package/src/utils/slashCommand.ts +152 -0
  427. package/src/utils/subagentJsonl.ts +52 -0
  428. package/src/utils/windowsCmdShim.ts +98 -0
  429. package/tests/__mocks__/claude-agent-sdk.ts +317 -0
  430. package/tests/__mocks__/codex-sdk.ts +88 -0
  431. package/tests/__mocks__/obsidian.ts +434 -0
  432. package/tests/helpers/mockElement.ts +403 -0
  433. package/tests/helpers/sdkMessages.ts +291 -0
  434. package/tests/integration/core/agent/ClaudianService.test.ts +1845 -0
  435. package/tests/integration/core/mcp/mcp.test.ts +905 -0
  436. package/tests/integration/features/chat/imagePersistence.test.ts +38 -0
  437. package/tests/integration/main.test.ts +1701 -0
  438. package/tests/setupWindow.ts +26 -0
  439. package/tests/tsconfig.json +7 -0
  440. package/tests/unit/core/commands/builtInCommands.test.ts +239 -0
  441. package/tests/unit/core/mcp/McpServerManager.test.ts +405 -0
  442. package/tests/unit/core/mcp/McpTester.test.ts +282 -0
  443. package/tests/unit/core/mcp/createNodeFetch.test.ts +188 -0
  444. package/tests/unit/core/providers/ProviderRegistry.test.ts +275 -0
  445. package/tests/unit/core/providers/ProviderSettingsCoordinator.test.ts +490 -0
  446. package/tests/unit/core/providers/ProviderWorkspaceRegistry.test.ts +84 -0
  447. package/tests/unit/core/providers/modelRouting.test.ts +91 -0
  448. package/tests/unit/core/providers/modelSelection.test.ts +155 -0
  449. package/tests/unit/core/providers/providerEnvironment.test.ts +162 -0
  450. package/tests/unit/core/providers/tabLifecycle.test.ts +217 -0
  451. package/tests/unit/core/security/ApprovalManager.test.ts +152 -0
  452. package/tests/unit/core/storage/VaultFileAdapter.test.ts +535 -0
  453. package/tests/unit/core/tools/todo.test.ts +227 -0
  454. package/tests/unit/core/tools/toolIcons.test.ts +75 -0
  455. package/tests/unit/core/tools/toolInput.test.ts +350 -0
  456. package/tests/unit/core/tools/toolNames.test.ts +464 -0
  457. package/tests/unit/core/types/mcp.test.ts +115 -0
  458. package/tests/unit/features/chat/ClaudianView.test.ts +404 -0
  459. package/tests/unit/features/chat/controllers/BrowserSelectionController.test.ts +179 -0
  460. package/tests/unit/features/chat/controllers/CanvasSelectionController.test.ts +216 -0
  461. package/tests/unit/features/chat/controllers/ConversationController.test.ts +2764 -0
  462. package/tests/unit/features/chat/controllers/InputController.test.ts +3188 -0
  463. package/tests/unit/features/chat/controllers/NavigationController.test.ts +640 -0
  464. package/tests/unit/features/chat/controllers/SelectionController.test.ts +695 -0
  465. package/tests/unit/features/chat/controllers/StreamController.test.ts +2534 -0
  466. package/tests/unit/features/chat/controllers/contextRowVisibility.test.ts +46 -0
  467. package/tests/unit/features/chat/controllers/index.test.ts +16 -0
  468. package/tests/unit/features/chat/rendering/DiffRenderer.test.ts +355 -0
  469. package/tests/unit/features/chat/rendering/InlineAskUserQuestion.test.ts +1035 -0
  470. package/tests/unit/features/chat/rendering/InlineExitPlanMode.test.ts +191 -0
  471. package/tests/unit/features/chat/rendering/InlinePlanApproval.test.ts +126 -0
  472. package/tests/unit/features/chat/rendering/MessageRenderer.test.ts +2004 -0
  473. package/tests/unit/features/chat/rendering/SubagentRenderer.test.ts +917 -0
  474. package/tests/unit/features/chat/rendering/ThinkingBlockRenderer.test.ts +124 -0
  475. package/tests/unit/features/chat/rendering/TodoListRenderer.test.ts +173 -0
  476. package/tests/unit/features/chat/rendering/ToolCallRenderer.test.ts +909 -0
  477. package/tests/unit/features/chat/rendering/WriteEditRenderer.test.ts +474 -0
  478. package/tests/unit/features/chat/rendering/collapsible.test.ts +158 -0
  479. package/tests/unit/features/chat/rendering/todoUtils.test.ts +105 -0
  480. package/tests/unit/features/chat/rewind.test.ts +56 -0
  481. package/tests/unit/features/chat/services/BangBashService.test.ts +142 -0
  482. package/tests/unit/features/chat/services/InstructionRefineService.test.ts +371 -0
  483. package/tests/unit/features/chat/services/SubagentManager.test.ts +1759 -0
  484. package/tests/unit/features/chat/services/TitleGenerationService.test.ts +480 -0
  485. package/tests/unit/features/chat/state/ChatState.test.ts +581 -0
  486. package/tests/unit/features/chat/tabs/Tab.test.ts +4287 -0
  487. package/tests/unit/features/chat/tabs/TabBar.test.ts +357 -0
  488. package/tests/unit/features/chat/tabs/TabManager.test.ts +2962 -0
  489. package/tests/unit/features/chat/tabs/index.test.ts +11 -0
  490. package/tests/unit/features/chat/ui/BangBashModeManager.test.ts +321 -0
  491. package/tests/unit/features/chat/ui/ExternalContextSelector.test.ts +555 -0
  492. package/tests/unit/features/chat/ui/FileContextManager.test.ts +876 -0
  493. package/tests/unit/features/chat/ui/ImageContext.test.ts +777 -0
  494. package/tests/unit/features/chat/ui/InputToolbar.test.ts +1139 -0
  495. package/tests/unit/features/chat/ui/InstructionModeManager.test.ts +243 -0
  496. package/tests/unit/features/chat/ui/NavigationSidebar.test.ts +570 -0
  497. package/tests/unit/features/chat/ui/StatusPanel.test.ts +953 -0
  498. package/tests/unit/features/chat/ui/file-context/state/FileContextState.test.ts +155 -0
  499. package/tests/unit/features/chat/ui/textareaResize.test.ts +102 -0
  500. package/tests/unit/features/chat/utils/usageInfo.test.ts +56 -0
  501. package/tests/unit/features/inline-edit/InlineEditService.test.ts +1199 -0
  502. package/tests/unit/features/inline-edit/ui/InlineEditModal.openAndWait.test.ts +1482 -0
  503. package/tests/unit/features/inline-edit/ui/InlineEditModal.test.ts +495 -0
  504. package/tests/unit/features/inline-edit/ui/inlineEditMarkdownPreview.test.ts +92 -0
  505. package/tests/unit/features/settings/AgentSettings.test.ts +82 -0
  506. package/tests/unit/features/settings/keyboardNavigation.test.ts +73 -0
  507. package/tests/unit/features/settings/ui/CodexSkillSettings.test.ts +294 -0
  508. package/tests/unit/features/settings/ui/CodexSubagentSettings.test.ts +207 -0
  509. package/tests/unit/i18n/constants.test.ts +43 -0
  510. package/tests/unit/i18n/i18n.test.ts +244 -0
  511. package/tests/unit/i18n/locales.test.ts +134 -0
  512. package/tests/unit/providers/acp/AcpClientConnection.test.ts +248 -0
  513. package/tests/unit/providers/acp/AcpJsonRpcTransport.test.ts +186 -0
  514. package/tests/unit/providers/acp/AcpSessionConfig.test.ts +247 -0
  515. package/tests/unit/providers/acp/AcpSessionUpdateNormalizer.test.ts +145 -0
  516. package/tests/unit/providers/acp/AcpSubprocess.test.ts +105 -0
  517. package/tests/unit/providers/acp/buildAcpUsageInfo.test.ts +51 -0
  518. package/tests/unit/providers/claude/agents/AgentManager.test.ts +590 -0
  519. package/tests/unit/providers/claude/agents/AgentStorage.test.ts +434 -0
  520. package/tests/unit/providers/claude/agents/index.test.ts +10 -0
  521. package/tests/unit/providers/claude/commands/ClaudeCommandCatalog.test.ts +396 -0
  522. package/tests/unit/providers/claude/commands/probeRuntimeCommands.test.ts +92 -0
  523. package/tests/unit/providers/claude/env/ClaudeSettingsReconciler.test.ts +57 -0
  524. package/tests/unit/providers/claude/env/claudeModelEnv.test.ts +228 -0
  525. package/tests/unit/providers/claude/hooks/SubagentHooks.test.ts +83 -0
  526. package/tests/unit/providers/claude/plugins/PluginManager.test.ts +832 -0
  527. package/tests/unit/providers/claude/plugins/index.test.ts +7 -0
  528. package/tests/unit/providers/claude/prompt/ClaudeTurnEncoder.test.ts +145 -0
  529. package/tests/unit/providers/claude/prompt/instructionRefine.test.ts +185 -0
  530. package/tests/unit/providers/claude/prompt/systemPrompt.test.ts +163 -0
  531. package/tests/unit/providers/claude/prompt/titleGeneration.test.ts +20 -0
  532. package/tests/unit/providers/claude/runtime/ClaudeTaskResultInterpreter.test.ts +28 -0
  533. package/tests/unit/providers/claude/runtime/ClaudianService.test.ts +3796 -0
  534. package/tests/unit/providers/claude/runtime/MessageChannel.test.ts +421 -0
  535. package/tests/unit/providers/claude/runtime/QueryOptionsBuilder.test.ts +775 -0
  536. package/tests/unit/providers/claude/runtime/SessionManager.test.ts +182 -0
  537. package/tests/unit/providers/claude/runtime/claudeColdStartQuery.test.ts +331 -0
  538. package/tests/unit/providers/claude/runtime/customSpawn.test.ts +374 -0
  539. package/tests/unit/providers/claude/runtime/index.test.ts +13 -0
  540. package/tests/unit/providers/claude/runtime/types.test.ts +190 -0
  541. package/tests/unit/providers/claude/sdk/typeGuards.test.ts +50 -0
  542. package/tests/unit/providers/claude/security/ClaudePermissionUpdates.test.ts +198 -0
  543. package/tests/unit/providers/claude/storage/AgentVaultStorage.test.ts +413 -0
  544. package/tests/unit/providers/claude/storage/CCSettingsStorage.test.ts +408 -0
  545. package/tests/unit/providers/claude/storage/ClaudianSettingsStorage.test.ts +653 -0
  546. package/tests/unit/providers/claude/storage/McpStorage.test.ts +619 -0
  547. package/tests/unit/providers/claude/storage/SessionStorage.test.ts +680 -0
  548. package/tests/unit/providers/claude/storage/SkillStorage.test.ts +275 -0
  549. package/tests/unit/providers/claude/storage/SlashCommandStorage.test.ts +612 -0
  550. package/tests/unit/providers/claude/storage/storage.test.ts +360 -0
  551. package/tests/unit/providers/claude/storage/storageService.convenience.test.ts +447 -0
  552. package/tests/unit/providers/claude/stream/transformSDKMessage.test.ts +1729 -0
  553. package/tests/unit/providers/claude/types/types.test.ts +726 -0
  554. package/tests/unit/providers/claude/ui/ClaudeChatUIConfig.test.ts +173 -0
  555. package/tests/unit/providers/claude/ui/ClaudeSettingsTab.test.ts +466 -0
  556. package/tests/unit/providers/codex/agents/CodexAgentMentionProvider.test.ts +89 -0
  557. package/tests/unit/providers/codex/auxiliary/CodexInstructionRefineService.test.ts +81 -0
  558. package/tests/unit/providers/codex/capabilities.test.ts +39 -0
  559. package/tests/unit/providers/codex/commands/CodexSkillCatalog.test.ts +413 -0
  560. package/tests/unit/providers/codex/env/CodexSettingsReconciler.test.ts +106 -0
  561. package/tests/unit/providers/codex/fixtures/codex-session-abort.jsonl +9 -0
  562. package/tests/unit/providers/codex/fixtures/codex-session-agent-lifecycle.jsonl +12 -0
  563. package/tests/unit/providers/codex/fixtures/codex-session-persisted-tools.jsonl +15 -0
  564. package/tests/unit/providers/codex/fixtures/codex-session-simple.jsonl +10 -0
  565. package/tests/unit/providers/codex/fixtures/codex-session-tools.jsonl +12 -0
  566. package/tests/unit/providers/codex/fixtures/codex-session-websearch-persisted.jsonl +3 -0
  567. package/tests/unit/providers/codex/fixtures/codex-session-websearch.jsonl +7 -0
  568. package/tests/unit/providers/codex/history/CodexConversationHistoryService.test.ts +690 -0
  569. package/tests/unit/providers/codex/history/CodexHistoryStore.test.ts +2202 -0
  570. package/tests/unit/providers/codex/normalization/codexSubagentNormalization.test.ts +81 -0
  571. package/tests/unit/providers/codex/normalization/codexToolNormalization.test.ts +322 -0
  572. package/tests/unit/providers/codex/prompt/encodeCodexTurn.test.ts +168 -0
  573. package/tests/unit/providers/codex/runtime/CodexAppServerProcess.test.ts +255 -0
  574. package/tests/unit/providers/codex/runtime/CodexAuxQueryRunner.test.ts +130 -0
  575. package/tests/unit/providers/codex/runtime/CodexBinaryLocator.test.ts +90 -0
  576. package/tests/unit/providers/codex/runtime/CodexChatRuntime.test.ts +2445 -0
  577. package/tests/unit/providers/codex/runtime/CodexCliResolver.test.ts +103 -0
  578. package/tests/unit/providers/codex/runtime/CodexExecutionTargetResolver.test.ts +105 -0
  579. package/tests/unit/providers/codex/runtime/CodexLaunchSpecBuilder.test.ts +150 -0
  580. package/tests/unit/providers/codex/runtime/CodexNotificationRouter.test.ts +1248 -0
  581. package/tests/unit/providers/codex/runtime/CodexPathMapper.test.ts +51 -0
  582. package/tests/unit/providers/codex/runtime/CodexRpcTransport.test.ts +220 -0
  583. package/tests/unit/providers/codex/runtime/CodexRuntimeContext.test.ts +107 -0
  584. package/tests/unit/providers/codex/runtime/CodexServerRequestRouter.test.ts +537 -0
  585. package/tests/unit/providers/codex/runtime/CodexSessionFileTail.test.ts +1305 -0
  586. package/tests/unit/providers/codex/runtime/CodexSessionManager.test.ts +82 -0
  587. package/tests/unit/providers/codex/runtime/codexAppServerTypes.test.ts +336 -0
  588. package/tests/unit/providers/codex/settings.test.ts +189 -0
  589. package/tests/unit/providers/codex/skills/CodexSkillListingService.test.ts +178 -0
  590. package/tests/unit/providers/codex/storage/CodexSkillStorage.test.ts +342 -0
  591. package/tests/unit/providers/codex/storage/CodexSubagentStorage.test.ts +376 -0
  592. package/tests/unit/providers/codex/ui/CodexChatUIConfig.test.ts +211 -0
  593. package/tests/unit/providers/codex/ui/CodexSettingsTab.test.ts +578 -0
  594. package/tests/unit/providers/defaultProviderConfigs.test.ts +18 -0
  595. package/tests/unit/providers/opencode/OpencodeAuxQueryRunner.test.ts +355 -0
  596. package/tests/unit/providers/opencode/OpencodeChatRuntime.test.ts +808 -0
  597. package/tests/unit/providers/opencode/OpencodeCliResolver.test.ts +93 -0
  598. package/tests/unit/providers/opencode/OpencodeCommandCatalog.test.ts +75 -0
  599. package/tests/unit/providers/opencode/OpencodeConversationHistoryService.test.ts +103 -0
  600. package/tests/unit/providers/opencode/OpencodeHistoryStore.test.ts +418 -0
  601. package/tests/unit/providers/opencode/OpencodeLaunchArtifacts.test.ts +268 -0
  602. package/tests/unit/providers/opencode/OpencodePaths.test.ts +35 -0
  603. package/tests/unit/providers/opencode/OpencodeRuntimeCommandLoader.test.ts +129 -0
  604. package/tests/unit/providers/opencode/OpencodeSettingsReconciler.test.ts +96 -0
  605. package/tests/unit/providers/opencode/OpencodeSettingsTab.test.ts +649 -0
  606. package/tests/unit/providers/opencode/OpencodeSqliteReader.test.ts +185 -0
  607. package/tests/unit/providers/opencode/agents/OpencodeAgentMentionProvider.test.ts +56 -0
  608. package/tests/unit/providers/opencode/buildOpencodePrompt.test.ts +87 -0
  609. package/tests/unit/providers/opencode/capabilities.test.ts +39 -0
  610. package/tests/unit/providers/opencode/models.test.ts +273 -0
  611. package/tests/unit/providers/opencode/modes.test.ts +151 -0
  612. package/tests/unit/providers/opencode/opencodeToolNormalization.test.ts +197 -0
  613. package/tests/unit/providers/opencode/settings.test.ts +469 -0
  614. package/tests/unit/providers/opencode/storage/OpencodeAgentStorage.test.ts +377 -0
  615. package/tests/unit/providers/opencode/ui/OpencodeAgentSettings.test.ts +91 -0
  616. package/tests/unit/providers/pi/PiRuntimeCommandLoader.test.ts +149 -0
  617. package/tests/unit/providers/pi/capabilities.test.ts +20 -0
  618. package/tests/unit/providers/pi/commands/PiCommandCatalog.test.ts +69 -0
  619. package/tests/unit/providers/pi/env/PiSettingsReconciler.test.ts +61 -0
  620. package/tests/unit/providers/pi/history/PiConversationHistoryService.test.ts +147 -0
  621. package/tests/unit/providers/pi/history/PiHistoryStore.test.ts +523 -0
  622. package/tests/unit/providers/pi/models.test.ts +96 -0
  623. package/tests/unit/providers/pi/registration.test.ts +54 -0
  624. package/tests/unit/providers/pi/runtime/PiAuxQueryRunner.test.ts +172 -0
  625. package/tests/unit/providers/pi/runtime/PiChatRuntime.test.ts +830 -0
  626. package/tests/unit/providers/pi/runtime/PiCliResolver.test.ts +105 -0
  627. package/tests/unit/providers/pi/runtime/PiEventNormalization.test.ts +147 -0
  628. package/tests/unit/providers/pi/runtime/PiExtensionUiBridge.test.ts +98 -0
  629. package/tests/unit/providers/pi/runtime/PiJsonl.test.ts +42 -0
  630. package/tests/unit/providers/pi/runtime/PiLaunchSpec.test.ts +92 -0
  631. package/tests/unit/providers/pi/runtime/PiModelDiscoveryService.test.ts +135 -0
  632. package/tests/unit/providers/pi/runtime/PiRpcPayloads.test.ts +15 -0
  633. package/tests/unit/providers/pi/runtime/PiRpcTransport.test.ts +116 -0
  634. package/tests/unit/providers/pi/runtime/PiSubprocess.test.ts +151 -0
  635. package/tests/unit/providers/pi/runtime/buildPiPrompt.test.ts +84 -0
  636. package/tests/unit/providers/pi/runtime/buildPiUsageInfo.test.ts +69 -0
  637. package/tests/unit/providers/pi/settings.test.ts +253 -0
  638. package/tests/unit/providers/pi/ui/PiChatUIConfig.test.ts +161 -0
  639. package/tests/unit/providers/pi/ui/PiSettingsTab.test.ts +492 -0
  640. package/tests/unit/scripts/rendererSafeUnref.test.ts +101 -0
  641. package/tests/unit/shared/components/ResumeSessionDropdown.test.ts +356 -0
  642. package/tests/unit/shared/components/SelectableDropdown.test.ts +406 -0
  643. package/tests/unit/shared/components/SlashCommandDropdown.provider.test.ts +354 -0
  644. package/tests/unit/shared/components/SlashCommandDropdown.test.ts +508 -0
  645. package/tests/unit/shared/icons.test.ts +56 -0
  646. package/tests/unit/shared/index.test.ts +46 -0
  647. package/tests/unit/shared/mention/MentionDropdownController.test.ts +823 -0
  648. package/tests/unit/shared/mention/VaultFileCache.test.ts +195 -0
  649. package/tests/unit/shared/mention/VaultFolderCache.test.ts +143 -0
  650. package/tests/unit/shared/mention/VaultMentionDataProvider.test.ts +87 -0
  651. package/tests/unit/shared/modals/ConfirmModal.test.ts +111 -0
  652. package/tests/unit/shared/modals/ForkTargetModal.test.ts +101 -0
  653. package/tests/unit/shared/modals/InstructionConfirmModal.test.ts +305 -0
  654. package/tests/unit/utils/agent.test.ts +395 -0
  655. package/tests/unit/utils/animationFrame.test.ts +59 -0
  656. package/tests/unit/utils/browser.test.ts +73 -0
  657. package/tests/unit/utils/canvas.test.ts +54 -0
  658. package/tests/unit/utils/claudeCli.test.ts +294 -0
  659. package/tests/unit/utils/cliBinaryLocator.test.ts +33 -0
  660. package/tests/unit/utils/context.test.ts +288 -0
  661. package/tests/unit/utils/contextMentionResolver.test.ts +270 -0
  662. package/tests/unit/utils/date.test.ts +80 -0
  663. package/tests/unit/utils/diff.test.ts +291 -0
  664. package/tests/unit/utils/editor.test.ts +249 -0
  665. package/tests/unit/utils/electronCompat.test.ts +87 -0
  666. package/tests/unit/utils/env.test.ts +1240 -0
  667. package/tests/unit/utils/externalContext.test.ts +336 -0
  668. package/tests/unit/utils/externalContextScanner.test.ts +186 -0
  669. package/tests/unit/utils/fileLink.dom.test.ts +273 -0
  670. package/tests/unit/utils/fileLink.handler.test.ts +64 -0
  671. package/tests/unit/utils/fileLink.test.ts +233 -0
  672. package/tests/unit/utils/frontmatter.test.ts +434 -0
  673. package/tests/unit/utils/imageEmbed.test.ts +407 -0
  674. package/tests/unit/utils/inlineEdit.test.ts +63 -0
  675. package/tests/unit/utils/interrupt.test.ts +73 -0
  676. package/tests/unit/utils/markdown.test.ts +35 -0
  677. package/tests/unit/utils/markdownMath.test.ts +54 -0
  678. package/tests/unit/utils/mcp.test.ts +256 -0
  679. package/tests/unit/utils/obsidianCompat.test.ts +18 -0
  680. package/tests/unit/utils/path.test.ts +677 -0
  681. package/tests/unit/utils/sdkSession.test.ts +2359 -0
  682. package/tests/unit/utils/session.test.ts +971 -0
  683. package/tests/unit/utils/slashCommand.test.ts +778 -0
  684. package/tests/unit/utils/utils.test.ts +809 -0
  685. package/tsconfig.jest.json +8 -0
  686. package/tsconfig.json +26 -0
  687. package/versions.json +4 -0
@@ -0,0 +1,1296 @@
1
+ import * as fs from 'fs';
2
+ import * as os from 'os';
3
+ import * as path from 'path';
4
+
5
+ import {
6
+ buildSystemPrompt,
7
+ computeSystemPromptKey,
8
+ type SystemPromptSettings,
9
+ } from '../../../core/prompt/mainAgent';
10
+ import { ProviderSettingsCoordinator } from '../../../core/providers/ProviderSettingsCoordinator';
11
+ import type { ProviderCapabilities, ProviderId } from '../../../core/providers/types';
12
+ import type { ChatRuntime } from '../../../core/runtime/ChatRuntime';
13
+ import type {
14
+ ApprovalCallback,
15
+ AskUserQuestionCallback,
16
+ AutoTurnCallback,
17
+ ChatRewindMode,
18
+ ChatRewindResult,
19
+ ChatRuntimeConversationState,
20
+ ChatRuntimeEnsureReadyOptions,
21
+ ChatRuntimeQueryOptions,
22
+ ChatTurnMetadata,
23
+ ChatTurnRequest,
24
+ ExitPlanModeCallback,
25
+ PreparedChatTurn,
26
+ SessionUpdateResult,
27
+ SubagentRuntimeState,
28
+ } from '../../../core/runtime/types';
29
+ import type { ChatMessage, Conversation, ForkSource, SlashCommand, StreamChunk } from '../../../core/types';
30
+ import type ClaudianPlugin from '../../../main';
31
+ import { getVaultPath } from '../../../utils/path';
32
+ import { buildContextFromHistory } from '../../../utils/session';
33
+ import { CODEX_PROVIDER_CAPABILITIES } from '../capabilities';
34
+ import {
35
+ deriveCodexMemoriesDirFromSessionsRoot,
36
+ deriveCodexSessionsRootFromSessionPath,
37
+ findCodexSessionFile,
38
+ } from '../history/CodexHistoryStore';
39
+ import { toCodexRuntimeModelId } from '../modelSelection';
40
+ import { encodeCodexTurn } from '../prompt/encodeCodexTurn';
41
+ import {
42
+ type CodexSafeMode,
43
+ getCodexProviderSettings,
44
+ getEffectiveCodexReasoningSummary,
45
+ } from '../settings';
46
+ import {
47
+ extractExplicitCodexSkillNames,
48
+ findPreferredCodexSkillByName,
49
+ } from '../skills/CodexSkillListingService';
50
+ import { type CodexProviderState, getCodexState } from '../types';
51
+ import { DEFAULT_CODEX_PRIMARY_MODEL, FAST_TIER_CODEX_MODEL } from '../types/models';
52
+ import { CodexAppServerProcess } from './CodexAppServerProcess';
53
+ import {
54
+ initializeCodexAppServerTransport,
55
+ resolveCodexAppServerLaunchSpec,
56
+ } from './codexAppServerSupport';
57
+ import type {
58
+ SandboxPolicy,
59
+ ServerRequestResolvedNotification,
60
+ SkillInput,
61
+ SkillsListResult,
62
+ ThreadCompactStartResult,
63
+ ThreadForkResult,
64
+ ThreadResumeResult,
65
+ ThreadRollbackResult,
66
+ ThreadStartResult,
67
+ TurnStartedNotification,
68
+ TurnStartResult,
69
+ TurnSteerResult,
70
+ UserInput,
71
+ } from './codexAppServerTypes';
72
+ import type { CodexLaunchSpec } from './codexLaunchTypes';
73
+ import { CodexNotificationRouter } from './CodexNotificationRouter';
74
+ import { CodexRpcTransport } from './CodexRpcTransport';
75
+ import { type CodexRuntimeContext, createCodexRuntimeContext } from './CodexRuntimeContext';
76
+ import { CodexServerRequestRouter } from './CodexServerRequestRouter';
77
+ import { CodexSessionManager } from './CodexSessionManager';
78
+
79
+ function resolveCodexSandboxConfig(
80
+ permissionMode: string,
81
+ codexSafeMode: CodexSafeMode = 'workspace-write',
82
+ ): { approvalPolicy: string; sandbox: string } {
83
+ if (permissionMode === 'yolo') {
84
+ return { approvalPolicy: 'never', sandbox: 'danger-full-access' };
85
+ }
86
+ if (permissionMode === 'plan') {
87
+ return { approvalPolicy: 'on-request', sandbox: 'workspace-write' };
88
+ }
89
+ // normal — resolve through the user's configured safe mode
90
+ return { approvalPolicy: 'on-request', sandbox: codexSafeMode };
91
+ }
92
+
93
+ function resolveCodexServiceTier(serviceTier: unknown, model: string | undefined): string | null {
94
+ if (model !== FAST_TIER_CODEX_MODEL) {
95
+ return null;
96
+ }
97
+ return serviceTier === 'fast' ? 'fast' : null;
98
+ }
99
+
100
+ const EFFORT_MAP: Record<string, string> = {
101
+ low: 'low',
102
+ medium: 'medium',
103
+ high: 'high',
104
+ xhigh: 'xhigh',
105
+ };
106
+
107
+ export class CodexChatRuntime implements ChatRuntime {
108
+ readonly providerId: ProviderId = 'codex';
109
+
110
+ private plugin: ClaudianPlugin;
111
+ private session = new CodexSessionManager();
112
+ private process: CodexAppServerProcess | null = null;
113
+ private transport: CodexRpcTransport | null = null;
114
+ private launchSpec: CodexLaunchSpec | null = null;
115
+ private runtimeContext: CodexRuntimeContext | null = null;
116
+ private notificationRouter: CodexNotificationRouter | null = null;
117
+ private serverRequestRouter = new CodexServerRequestRouter();
118
+ private ready = false;
119
+ private readyListeners = new Set<(ready: boolean) => void>();
120
+ private clientConfigKey: string | null = null;
121
+ private currentTurnId: string | null = null;
122
+ private currentQueryThreadId: string | null = null;
123
+ private loadedThreadId: string | null = null;
124
+ private currentThreadPath: string | null = null;
125
+ private pendingTurnNotifications: Array<{ method: string; params: unknown }> = [];
126
+
127
+ // Chunk buffer: notifications push here, query() drains
128
+ private chunkBuffer: StreamChunk[] = [];
129
+ private chunkResolve: (() => void) | null = null;
130
+
131
+ private approvalCallback: ApprovalCallback | null = null;
132
+ private approvalDismisser: (() => void) | null = null;
133
+ private askUserCallback: AskUserQuestionCallback | null = null;
134
+ private exitPlanModeCallback: ExitPlanModeCallback | null = null;
135
+ private permissionModeSyncCallback: ((sdkMode: string) => void) | null = null;
136
+ private subagentHookProvider: (() => SubagentRuntimeState) | null = null;
137
+ private autoTurnCallback: AutoTurnCallback | null = null;
138
+ private resumeCheckpoint: string | undefined;
139
+ private activeInputBundles = new Set<CodexInputBundle>();
140
+
141
+ // Fork state
142
+ private pendingFork: ForkSource | null = null;
143
+
144
+ // Cancellation
145
+ private canceled = false;
146
+ private turnMetadata: ChatTurnMetadata = {};
147
+
148
+ constructor(plugin: ClaudianPlugin) {
149
+ this.plugin = plugin;
150
+ }
151
+
152
+ getCapabilities(): Readonly<ProviderCapabilities> {
153
+ return CODEX_PROVIDER_CAPABILITIES;
154
+ }
155
+
156
+ prepareTurn(request: ChatTurnRequest): PreparedChatTurn {
157
+ return encodeCodexTurn(request);
158
+ }
159
+
160
+ consumeTurnMetadata(): ChatTurnMetadata {
161
+ const metadata = { ...this.turnMetadata };
162
+ this.turnMetadata = {};
163
+ return metadata;
164
+ }
165
+
166
+ onReadyStateChange(listener: (ready: boolean) => void): () => void {
167
+ this.readyListeners.add(listener);
168
+ return () => {
169
+ this.readyListeners.delete(listener);
170
+ };
171
+ }
172
+
173
+ setResumeCheckpoint(checkpointId: string | undefined): void {
174
+ this.resumeCheckpoint = checkpointId;
175
+ }
176
+
177
+ syncConversationState(
178
+ conversation: ChatRuntimeConversationState | null,
179
+ _externalContextPaths?: string[],
180
+ ): void {
181
+ if (!conversation) {
182
+ this.session.reset();
183
+ this.loadedThreadId = null;
184
+ this.currentThreadPath = null;
185
+ this.pendingFork = null;
186
+ return;
187
+ }
188
+
189
+ const state = getCodexState(conversation.providerState);
190
+
191
+ // Pending fork: store fork metadata, don't set the source thread as our session
192
+ if (state.forkSource && !state.threadId && !conversation.sessionId) {
193
+ this.pendingFork = state.forkSource;
194
+ this.session.reset();
195
+ this.loadedThreadId = null;
196
+ this.currentThreadPath = null;
197
+ return;
198
+ }
199
+
200
+ this.pendingFork = null;
201
+ const threadId = state.threadId ?? conversation.sessionId ?? null;
202
+
203
+ if (!threadId) {
204
+ this.session.reset();
205
+ this.loadedThreadId = null;
206
+ this.currentThreadPath = null;
207
+ return;
208
+ }
209
+
210
+ this.session.setThread(threadId, state.sessionFilePath);
211
+ }
212
+
213
+ async reloadMcpServers(): Promise<void> {
214
+ // No-op: Codex handles MCP internally
215
+ }
216
+
217
+ async ensureReady(options?: ChatRuntimeEnsureReadyOptions): Promise<boolean> {
218
+ const promptSettings = this.getSystemPromptSettings();
219
+ const promptKey = computeSystemPromptKey(promptSettings);
220
+ const launchSpec = resolveCodexAppServerLaunchSpec(this.plugin, this.providerId);
221
+ const clientConfigKey = [promptKey, JSON.stringify({
222
+ command: launchSpec.command,
223
+ args: launchSpec.args,
224
+ spawnCwd: launchSpec.spawnCwd,
225
+ targetCwd: launchSpec.targetCwd,
226
+ target: launchSpec.target,
227
+ })].join('::');
228
+ const shouldRebuild = !this.process
229
+ || !this.transport
230
+ || !this.process.isAlive()
231
+ || options?.force === true
232
+ || this.clientConfigKey !== clientConfigKey;
233
+
234
+ if (shouldRebuild) {
235
+ await this.shutdownProcess();
236
+ await this.startAppServer(launchSpec, clientConfigKey);
237
+ }
238
+
239
+ this.setReady(true);
240
+ return shouldRebuild;
241
+ }
242
+
243
+ async *query(
244
+ originalTurn: PreparedChatTurn,
245
+ _conversationHistory?: ChatMessage[],
246
+ queryOptions?: ChatRuntimeQueryOptions,
247
+ ): AsyncGenerator<StreamChunk> {
248
+ this.resetTurnMetadata();
249
+ let turn = originalTurn;
250
+ await this.ensureReady();
251
+
252
+ this.canceled = false;
253
+ this.cleanupActiveInputBundles();
254
+ this.chunkBuffer = [];
255
+ this.chunkResolve = null;
256
+ this.currentQueryThreadId = null;
257
+ this.pendingTurnNotifications = [];
258
+
259
+ const model = this.resolveModel(queryOptions);
260
+ const promptSettings = this.getSystemPromptSettings();
261
+ const promptText = buildSystemPrompt(promptSettings);
262
+
263
+ const enqueueChunk = (chunk: StreamChunk): void => {
264
+ this.chunkBuffer.push(chunk);
265
+ if (this.chunkResolve) {
266
+ this.chunkResolve();
267
+ this.chunkResolve = null;
268
+ }
269
+ };
270
+
271
+ // Set up notification router to push chunks
272
+ this.notificationRouter = new CodexNotificationRouter(
273
+ (chunk) => enqueueChunk(chunk),
274
+ (update) => this.recordTurnMetadata(update),
275
+ );
276
+
277
+ this.wireTransportHandlers();
278
+
279
+ const compactValidationError = this.validateCompactTurn(originalTurn);
280
+ if (compactValidationError) {
281
+ yield { type: 'error', content: compactValidationError };
282
+ yield { type: 'done' };
283
+ return;
284
+ }
285
+
286
+ try {
287
+ // Thread lifecycle
288
+ const existingThreadId = this.session.getThreadId();
289
+ let threadId: string;
290
+ let threadPath: string | null = null;
291
+ let threadTargetPath: string | null = null;
292
+ let completedPendingFork = false;
293
+
294
+ if (this.pendingFork) {
295
+ // Pending fork: fork the source thread, optionally roll back, then start a turn
296
+ const fork = this.pendingFork;
297
+
298
+ const forkResult = await this.transport!.request<ThreadForkResult>('thread/fork', {
299
+ threadId: fork.sessionId,
300
+ });
301
+ threadId = forkResult.thread.id;
302
+ threadTargetPath = forkResult.thread.path ?? null;
303
+ threadPath = this.toHostSessionPath(threadTargetPath);
304
+
305
+ // Compute rollback: count turns after the resumeAt checkpoint
306
+ const forkTurns = forkResult.thread.turns ?? [];
307
+ const checkpointIndex = forkTurns.findIndex(t => t.id === fork.resumeAt);
308
+ if (checkpointIndex < 0) {
309
+ throw new Error(`Fork checkpoint not found: ${fork.resumeAt}`);
310
+ }
311
+ const numTurnsToRollback = forkTurns.length - checkpointIndex - 1;
312
+
313
+ // Resume the forked thread (required before rollback and turn/start)
314
+ const permissionMode = this.resolveSandboxConfig();
315
+ await this.transport!.request<ThreadResumeResult>('thread/resume', {
316
+ threadId,
317
+ model: model ?? DEFAULT_CODEX_PRIMARY_MODEL,
318
+ approvalPolicy: permissionMode.approvalPolicy,
319
+ sandbox: permissionMode.sandbox,
320
+ serviceTier: resolveCodexServiceTier(this.getProviderSettings().serviceTier, model ?? DEFAULT_CODEX_PRIMARY_MODEL),
321
+ baseInstructions: promptText,
322
+ experimentalRawEvents: true,
323
+ persistExtendedHistory: true,
324
+ });
325
+
326
+ if (numTurnsToRollback > 0) {
327
+ await this.transport!.request<ThreadRollbackResult>('thread/rollback', {
328
+ threadId,
329
+ numTurns: numTurnsToRollback,
330
+ });
331
+ }
332
+
333
+ this.loadedThreadId = threadId;
334
+ completedPendingFork = true;
335
+
336
+ // Build replay suffix from conversation history after the checkpoint
337
+ if (_conversationHistory && _conversationHistory.length > 0) {
338
+ const checkpointIdx = _conversationHistory.findIndex(
339
+ m => m.assistantMessageId === fork.resumeAt,
340
+ );
341
+ if (checkpointIdx >= 0 && checkpointIdx < _conversationHistory.length - 1) {
342
+ const suffix = _conversationHistory.slice(checkpointIdx + 1);
343
+ const replayContext = buildContextFromHistory(suffix);
344
+ if (replayContext.trim()) {
345
+ turn = {
346
+ ...turn,
347
+ prompt: `${replayContext}\n\nUser: ${turn.prompt}`,
348
+ };
349
+ }
350
+ }
351
+ }
352
+ } else if (existingThreadId && existingThreadId !== this.loadedThreadId) {
353
+ // Resume a persisted thread not yet loaded in this daemon
354
+ const permissionMode = this.resolveSandboxConfig();
355
+ const resumeResult = await this.transport!.request<ThreadResumeResult>('thread/resume', {
356
+ threadId: existingThreadId,
357
+ model: model ?? DEFAULT_CODEX_PRIMARY_MODEL,
358
+ approvalPolicy: permissionMode.approvalPolicy,
359
+ sandbox: permissionMode.sandbox,
360
+ serviceTier: resolveCodexServiceTier(this.getProviderSettings().serviceTier, model ?? DEFAULT_CODEX_PRIMARY_MODEL),
361
+ baseInstructions: promptText,
362
+ experimentalRawEvents: true,
363
+ persistExtendedHistory: true,
364
+ });
365
+ threadId = resumeResult.thread.id;
366
+ threadTargetPath = resumeResult.thread.path ?? null;
367
+ threadPath = this.toHostSessionPath(threadTargetPath);
368
+ this.loadedThreadId = threadId;
369
+ } else if (existingThreadId && existingThreadId === this.loadedThreadId) {
370
+ // Thread already loaded — just start a new turn
371
+ threadId = existingThreadId;
372
+ } else {
373
+ // New thread
374
+ const permissionMode = this.resolveSandboxConfig();
375
+ const startResult = await this.transport!.request<ThreadStartResult>('thread/start', {
376
+ model: model ?? DEFAULT_CODEX_PRIMARY_MODEL,
377
+ cwd: this.launchSpec?.targetCwd ?? getVaultPath(this.plugin.app) ?? undefined,
378
+ approvalPolicy: permissionMode.approvalPolicy,
379
+ sandbox: permissionMode.sandbox,
380
+ serviceTier: resolveCodexServiceTier(this.getProviderSettings().serviceTier, model ?? DEFAULT_CODEX_PRIMARY_MODEL),
381
+ baseInstructions: promptText,
382
+ experimentalRawEvents: true,
383
+ persistExtendedHistory: true,
384
+ });
385
+ threadId = startResult.thread.id;
386
+ threadTargetPath = startResult.thread.path ?? null;
387
+ threadPath = this.toHostSessionPath(threadTargetPath);
388
+ this.loadedThreadId = threadId;
389
+ }
390
+
391
+ // Update session with thread info
392
+ this.session.setThread(threadId, threadPath ?? this.currentThreadPath ?? undefined);
393
+ if (threadPath) this.currentThreadPath = threadPath;
394
+ this.currentQueryThreadId = threadId;
395
+ if (completedPendingFork) {
396
+ this.pendingFork = null;
397
+ }
398
+
399
+ if (turn.isCompact) {
400
+ // --- Manual compact path: thread/compact/start ---
401
+ this.notificationRouter?.beginTurn({ isPlanTurn: false });
402
+
403
+ await this.transport!.request<ThreadCompactStartResult>(
404
+ 'thread/compact/start',
405
+ { threadId },
406
+ );
407
+ this.recordTurnMetadata({ wasSent: true });
408
+ // currentTurnId will be set by turn/started notification
409
+ } else {
410
+ // --- Normal turn path ---
411
+ const sessionFilePathHint = threadPath ?? this.session.getSessionFilePath() ?? null;
412
+
413
+ // Build input
414
+ const skillInputs = await this.resolveSkillInputs(turn.request.text);
415
+ const turnInputBundle = this.buildInput(turn.prompt, turn.request.images, skillInputs);
416
+ this.registerActiveInputBundle(turnInputBundle);
417
+
418
+ // Start turn
419
+ const providerSettings = this.getProviderSettings();
420
+ const effort = EFFORT_MAP[providerSettings.effortLevel as string] ?? 'medium';
421
+ const resolvedModel = model ?? DEFAULT_CODEX_PRIMARY_MODEL;
422
+ const isPlanMode = providerSettings.permissionMode === 'plan';
423
+ const externalContextPaths = this.resolveExternalContextPaths(turn, queryOptions);
424
+ const permissionMode = this.resolveSandboxConfig();
425
+ const transcriptRootTarget = this.runtimeContext?.sessionsDirTarget
426
+ ?? deriveCodexSessionsRootFromSessionPath(threadTargetPath)
427
+ ?? this.resolveTranscriptRootTarget(sessionFilePathHint);
428
+ const sandboxPolicy = this.buildTurnSandboxPolicy(
429
+ externalContextPaths,
430
+ permissionMode.sandbox,
431
+ transcriptRootTarget,
432
+ sessionFilePathHint,
433
+ );
434
+
435
+ const collaborationMode = {
436
+ mode: isPlanMode ? 'plan' as const : 'default' as const,
437
+ settings: {
438
+ model: resolvedModel,
439
+ reasoning_effort: effort,
440
+ developer_instructions: null,
441
+ },
442
+ };
443
+
444
+ const summary = getEffectiveCodexReasoningSummary(providerSettings, resolvedModel);
445
+ const serviceTier = resolveCodexServiceTier(providerSettings.serviceTier, resolvedModel);
446
+
447
+ // Configure router plan state before turn/start so buffered notifications
448
+ // that arrive before currentTurnId is set already see the correct state.
449
+ this.notificationRouter?.beginTurn({ isPlanTurn: isPlanMode });
450
+
451
+ const turnResult = await this.transport!.request<TurnStartResult>('turn/start', {
452
+ threadId,
453
+ input: turnInputBundle.input,
454
+ approvalPolicy: permissionMode.approvalPolicy,
455
+ model: resolvedModel,
456
+ serviceTier,
457
+ effort,
458
+ summary,
459
+ sandboxPolicy,
460
+ collaborationMode,
461
+ });
462
+ this.currentTurnId = turnResult.turn.id;
463
+ this.recordTurnMetadata({
464
+ userMessageId: turnResult.turn.id,
465
+ wasSent: true,
466
+ });
467
+ this.flushPendingTurnNotifications();
468
+ }
469
+
470
+ // Yield chunks until done or canceled
471
+ while (true) {
472
+ if (this.canceled) {
473
+ // Drain remaining chunks before exiting
474
+ while (this.chunkBuffer.length > 0) {
475
+ const chunk = this.chunkBuffer.shift()!;
476
+ yield chunk;
477
+ if (chunk.type === 'done') return;
478
+ }
479
+ yield { type: 'done' };
480
+ return;
481
+ }
482
+
483
+ if (this.chunkBuffer.length === 0) {
484
+ await new Promise<void>((resolve) => {
485
+ this.chunkResolve = resolve;
486
+ if (this.chunkBuffer.length > 0 || this.canceled) {
487
+ resolve();
488
+ this.chunkResolve = null;
489
+ }
490
+ });
491
+ }
492
+
493
+ while (this.chunkBuffer.length > 0) {
494
+ const chunk = this.chunkBuffer.shift()!;
495
+ yield chunk;
496
+ if (chunk.type === 'done') {
497
+ return;
498
+ }
499
+ }
500
+ }
501
+ } catch (err: unknown) {
502
+ if (this.canceled) {
503
+ yield { type: 'done' };
504
+ return;
505
+ }
506
+ const message = err instanceof Error ? err.message : 'Unknown Codex error';
507
+ yield { type: 'error', content: message };
508
+ yield { type: 'done' };
509
+ return;
510
+ } finally {
511
+ this.notificationRouter?.endTurn();
512
+
513
+ this.cleanupActiveInputBundles();
514
+ this.currentTurnId = null;
515
+ this.currentQueryThreadId = null;
516
+ this.pendingTurnNotifications = [];
517
+
518
+ // Session file discovery fallback
519
+ if (!this.session.getSessionFilePath()) {
520
+ const threadId = this.session.getThreadId();
521
+ if (threadId) {
522
+ const sessionFilePath = findCodexSessionFile(
523
+ threadId,
524
+ this.resolveTranscriptRootHost(this.session.getSessionFilePath() ?? this.currentThreadPath) ?? undefined,
525
+ );
526
+ if (sessionFilePath) {
527
+ this.session.setThread(threadId, sessionFilePath);
528
+ }
529
+ }
530
+ }
531
+ }
532
+ }
533
+
534
+ async steer(turn: PreparedChatTurn): Promise<boolean> {
535
+ if (turn.isCompact || this.canceled) {
536
+ return false;
537
+ }
538
+
539
+ const transport = this.transport;
540
+ const threadId = this.currentQueryThreadId;
541
+ const turnId = this.currentTurnId;
542
+ if (!transport || !threadId || !turnId) {
543
+ return false;
544
+ }
545
+
546
+ const skillInputs = await this.resolveSkillInputs(turn.request.text);
547
+ const inputBundle = this.buildInput(turn.prompt, turn.request.images, skillInputs);
548
+ this.registerActiveInputBundle(inputBundle);
549
+
550
+ try {
551
+ const result = await transport.request<TurnSteerResult>('turn/steer', {
552
+ threadId,
553
+ input: inputBundle.input,
554
+ expectedTurnId: turnId,
555
+ });
556
+
557
+ if (result.turnId !== turnId) {
558
+ return false;
559
+ }
560
+
561
+ return this.currentQueryThreadId === threadId
562
+ && this.currentTurnId === turnId
563
+ && !this.canceled;
564
+ } catch (error) {
565
+ this.disposeInputBundle(inputBundle);
566
+ throw error;
567
+ }
568
+ }
569
+
570
+ cancel(): void {
571
+ this.canceled = true;
572
+ this.dismissAllPendingPrompts();
573
+
574
+ const threadId = this.session.getThreadId();
575
+ const turnId = this.currentTurnId;
576
+
577
+ if (this.transport && threadId && turnId) {
578
+ this.transport.request('turn/interrupt', { threadId, turnId }).catch(() => {
579
+ // best-effort
580
+ });
581
+ }
582
+
583
+ // Unblock the chunk-wait loop
584
+ if (this.chunkResolve) {
585
+ this.chunkResolve();
586
+ this.chunkResolve = null;
587
+ }
588
+ }
589
+
590
+ resetSession(): void {
591
+ this.teardownState();
592
+ }
593
+
594
+ getSessionId(): string | null {
595
+ return this.session.getThreadId();
596
+ }
597
+
598
+ consumeSessionInvalidation(): boolean {
599
+ return this.session.consumeInvalidation();
600
+ }
601
+
602
+ isReady(): boolean {
603
+ return this.ready;
604
+ }
605
+
606
+ private resetTurnMetadata(): void {
607
+ this.turnMetadata = {};
608
+ }
609
+
610
+ private recordTurnMetadata(update: Partial<ChatTurnMetadata>): void {
611
+ this.turnMetadata = {
612
+ ...this.turnMetadata,
613
+ ...update,
614
+ };
615
+ }
616
+
617
+ async getSupportedCommands(): Promise<SlashCommand[]> {
618
+ return [];
619
+ }
620
+
621
+ cleanup(): void {
622
+ this.cancel();
623
+ this.teardownState();
624
+ this.readyListeners.clear();
625
+ }
626
+
627
+ async rewind(
628
+ _userMessageId: string,
629
+ _assistantMessageId: string,
630
+ _mode?: ChatRewindMode,
631
+ ): Promise<ChatRewindResult> {
632
+ return { canRewind: false, error: 'Codex does not support rewind' };
633
+ }
634
+
635
+ setApprovalCallback(callback: ApprovalCallback | null): void {
636
+ this.approvalCallback = callback;
637
+ this.serverRequestRouter.setApprovalCallback(callback);
638
+ }
639
+
640
+ setApprovalDismisser(dismisser: (() => void) | null): void {
641
+ this.approvalDismisser = dismisser;
642
+ }
643
+
644
+ setAskUserQuestionCallback(callback: AskUserQuestionCallback | null): void {
645
+ this.askUserCallback = callback;
646
+ this.serverRequestRouter.setAskUserCallback(callback);
647
+ }
648
+
649
+ setExitPlanModeCallback(callback: ExitPlanModeCallback | null): void {
650
+ this.exitPlanModeCallback = callback;
651
+ }
652
+
653
+ setPermissionModeSyncCallback(callback: ((sdkMode: string) => void) | null): void {
654
+ this.permissionModeSyncCallback = callback;
655
+ }
656
+
657
+ setSubagentHookProvider(getState: () => SubagentRuntimeState): void {
658
+ this.subagentHookProvider = getState;
659
+ }
660
+
661
+ setAutoTurnCallback(callback: AutoTurnCallback | null): void {
662
+ this.autoTurnCallback = callback;
663
+ }
664
+
665
+ buildSessionUpdates(params: {
666
+ conversation: Conversation | null;
667
+ sessionInvalidated: boolean;
668
+ }): SessionUpdateResult {
669
+ const threadId = this.session.getThreadId();
670
+ const sessionFilePath = this.session.getSessionFilePath() ?? this.currentThreadPath;
671
+ const transcriptRootPath = this.resolveTranscriptRootHost(sessionFilePath);
672
+
673
+ // Preserve forkSource from existing conversation state
674
+ const existingState = params.conversation
675
+ ? getCodexState(params.conversation.providerState)
676
+ : null;
677
+
678
+ const providerState: CodexProviderState = {
679
+ ...(threadId ? { threadId } : {}),
680
+ ...(sessionFilePath ? { sessionFilePath } : {}),
681
+ ...(
682
+ transcriptRootPath || existingState?.transcriptRootPath
683
+ ? { transcriptRootPath: transcriptRootPath ?? existingState?.transcriptRootPath }
684
+ : {}
685
+ ),
686
+ ...(existingState?.forkSource ? { forkSource: existingState.forkSource } : {}),
687
+ ...(
688
+ existingState?.forkSourceSessionFilePath
689
+ ? { forkSourceSessionFilePath: existingState.forkSourceSessionFilePath }
690
+ : {}
691
+ ),
692
+ ...(
693
+ existingState?.forkSourceTranscriptRootPath
694
+ ? { forkSourceTranscriptRootPath: existingState.forkSourceTranscriptRootPath }
695
+ : {}
696
+ ),
697
+ };
698
+
699
+ const updates: Partial<Conversation> = {
700
+ sessionId: threadId,
701
+ providerState: providerState as Record<string, unknown>,
702
+ };
703
+
704
+ if (params.sessionInvalidated && params.conversation) {
705
+ updates.sessionId = null;
706
+ updates.providerState = undefined;
707
+ }
708
+
709
+ return { updates };
710
+ }
711
+
712
+ resolveSessionIdForFork(conversation: Conversation | null): string | null {
713
+ const threadId = this.session.getThreadId();
714
+ if (threadId) return threadId;
715
+
716
+ if (!conversation) return null;
717
+ const state = getCodexState(conversation.providerState);
718
+ return state.threadId ?? conversation.sessionId ?? state.forkSource?.sessionId ?? null;
719
+ }
720
+
721
+ // -----------------------------------------------------------------------
722
+ // Private helpers
723
+ // -----------------------------------------------------------------------
724
+
725
+ private teardownState(): void {
726
+ this.cleanupActiveInputBundles();
727
+ this.session.reset();
728
+ this.launchSpec = null;
729
+ this.runtimeContext = null;
730
+ this.loadedThreadId = null;
731
+ this.currentThreadPath = null;
732
+ this.currentTurnId = null;
733
+ this.currentQueryThreadId = null;
734
+ this.pendingTurnNotifications = [];
735
+ this.pendingFork = null;
736
+ this.clientConfigKey = null;
737
+ this.shutdownProcess().catch(() => {});
738
+ this.setReady(false);
739
+ }
740
+
741
+ private dismissApprovalUI(): void {
742
+ if (this.approvalDismisser) {
743
+ this.approvalDismisser();
744
+ }
745
+ }
746
+
747
+ private dismissAllPendingPrompts(): void {
748
+ this.dismissApprovalUI();
749
+ this.serverRequestRouter.abortPendingAskUser();
750
+ }
751
+
752
+ private registerActiveInputBundle(bundle: CodexInputBundle): void {
753
+ this.activeInputBundles.add(bundle);
754
+ }
755
+
756
+ private disposeInputBundle(bundle: CodexInputBundle): void {
757
+ if (this.activeInputBundles.delete(bundle)) {
758
+ bundle.cleanup();
759
+ return;
760
+ }
761
+
762
+ bundle.cleanup();
763
+ }
764
+
765
+ private cleanupActiveInputBundles(): void {
766
+ for (const bundle of this.activeInputBundles) {
767
+ bundle.cleanup();
768
+ }
769
+ this.activeInputBundles.clear();
770
+ }
771
+
772
+ private setReady(ready: boolean): void {
773
+ this.ready = ready;
774
+ for (const listener of this.readyListeners) {
775
+ listener(ready);
776
+ }
777
+ }
778
+
779
+ private getSystemPromptSettings(): SystemPromptSettings {
780
+ const settings = this.plugin.settings;
781
+ return {
782
+ mediaFolder: settings.mediaFolder,
783
+ customPrompt: settings.systemPrompt,
784
+ vaultPath: getVaultPath(this.plugin.app) ?? undefined,
785
+ userName: settings.userName,
786
+ };
787
+ }
788
+
789
+ private getProviderSettings(): Record<string, unknown> {
790
+ return ProviderSettingsCoordinator.getProviderSettingsSnapshot(
791
+ this.plugin.settings,
792
+ this.providerId,
793
+ );
794
+ }
795
+
796
+ getAuxiliaryModel(): string | null {
797
+ return this.resolveModel() ?? null;
798
+ }
799
+
800
+ private resolveModel(queryOptions?: ChatRuntimeQueryOptions): string | undefined {
801
+ const providerSettings = this.getProviderSettings();
802
+ const model = queryOptions?.model ?? providerSettings.model as string | undefined;
803
+ return model ? toCodexRuntimeModelId(model) : undefined;
804
+ }
805
+
806
+ private resolveSandboxConfig(): { approvalPolicy: string; sandbox: string } {
807
+ const providerSettings = this.getProviderSettings();
808
+ return resolveCodexSandboxConfig(
809
+ providerSettings.permissionMode as string,
810
+ getCodexProviderSettings(providerSettings).safeMode,
811
+ );
812
+ }
813
+
814
+ private async startAppServer(launchSpec: CodexLaunchSpec, clientConfigKey: string): Promise<void> {
815
+ this.launchSpec = launchSpec;
816
+ this.process = new CodexAppServerProcess(launchSpec);
817
+ this.process.start();
818
+
819
+ this.transport = new CodexRpcTransport(this.process);
820
+ this.transport.start();
821
+
822
+ const initializeResult = await initializeCodexAppServerTransport(this.transport);
823
+ this.runtimeContext = createCodexRuntimeContext(launchSpec, initializeResult);
824
+ this.clientConfigKey = clientConfigKey;
825
+ }
826
+
827
+ private wireTransportHandlers(): void {
828
+ if (!this.transport || !this.notificationRouter) return;
829
+
830
+ const router = this.notificationRouter;
831
+ const methods = [
832
+ 'item/agentMessage/delta',
833
+ 'item/started',
834
+ 'item/completed',
835
+ 'item/plan/delta',
836
+ 'item/reasoning/textDelta',
837
+ 'item/reasoning/summaryTextDelta',
838
+ 'item/reasoning/summaryPartAdded',
839
+ 'thread/tokenUsage/updated',
840
+ 'turn/plan/updated',
841
+ 'turn/completed',
842
+ 'error',
843
+ 'thread/started',
844
+ 'thread/status/changed',
845
+ 'turn/started',
846
+ 'serverRequest/resolved',
847
+ 'item/commandExecution/outputDelta',
848
+ 'item/fileChange/outputDelta',
849
+ 'item/fileChange/patchUpdated',
850
+ 'rawResponseItem/completed',
851
+ 'event_msg',
852
+ ];
853
+
854
+ for (const method of methods) {
855
+ this.transport.onNotification(method, (params) => {
856
+ if (method === 'serverRequest/resolved') {
857
+ this.handleServerRequestResolved(params as ServerRequestResolvedNotification);
858
+ return;
859
+ }
860
+ if (!this.routeNotification(method, params)) {
861
+ return;
862
+ }
863
+ router.handleNotification(method, params);
864
+ });
865
+ }
866
+
867
+ // Server requests (approvals, ask-user)
868
+ const requestMethods = [
869
+ 'item/commandExecution/requestApproval',
870
+ 'item/fileChange/requestApproval',
871
+ 'item/permissions/requestApproval',
872
+ 'item/tool/requestUserInput',
873
+ ];
874
+
875
+ for (const method of requestMethods) {
876
+ this.transport.onServerRequest(method, (requestId, params) => {
877
+ return this.serverRequestRouter.handleServerRequest(requestId, method, params);
878
+ });
879
+ }
880
+ }
881
+
882
+ private async shutdownProcess(): Promise<void> {
883
+ if (this.transport) {
884
+ this.transport.dispose();
885
+ this.transport = null;
886
+ }
887
+ if (this.process) {
888
+ await this.process.shutdown();
889
+ this.process = null;
890
+ }
891
+ this.launchSpec = null;
892
+ this.runtimeContext = null;
893
+ this.notificationRouter = null;
894
+ this.currentTurnId = null;
895
+ this.currentQueryThreadId = null;
896
+ this.pendingTurnNotifications = [];
897
+ this.loadedThreadId = null;
898
+ }
899
+
900
+ private resolveExternalContextPaths(
901
+ turn: PreparedChatTurn,
902
+ queryOptions?: ChatRuntimeQueryOptions,
903
+ ): string[] {
904
+ const externalContextPaths = turn.request.externalContextPaths ?? queryOptions?.externalContextPaths ?? [];
905
+ return [...new Set(externalContextPaths.filter((value): value is string => typeof value === 'string' && value.trim().length > 0))];
906
+ }
907
+
908
+ private buildTurnSandboxPolicy(
909
+ externalContextPaths: string[],
910
+ sandboxMode: string,
911
+ transcriptRootTargetHint?: string | null,
912
+ sessionFilePathHint?: string | null,
913
+ ): SandboxPolicy | undefined {
914
+ if (sandboxMode === 'danger-full-access') {
915
+ return { type: 'dangerFullAccess' };
916
+ }
917
+
918
+ if (sandboxMode === 'read-only') {
919
+ return {
920
+ type: 'readOnly',
921
+ access: { type: 'fullAccess' },
922
+ networkAccess: false,
923
+ };
924
+ }
925
+
926
+ if (sandboxMode !== 'workspace-write') {
927
+ return undefined;
928
+ }
929
+
930
+ const mappedExternalContextPaths = this.mapRequiredHostPathsToTarget(
931
+ externalContextPaths,
932
+ 'external context path',
933
+ );
934
+ const memoriesDirTarget = deriveCodexMemoriesDirFromSessionsRoot(transcriptRootTargetHint)
935
+ ?? this.resolveMemoriesDirTarget(sessionFilePathHint)
936
+ ?? (
937
+ this.launchSpec?.target.method === 'wsl'
938
+ ? null
939
+ : path.join(os.homedir(), '.codex', 'memories')
940
+ );
941
+
942
+ const writableRoots = [
943
+ this.launchSpec?.targetCwd ?? getVaultPath(this.plugin.app),
944
+ ...mappedExternalContextPaths,
945
+ memoriesDirTarget,
946
+ this.mapHostPathToTarget(os.tmpdir()),
947
+ this.launchSpec?.target.platformFamily === 'unix' ? '/tmp' : null,
948
+ this.mapHostPathToTarget(process.env.TMPDIR),
949
+ ].filter((value): value is string => typeof value === 'string' && value.trim().length > 0);
950
+
951
+ return {
952
+ type: 'workspaceWrite',
953
+ writableRoots: [...new Set(writableRoots)],
954
+ readOnlyAccess: { type: 'fullAccess' },
955
+ networkAccess: false,
956
+ excludeTmpdirEnvVar: false,
957
+ excludeSlashTmp: false,
958
+ };
959
+ }
960
+
961
+ private handleServerRequestResolved(params: ServerRequestResolvedNotification): void {
962
+ if (this.serverRequestRouter.hasPendingApprovalRequest(params.requestId, params.threadId)) {
963
+ this.dismissApprovalUI();
964
+ return;
965
+ }
966
+
967
+ this.serverRequestRouter.abortPendingAskUser(params.requestId, params.threadId);
968
+ }
969
+
970
+ private routeNotification(
971
+ method: string,
972
+ params: unknown,
973
+ ): boolean {
974
+ // turn/started can establish the active turn ID when the query didn't
975
+ // receive one from the RPC response (e.g. thread/compact/start).
976
+ if (method === 'turn/started') {
977
+ this.handleTurnStartedNotification(params);
978
+ return false;
979
+ }
980
+
981
+ const scope = this.extractNotificationScope(method, params);
982
+ if (!scope) {
983
+ return true;
984
+ }
985
+
986
+ if (!this.currentQueryThreadId || scope.threadId !== this.currentQueryThreadId) {
987
+ return false;
988
+ }
989
+
990
+ if (!this.currentTurnId) {
991
+ this.pendingTurnNotifications.push({ method, params });
992
+ return false;
993
+ }
994
+
995
+ if (scope.turnId !== this.currentTurnId) {
996
+ return false;
997
+ }
998
+
999
+ return true;
1000
+ }
1001
+
1002
+ private handleTurnStartedNotification(params: unknown): void {
1003
+ if (!params || typeof params !== 'object') return;
1004
+
1005
+ const notification = params as TurnStartedNotification;
1006
+ const threadId = notification.threadId;
1007
+ const turnId = notification.turn?.id;
1008
+
1009
+ if (!threadId || !turnId) return;
1010
+ if (threadId !== this.currentQueryThreadId) return;
1011
+
1012
+ // Only establish the turn ID if the current query doesn't have one yet.
1013
+ // Normal turn/start responses already set it; this path covers
1014
+ // thread/compact/start which returns {} without a turn.
1015
+ if (!this.currentTurnId) {
1016
+ this.currentTurnId = turnId;
1017
+ this.flushPendingTurnNotifications();
1018
+ }
1019
+ }
1020
+
1021
+ private validateCompactTurn(turn: PreparedChatTurn): string | null {
1022
+ if (!turn.isCompact) {
1023
+ return null;
1024
+ }
1025
+
1026
+ if (turn.request.text.trim() !== '/compact') {
1027
+ return '/compact does not accept arguments';
1028
+ }
1029
+
1030
+ return null;
1031
+ }
1032
+
1033
+ private flushPendingTurnNotifications(): void {
1034
+ if (!this.notificationRouter || !this.currentTurnId) {
1035
+ this.pendingTurnNotifications = [];
1036
+ return;
1037
+ }
1038
+
1039
+ const pending = this.pendingTurnNotifications;
1040
+ this.pendingTurnNotifications = [];
1041
+
1042
+ for (const notification of pending) {
1043
+ const scope = this.extractNotificationScope(notification.method, notification.params);
1044
+ if (!scope) {
1045
+ this.notificationRouter.handleNotification(notification.method, notification.params);
1046
+ continue;
1047
+ }
1048
+
1049
+ if (
1050
+ scope.threadId === this.currentQueryThreadId
1051
+ && scope.turnId === this.currentTurnId
1052
+ ) {
1053
+ this.notificationRouter.handleNotification(notification.method, notification.params);
1054
+ }
1055
+ }
1056
+ }
1057
+
1058
+ private extractNotificationScope(
1059
+ method: string,
1060
+ params: unknown,
1061
+ ): { threadId: string; turnId: string } | null {
1062
+ if (!params || typeof params !== 'object') {
1063
+ return null;
1064
+ }
1065
+
1066
+ const notification = params as Record<string, unknown>;
1067
+ const threadId = typeof notification.threadId === 'string' ? notification.threadId : null;
1068
+
1069
+ if (method === 'turn/completed') {
1070
+ const turn = notification.turn;
1071
+ const turnId = turn && typeof turn === 'object' && typeof (turn as Record<string, unknown>).id === 'string'
1072
+ ? (turn as Record<string, unknown>).id as string
1073
+ : null;
1074
+
1075
+ return threadId && turnId ? { threadId, turnId } : null;
1076
+ }
1077
+
1078
+ const turnId = typeof notification.turnId === 'string'
1079
+ ? notification.turnId
1080
+ : typeof notification.turn_id === 'string'
1081
+ ? notification.turn_id
1082
+ : null;
1083
+ return threadId && turnId ? { threadId, turnId } : null;
1084
+ }
1085
+
1086
+ private async resolveSkillInputs(text: string): Promise<SkillInput[]> {
1087
+ const skillNames = extractExplicitCodexSkillNames(text);
1088
+ if (skillNames.length === 0 || !this.transport) {
1089
+ return [];
1090
+ }
1091
+
1092
+ try {
1093
+ const cwd = this.launchSpec?.targetCwd ?? getVaultPath(this.plugin.app) ?? process.cwd();
1094
+ const result = await this.transport.request<SkillsListResult>('skills/list', {
1095
+ cwds: [cwd],
1096
+ });
1097
+ const skills = result.data.find(entry => entry.cwd === cwd)?.skills ?? result.data[0]?.skills ?? [];
1098
+ const resolvedInputs: SkillInput[] = [];
1099
+
1100
+ for (const skillName of skillNames) {
1101
+ const resolvedSkill = findPreferredCodexSkillByName(skills, skillName);
1102
+ if (!resolvedSkill) {
1103
+ continue;
1104
+ }
1105
+
1106
+ resolvedInputs.push({
1107
+ type: 'skill',
1108
+ name: resolvedSkill.name,
1109
+ path: resolvedSkill.path,
1110
+ });
1111
+ }
1112
+
1113
+ return resolvedInputs;
1114
+ } catch {
1115
+ return [];
1116
+ }
1117
+ }
1118
+
1119
+ private buildInput(text: string, images?: ImageAttachment[], skills?: SkillInput[]): CodexInputBundle {
1120
+ const input: UserInput[] = [];
1121
+ let tempDir: string | null = null;
1122
+
1123
+ const cleanup = (): void => {
1124
+ if (!tempDir) {
1125
+ return;
1126
+ }
1127
+
1128
+ try {
1129
+ fs.rmSync(tempDir, { recursive: true, force: true });
1130
+ } catch {
1131
+ // best-effort cleanup
1132
+ }
1133
+ };
1134
+
1135
+ try {
1136
+ if (images && images.length > 0) {
1137
+ tempDir = fs.mkdtempSync(path.join(os.tmpdir(), 'claudian-codex-images-'));
1138
+ for (let i = 0; i < images.length; i++) {
1139
+ const img = images[i];
1140
+ if (!img.mediaType.startsWith('image/')) continue;
1141
+
1142
+ const filename = toAttachmentFilename(img, i);
1143
+ const filePath = path.join(tempDir, `${i + 1}-${filename}`);
1144
+ fs.writeFileSync(filePath, Buffer.from(img.data, 'base64'));
1145
+ const targetFilePath = this.mapHostPathToTarget(filePath);
1146
+ if (!targetFilePath) {
1147
+ throw new Error(`Codex cannot access image attachment path from the selected target: ${filePath}`);
1148
+ }
1149
+ input.push({ type: 'localImage', path: targetFilePath });
1150
+ }
1151
+ }
1152
+
1153
+ if (text) {
1154
+ input.push({ type: 'text', text, text_elements: [] });
1155
+ }
1156
+
1157
+ if (skills && skills.length > 0) {
1158
+ input.push(...skills);
1159
+ }
1160
+
1161
+ return { input, cleanup };
1162
+ } catch (error) {
1163
+ cleanup();
1164
+ throw error;
1165
+ }
1166
+ }
1167
+
1168
+ private toHostSessionPath(targetPath: string | null | undefined): string | null {
1169
+ if (!targetPath) {
1170
+ return null;
1171
+ }
1172
+
1173
+ return this.launchSpec?.pathMapper.toHostPath(targetPath) ?? targetPath;
1174
+ }
1175
+
1176
+ private toTargetSessionPath(sessionPath: string | null | undefined): string | null {
1177
+ if (!sessionPath) {
1178
+ return null;
1179
+ }
1180
+
1181
+ if (!this.launchSpec) {
1182
+ return sessionPath;
1183
+ }
1184
+
1185
+ if (this.launchSpec.target.platformFamily === 'unix' && sessionPath.startsWith('/')) {
1186
+ return sessionPath;
1187
+ }
1188
+
1189
+ if (
1190
+ this.launchSpec.target.platformFamily === 'windows'
1191
+ && (/^[A-Za-z]:[\\/]/.test(sessionPath) || sessionPath.startsWith('\\\\'))
1192
+ ) {
1193
+ return sessionPath;
1194
+ }
1195
+
1196
+ return this.launchSpec.pathMapper.toTargetPath(sessionPath) ?? sessionPath;
1197
+ }
1198
+
1199
+ private mapHostPathToTarget(hostPath: string | null | undefined): string | null {
1200
+ if (!hostPath) {
1201
+ return null;
1202
+ }
1203
+
1204
+ return this.launchSpec?.pathMapper.toTargetPath(hostPath) ?? hostPath;
1205
+ }
1206
+
1207
+ private mapRequiredHostPathsToTarget(hostPaths: string[], label: string): string[] {
1208
+ if (!this.launchSpec) {
1209
+ return hostPaths;
1210
+ }
1211
+
1212
+ return hostPaths.map((hostPath) => {
1213
+ const targetPath = this.launchSpec!.pathMapper.toTargetPath(hostPath);
1214
+ if (!targetPath) {
1215
+ throw new Error(`Codex cannot access ${label} from the selected target: ${hostPath}`);
1216
+ }
1217
+ return targetPath;
1218
+ });
1219
+ }
1220
+
1221
+ private resolveTranscriptRootHost(sessionFilePath?: string | null): string | null {
1222
+ return this.runtimeContext?.sessionsDirHost
1223
+ ?? deriveCodexSessionsRootFromSessionPath(
1224
+ sessionFilePath ?? this.session.getSessionFilePath() ?? this.currentThreadPath,
1225
+ );
1226
+ }
1227
+
1228
+ private resolveTranscriptRootTarget(sessionFilePath?: string | null): string | null {
1229
+ if (this.runtimeContext?.sessionsDirTarget) {
1230
+ return this.runtimeContext.sessionsDirTarget;
1231
+ }
1232
+
1233
+ const targetSessionPath = this.toTargetSessionPath(
1234
+ sessionFilePath ?? this.session.getSessionFilePath() ?? this.currentThreadPath,
1235
+ );
1236
+ return deriveCodexSessionsRootFromSessionPath(targetSessionPath);
1237
+ }
1238
+
1239
+ private resolveMemoriesDirTarget(sessionFilePath?: string | null): string | null {
1240
+ if (this.runtimeContext?.memoriesDirTarget) {
1241
+ return this.runtimeContext.memoriesDirTarget;
1242
+ }
1243
+
1244
+ return deriveCodexMemoriesDirFromSessionsRoot(
1245
+ this.resolveTranscriptRootTarget(sessionFilePath),
1246
+ );
1247
+ }
1248
+ }
1249
+
1250
+ // ---------------------------------------------------------------------------
1251
+ // Image attachment helpers
1252
+ // ---------------------------------------------------------------------------
1253
+
1254
+ interface ImageAttachment {
1255
+ data: string;
1256
+ mediaType: string;
1257
+ filename?: string;
1258
+ }
1259
+
1260
+ interface CodexInputBundle {
1261
+ input: UserInput[];
1262
+ cleanup: () => void;
1263
+ }
1264
+
1265
+ function toAttachmentFilename(attachment: ImageAttachment, index: number): string {
1266
+ const base = (attachment.filename ?? '').trim().replace(/[^A-Za-z0-9._-]/g, '_') || `image-${index + 1}`;
1267
+ if (base.includes('.')) return base;
1268
+ const subtype = attachment.mediaType.split('/')[1] ?? 'img';
1269
+ const extension = subtype === 'jpeg' ? 'jpg' : subtype;
1270
+ return `${base}.${extension}`;
1271
+ }
1272
+
1273
+ export { toAttachmentFilename as _toAttachmentFilename };
1274
+
1275
+ // ---------------------------------------------------------------------------
1276
+ // Interrupt kind classification (preserved for history parsing)
1277
+ // ---------------------------------------------------------------------------
1278
+
1279
+ export type CodexInterruptKind = 'user_request' | 'tool_use' | 'compaction_canceled';
1280
+
1281
+ export function mapCodexAbortReasonToInterruptKind(reason: string): CodexInterruptKind | undefined {
1282
+ const normalized = reason.trim().toLowerCase();
1283
+ if (!normalized) return undefined;
1284
+
1285
+ if (normalized === 'interrupted' || normalized === 'cancelled' || normalized === 'canceled') {
1286
+ return 'user_request';
1287
+ }
1288
+ if (normalized.includes('tool')) {
1289
+ return 'tool_use';
1290
+ }
1291
+ if (normalized.includes('compact')) {
1292
+ return 'compaction_canceled';
1293
+ }
1294
+
1295
+ return undefined;
1296
+ }