@within-7/minto 0.1.6 → 0.2.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.
Files changed (487) hide show
  1. package/cli.js +155 -37
  2. package/dist/Tool.js +38 -0
  3. package/dist/Tool.js.map +3 -3
  4. package/dist/commands/agents/AgentsCommand.js +52 -26
  5. package/dist/commands/agents/AgentsCommand.js.map +2 -2
  6. package/dist/commands/agents/constants.js +1 -1
  7. package/dist/commands/agents/constants.js.map +1 -1
  8. package/dist/commands/agents/index.js +1 -1
  9. package/dist/commands/bug.js +74 -7
  10. package/dist/commands/bug.js.map +3 -3
  11. package/dist/commands/clear.js +3 -0
  12. package/dist/commands/clear.js.map +2 -2
  13. package/dist/commands/compact.js +37 -0
  14. package/dist/commands/compact.js.map +2 -2
  15. package/dist/commands/context.js +84 -0
  16. package/dist/commands/context.js.map +7 -0
  17. package/dist/commands/ctx_viz.js +18 -10
  18. package/dist/commands/ctx_viz.js.map +2 -2
  19. package/dist/commands/doctor.js +158 -12
  20. package/dist/commands/doctor.js.map +2 -2
  21. package/dist/commands/export.js +156 -0
  22. package/dist/commands/export.js.map +7 -0
  23. package/dist/commands/mcp-interactive.js +21 -12
  24. package/dist/commands/mcp-interactive.js.map +2 -2
  25. package/dist/commands/model.js +6 -5
  26. package/dist/commands/model.js.map +2 -2
  27. package/dist/commands/permissions.js +86 -0
  28. package/dist/commands/permissions.js.map +7 -0
  29. package/dist/commands/quit.js +3 -1
  30. package/dist/commands/quit.js.map +2 -2
  31. package/dist/commands/sandbox.js +104 -0
  32. package/dist/commands/sandbox.js.map +7 -0
  33. package/dist/commands/status.js +58 -0
  34. package/dist/commands/status.js.map +7 -0
  35. package/dist/commands/tasks.js +108 -0
  36. package/dist/commands/tasks.js.map +7 -0
  37. package/dist/commands/todos.js +123 -0
  38. package/dist/commands/todos.js.map +7 -0
  39. package/dist/commands.js +20 -2
  40. package/dist/commands.js.map +2 -2
  41. package/dist/components/AgentThinkingBlock.js +10 -18
  42. package/dist/components/AgentThinkingBlock.js.map +2 -2
  43. package/dist/components/BackgroundTasksPanel.js +78 -29
  44. package/dist/components/BackgroundTasksPanel.js.map +2 -2
  45. package/dist/components/BashStreamingProgress.js +24 -0
  46. package/dist/components/BashStreamingProgress.js.map +7 -0
  47. package/dist/components/CollapsibleHint.js +14 -0
  48. package/dist/components/CollapsibleHint.js.map +7 -0
  49. package/dist/components/FileEditToolUpdatedMessage.js +1 -1
  50. package/dist/components/FileEditToolUpdatedMessage.js.map +2 -2
  51. package/dist/components/HotkeyHelpPanel.js +137 -0
  52. package/dist/components/HotkeyHelpPanel.js.map +7 -0
  53. package/dist/components/Logo.js +5 -5
  54. package/dist/components/Logo.js.map +2 -2
  55. package/dist/components/Message.js +23 -7
  56. package/dist/components/Message.js.map +3 -3
  57. package/dist/components/ModelConfig.js +16 -3
  58. package/dist/components/ModelConfig.js.map +2 -2
  59. package/dist/components/ModelListManager.js +3 -3
  60. package/dist/components/ModelListManager.js.map +2 -2
  61. package/dist/components/ModelSelector/ModelSelector.js +1 -1
  62. package/dist/components/Onboarding.js +19 -14
  63. package/dist/components/Onboarding.js.map +2 -2
  64. package/dist/components/ProgressBar.js +74 -0
  65. package/dist/components/ProgressBar.js.map +7 -0
  66. package/dist/components/PromptInput.js +156 -46
  67. package/dist/components/PromptInput.js.map +2 -2
  68. package/dist/components/RequestStatusIndicator.js +194 -0
  69. package/dist/components/RequestStatusIndicator.js.map +7 -0
  70. package/dist/components/Spinner.js +92 -27
  71. package/dist/components/Spinner.js.map +2 -2
  72. package/dist/components/SpinnerSymbol.js +21 -27
  73. package/dist/components/SpinnerSymbol.js.map +2 -2
  74. package/dist/components/StreamingBashOutput.js +9 -8
  75. package/dist/components/StreamingBashOutput.js.map +2 -2
  76. package/dist/components/SubagentBlock.js +1 -1
  77. package/dist/components/SubagentBlock.js.map +1 -1
  78. package/dist/components/SubagentProgress.js +10 -11
  79. package/dist/components/SubagentProgress.js.map +2 -2
  80. package/dist/components/TaskCard.js +16 -13
  81. package/dist/components/TaskCard.js.map +2 -2
  82. package/dist/components/TodoChangeBlock.js +1 -1
  83. package/dist/components/TodoChangeBlock.js.map +2 -2
  84. package/dist/components/TodoPanel.js +120 -29
  85. package/dist/components/TodoPanel.js.map +3 -3
  86. package/dist/components/TokenCounter.js +74 -0
  87. package/dist/components/TokenCounter.js.map +7 -0
  88. package/dist/components/TokenWarning.js +2 -1
  89. package/dist/components/TokenWarning.js.map +2 -2
  90. package/dist/components/TreeConnector.js +25 -0
  91. package/dist/components/TreeConnector.js.map +7 -0
  92. package/dist/components/TurnCompletionIndicator.js +18 -0
  93. package/dist/components/TurnCompletionIndicator.js.map +7 -0
  94. package/dist/components/messages/AssistantTextMessage.js +5 -2
  95. package/dist/components/messages/AssistantTextMessage.js.map +2 -2
  96. package/dist/components/messages/AssistantThinkingMessage.js +18 -3
  97. package/dist/components/messages/AssistantThinkingMessage.js.map +2 -2
  98. package/dist/components/messages/AssistantToolUseMessage.js +11 -8
  99. package/dist/components/messages/AssistantToolUseMessage.js.map +2 -2
  100. package/dist/components/messages/GroupRenderer.js +53 -0
  101. package/dist/components/messages/GroupRenderer.js.map +7 -0
  102. package/dist/components/messages/NestedTasksPreview.js +12 -0
  103. package/dist/components/messages/NestedTasksPreview.js.map +7 -0
  104. package/dist/components/messages/ParallelTasksGroupView.js +92 -0
  105. package/dist/components/messages/ParallelTasksGroupView.js.map +7 -0
  106. package/dist/components/messages/TaskInModuleView.js +198 -0
  107. package/dist/components/messages/TaskInModuleView.js.map +7 -0
  108. package/dist/components/messages/TaskOutputContent.js +53 -0
  109. package/dist/components/messages/TaskOutputContent.js.map +7 -0
  110. package/dist/components/messages/UserPromptMessage.js +1 -1
  111. package/dist/components/messages/UserPromptMessage.js.map +2 -2
  112. package/dist/components/messages/UserToolResultMessage/UserToolSuccessMessage.js +2 -3
  113. package/dist/components/messages/UserToolResultMessage/UserToolSuccessMessage.js.map +2 -2
  114. package/dist/components/permissions/FallbackPermissionRequest.js +4 -4
  115. package/dist/components/permissions/FallbackPermissionRequest.js.map +2 -2
  116. package/dist/components/permissions/FilesystemPermissionRequest/FilesystemPermissionRequest.js +4 -4
  117. package/dist/components/permissions/FilesystemPermissionRequest/FilesystemPermissionRequest.js.map +2 -2
  118. package/dist/constants/colors.js +48 -0
  119. package/dist/constants/colors.js.map +2 -2
  120. package/dist/constants/formatRules.js +102 -0
  121. package/dist/constants/formatRules.js.map +7 -0
  122. package/dist/constants/prompts.js +12 -34
  123. package/dist/constants/prompts.js.map +2 -2
  124. package/dist/constants/symbols.js +64 -6
  125. package/dist/constants/symbols.js.map +2 -2
  126. package/dist/constants/timing.js +5 -0
  127. package/dist/constants/timing.js.map +2 -2
  128. package/dist/core/config/defaults.js +84 -0
  129. package/dist/core/config/defaults.js.map +7 -0
  130. package/dist/core/config/index.js +111 -0
  131. package/dist/core/config/index.js.map +7 -0
  132. package/dist/core/config/loader.js +221 -0
  133. package/dist/core/config/loader.js.map +7 -0
  134. package/dist/core/config/migrations.js +128 -0
  135. package/dist/core/config/migrations.js.map +7 -0
  136. package/dist/core/config/schema.js +178 -0
  137. package/dist/core/config/schema.js.map +7 -0
  138. package/dist/core/costTracker.js +138 -0
  139. package/dist/core/costTracker.js.map +7 -0
  140. package/dist/core/index.js +5 -0
  141. package/dist/core/index.js.map +7 -0
  142. package/dist/core/permissions/auditLog.js +204 -0
  143. package/dist/core/permissions/auditLog.js.map +7 -0
  144. package/dist/core/permissions/engine/index.js +3 -0
  145. package/dist/core/permissions/engine/index.js.map +7 -0
  146. package/dist/core/permissions/engine/permissionEngine.js +106 -0
  147. package/dist/core/permissions/engine/permissionEngine.js.map +7 -0
  148. package/dist/core/permissions/engine/types.js +1 -0
  149. package/dist/core/permissions/engine/types.js.map +7 -0
  150. package/dist/core/permissions/index.js +84 -0
  151. package/dist/core/permissions/index.js.map +7 -0
  152. package/dist/core/permissions/ruleEngine.js +259 -0
  153. package/dist/core/permissions/ruleEngine.js.map +7 -0
  154. package/dist/core/permissions/rules/allowedToolsRule.js +62 -0
  155. package/dist/core/permissions/rules/allowedToolsRule.js.map +7 -0
  156. package/dist/core/permissions/rules/autoEscalationRule.js +291 -0
  157. package/dist/core/permissions/rules/autoEscalationRule.js.map +7 -0
  158. package/dist/core/permissions/rules/index.js +46 -0
  159. package/dist/core/permissions/rules/index.js.map +7 -0
  160. package/dist/core/permissions/rules/planModeRule.js +55 -0
  161. package/dist/core/permissions/rules/planModeRule.js.map +7 -0
  162. package/dist/core/permissions/rules/projectBoundaryRule.js +168 -0
  163. package/dist/core/permissions/rules/projectBoundaryRule.js.map +7 -0
  164. package/dist/core/permissions/rules/safeModeRule.js +65 -0
  165. package/dist/core/permissions/rules/safeModeRule.js.map +7 -0
  166. package/dist/core/permissions/rules/sensitivePathsRule.js +340 -0
  167. package/dist/core/permissions/rules/sensitivePathsRule.js.map +7 -0
  168. package/dist/core/permissions/types.js +127 -0
  169. package/dist/core/permissions/types.js.map +7 -0
  170. package/dist/core/tools/executor.js +143 -0
  171. package/dist/core/tools/executor.js.map +7 -0
  172. package/dist/core/tools/index.js +15 -0
  173. package/dist/core/tools/index.js.map +7 -0
  174. package/dist/core/tools/registry.js +183 -0
  175. package/dist/core/tools/registry.js.map +7 -0
  176. package/dist/core/tools/types.js +1 -0
  177. package/dist/core/tools/types.js.map +7 -0
  178. package/dist/cost-tracker.js +23 -15
  179. package/dist/cost-tracker.js.map +2 -2
  180. package/dist/entrypoints/cli.js +43 -43
  181. package/dist/entrypoints/cli.js.map +2 -2
  182. package/dist/entrypoints/mcp.js +12 -4
  183. package/dist/entrypoints/mcp.js.map +2 -2
  184. package/dist/history.js +14 -3
  185. package/dist/history.js.map +2 -2
  186. package/dist/hooks/useAgentTranscripts.js +116 -0
  187. package/dist/hooks/useAgentTranscripts.js.map +7 -0
  188. package/dist/hooks/useAnimationSync.js +53 -0
  189. package/dist/hooks/useAnimationSync.js.map +7 -0
  190. package/dist/hooks/useArrowKeyHistory.js +4 -2
  191. package/dist/hooks/useArrowKeyHistory.js.map +2 -2
  192. package/dist/hooks/useCanUseTool.js +3 -1
  193. package/dist/hooks/useCanUseTool.js.map +2 -2
  194. package/dist/hooks/useCancelRequest.js +4 -1
  195. package/dist/hooks/useCancelRequest.js.map +2 -2
  196. package/dist/hooks/useExitOnCtrlCD.js +9 -5
  197. package/dist/hooks/useExitOnCtrlCD.js.map +2 -2
  198. package/dist/hooks/useHookStatus.js +40 -0
  199. package/dist/hooks/useHookStatus.js.map +7 -0
  200. package/dist/hooks/useLogMessages.js +17 -1
  201. package/dist/hooks/useLogMessages.js.map +2 -2
  202. package/dist/hooks/useMessageGroups.js +43 -0
  203. package/dist/hooks/useMessageGroups.js.map +7 -0
  204. package/dist/hooks/useTerminalSize.js +62 -6
  205. package/dist/hooks/useTerminalSize.js.map +2 -2
  206. package/dist/hooks/useUnifiedCompletion.js +69 -0
  207. package/dist/hooks/useUnifiedCompletion.js.map +2 -2
  208. package/dist/i18n/index.js +109 -0
  209. package/dist/i18n/index.js.map +7 -0
  210. package/dist/i18n/locales/en.js +347 -0
  211. package/dist/i18n/locales/en.js.map +7 -0
  212. package/dist/i18n/locales/index.js +7 -0
  213. package/dist/i18n/locales/index.js.map +7 -0
  214. package/dist/i18n/locales/zh-CN.js +347 -0
  215. package/dist/i18n/locales/zh-CN.js.map +7 -0
  216. package/dist/i18n/types.js +8 -0
  217. package/dist/i18n/types.js.map +7 -0
  218. package/dist/query.js +175 -17
  219. package/dist/query.js.map +3 -3
  220. package/dist/screens/REPL.js +501 -192
  221. package/dist/screens/REPL.js.map +3 -3
  222. package/dist/services/adapters/chatCompletions.js +3 -1
  223. package/dist/services/adapters/chatCompletions.js.map +2 -2
  224. package/dist/services/adapters/messageNormalizer.js +354 -0
  225. package/dist/services/adapters/messageNormalizer.js.map +7 -0
  226. package/dist/services/adapters/responsesAPI.js +6 -3
  227. package/dist/services/adapters/responsesAPI.js.map +2 -2
  228. package/dist/services/checkpointManager.js +386 -0
  229. package/dist/services/checkpointManager.js.map +7 -0
  230. package/dist/services/claude.js +138 -11
  231. package/dist/services/claude.js.map +3 -3
  232. package/dist/services/compressionService.js +50 -1
  233. package/dist/services/compressionService.js.map +2 -2
  234. package/dist/services/contextMonitor.js +162 -0
  235. package/dist/services/contextMonitor.js.map +7 -0
  236. package/dist/services/customCommands.js +60 -41
  237. package/dist/services/customCommands.js.map +2 -2
  238. package/dist/services/hookExecutor.js +173 -1
  239. package/dist/services/hookExecutor.js.map +2 -2
  240. package/dist/services/intelligentCompactor.js +281 -0
  241. package/dist/services/intelligentCompactor.js.map +7 -0
  242. package/dist/services/lspConfig.js +109 -0
  243. package/dist/services/lspConfig.js.map +7 -0
  244. package/dist/services/mcpClient.js +273 -34
  245. package/dist/services/mcpClient.js.map +2 -2
  246. package/dist/services/modelOrchestrator.js +310 -0
  247. package/dist/services/modelOrchestrator.js.map +7 -0
  248. package/dist/services/openai.js +8 -1
  249. package/dist/services/openai.js.map +2 -2
  250. package/dist/services/outputStyles.js +138 -0
  251. package/dist/services/outputStyles.js.map +7 -0
  252. package/dist/services/plugins/index.js +5 -0
  253. package/dist/services/plugins/index.js.map +7 -0
  254. package/dist/services/plugins/lspServers.js +188 -0
  255. package/dist/services/plugins/lspServers.js.map +7 -0
  256. package/dist/services/plugins/pluginRuntime.js +229 -0
  257. package/dist/services/plugins/pluginRuntime.js.map +7 -0
  258. package/dist/services/plugins/pluginValidation.js +219 -0
  259. package/dist/services/plugins/pluginValidation.js.map +7 -0
  260. package/dist/services/plugins/skillMarketplace.js +556 -0
  261. package/dist/services/plugins/skillMarketplace.js.map +7 -0
  262. package/dist/services/responseStateManager.js +37 -3
  263. package/dist/services/responseStateManager.js.map +2 -2
  264. package/dist/services/sandbox/filesystemBoundary.js +300 -0
  265. package/dist/services/sandbox/filesystemBoundary.js.map +7 -0
  266. package/dist/services/sandbox/index.js +14 -0
  267. package/dist/services/sandbox/index.js.map +7 -0
  268. package/dist/services/sandbox/networkProxy.js +293 -0
  269. package/dist/services/sandbox/networkProxy.js.map +7 -0
  270. package/dist/services/sandbox/sandboxController.js +574 -0
  271. package/dist/services/sandbox/sandboxController.js.map +7 -0
  272. package/dist/services/sandbox/types.js +50 -0
  273. package/dist/services/sandbox/types.js.map +7 -0
  274. package/dist/services/sessionMemory.js +266 -0
  275. package/dist/services/sessionMemory.js.map +7 -0
  276. package/dist/services/taskRouter.js +324 -0
  277. package/dist/services/taskRouter.js.map +7 -0
  278. package/dist/tools/ArchitectTool/ArchitectTool.js +10 -3
  279. package/dist/tools/ArchitectTool/ArchitectTool.js.map +2 -2
  280. package/dist/tools/AskExpertModelTool/AskExpertModelTool.js +3 -0
  281. package/dist/tools/AskExpertModelTool/AskExpertModelTool.js.map +2 -2
  282. package/dist/tools/AskUserQuestionTool/AskUserQuestionTool.js +8 -1
  283. package/dist/tools/AskUserQuestionTool/AskUserQuestionTool.js.map +2 -2
  284. package/dist/tools/BaseTool.js +72 -0
  285. package/dist/tools/BaseTool.js.map +7 -0
  286. package/dist/tools/BashOutputTool/BashOutputToolResultMessage.js +3 -0
  287. package/dist/tools/BashOutputTool/BashOutputToolResultMessage.js.map +2 -2
  288. package/dist/tools/BashTool/BashTool.js +60 -3
  289. package/dist/tools/BashTool/BashTool.js.map +2 -2
  290. package/dist/tools/BashTool/BashToolResultMessage.js +3 -0
  291. package/dist/tools/BashTool/BashToolResultMessage.js.map +2 -2
  292. package/dist/tools/BashTool/OutputLine.js +54 -0
  293. package/dist/tools/BashTool/OutputLine.js.map +2 -2
  294. package/dist/tools/BashTool/prompt.js +192 -3
  295. package/dist/tools/BashTool/prompt.js.map +2 -2
  296. package/dist/tools/FileEditTool/FileEditTool.js +29 -4
  297. package/dist/tools/FileEditTool/FileEditTool.js.map +2 -2
  298. package/dist/tools/FileReadTool/FileReadTool.js +23 -4
  299. package/dist/tools/FileReadTool/FileReadTool.js.map +2 -2
  300. package/dist/tools/FileWriteTool/FileWriteTool.js +5 -5
  301. package/dist/tools/FileWriteTool/FileWriteTool.js.map +2 -2
  302. package/dist/tools/GlobTool/GlobTool.js +14 -3
  303. package/dist/tools/GlobTool/GlobTool.js.map +2 -2
  304. package/dist/tools/GrepTool/GrepTool.js +41 -7
  305. package/dist/tools/GrepTool/GrepTool.js.map +2 -2
  306. package/dist/tools/KillShellTool/KillShellToolResultMessage.js +3 -0
  307. package/dist/tools/KillShellTool/KillShellToolResultMessage.js.map +2 -2
  308. package/dist/tools/ListMcpResourcesTool/ListMcpResourcesTool.js +109 -0
  309. package/dist/tools/ListMcpResourcesTool/ListMcpResourcesTool.js.map +7 -0
  310. package/dist/tools/ListMcpResourcesTool/prompt.js +19 -0
  311. package/dist/tools/ListMcpResourcesTool/prompt.js.map +7 -0
  312. package/dist/tools/LspTool/LspTool.js +664 -0
  313. package/dist/tools/LspTool/LspTool.js.map +7 -0
  314. package/dist/tools/LspTool/prompt.js +27 -0
  315. package/dist/tools/LspTool/prompt.js.map +7 -0
  316. package/dist/tools/MCPTool/MCPTool.js +11 -4
  317. package/dist/tools/MCPTool/MCPTool.js.map +2 -2
  318. package/dist/tools/MemoryReadTool/MemoryReadTool.js +19 -6
  319. package/dist/tools/MemoryReadTool/MemoryReadTool.js.map +2 -2
  320. package/dist/tools/MemoryWriteTool/MemoryWriteTool.js +6 -6
  321. package/dist/tools/MemoryWriteTool/MemoryWriteTool.js.map +2 -2
  322. package/dist/tools/MultiEditTool/MultiEditTool.js +19 -2
  323. package/dist/tools/MultiEditTool/MultiEditTool.js.map +2 -2
  324. package/dist/tools/NotebookEditTool/NotebookEditTool.js +5 -1
  325. package/dist/tools/NotebookEditTool/NotebookEditTool.js.map +2 -2
  326. package/dist/tools/NotebookReadTool/NotebookReadTool.js +8 -4
  327. package/dist/tools/NotebookReadTool/NotebookReadTool.js.map +2 -2
  328. package/dist/tools/PlanModeTool/EnterPlanModeTool.js +74 -0
  329. package/dist/tools/PlanModeTool/EnterPlanModeTool.js.map +7 -0
  330. package/dist/tools/PlanModeTool/ExitPlanModeTool.js +108 -0
  331. package/dist/tools/PlanModeTool/ExitPlanModeTool.js.map +7 -0
  332. package/dist/tools/PlanModeTool/prompt.js +94 -0
  333. package/dist/tools/PlanModeTool/prompt.js.map +7 -0
  334. package/dist/tools/ReadMcpResourceTool/ReadMcpResourceTool.js +130 -0
  335. package/dist/tools/ReadMcpResourceTool/ReadMcpResourceTool.js.map +7 -0
  336. package/dist/tools/ReadMcpResourceTool/prompt.js +17 -0
  337. package/dist/tools/ReadMcpResourceTool/prompt.js.map +7 -0
  338. package/dist/tools/SkillTool/SkillTool.js +14 -3
  339. package/dist/tools/SkillTool/SkillTool.js.map +2 -2
  340. package/dist/tools/SlashCommandTool/SlashCommandTool.js +260 -0
  341. package/dist/tools/SlashCommandTool/SlashCommandTool.js.map +7 -0
  342. package/dist/tools/SlashCommandTool/prompt.js +35 -0
  343. package/dist/tools/SlashCommandTool/prompt.js.map +7 -0
  344. package/dist/tools/TaskOutputTool/TaskOutputTool.js +189 -0
  345. package/dist/tools/TaskOutputTool/TaskOutputTool.js.map +7 -0
  346. package/dist/tools/TaskOutputTool/prompt.js +15 -0
  347. package/dist/tools/TaskOutputTool/prompt.js.map +7 -0
  348. package/dist/tools/TaskTool/TaskTool.js +321 -146
  349. package/dist/tools/TaskTool/TaskTool.js.map +2 -2
  350. package/dist/tools/TaskTool/prompt.js.map +2 -2
  351. package/dist/tools/TodoWriteTool/TodoWriteTool.js +42 -73
  352. package/dist/tools/TodoWriteTool/TodoWriteTool.js.map +2 -2
  353. package/dist/tools/URLFetcherTool/URLFetcherTool.js +7 -1
  354. package/dist/tools/URLFetcherTool/URLFetcherTool.js.map +2 -2
  355. package/dist/tools/URLFetcherTool/cache.js +55 -8
  356. package/dist/tools/URLFetcherTool/cache.js.map +2 -2
  357. package/dist/tools/WebSearchTool/WebSearchTool.js +6 -1
  358. package/dist/tools/WebSearchTool/WebSearchTool.js.map +2 -2
  359. package/dist/tools.js +31 -2
  360. package/dist/tools.js.map +2 -2
  361. package/dist/types/hooks.js +4 -0
  362. package/dist/types/hooks.js.map +2 -2
  363. package/dist/types/marketplace.js.map +2 -2
  364. package/dist/types/messageGroup.js +36 -0
  365. package/dist/types/messageGroup.js.map +7 -0
  366. package/dist/types/plugin.js.map +2 -2
  367. package/dist/types/thinking.js +1 -0
  368. package/dist/types/thinking.js.map +7 -0
  369. package/dist/utils/BackgroundShellManager.js +136 -39
  370. package/dist/utils/BackgroundShellManager.js.map +2 -2
  371. package/dist/utils/MessageBatchBuffer.js +102 -0
  372. package/dist/utils/MessageBatchBuffer.js.map +7 -0
  373. package/dist/utils/PersistentShell.js +151 -1
  374. package/dist/utils/PersistentShell.js.map +2 -2
  375. package/dist/utils/agentLoader.js +1 -23
  376. package/dist/utils/agentLoader.js.map +2 -2
  377. package/dist/utils/agentTranscripts.js +641 -0
  378. package/dist/utils/agentTranscripts.js.map +7 -0
  379. package/dist/utils/animationManager.js +213 -0
  380. package/dist/utils/animationManager.js.map +7 -0
  381. package/dist/utils/animationSync.js +110 -0
  382. package/dist/utils/animationSync.js.map +7 -0
  383. package/dist/utils/asyncFile.js +215 -0
  384. package/dist/utils/asyncFile.js.map +7 -0
  385. package/dist/utils/backgroundAgentManager.js +231 -0
  386. package/dist/utils/backgroundAgentManager.js.map +7 -0
  387. package/dist/utils/config.js +63 -7
  388. package/dist/utils/config.js.map +2 -2
  389. package/dist/utils/conversationRecovery.js +19 -0
  390. package/dist/utils/conversationRecovery.js.map +2 -2
  391. package/dist/utils/exit.js +73 -0
  392. package/dist/utils/exit.js.map +7 -0
  393. package/dist/utils/format.js +73 -5
  394. package/dist/utils/format.js.map +2 -2
  395. package/dist/utils/generators.js +76 -6
  396. package/dist/utils/generators.js.map +2 -2
  397. package/dist/utils/globalErrorHandler.js +149 -0
  398. package/dist/utils/globalErrorHandler.js.map +7 -0
  399. package/dist/utils/groupHandlers/index.js +8 -0
  400. package/dist/utils/groupHandlers/index.js.map +7 -0
  401. package/dist/utils/groupHandlers/parallelTasksHandler.js +140 -0
  402. package/dist/utils/groupHandlers/parallelTasksHandler.js.map +7 -0
  403. package/dist/utils/groupHandlers/taskHandler.js +104 -0
  404. package/dist/utils/groupHandlers/taskHandler.js.map +7 -0
  405. package/dist/utils/groupHandlers/types.js +1 -0
  406. package/dist/utils/groupHandlers/types.js.map +7 -0
  407. package/dist/utils/logRotation.js +224 -0
  408. package/dist/utils/logRotation.js.map +7 -0
  409. package/dist/utils/marketplaceManager.js +3 -5
  410. package/dist/utils/marketplaceManager.js.map +2 -2
  411. package/dist/utils/memSafety.js +264 -0
  412. package/dist/utils/memSafety.js.map +7 -0
  413. package/dist/utils/messageGroupManager.js +274 -0
  414. package/dist/utils/messageGroupManager.js.map +7 -0
  415. package/dist/utils/messages.js +13 -4
  416. package/dist/utils/messages.js.map +2 -2
  417. package/dist/utils/model.js +119 -15
  418. package/dist/utils/model.js.map +3 -3
  419. package/dist/utils/permissions/filesystem.js +157 -5
  420. package/dist/utils/permissions/filesystem.js.map +2 -2
  421. package/dist/utils/plan/planMode.js +143 -0
  422. package/dist/utils/plan/planMode.js.map +7 -0
  423. package/dist/utils/pluginLoader.js +17 -21
  424. package/dist/utils/pluginLoader.js.map +2 -2
  425. package/dist/utils/ripgrep.js +55 -2
  426. package/dist/utils/ripgrep.js.map +2 -2
  427. package/dist/utils/sanitizeInput.js +32 -0
  428. package/dist/utils/sanitizeInput.js.map +7 -0
  429. package/dist/utils/secureKeyStorage.js +312 -0
  430. package/dist/utils/secureKeyStorage.js.map +7 -0
  431. package/dist/utils/session/sessionPlugins.js +67 -0
  432. package/dist/utils/session/sessionPlugins.js.map +7 -0
  433. package/dist/utils/taskDisplayUtils.js +257 -0
  434. package/dist/utils/taskDisplayUtils.js.map +7 -0
  435. package/dist/utils/teamConfig.js +2 -1
  436. package/dist/utils/teamConfig.js.map +2 -2
  437. package/dist/utils/todoStorage.js +92 -2
  438. package/dist/utils/todoStorage.js.map +2 -2
  439. package/dist/utils/toolTimeout.js +136 -0
  440. package/dist/utils/toolTimeout.js.map +7 -0
  441. package/dist/utils/tooling/safeRender.js +115 -0
  442. package/dist/utils/tooling/safeRender.js.map +7 -0
  443. package/dist/utils/userFriendlyError.js +346 -0
  444. package/dist/utils/userFriendlyError.js.map +7 -0
  445. package/dist/utils/vendor/ripgrep/arm64-darwin/rg +0 -0
  446. package/dist/version.js +2 -2
  447. package/dist/version.js.map +1 -1
  448. package/package.json +14 -4
  449. package/scripts/postinstall.js +128 -38
  450. package/dist/commands/agents.js +0 -2086
  451. package/dist/commands/agents.js.map +0 -7
  452. package/dist/commands/build.js +0 -74
  453. package/dist/commands/build.js.map +0 -7
  454. package/dist/commands/compression.js +0 -57
  455. package/dist/commands/compression.js.map +0 -7
  456. package/dist/commands/listen.js +0 -37
  457. package/dist/commands/listen.js.map +0 -7
  458. package/dist/commands/login.js +0 -37
  459. package/dist/commands/login.js.map +0 -7
  460. package/dist/commands/logout.js +0 -33
  461. package/dist/commands/logout.js.map +0 -7
  462. package/dist/commands/mcp.js +0 -40
  463. package/dist/commands/mcp.js.map +0 -7
  464. package/dist/commands/mcp_refresh.js +0 -40
  465. package/dist/commands/mcp_refresh.js.map +0 -7
  466. package/dist/commands/modelstatus.js +0 -21
  467. package/dist/commands/modelstatus.js.map +0 -7
  468. package/dist/commands/onboarding.js +0 -36
  469. package/dist/commands/onboarding.js.map +0 -7
  470. package/dist/commands/plugin-interactive.js +0 -446
  471. package/dist/commands/plugin-interactive.js.map +0 -7
  472. package/dist/commands/pr_comments.js +0 -61
  473. package/dist/commands/pr_comments.js.map +0 -7
  474. package/dist/commands/release-notes.js +0 -30
  475. package/dist/commands/release-notes.js.map +0 -7
  476. package/dist/commands/review.js +0 -51
  477. package/dist/commands/review.js.map +0 -7
  478. package/dist/components/Bug.js +0 -147
  479. package/dist/components/Bug.js.map +0 -7
  480. package/dist/components/ModelSelector.js +0 -2062
  481. package/dist/components/ModelSelector.js.map +0 -7
  482. package/dist/components/ModelStatusDisplay.js +0 -87
  483. package/dist/components/ModelStatusDisplay.js.map +0 -7
  484. package/dist/entrypoints/cli-wrapper.js +0 -61
  485. package/dist/entrypoints/cli-wrapper.js.map +0 -7
  486. package/dist/screens/Doctor.js +0 -22
  487. package/dist/screens/Doctor.js.map +0 -7
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../src/services/compressionService.ts"],
4
- "sourcesContent": ["/**\n * Compression Service - Stateless Context Compression Core\n *\n * This service provides stateless compression functionality that can be used by\n * both the main agent and subagents without side effects. It is the foundation\n * of the layered compression architecture.\n *\n * **Architecture Design:**\n *\n * ```\n * \u250C\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2510\n * \u2502 Compression System Architecture \u2502\n * \u251C\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2524\n * \u2502 Layer 1: MainAgentCompressor \u2502\n * \u2502 - Handles /compact command and main agent compression \u2502\n * \u2502 - Manages global state (messages, cache) \u2502\n * \u2502 - Recovers recently accessed files \u2502\n * \u251C\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2524\n * \u2502 Layer 2: SubagentCompressor \u2502\n * \u2502 - Handles subagent automatic compression \u2502\n * \u2502 - Operates only on local message arrays \u2502\n * \u2502 - Zero side effects, no global state modification \u2502\n * \u251C\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2524\n * \u2502 Layer 3: CompressionService (THIS FILE) \u2502\n * \u2502 - Stateless compression algorithm \u2502\n * \u2502 - LLM-based summarization \u2502\n * \u2502 - Shared by both layers above \u2502\n * \u2514\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2518\n * ```\n *\n * **Key Principles:**\n * - \u2705 Completely stateless - no global variable access\n * - \u2705 Pure functions - same input always produces same output\n * - \u2705 No side effects - does not modify messages parameter\n * - \u2705 Reusable - shared by main agent and subagent compressors\n *\n * @module CompressionService\n * @see src/services/mainAgentCompressor.ts - Main agent compression layer\n * @see src/services/subagentCompressor.ts - Subagent compression layer\n * @see CONTEXT_COMPRESSION_REFACTORING_GUIDE.md - Architecture documentation\n */\n\nimport { Message } from '@query'\nimport { queryLLM } from '@services/claude'\nimport { createUserMessage, normalizeMessagesForAPI } from '@utils/messages'\nimport { countTokens } from '@utils/tokens'\nimport {\n getCompressionPrompt,\n CompressionMode,\n} from '@constants/compressionPrompts'\nimport { getCompressionMode } from '@utils/compressionMode'\n\n/**\n * Configuration options for compression behavior\n */\nexport interface CompressionOptions {\n /**\n * Compression mode: determines the structure of the summary\n * - 'business': Optimized for strategic consulting, research, and management\n * - 'code': Optimized for software development, debugging, and coding\n *\n * If not specified, uses the global compression mode setting\n */\n mode?: CompressionMode\n\n /**\n * Threshold ratio for triggering compression (0-1)\n *\n * Compression triggers when: tokenCount >= contextLength * threshold\n * Default: 0.92 (92% of context length)\n *\n * @example\n * threshold: 0.92 // Compress at 92% capacity\n * threshold: 0.80 // Compress earlier at 80% capacity\n */\n threshold?: number\n\n /**\n * Target model's maximum context length in tokens\n *\n * This should match the context length of the model being used\n * by the agent (main agent or subagent).\n *\n * @example\n * contextLength: 200_000 // Claude Sonnet 3.5\n * contextLength: 128_000 // GPT-4\n */\n contextLength: number\n\n /**\n * Whether to recover recently accessed files (main agent only)\n *\n * File recovery adds recently accessed files back into the compressed\n * context to maintain development workflow continuity. This should\n * only be enabled for the main agent.\n *\n * - Main agent: true (preserve coding context)\n * - Subagent: false (keep lightweight)\n *\n * Default: false\n */\n recoverFiles?: boolean\n\n /**\n * Maximum number of files to recover (if recoverFiles is true)\n *\n * Limits the number of files added to prevent context bloat.\n * Default: 5\n */\n maxRecoveredFiles?: number\n}\n\n/**\n * Result of a compression operation\n */\nexport interface CompressionResult {\n /**\n * Compressed message array\n *\n * If compression was performed:\n * - Index 0: User message explaining compression occurred\n * - Index 1: Assistant message containing the summary\n * - Index 2+: Recovered file messages (if recoverFiles was true)\n *\n * If compression was not needed:\n * - Returns the original messages array unchanged\n */\n compressedMessages: Message[]\n\n /**\n * Whether compression was actually performed\n *\n * false if:\n * - Token count below threshold\n * - Message count < 3\n * - Compression failed (error caught)\n */\n wasCompacted: boolean\n\n /**\n * The generated summary text (only if wasCompacted is true)\n *\n * Contains the LLM-generated conversation summary structured\n * according to the compression mode (business or code).\n */\n summary?: string\n\n /**\n * Recovered files data (only if recoverFiles was true and wasCompacted is true)\n *\n * Each entry contains:\n * - path: Absolute file path\n * - content: File content (possibly truncated)\n * - tokens: Estimated token count\n * - truncated: Whether content was truncated due to token limits\n */\n recoveredFiles?: Array<{\n path: string\n content: string\n tokens: number\n truncated: boolean\n }>\n}\n\n/**\n * Stateless Compression Service\n *\n * Provides pure compression functionality without touching any global state.\n * Used as the foundation for both main agent and subagent compression.\n */\nexport class CompressionService {\n /**\n * Determine if compression should be triggered\n *\n * Compression is triggered when:\n * 1. Message count >= 3 (need enough context to summarize)\n * 2. Token count >= contextLength * threshold\n *\n * @param messages - Messages to evaluate for compression\n * @param options - Compression configuration\n * @returns true if compression should be triggered\n *\n * @example\n * ```typescript\n * const shouldCompress = CompressionService.shouldCompress(messages, {\n * contextLength: 200_000,\n * threshold: 0.92,\n * })\n * if (shouldCompress) {\n * // Proceed with compression\n * }\n * ```\n */\n static shouldCompress(\n messages: Message[],\n options: CompressionOptions,\n ): boolean {\n // Need at least 3 messages to create meaningful summary\n if (messages.length < 3) {\n return false\n }\n\n const tokenCount = countTokens(messages)\n const threshold = options.contextLength * (options.threshold ?? 0.92)\n\n return tokenCount >= threshold\n }\n\n /**\n * Execute context compression\n *\n * This method performs the actual compression by:\n * 1. Checking if compression is needed (via shouldCompress)\n * 2. Generating a comprehensive summary using LLM\n * 3. Optionally recovering recently accessed files\n * 4. Building the compressed message array\n *\n * **Important:** This method is completely stateless and has no side effects.\n * It does NOT:\n * - Modify the input messages array\n * - Update any global state (messages, cache, etc.)\n * - Call any state setters\n *\n * State management is the responsibility of the calling layer\n * (MainAgentCompressor or SubagentCompressor).\n *\n * @param messages - Messages to compress\n * @param tools - Available tools for LLM\n * @param abortSignal - Abort signal for cancellation\n * @param options - Compression configuration\n * @returns Compression result with compressed messages and metadata\n *\n * @throws Never throws - errors are caught and returned as wasCompacted: false\n *\n * @example\n * ```typescript\n * // Subagent compression (no file recovery)\n * const result = await CompressionService.compress(\n * subagentMessages,\n * tools,\n * abortSignal,\n * {\n * contextLength: 128_000,\n * recoverFiles: false,\n * }\n * )\n *\n * if (result.wasCompacted) {\n * // Use result.compressedMessages\n * subagentMessages = result.compressedMessages\n * }\n * ```\n *\n * @example\n * ```typescript\n * // Main agent compression (with file recovery)\n * const result = await CompressionService.compress(\n * mainMessages,\n * tools,\n * abortSignal,\n * {\n * contextLength: 200_000,\n * recoverFiles: true,\n * maxRecoveredFiles: 5,\n * }\n * )\n *\n * if (result.wasCompacted) {\n * // Main agent layer will handle global state updates\n * // (messages, cache clearing, etc.)\n * }\n * ```\n */\n static async compress(\n messages: Message[],\n tools: any[],\n abortSignal: AbortSignal,\n options: CompressionOptions,\n ): Promise<CompressionResult> {\n // Early return if compression not needed\n if (!this.shouldCompress(messages, options)) {\n return {\n compressedMessages: messages,\n wasCompacted: false,\n }\n }\n\n try {\n // Step 1: Generate comprehensive conversation summary using LLM\n const mode = options.mode ?? getCompressionMode()\n const compressionPrompt = getCompressionPrompt(mode)\n const summaryRequest = createUserMessage(compressionPrompt)\n\n const summaryResponse = await queryLLM(\n normalizeMessagesForAPI([...messages, summaryRequest]),\n [\n 'You are a helpful AI assistant tasked with creating comprehensive conversation summaries that preserve all essential context for continuing work.',\n ],\n 0,\n tools,\n abortSignal,\n {\n safeMode: false,\n model: 'main', // Use main model for high-quality summarization\n prependCLISysprompt: true,\n },\n )\n\n // Step 2: Extract summary text from response\n const content = summaryResponse.message.content\n const summary =\n typeof content === 'string'\n ? content\n : content.length > 0 && content[0]?.type === 'text'\n ? content[0].text\n : null\n\n if (!summary) {\n throw new Error(\n 'Failed to generate conversation summary - response did not contain valid text content',\n )\n }\n\n // Step 3: Reset token usage for summary response\n // The summary generation itself shouldn't count against the budget\n // since it's an internal operation, not user-requested work\n summaryResponse.message.usage = {\n input_tokens: 0,\n output_tokens: summaryResponse.message.usage.output_tokens,\n cache_creation_input_tokens: 0,\n cache_read_input_tokens: 0,\n }\n\n // Step 4: Build compressed messages array\n const compressedMessages: Message[] = [\n createUserMessage(\n 'Context automatically compressed due to token limit. Essential information preserved.',\n ),\n summaryResponse,\n ]\n\n // Step 5: File recovery (optional, main agent only)\n let recoveredFiles: CompressionResult['recoveredFiles']\n\n if (options.recoverFiles) {\n // TODO: Implement file recovery\n // This should be migrated from src/utils/fileRecoveryCore.ts\n // and integrated here to maintain recently accessed files\n // in the compressed context.\n //\n // Requirements:\n // - Use fileFreshnessService to get important files\n // - Respect maxRecoveredFiles limit\n // - Apply per-file and total token budgets\n // - Add recovered files as user messages to compressedMessages\n //\n // See: src/utils/fileRecoveryCore.ts:selectAndReadFiles()\n recoveredFiles = await this.recoverRecentFiles(\n options.maxRecoveredFiles ?? 5,\n )\n\n // Note: File recovery is intentionally left as empty for now\n // When implemented, recovered files should be appended to\n // compressedMessages as formatted user messages\n }\n\n return {\n compressedMessages,\n wasCompacted: true,\n summary,\n recoveredFiles,\n }\n } catch (error) {\n // Graceful degradation: if compression fails, continue with original messages\n // This ensures the system remains functional even if compression encounters issues\n console.error('Compression failed:', error)\n return {\n compressedMessages: messages,\n wasCompacted: false,\n }\n }\n }\n\n /**\n * Recover recently accessed files for context preservation\n *\n * **Status: Not Implemented (TODO)**\n *\n * This method should be migrated from src/utils/fileRecoveryCore.ts\n * to maintain development context after compression. The implementation\n * should:\n *\n * 1. Query fileFreshnessService for important files\n * 2. Read file contents with proper error handling\n * 3. Apply token budgets (per-file and total)\n * 4. Truncate large files while preserving essential content\n * 5. Return structured file data\n *\n * **Why this is private:**\n * File recovery is an internal implementation detail of the compression\n * process. External code should only interact with the public compress()\n * method.\n *\n * **Why this is only for main agent:**\n * - Main agent needs file context for coding workflows\n * - Subagents should remain lightweight and stateless\n * - File recovery adds significant token overhead\n *\n * @param maxFiles - Maximum number of files to recover\n * @returns Array of recovered file data\n *\n * @private\n * @internal\n *\n * @see src/utils/fileRecoveryCore.ts - Current implementation to migrate\n * @see src/services/fileFreshness.ts - File tracking service\n */\n private static async recoverRecentFiles(\n maxFiles: number,\n ): Promise<NonNullable<CompressionResult['recoveredFiles']>> {\n // TODO: Implement file recovery logic\n //\n // Migration checklist:\n // - [ ] Import fileFreshnessService from @services/fileFreshness\n // - [ ] Call getImportantFiles(maxFiles)\n // - [ ] Read each file using readTextContent from @utils/file\n // - [ ] Estimate tokens (content.length * 0.25)\n // - [ ] Apply MAX_TOKENS_PER_FILE limit (10,000)\n // - [ ] Apply MAX_TOTAL_FILE_TOKENS limit (50,000)\n // - [ ] Handle read errors gracefully (skip failed files)\n // - [ ] Return structured array matching CompressionResult['recoveredFiles']\n //\n // Reference implementation: src/utils/fileRecoveryCore.ts:selectAndReadFiles()\n\n return []\n }\n}\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["/**\n * Compression Service - Stateless Context Compression Core\n *\n * This service provides stateless compression functionality that can be used by\n * both the main agent and subagents without side effects. It is the foundation\n * of the layered compression architecture.\n *\n * **Architecture Design:**\n *\n * ```\n * \u250C\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2510\n * \u2502 Compression System Architecture \u2502\n * \u251C\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2524\n * \u2502 Layer 1: MainAgentCompressor \u2502\n * \u2502 - Handles /compact command and main agent compression \u2502\n * \u2502 - Manages global state (messages, cache) \u2502\n * \u2502 - Recovers recently accessed files \u2502\n * \u251C\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2524\n * \u2502 Layer 2: SubagentCompressor \u2502\n * \u2502 - Handles subagent automatic compression \u2502\n * \u2502 - Operates only on local message arrays \u2502\n * \u2502 - Zero side effects, no global state modification \u2502\n * \u251C\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2524\n * \u2502 Layer 3: CompressionService (THIS FILE) \u2502\n * \u2502 - Stateless compression algorithm \u2502\n * \u2502 - LLM-based summarization \u2502\n * \u2502 - Shared by both layers above \u2502\n * \u2514\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2518\n * ```\n *\n * **Key Principles:**\n * - \u2705 Completely stateless - no global variable access\n * - \u2705 Pure functions - same input always produces same output\n * - \u2705 No side effects - does not modify messages parameter\n * - \u2705 Reusable - shared by main agent and subagent compressors\n *\n * @module CompressionService\n * @see src/services/mainAgentCompressor.ts - Main agent compression layer\n * @see src/services/subagentCompressor.ts - Subagent compression layer\n * @see CONTEXT_COMPRESSION_REFACTORING_GUIDE.md - Architecture documentation\n */\n\nimport { Message } from '@query'\nimport { queryLLM } from '@services/claude'\nimport { createUserMessage, normalizeMessagesForAPI } from '@utils/messages'\nimport { countTokens } from '@utils/tokens'\nimport {\n getCompressionPrompt,\n CompressionMode,\n} from '@constants/compressionPrompts'\nimport { getCompressionMode } from '@utils/compressionMode'\nimport { fileFreshnessService } from '@services/fileFreshness'\nimport { readTextContent } from '@utils/file'\n\n// File recovery configuration constants\nconst MAX_TOKENS_PER_FILE = 10_000\nconst MAX_TOTAL_FILE_TOKENS = 50_000\n\n/**\n * Configuration options for compression behavior\n */\nexport interface CompressionOptions {\n /**\n * Compression mode: determines the structure of the summary\n * - 'business': Optimized for strategic consulting, research, and management\n * - 'code': Optimized for software development, debugging, and coding\n *\n * If not specified, uses the global compression mode setting\n */\n mode?: CompressionMode\n\n /**\n * Threshold ratio for triggering compression (0-1)\n *\n * Compression triggers when: tokenCount >= contextLength * threshold\n * Default: 0.92 (92% of context length)\n *\n * @example\n * threshold: 0.92 // Compress at 92% capacity\n * threshold: 0.80 // Compress earlier at 80% capacity\n */\n threshold?: number\n\n /**\n * Target model's maximum context length in tokens\n *\n * This should match the context length of the model being used\n * by the agent (main agent or subagent).\n *\n * @example\n * contextLength: 200_000 // Claude Sonnet 3.5\n * contextLength: 128_000 // GPT-4\n */\n contextLength: number\n\n /**\n * Whether to recover recently accessed files (main agent only)\n *\n * File recovery adds recently accessed files back into the compressed\n * context to maintain development workflow continuity. This should\n * only be enabled for the main agent.\n *\n * - Main agent: true (preserve coding context)\n * - Subagent: false (keep lightweight)\n *\n * Default: false\n */\n recoverFiles?: boolean\n\n /**\n * Maximum number of files to recover (if recoverFiles is true)\n *\n * Limits the number of files added to prevent context bloat.\n * Default: 5\n */\n maxRecoveredFiles?: number\n}\n\n/**\n * Result of a compression operation\n */\nexport interface CompressionResult {\n /**\n * Compressed message array\n *\n * If compression was performed:\n * - Index 0: User message explaining compression occurred\n * - Index 1: Assistant message containing the summary\n * - Index 2+: Recovered file messages (if recoverFiles was true)\n *\n * If compression was not needed:\n * - Returns the original messages array unchanged\n */\n compressedMessages: Message[]\n\n /**\n * Whether compression was actually performed\n *\n * false if:\n * - Token count below threshold\n * - Message count < 3\n * - Compression failed (error caught)\n */\n wasCompacted: boolean\n\n /**\n * The generated summary text (only if wasCompacted is true)\n *\n * Contains the LLM-generated conversation summary structured\n * according to the compression mode (business or code).\n */\n summary?: string\n\n /**\n * Recovered files data (only if recoverFiles was true and wasCompacted is true)\n *\n * Each entry contains:\n * - path: Absolute file path\n * - content: File content (possibly truncated)\n * - tokens: Estimated token count\n * - truncated: Whether content was truncated due to token limits\n */\n recoveredFiles?: Array<{\n path: string\n content: string\n tokens: number\n truncated: boolean\n }>\n}\n\n/**\n * Stateless Compression Service\n *\n * Provides pure compression functionality without touching any global state.\n * Used as the foundation for both main agent and subagent compression.\n */\nexport class CompressionService {\n /**\n * Determine if compression should be triggered\n *\n * Compression is triggered when:\n * 1. Message count >= 3 (need enough context to summarize)\n * 2. Token count >= contextLength * threshold\n *\n * @param messages - Messages to evaluate for compression\n * @param options - Compression configuration\n * @returns true if compression should be triggered\n *\n * @example\n * ```typescript\n * const shouldCompress = CompressionService.shouldCompress(messages, {\n * contextLength: 200_000,\n * threshold: 0.92,\n * })\n * if (shouldCompress) {\n * // Proceed with compression\n * }\n * ```\n */\n static shouldCompress(\n messages: Message[],\n options: CompressionOptions,\n ): boolean {\n // Need at least 3 messages to create meaningful summary\n if (messages.length < 3) {\n return false\n }\n\n const tokenCount = countTokens(messages)\n const threshold = options.contextLength * (options.threshold ?? 0.92)\n\n return tokenCount >= threshold\n }\n\n /**\n * Execute context compression\n *\n * This method performs the actual compression by:\n * 1. Checking if compression is needed (via shouldCompress)\n * 2. Generating a comprehensive summary using LLM\n * 3. Optionally recovering recently accessed files\n * 4. Building the compressed message array\n *\n * **Important:** This method is completely stateless and has no side effects.\n * It does NOT:\n * - Modify the input messages array\n * - Update any global state (messages, cache, etc.)\n * - Call any state setters\n *\n * State management is the responsibility of the calling layer\n * (MainAgentCompressor or SubagentCompressor).\n *\n * @param messages - Messages to compress\n * @param tools - Available tools for LLM\n * @param abortSignal - Abort signal for cancellation\n * @param options - Compression configuration\n * @returns Compression result with compressed messages and metadata\n *\n * @throws Never throws - errors are caught and returned as wasCompacted: false\n *\n * @example\n * ```typescript\n * // Subagent compression (no file recovery)\n * const result = await CompressionService.compress(\n * subagentMessages,\n * tools,\n * abortSignal,\n * {\n * contextLength: 128_000,\n * recoverFiles: false,\n * }\n * )\n *\n * if (result.wasCompacted) {\n * // Use result.compressedMessages\n * subagentMessages = result.compressedMessages\n * }\n * ```\n *\n * @example\n * ```typescript\n * // Main agent compression (with file recovery)\n * const result = await CompressionService.compress(\n * mainMessages,\n * tools,\n * abortSignal,\n * {\n * contextLength: 200_000,\n * recoverFiles: true,\n * maxRecoveredFiles: 5,\n * }\n * )\n *\n * if (result.wasCompacted) {\n * // Main agent layer will handle global state updates\n * // (messages, cache clearing, etc.)\n * }\n * ```\n */\n static async compress(\n messages: Message[],\n tools: any[],\n abortSignal: AbortSignal,\n options: CompressionOptions,\n ): Promise<CompressionResult> {\n // Early return if compression not needed\n if (!this.shouldCompress(messages, options)) {\n return {\n compressedMessages: messages,\n wasCompacted: false,\n }\n }\n\n try {\n // Step 1: Generate comprehensive conversation summary using LLM\n const mode = options.mode ?? getCompressionMode()\n const compressionPrompt = getCompressionPrompt(mode)\n const summaryRequest = createUserMessage(compressionPrompt)\n\n const summaryResponse = await queryLLM(\n normalizeMessagesForAPI([...messages, summaryRequest]),\n [\n 'You are a helpful AI assistant tasked with creating comprehensive conversation summaries that preserve all essential context for continuing work.',\n ],\n 0,\n tools,\n abortSignal,\n {\n safeMode: false,\n model: 'main', // Use main model for high-quality summarization\n prependCLISysprompt: true,\n },\n )\n\n // Step 2: Extract summary text from response\n const content = summaryResponse.message.content\n const summary =\n typeof content === 'string'\n ? content\n : content.length > 0 && content[0]?.type === 'text'\n ? content[0].text\n : null\n\n if (!summary) {\n throw new Error(\n 'Failed to generate conversation summary - response did not contain valid text content',\n )\n }\n\n // Step 3: Reset token usage for summary response\n // The summary generation itself shouldn't count against the budget\n // since it's an internal operation, not user-requested work\n summaryResponse.message.usage = {\n input_tokens: 0,\n output_tokens: summaryResponse.message.usage.output_tokens,\n cache_creation_input_tokens: 0,\n cache_read_input_tokens: 0,\n }\n\n // Step 4: Build compressed messages array\n const compressedMessages: Message[] = [\n createUserMessage(\n 'Context automatically compressed due to token limit. Essential information preserved.',\n ),\n summaryResponse,\n ]\n\n // Step 5: File recovery (optional, main agent only)\n let recoveredFiles: CompressionResult['recoveredFiles']\n\n if (options.recoverFiles) {\n recoveredFiles = await this.recoverRecentFiles(\n options.maxRecoveredFiles ?? 5,\n )\n\n // Add recovered files as user messages to preserve context\n if (recoveredFiles && recoveredFiles.length > 0) {\n for (const file of recoveredFiles) {\n const truncatedNote = file.truncated ? ' (truncated)' : ''\n const fileMessage = createUserMessage(\n `[Recovered File${truncatedNote}] ${file.path}\\n\\n\\`\\`\\`\\n${file.content}\\n\\`\\`\\``,\n )\n compressedMessages.push(fileMessage)\n }\n }\n }\n\n return {\n compressedMessages,\n wasCompacted: true,\n summary,\n recoveredFiles,\n }\n } catch (error) {\n // Graceful degradation: if compression fails, continue with original messages\n // This ensures the system remains functional even if compression encounters issues\n console.error('Compression failed:', error)\n return {\n compressedMessages: messages,\n wasCompacted: false,\n }\n }\n }\n\n /**\n * Recover recently accessed files for context preservation\n *\n * **Status: Not Implemented (TODO)**\n *\n * This method should be migrated from src/utils/fileRecoveryCore.ts\n * to maintain development context after compression. The implementation\n * should:\n *\n * 1. Query fileFreshnessService for important files\n * 2. Read file contents with proper error handling\n * 3. Apply token budgets (per-file and total)\n * 4. Truncate large files while preserving essential content\n * 5. Return structured file data\n *\n * **Why this is private:**\n * File recovery is an internal implementation detail of the compression\n * process. External code should only interact with the public compress()\n * method.\n *\n * **Why this is only for main agent:**\n * - Main agent needs file context for coding workflows\n * - Subagents should remain lightweight and stateless\n * - File recovery adds significant token overhead\n *\n * @param maxFiles - Maximum number of files to recover\n * @returns Array of recovered file data\n *\n * @private\n * @internal\n *\n * @see src/utils/fileRecoveryCore.ts - Current implementation to migrate\n * @see src/services/fileFreshness.ts - File tracking service\n */\n private static async recoverRecentFiles(\n maxFiles: number,\n ): Promise<NonNullable<CompressionResult['recoveredFiles']>> {\n const importantFiles = fileFreshnessService.getImportantFiles(maxFiles)\n const results: NonNullable<CompressionResult['recoveredFiles']> = []\n let totalTokens = 0\n\n for (const fileInfo of importantFiles) {\n try {\n const { content } = readTextContent(fileInfo.path)\n const estimatedTokens = Math.ceil(content.length * 0.25)\n\n // Apply per-file token limit to prevent any single file from dominating context\n let finalContent = content\n let truncated = false\n\n if (estimatedTokens > MAX_TOKENS_PER_FILE) {\n const maxChars = Math.floor(MAX_TOKENS_PER_FILE / 0.25)\n finalContent = content.substring(0, maxChars)\n truncated = true\n }\n\n const finalTokens = Math.min(estimatedTokens, MAX_TOKENS_PER_FILE)\n\n // Enforce total token budget to maintain compression effectiveness\n if (totalTokens + finalTokens > MAX_TOTAL_FILE_TOKENS) {\n break\n }\n\n totalTokens += finalTokens\n results.push({\n path: fileInfo.path,\n content: finalContent,\n tokens: finalTokens,\n truncated,\n })\n } catch (error) {\n // Skip files that cannot be read - don't let one failure stop the process\n console.error(\n `Failed to read file for recovery: ${fileInfo.path}`,\n error,\n )\n }\n }\n\n return results\n }\n}\n"],
5
+ "mappings": "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",
6
6
  "names": []
7
7
  }
@@ -0,0 +1,162 @@
1
+ import { countTokens } from "../utils/tokens.js";
2
+ import { getModelManager } from "../utils/model.js";
3
+ import { debug } from "../utils/debugLogger.js";
4
+ const DEFAULT_CONFIG = {
5
+ warningThreshold: 0.7,
6
+ autoCompactThreshold: 0.95,
7
+ avgTokensPerTurn: 2e3,
8
+ minTurnsBeforeWarning: 3
9
+ };
10
+ class ContextMonitor {
11
+ config;
12
+ lastSuggestTime = 0;
13
+ suggestCooldownMs = 6e4;
14
+ // 1 minute cooldown
15
+ turnsSinceSuggest = 0;
16
+ constructor(config) {
17
+ this.config = { ...DEFAULT_CONFIG, ...config };
18
+ }
19
+ /**
20
+ * Update configuration
21
+ */
22
+ updateConfig(config) {
23
+ this.config = { ...this.config, ...config };
24
+ }
25
+ /**
26
+ * Get current configuration
27
+ */
28
+ getConfig() {
29
+ return { ...this.config };
30
+ }
31
+ /**
32
+ * Check context usage and return a comprehensive report
33
+ */
34
+ checkUsage(messages, modelPointer = "main") {
35
+ const tokenCount = countTokens(messages);
36
+ const maxTokens = this.getContextLength(modelPointer);
37
+ const usagePercent = maxTokens > 0 ? tokenCount / maxTokens : 0;
38
+ let status = "normal";
39
+ let action = "none";
40
+ let message = "";
41
+ if (usagePercent >= this.config.autoCompactThreshold) {
42
+ status = "critical";
43
+ action = "auto-compact";
44
+ message = `Context critical (${this.formatPercent(usagePercent)}). Auto-compacting to preserve quality.`;
45
+ } else if (usagePercent >= this.config.warningThreshold) {
46
+ status = "warning";
47
+ const now = Date.now();
48
+ const cooldownPassed = now - this.lastSuggestTime >= this.suggestCooldownMs;
49
+ const enoughTurnsPassed = this.turnsSinceSuggest >= this.config.minTurnsBeforeWarning;
50
+ if (cooldownPassed && enoughTurnsPassed) {
51
+ action = "suggest-compact";
52
+ message = `Context at ${this.formatPercent(usagePercent)}. Consider running /compact to preserve response quality.`;
53
+ this.lastSuggestTime = now;
54
+ this.turnsSinceSuggest = 0;
55
+ } else {
56
+ message = `Context at ${this.formatPercent(usagePercent)}. Running low on context space.`;
57
+ }
58
+ } else {
59
+ message = `Context usage normal (${this.formatPercent(usagePercent)}).`;
60
+ }
61
+ const remainingTokens = maxTokens - tokenCount;
62
+ const estimatedTurnsRemaining = Math.max(
63
+ 0,
64
+ Math.floor(remainingTokens / this.config.avgTokensPerTurn)
65
+ );
66
+ const report = {
67
+ tokenCount,
68
+ maxTokens,
69
+ usagePercent,
70
+ status,
71
+ action,
72
+ estimatedTurnsRemaining,
73
+ compactSuggested: action === "suggest-compact",
74
+ message
75
+ };
76
+ debug.state("CONTEXT_MONITOR_CHECK", {
77
+ tokenCount: String(tokenCount),
78
+ maxTokens: String(maxTokens),
79
+ usagePercent: this.formatPercent(usagePercent),
80
+ status,
81
+ action,
82
+ estimatedTurns: String(estimatedTurnsRemaining)
83
+ });
84
+ return report;
85
+ }
86
+ /**
87
+ * Record that a turn was completed (for cooldown tracking)
88
+ */
89
+ recordTurn() {
90
+ this.turnsSinceSuggest++;
91
+ }
92
+ /**
93
+ * Reset suggest cooldown (e.g., after user manually compacts)
94
+ */
95
+ resetCooldown() {
96
+ this.lastSuggestTime = 0;
97
+ this.turnsSinceSuggest = 0;
98
+ }
99
+ /**
100
+ * Get context length for a model
101
+ */
102
+ getContextLength(modelPointer) {
103
+ try {
104
+ const modelManager = getModelManager();
105
+ const profile = modelManager.resolveModel(modelPointer);
106
+ if (profile) {
107
+ return profile.contextLength;
108
+ }
109
+ return 2e5;
110
+ } catch {
111
+ return 2e5;
112
+ }
113
+ }
114
+ /**
115
+ * Format percentage for display
116
+ */
117
+ formatPercent(value) {
118
+ return `${Math.round(value * 100)}%`;
119
+ }
120
+ }
121
+ function getContextVisualization(report) {
122
+ const progressPercent = Math.round(report.usagePercent * 100);
123
+ let color = "green";
124
+ if (report.status === "critical") {
125
+ color = "red";
126
+ } else if (report.status === "warning") {
127
+ color = "yellow";
128
+ }
129
+ const formatTokens = (n) => {
130
+ if (n >= 1e6) return `${(n / 1e6).toFixed(1)}M`;
131
+ if (n >= 1e3) return `${(n / 1e3).toFixed(1)}K`;
132
+ return String(n);
133
+ };
134
+ return {
135
+ progressPercent,
136
+ color,
137
+ label: `Context: ${progressPercent}%`,
138
+ stats: {
139
+ used: formatTokens(report.tokenCount),
140
+ total: formatTokens(report.maxTokens),
141
+ remaining: formatTokens(report.maxTokens - report.tokenCount),
142
+ turns: report.estimatedTurnsRemaining
143
+ }
144
+ };
145
+ }
146
+ let globalMonitor = null;
147
+ function getContextMonitor() {
148
+ if (!globalMonitor) {
149
+ globalMonitor = new ContextMonitor();
150
+ }
151
+ return globalMonitor;
152
+ }
153
+ function resetContextMonitor() {
154
+ globalMonitor = null;
155
+ }
156
+ export {
157
+ ContextMonitor,
158
+ getContextMonitor,
159
+ getContextVisualization,
160
+ resetContextMonitor
161
+ };
162
+ //# sourceMappingURL=contextMonitor.js.map
@@ -0,0 +1,7 @@
1
+ {
2
+ "version": 3,
3
+ "sources": ["../../src/services/contextMonitor.ts"],
4
+ "sourcesContent": ["/**\n * Context Monitor Service\n *\n * Monitors token usage and provides intelligent compression recommendations.\n * Implements the 70%/95% threshold system from the optimization roadmap.\n *\n * Features:\n * - Real-time token usage tracking\n * - 70% threshold warning (suggest manual /compact)\n * - 95% threshold auto-compact trigger\n * - Estimated turns remaining calculation\n * - Context health status reporting\n */\n\nimport { Message } from '@query'\nimport { countTokens } from '@utils/tokens'\nimport { getModelManager } from '@utils/model'\nimport { debug } from '@utils/debugLogger'\n\n/**\n * Context status levels\n */\nexport type ContextStatus = 'normal' | 'warning' | 'critical'\n\n/**\n * Recommended action based on context status\n */\nexport type ContextAction = 'none' | 'suggest-compact' | 'auto-compact'\n\n/**\n * Context usage report\n */\nexport interface ContextUsageReport {\n /** Current token count */\n tokenCount: number\n /** Maximum context length for current model */\n maxTokens: number\n /** Usage percentage (0-1) */\n usagePercent: number\n /** Context health status */\n status: ContextStatus\n /** Recommended action */\n action: ContextAction\n /** Estimated turns remaining at current pace */\n estimatedTurnsRemaining: number\n /** Whether compact was recently suggested */\n compactSuggested: boolean\n /** Human-readable status message */\n message: string\n}\n\n/**\n * Context Monitor Configuration\n */\nexport interface ContextMonitorConfig {\n /** Warning threshold (0-1), default 0.70 */\n warningThreshold: number\n /** Auto-compact threshold (0-1), default 0.95 */\n autoCompactThreshold: number\n /** Average tokens per turn estimate, default 2000 */\n avgTokensPerTurn: number\n /** Minimum turns before suggesting compact, default 3 */\n minTurnsBeforeWarning: number\n}\n\n/**\n * Default configuration\n */\nconst DEFAULT_CONFIG: ContextMonitorConfig = {\n warningThreshold: 0.7,\n autoCompactThreshold: 0.95,\n avgTokensPerTurn: 2000,\n minTurnsBeforeWarning: 3,\n}\n\n/**\n * Context Monitor\n *\n * Tracks context usage and provides compression recommendations.\n */\nexport class ContextMonitor {\n private config: ContextMonitorConfig\n private lastSuggestTime: number = 0\n private suggestCooldownMs: number = 60000 // 1 minute cooldown\n private turnsSinceSuggest: number = 0\n\n constructor(config?: Partial<ContextMonitorConfig>) {\n this.config = { ...DEFAULT_CONFIG, ...config }\n }\n\n /**\n * Update configuration\n */\n updateConfig(config: Partial<ContextMonitorConfig>): void {\n this.config = { ...this.config, ...config }\n }\n\n /**\n * Get current configuration\n */\n getConfig(): ContextMonitorConfig {\n return { ...this.config }\n }\n\n /**\n * Check context usage and return a comprehensive report\n */\n checkUsage(\n messages: Message[],\n modelPointer: string = 'main',\n ): ContextUsageReport {\n const tokenCount = countTokens(messages)\n const maxTokens = this.getContextLength(modelPointer)\n const usagePercent = maxTokens > 0 ? tokenCount / maxTokens : 0\n\n // Determine status\n let status: ContextStatus = 'normal'\n let action: ContextAction = 'none'\n let message = ''\n\n if (usagePercent >= this.config.autoCompactThreshold) {\n status = 'critical'\n action = 'auto-compact'\n message = `Context critical (${this.formatPercent(usagePercent)}). Auto-compacting to preserve quality.`\n } else if (usagePercent >= this.config.warningThreshold) {\n status = 'warning'\n\n // Only suggest compact if cooldown has passed\n const now = Date.now()\n const cooldownPassed =\n now - this.lastSuggestTime >= this.suggestCooldownMs\n const enoughTurnsPassed =\n this.turnsSinceSuggest >= this.config.minTurnsBeforeWarning\n\n if (cooldownPassed && enoughTurnsPassed) {\n action = 'suggest-compact'\n message = `Context at ${this.formatPercent(usagePercent)}. Consider running /compact to preserve response quality.`\n this.lastSuggestTime = now\n this.turnsSinceSuggest = 0\n } else {\n message = `Context at ${this.formatPercent(usagePercent)}. Running low on context space.`\n }\n } else {\n message = `Context usage normal (${this.formatPercent(usagePercent)}).`\n }\n\n // Calculate estimated turns remaining\n const remainingTokens = maxTokens - tokenCount\n const estimatedTurnsRemaining = Math.max(\n 0,\n Math.floor(remainingTokens / this.config.avgTokensPerTurn),\n )\n\n const report: ContextUsageReport = {\n tokenCount,\n maxTokens,\n usagePercent,\n status,\n action,\n estimatedTurnsRemaining,\n compactSuggested: action === 'suggest-compact',\n message,\n }\n\n debug.state('CONTEXT_MONITOR_CHECK', {\n tokenCount: String(tokenCount),\n maxTokens: String(maxTokens),\n usagePercent: this.formatPercent(usagePercent),\n status,\n action,\n estimatedTurns: String(estimatedTurnsRemaining),\n })\n\n return report\n }\n\n /**\n * Record that a turn was completed (for cooldown tracking)\n */\n recordTurn(): void {\n this.turnsSinceSuggest++\n }\n\n /**\n * Reset suggest cooldown (e.g., after user manually compacts)\n */\n resetCooldown(): void {\n this.lastSuggestTime = 0\n this.turnsSinceSuggest = 0\n }\n\n /**\n * Get context length for a model\n */\n private getContextLength(modelPointer: string): number {\n try {\n const modelManager = getModelManager()\n // Use resolveModel which handles both pointer types and model names\n const profile = modelManager.resolveModel(modelPointer)\n if (profile) {\n return profile.contextLength\n }\n // Default fallback for Claude\n return 200_000\n } catch {\n return 200_000\n }\n }\n\n /**\n * Format percentage for display\n */\n private formatPercent(value: number): string {\n return `${Math.round(value * 100)}%`\n }\n}\n\n/**\n * Context usage visualization data for UI\n */\nexport interface ContextVisualization {\n /** Progress bar segments (0-100) */\n progressPercent: number\n /** Color for progress bar */\n color: 'green' | 'yellow' | 'red'\n /** Label text */\n label: string\n /** Detailed stats */\n stats: {\n used: string\n total: string\n remaining: string\n turns: number\n }\n}\n\n/**\n * Generate visualization data for context usage\n */\nexport function getContextVisualization(\n report: ContextUsageReport,\n): ContextVisualization {\n const progressPercent = Math.round(report.usagePercent * 100)\n\n let color: ContextVisualization['color'] = 'green'\n if (report.status === 'critical') {\n color = 'red'\n } else if (report.status === 'warning') {\n color = 'yellow'\n }\n\n const formatTokens = (n: number): string => {\n if (n >= 1000000) return `${(n / 1000000).toFixed(1)}M`\n if (n >= 1000) return `${(n / 1000).toFixed(1)}K`\n return String(n)\n }\n\n return {\n progressPercent,\n color,\n label: `Context: ${progressPercent}%`,\n stats: {\n used: formatTokens(report.tokenCount),\n total: formatTokens(report.maxTokens),\n remaining: formatTokens(report.maxTokens - report.tokenCount),\n turns: report.estimatedTurnsRemaining,\n },\n }\n}\n\n// Global singleton instance\nlet globalMonitor: ContextMonitor | null = null\n\n/**\n * Get the global context monitor instance\n */\nexport function getContextMonitor(): ContextMonitor {\n if (!globalMonitor) {\n globalMonitor = new ContextMonitor()\n }\n return globalMonitor\n}\n\n/**\n * Reset the global context monitor (for testing)\n */\nexport function resetContextMonitor(): void {\n globalMonitor = null\n}\n"],
5
+ "mappings": "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",
6
+ "names": []
7
+ }
@@ -6,7 +6,49 @@ import { getCwd } from "../utils/state.js";
6
6
  import { execFile } from "child_process";
7
7
  import { promisify } from "util";
8
8
  import { loadAllPlugins } from "../utils/pluginLoader.js";
9
+ import {
10
+ getCurrentProjectConfig
11
+ } from "../utils/config.js";
9
12
  const execFileAsync = promisify(execFile);
13
+ function isBashCommandAllowed(command) {
14
+ const projectConfig = getCurrentProjectConfig();
15
+ const allowedTools = projectConfig?.allowedTools || [];
16
+ const parts = command.split(/\s+/);
17
+ const baseCommand = parts[0];
18
+ if (allowedTools.includes(`Bash(${command})`)) {
19
+ return true;
20
+ }
21
+ if (allowedTools.includes(`Bash(${baseCommand}:*)`)) {
22
+ return true;
23
+ }
24
+ const safeCommands = /* @__PURE__ */ new Set([
25
+ "git",
26
+ "ls",
27
+ "pwd",
28
+ "date",
29
+ "which",
30
+ "echo",
31
+ "cat",
32
+ "head",
33
+ "tail",
34
+ "wc",
35
+ "sort",
36
+ "uniq",
37
+ "grep",
38
+ "find",
39
+ "dirname",
40
+ "basename"
41
+ ]);
42
+ if (safeCommands.has(baseCommand)) {
43
+ return true;
44
+ }
45
+ return false;
46
+ }
47
+ function isFileReadAllowed(filePath) {
48
+ const projectConfig = getCurrentProjectConfig();
49
+ const allowedTools = projectConfig?.allowedTools || [];
50
+ return true;
51
+ }
10
52
  async function executeBashCommands(content) {
11
53
  const bashCommandRegex = /!\`([^`]+)\`/g;
12
54
  const matches = [...content.matchAll(bashCommandRegex)];
@@ -17,6 +59,13 @@ async function executeBashCommands(content) {
17
59
  for (const match of matches) {
18
60
  const fullMatch = match[0];
19
61
  const command = match[1].trim();
62
+ if (!isBashCommandAllowed(command)) {
63
+ console.warn(
64
+ `Custom command bash execution denied: "${command}" - not in allowed list`
65
+ );
66
+ result = result.replace(fullMatch, `(permission denied: ${command})`);
67
+ continue;
68
+ }
20
69
  try {
21
70
  const parts = command.split(/\s+/);
22
71
  const cmd = parts[0];
@@ -72,8 +121,6 @@ ${fileContent}
72
121
  return result;
73
122
  }
74
123
  function validateAllowedTools(allowedTools) {
75
- if (allowedTools && allowedTools.length > 0) {
76
- }
77
124
  return true;
78
125
  }
79
126
  function parseFrontmatter(content) {
@@ -147,9 +194,8 @@ function createCustomCommand(frontmatter, content, filePath, baseDir) {
147
194
  const relativePath = filePath.replace(baseDir + "/", "");
148
195
  const pathParts = relativePath.split("/");
149
196
  const fileName = pathParts[pathParts.length - 1].replace(".md", "");
150
- const userClaudeDir = join(homedir(), ".claude", "commands");
151
197
  const userMintoDir = join(homedir(), ".minto", "commands");
152
- const scope = baseDir === userClaudeDir || baseDir === userMintoDir ? "user" : "project";
198
+ const scope = baseDir === userMintoDir ? "user" : "project";
153
199
  const prefix = scope === "user" ? "user" : "project";
154
200
  let finalName;
155
201
  if (frontmatter.name) {
@@ -221,47 +267,26 @@ IMPORTANT: You are restricted to using only these tools: ${allowedToolsStr}. Do
221
267
  }
222
268
  const loadCustomCommands = memoize(
223
269
  async () => {
224
- const userClaudeDir = join(homedir(), ".claude", "commands");
225
- const projectClaudeDir = join(getCwd(), ".claude", "commands");
226
270
  const userMintoDir = join(homedir(), ".minto", "commands");
227
271
  const projectMintoDir = join(getCwd(), ".minto", "commands");
228
272
  const abortController = new AbortController();
229
273
  const timeout = setTimeout(() => abortController.abort(), 3e3);
230
274
  try {
231
275
  const startTime = Date.now();
232
- const [
233
- projectClaudeFiles,
234
- userClaudeFiles,
235
- projectMintoFiles,
236
- userMintoFiles
237
- ] = await Promise.all([
238
- existsSync(projectClaudeDir) ? scanMarkdownFiles(
239
- ["--files", "--hidden", "--glob", "*.md"],
240
- // Legacy args for ripgrep compatibility
241
- projectClaudeDir,
242
- abortController.signal
243
- ) : Promise.resolve([]),
244
- existsSync(userClaudeDir) ? scanMarkdownFiles(
245
- ["--files", "--glob", "*.md"],
246
- // Legacy args for ripgrep compatibility
247
- userClaudeDir,
248
- abortController.signal
249
- ) : Promise.resolve([]),
276
+ const [projectMintoFiles, userMintoFiles] = await Promise.all([
250
277
  existsSync(projectMintoDir) ? scanMarkdownFiles(
251
278
  ["--files", "--hidden", "--glob", "*.md"],
252
- // Legacy args for ripgrep compatibility
253
279
  projectMintoDir,
254
280
  abortController.signal
255
281
  ) : Promise.resolve([]),
256
282
  existsSync(userMintoDir) ? scanMarkdownFiles(
257
283
  ["--files", "--glob", "*.md"],
258
- // Legacy args for ripgrep compatibility
259
284
  userMintoDir,
260
285
  abortController.signal
261
286
  ) : Promise.resolve([])
262
287
  ]);
263
- const projectFiles = [...projectMintoFiles, ...projectClaudeFiles];
264
- const userFiles = [...userMintoFiles, ...userClaudeFiles];
288
+ const projectFiles = [...projectMintoFiles];
289
+ const userFiles = [...userMintoFiles];
265
290
  const allFiles = [...projectFiles, ...userFiles];
266
291
  const duration = Date.now() - startTime;
267
292
  const commands = [];
@@ -269,12 +294,11 @@ const loadCustomCommands = memoize(
269
294
  try {
270
295
  const content = readFileSync(filePath, { encoding: "utf-8" });
271
296
  const { frontmatter, content: commandContent } = parseFrontmatter(content);
272
- const baseDir = filePath.includes(".minto/commands") ? projectMintoDir : projectClaudeDir;
273
297
  const command = createCustomCommand(
274
298
  frontmatter,
275
299
  commandContent,
276
300
  filePath,
277
- baseDir
301
+ projectMintoDir
278
302
  );
279
303
  if (command) {
280
304
  commands.push(command);
@@ -287,12 +311,11 @@ const loadCustomCommands = memoize(
287
311
  try {
288
312
  const content = readFileSync(filePath, { encoding: "utf-8" });
289
313
  const { frontmatter, content: commandContent } = parseFrontmatter(content);
290
- const baseDir = filePath.includes(".minto/commands") ? userMintoDir : userClaudeDir;
291
314
  const command = createCustomCommand(
292
315
  frontmatter,
293
316
  commandContent,
294
317
  filePath,
295
- baseDir
318
+ userMintoDir
296
319
  );
297
320
  if (command) {
298
321
  commands.push(command);
@@ -314,11 +337,9 @@ const loadCustomCommands = memoize(
314
337
  // This ensures cache invalidation when directories change
315
338
  () => {
316
339
  const cwd = getCwd();
317
- const userClaudeDir = join(homedir(), ".claude", "commands");
318
- const projectClaudeDir = join(cwd, ".claude", "commands");
319
340
  const userMintoDir = join(homedir(), ".minto", "commands");
320
341
  const projectMintoDir = join(cwd, ".minto", "commands");
321
- return `${cwd}:${existsSync(userClaudeDir)}:${existsSync(projectClaudeDir)}:${existsSync(userMintoDir)}:${existsSync(projectMintoDir)}:${Math.floor(Date.now() / 6e4)}`;
342
+ return `${cwd}:${existsSync(userMintoDir)}:${existsSync(projectMintoDir)}:${Math.floor(Date.now() / 6e4)}`;
322
343
  }
323
344
  );
324
345
  const reloadCustomCommands = () => {
@@ -326,15 +347,13 @@ const reloadCustomCommands = () => {
326
347
  };
327
348
  function getCustomCommandDirectories() {
328
349
  return {
329
- userClaude: join(homedir(), ".claude", "commands"),
330
- projectClaude: join(getCwd(), ".claude", "commands"),
331
- userMinto: join(homedir(), ".minto", "commands"),
332
- projectMinto: join(getCwd(), ".minto", "commands")
350
+ user: join(homedir(), ".minto", "commands"),
351
+ project: join(getCwd(), ".minto", "commands")
333
352
  };
334
353
  }
335
354
  function hasCustomCommands() {
336
- const { userClaude, projectClaude, userMinto, projectMinto } = getCustomCommandDirectories();
337
- return existsSync(userClaude) || existsSync(projectClaude) || existsSync(userMinto) || existsSync(projectMinto);
355
+ const { user, project } = getCustomCommandDirectories();
356
+ return existsSync(user) || existsSync(project);
338
357
  }
339
358
  const loadPluginCommands = memoize(
340
359
  async () => {