@oh-my-pi/pi-coding-agent 8.0.20 → 8.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 (421) hide show
  1. package/CHANGELOG.md +125 -0
  2. package/docs/session.md +111 -46
  3. package/examples/custom-tools/hello/index.ts +1 -1
  4. package/examples/custom-tools/todo/index.ts +3 -4
  5. package/examples/extensions/api-demo.ts +0 -1
  6. package/examples/extensions/chalk-logger.ts +2 -3
  7. package/examples/extensions/hello.ts +0 -1
  8. package/examples/extensions/pirate.ts +0 -1
  9. package/examples/extensions/plan-mode.ts +15 -16
  10. package/examples/extensions/todo.ts +3 -4
  11. package/examples/extensions/tools.ts +1 -2
  12. package/examples/extensions/with-deps/index.ts +0 -1
  13. package/examples/hooks/auto-commit-on-exit.ts +1 -2
  14. package/examples/hooks/confirm-destructive.ts +0 -1
  15. package/examples/hooks/custom-compaction.ts +1 -2
  16. package/examples/hooks/dirty-repo-guard.ts +0 -1
  17. package/examples/hooks/file-trigger.ts +3 -4
  18. package/examples/hooks/git-checkpoint.ts +0 -1
  19. package/examples/hooks/handoff.ts +3 -4
  20. package/examples/hooks/permission-gate.ts +1 -2
  21. package/examples/hooks/protected-paths.ts +1 -2
  22. package/examples/hooks/qna.ts +2 -3
  23. package/examples/hooks/snake.ts +4 -5
  24. package/examples/hooks/status-line.ts +0 -1
  25. package/examples/sdk/01-minimal.ts +2 -3
  26. package/examples/sdk/02-custom-model.ts +2 -3
  27. package/examples/sdk/03-custom-prompt.ts +3 -4
  28. package/examples/sdk/04-skills.ts +2 -3
  29. package/examples/sdk/06-extensions.ts +1 -2
  30. package/examples/sdk/06-hooks.ts +6 -7
  31. package/examples/sdk/07-context-files.ts +0 -1
  32. package/examples/sdk/08-prompt-templates.ts +0 -1
  33. package/examples/sdk/08-slash-commands.ts +0 -1
  34. package/examples/sdk/09-api-keys-and-oauth.ts +0 -1
  35. package/examples/sdk/10-settings.ts +0 -1
  36. package/examples/sdk/11-sessions.ts +0 -1
  37. package/package.json +54 -23
  38. package/scripts/format-prompts.ts +0 -1
  39. package/src/capability/context-file.ts +3 -4
  40. package/src/capability/extension-module.ts +3 -4
  41. package/src/capability/extension.ts +3 -4
  42. package/src/capability/fs.ts +20 -21
  43. package/src/capability/hook.ts +3 -4
  44. package/src/capability/index.ts +15 -16
  45. package/src/capability/instruction.ts +3 -4
  46. package/src/capability/mcp.ts +3 -4
  47. package/src/capability/prompt.ts +3 -4
  48. package/src/capability/rule.ts +3 -4
  49. package/src/capability/settings.ts +2 -3
  50. package/src/capability/skill.ts +3 -4
  51. package/src/capability/slash-command.ts +3 -4
  52. package/src/capability/ssh.ts +3 -4
  53. package/src/capability/system-prompt.ts +3 -4
  54. package/src/capability/tool.ts +3 -4
  55. package/src/cli/args.ts +5 -6
  56. package/src/cli/config-cli.ts +6 -7
  57. package/src/cli/file-processor.ts +19 -17
  58. package/src/cli/jupyter-cli.ts +105 -0
  59. package/src/cli/list-models.ts +10 -11
  60. package/src/cli/plugin-cli.ts +20 -25
  61. package/src/cli/session-picker.ts +2 -3
  62. package/src/cli/setup-cli.ts +2 -3
  63. package/src/cli/stats-cli.ts +2 -3
  64. package/src/cli/update-cli.ts +25 -22
  65. package/src/commit/agentic/agent.ts +307 -0
  66. package/src/commit/agentic/fallback.ts +96 -0
  67. package/src/commit/agentic/index.ts +351 -0
  68. package/src/commit/agentic/prompts/analyze-file.md +22 -0
  69. package/src/commit/agentic/prompts/session-user.md +26 -0
  70. package/src/commit/agentic/prompts/split-confirm.md +1 -0
  71. package/src/commit/agentic/prompts/system.md +40 -0
  72. package/src/commit/agentic/state.ts +69 -0
  73. package/src/commit/agentic/tools/analyze-file.ts +131 -0
  74. package/src/commit/agentic/tools/git-file-diff.ts +194 -0
  75. package/src/commit/agentic/tools/git-hunk.ts +50 -0
  76. package/src/commit/agentic/tools/git-overview.ts +84 -0
  77. package/src/commit/agentic/tools/index.ts +56 -0
  78. package/src/commit/agentic/tools/propose-changelog.ts +128 -0
  79. package/src/commit/agentic/tools/propose-commit.ts +154 -0
  80. package/src/commit/agentic/tools/recent-commits.ts +81 -0
  81. package/src/commit/agentic/tools/split-commit.ts +280 -0
  82. package/src/commit/agentic/topo-sort.ts +44 -0
  83. package/src/commit/agentic/trivial.ts +51 -0
  84. package/src/commit/agentic/validation.ts +200 -0
  85. package/src/commit/analysis/conventional.ts +165 -0
  86. package/src/commit/analysis/index.ts +4 -0
  87. package/src/commit/analysis/scope.ts +242 -0
  88. package/src/commit/analysis/summary.ts +112 -0
  89. package/src/commit/analysis/validation.ts +66 -0
  90. package/src/commit/changelog/detect.ts +36 -0
  91. package/src/commit/changelog/generate.ts +110 -0
  92. package/src/commit/changelog/index.ts +233 -0
  93. package/src/commit/changelog/parse.ts +44 -0
  94. package/src/commit/cli.ts +93 -0
  95. package/src/commit/git/diff.ts +148 -0
  96. package/src/commit/git/errors.ts +11 -0
  97. package/src/commit/git/index.ts +212 -0
  98. package/src/commit/git/operations.ts +53 -0
  99. package/src/commit/index.ts +5 -0
  100. package/src/commit/map-reduce/index.ts +63 -0
  101. package/src/commit/map-reduce/map-phase.ts +178 -0
  102. package/src/commit/map-reduce/reduce-phase.ts +145 -0
  103. package/src/commit/map-reduce/utils.ts +9 -0
  104. package/src/commit/message.ts +11 -0
  105. package/src/commit/model-selection.ts +80 -0
  106. package/src/commit/pipeline.ts +240 -0
  107. package/src/commit/prompts/analysis-system.md +155 -0
  108. package/src/commit/prompts/analysis-user.md +41 -0
  109. package/src/commit/prompts/changelog-system.md +56 -0
  110. package/src/commit/prompts/changelog-user.md +19 -0
  111. package/src/commit/prompts/file-observer-system.md +26 -0
  112. package/src/commit/prompts/file-observer-user.md +9 -0
  113. package/src/commit/prompts/reduce-system.md +60 -0
  114. package/src/commit/prompts/reduce-user.md +17 -0
  115. package/src/commit/prompts/summary-retry.md +4 -0
  116. package/src/commit/prompts/summary-system.md +52 -0
  117. package/src/commit/prompts/summary-user.md +13 -0
  118. package/src/commit/prompts/types-description.md +2 -0
  119. package/src/commit/types.ts +109 -0
  120. package/src/commit/utils/exclusions.ts +42 -0
  121. package/src/config/file-lock.ts +121 -0
  122. package/src/config/keybindings.ts +6 -8
  123. package/src/config/model-registry.ts +65 -38
  124. package/src/config/model-resolver.ts +18 -19
  125. package/src/config/prompt-templates.ts +11 -11
  126. package/src/config/settings-manager.ts +141 -50
  127. package/src/config.ts +64 -66
  128. package/src/cursor.ts +11 -9
  129. package/src/discovery/agents-md.ts +11 -12
  130. package/src/discovery/builtin.ts +68 -73
  131. package/src/discovery/claude.ts +41 -42
  132. package/src/discovery/cline.ts +11 -12
  133. package/src/discovery/codex.ts +52 -53
  134. package/src/discovery/cursor.ts +9 -10
  135. package/src/discovery/gemini.ts +17 -22
  136. package/src/discovery/github.ts +13 -14
  137. package/src/discovery/helpers.ts +35 -34
  138. package/src/discovery/index.ts +22 -24
  139. package/src/discovery/mcp-json.ts +8 -9
  140. package/src/discovery/ssh.ts +8 -9
  141. package/src/discovery/vscode.ts +4 -5
  142. package/src/discovery/windsurf.ts +6 -7
  143. package/src/exa/company.ts +1 -2
  144. package/src/exa/index.ts +2 -3
  145. package/src/exa/linkedin.ts +1 -2
  146. package/src/exa/mcp-client.ts +14 -16
  147. package/src/exa/render.ts +10 -11
  148. package/src/exa/researcher.ts +1 -2
  149. package/src/exa/search.ts +1 -2
  150. package/src/exa/types.ts +0 -1
  151. package/src/exa/websets.ts +1 -2
  152. package/src/exec/bash-executor.ts +3 -4
  153. package/src/exec/exec.ts +0 -1
  154. package/src/export/custom-share.ts +5 -6
  155. package/src/export/html/index.ts +24 -21
  156. package/src/export/ttsr.ts +2 -3
  157. package/src/extensibility/custom-commands/bundled/review/index.ts +7 -8
  158. package/src/extensibility/custom-commands/loader.ts +18 -15
  159. package/src/extensibility/custom-commands/types.ts +2 -3
  160. package/src/extensibility/custom-tools/loader.ts +11 -12
  161. package/src/extensibility/custom-tools/types.ts +7 -8
  162. package/src/extensibility/custom-tools/wrapper.ts +2 -3
  163. package/src/extensibility/extensions/loader.ts +76 -54
  164. package/src/extensibility/extensions/runner.ts +11 -12
  165. package/src/extensibility/extensions/types.ts +20 -27
  166. package/src/extensibility/extensions/wrapper.ts +3 -4
  167. package/src/extensibility/hooks/index.ts +1 -1
  168. package/src/extensibility/hooks/loader.ts +9 -10
  169. package/src/extensibility/hooks/runner.ts +7 -8
  170. package/src/extensibility/hooks/tool-wrapper.ts +0 -1
  171. package/src/extensibility/hooks/types.ts +11 -18
  172. package/src/extensibility/plugins/doctor.ts +3 -3
  173. package/src/extensibility/plugins/installer.ts +27 -27
  174. package/src/extensibility/plugins/loader.ts +59 -56
  175. package/src/extensibility/plugins/manager.ts +211 -171
  176. package/src/extensibility/plugins/parser.ts +1 -1
  177. package/src/extensibility/plugins/paths.ts +8 -8
  178. package/src/extensibility/skills.ts +63 -60
  179. package/src/extensibility/slash-commands.ts +10 -10
  180. package/src/index.ts +54 -54
  181. package/src/internal-urls/agent-protocol.ts +21 -11
  182. package/src/internal-urls/artifact-protocol.ts +17 -13
  183. package/src/internal-urls/router.ts +1 -2
  184. package/src/internal-urls/rule-protocol.ts +3 -4
  185. package/src/internal-urls/skill-protocol.ts +3 -4
  186. package/src/ipy/executor.ts +109 -9
  187. package/src/ipy/gateway-coordinator.ts +79 -90
  188. package/src/ipy/kernel.ts +32 -30
  189. package/src/ipy/modules.ts +13 -13
  190. package/src/lsp/client.ts +21 -10
  191. package/src/lsp/clients/biome-client.ts +1 -2
  192. package/src/lsp/clients/index.ts +3 -3
  193. package/src/lsp/clients/lsp-linter-client.ts +4 -5
  194. package/src/lsp/config.ts +15 -15
  195. package/src/lsp/edits.ts +4 -5
  196. package/src/lsp/index.ts +43 -44
  197. package/src/lsp/lspmux.ts +8 -8
  198. package/src/lsp/render.ts +99 -61
  199. package/src/lsp/utils.ts +3 -3
  200. package/src/main.ts +71 -37
  201. package/src/mcp/client.ts +2 -3
  202. package/src/mcp/config.ts +5 -6
  203. package/src/mcp/json-rpc.ts +0 -1
  204. package/src/mcp/loader.ts +6 -7
  205. package/src/mcp/manager.ts +17 -18
  206. package/src/mcp/tool-bridge.ts +4 -9
  207. package/src/mcp/tool-cache.ts +2 -3
  208. package/src/mcp/transports/http.ts +2 -4
  209. package/src/mcp/transports/stdio.ts +1 -2
  210. package/src/migrations.ts +63 -52
  211. package/src/modes/components/armin.ts +4 -5
  212. package/src/modes/components/assistant-message.ts +33 -5
  213. package/src/modes/components/bash-execution.ts +7 -8
  214. package/src/modes/components/bordered-loader.ts +3 -3
  215. package/src/modes/components/branch-summary-message.ts +3 -3
  216. package/src/modes/components/compaction-summary-message.ts +3 -3
  217. package/src/modes/components/countdown-timer.ts +0 -1
  218. package/src/modes/components/custom-message.ts +5 -5
  219. package/src/modes/components/diff.ts +1 -1
  220. package/src/modes/components/dynamic-border.ts +2 -2
  221. package/src/modes/components/extensions/extension-dashboard.ts +6 -7
  222. package/src/modes/components/extensions/extension-list.ts +2 -3
  223. package/src/modes/components/extensions/inspector-panel.ts +3 -4
  224. package/src/modes/components/extensions/state-manager.ts +25 -26
  225. package/src/modes/components/extensions/types.ts +1 -2
  226. package/src/modes/components/footer.ts +47 -43
  227. package/src/modes/components/history-search.ts +2 -2
  228. package/src/modes/components/hook-editor.ts +3 -4
  229. package/src/modes/components/hook-input.ts +2 -3
  230. package/src/modes/components/hook-message.ts +5 -5
  231. package/src/modes/components/hook-selector.ts +2 -3
  232. package/src/modes/components/keybinding-hints.ts +2 -3
  233. package/src/modes/components/login-dialog.ts +2 -2
  234. package/src/modes/components/model-selector.ts +12 -12
  235. package/src/modes/components/oauth-selector.ts +2 -2
  236. package/src/modes/components/plugin-settings.ts +20 -20
  237. package/src/modes/components/python-execution.ts +7 -8
  238. package/src/modes/components/queue-mode-selector.ts +3 -3
  239. package/src/modes/components/read-tool-group.ts +2 -2
  240. package/src/modes/components/session-selector.ts +4 -4
  241. package/src/modes/components/settings-defs.ts +77 -69
  242. package/src/modes/components/settings-selector.ts +16 -16
  243. package/src/modes/components/show-images-selector.ts +2 -2
  244. package/src/modes/components/status-line/segments.ts +4 -4
  245. package/src/modes/components/status-line/separators.ts +1 -1
  246. package/src/modes/components/status-line/types.ts +2 -2
  247. package/src/modes/components/status-line-segment-editor.ts +7 -8
  248. package/src/modes/components/status-line.ts +12 -12
  249. package/src/modes/components/theme-selector.ts +8 -7
  250. package/src/modes/components/thinking-selector.ts +4 -4
  251. package/src/modes/components/todo-display.ts +2 -2
  252. package/src/modes/components/todo-reminder.ts +4 -4
  253. package/src/modes/components/tool-execution.ts +16 -19
  254. package/src/modes/components/tree-selector.ts +12 -12
  255. package/src/modes/components/ttsr-notification.ts +5 -5
  256. package/src/modes/components/user-message-selector.ts +1 -1
  257. package/src/modes/components/user-message.ts +1 -1
  258. package/src/modes/components/visual-truncate.ts +0 -1
  259. package/src/modes/components/welcome.ts +4 -4
  260. package/src/modes/controllers/command-controller.ts +46 -47
  261. package/src/modes/controllers/event-controller.ts +16 -20
  262. package/src/modes/controllers/extension-ui-controller.ts +40 -46
  263. package/src/modes/controllers/input-controller.ts +17 -18
  264. package/src/modes/controllers/selector-controller.ts +103 -91
  265. package/src/modes/index.ts +3 -3
  266. package/src/modes/interactive-mode.ts +31 -31
  267. package/src/modes/print-mode.ts +12 -13
  268. package/src/modes/rpc/rpc-client.ts +7 -8
  269. package/src/modes/rpc/rpc-mode.ts +24 -28
  270. package/src/modes/rpc/rpc-types.ts +3 -4
  271. package/src/modes/theme/mermaid-cache.ts +89 -0
  272. package/src/modes/theme/theme.ts +130 -53
  273. package/src/modes/types.ts +10 -10
  274. package/src/modes/utils/ui-helpers.ts +17 -17
  275. package/src/patch/applicator.ts +18 -19
  276. package/src/patch/diff.ts +1 -2
  277. package/src/patch/fuzzy.ts +1 -2
  278. package/src/patch/index.ts +11 -18
  279. package/src/patch/normalize.ts +4 -4
  280. package/src/patch/normative.ts +1 -2
  281. package/src/patch/parser.ts +8 -9
  282. package/src/patch/shared.ts +43 -16
  283. package/src/prompts/tools/task.md +2 -0
  284. package/src/sdk.ts +100 -65
  285. package/src/session/agent-session.ts +84 -85
  286. package/src/session/agent-storage.ts +43 -39
  287. package/src/session/artifacts.ts +32 -10
  288. package/src/session/auth-storage.ts +50 -39
  289. package/src/session/compaction/branch-summarization.ts +7 -10
  290. package/src/session/compaction/compaction.ts +8 -19
  291. package/src/session/compaction/utils.ts +6 -9
  292. package/src/session/history-storage.ts +10 -10
  293. package/src/session/messages.ts +4 -5
  294. package/src/session/session-manager.ts +76 -65
  295. package/src/session/session-storage.ts +57 -69
  296. package/src/session/storage-migration.ts +14 -56
  297. package/src/session/streaming-output.ts +2 -2
  298. package/src/ssh/connection-manager.ts +43 -50
  299. package/src/ssh/ssh-executor.ts +2 -2
  300. package/src/ssh/sshfs-mount.ts +11 -18
  301. package/src/system-prompt.ts +28 -35
  302. package/src/task/agents.ts +45 -30
  303. package/src/task/commands.ts +6 -7
  304. package/src/task/discovery.ts +39 -76
  305. package/src/task/executor.ts +14 -15
  306. package/src/task/index.ts +40 -34
  307. package/src/task/output-manager.ts +93 -0
  308. package/src/task/parallel.ts +0 -1
  309. package/src/task/render.ts +24 -30
  310. package/src/task/subprocess-tool-registry.ts +1 -2
  311. package/src/task/worker-protocol.ts +3 -3
  312. package/src/task/worker.ts +33 -39
  313. package/src/task/worktree.ts +19 -19
  314. package/src/tools/ask.ts +41 -20
  315. package/src/tools/bash-interceptor.ts +1 -5
  316. package/src/tools/bash.ts +91 -97
  317. package/src/tools/calculator.ts +49 -47
  318. package/src/tools/complete.ts +4 -5
  319. package/src/tools/context.ts +2 -2
  320. package/src/tools/fetch.ts +84 -124
  321. package/src/tools/find.ts +94 -98
  322. package/src/tools/gemini-image.ts +14 -14
  323. package/src/tools/grep.ts +100 -116
  324. package/src/tools/index.ts +80 -55
  325. package/src/tools/list-limit.ts +1 -1
  326. package/src/tools/ls.ts +44 -70
  327. package/src/tools/notebook.ts +51 -67
  328. package/src/tools/output-meta.ts +3 -4
  329. package/src/tools/output-utils.ts +2 -2
  330. package/src/tools/path-utils.ts +5 -5
  331. package/src/tools/python.ts +104 -217
  332. package/src/tools/read.ts +92 -33
  333. package/src/tools/render-utils.ts +8 -23
  334. package/src/tools/renderers.ts +6 -7
  335. package/src/tools/review.ts +8 -11
  336. package/src/tools/ssh.ts +69 -49
  337. package/src/tools/todo-write.ts +37 -25
  338. package/src/tools/tool-errors.ts +3 -3
  339. package/src/tools/tool-result.ts +3 -8
  340. package/src/tools/write.ts +99 -75
  341. package/src/tui/code-cell.ts +109 -0
  342. package/src/tui/file-list.ts +47 -0
  343. package/src/tui/index.ts +11 -0
  344. package/src/tui/output-block.ts +72 -0
  345. package/src/tui/status-line.ts +39 -0
  346. package/src/tui/tree-list.ts +55 -0
  347. package/src/tui/types.ts +16 -0
  348. package/src/tui/utils.ts +48 -0
  349. package/src/utils/changelog.ts +9 -10
  350. package/src/utils/clipboard.ts +11 -11
  351. package/src/utils/file-mentions.ts +4 -10
  352. package/src/utils/frontmatter.ts +6 -3
  353. package/src/utils/fuzzy.ts +2 -2
  354. package/src/utils/image-convert.ts +1 -1
  355. package/src/utils/image-resize.ts +1 -1
  356. package/src/utils/mime.ts +2 -2
  357. package/src/utils/shell-snapshot.ts +11 -13
  358. package/src/utils/shell.ts +4 -5
  359. package/src/utils/title-generator.ts +8 -9
  360. package/src/utils/tools-manager.ts +23 -23
  361. package/src/vendor/photon/index.js +1099 -1059
  362. package/src/vendor/photon/photon_rs_bg.wasm +0 -0
  363. package/src/web/scrapers/artifacthub.ts +1 -1
  364. package/src/web/scrapers/arxiv.ts +2 -2
  365. package/src/web/scrapers/bluesky.ts +2 -2
  366. package/src/web/scrapers/cheatsh.ts +1 -1
  367. package/src/web/scrapers/chocolatey.ts +2 -2
  368. package/src/web/scrapers/choosealicense.ts +5 -5
  369. package/src/web/scrapers/cisa-kev.ts +1 -1
  370. package/src/web/scrapers/crossref.ts +2 -2
  371. package/src/web/scrapers/devto.ts +3 -3
  372. package/src/web/scrapers/discogs.ts +3 -4
  373. package/src/web/scrapers/discourse.ts +1 -1
  374. package/src/web/scrapers/dockerhub.ts +1 -1
  375. package/src/web/scrapers/fdroid.ts +2 -2
  376. package/src/web/scrapers/firefox-addons.ts +3 -3
  377. package/src/web/scrapers/flathub.ts +1 -1
  378. package/src/web/scrapers/github.ts +3 -3
  379. package/src/web/scrapers/gitlab.ts +4 -4
  380. package/src/web/scrapers/hackernews.ts +2 -2
  381. package/src/web/scrapers/huggingface.ts +1 -1
  382. package/src/web/scrapers/iacr.ts +2 -2
  383. package/src/web/scrapers/index.ts +0 -1
  384. package/src/web/scrapers/jetbrains-marketplace.ts +1 -1
  385. package/src/web/scrapers/lemmy.ts +2 -2
  386. package/src/web/scrapers/maven.ts +2 -2
  387. package/src/web/scrapers/mdn.ts +2 -4
  388. package/src/web/scrapers/metacpan.ts +2 -2
  389. package/src/web/scrapers/musicbrainz.ts +1 -2
  390. package/src/web/scrapers/npm.ts +1 -1
  391. package/src/web/scrapers/nuget.ts +2 -2
  392. package/src/web/scrapers/nvd.ts +3 -3
  393. package/src/web/scrapers/ollama.ts +7 -9
  394. package/src/web/scrapers/opencorporates.ts +2 -2
  395. package/src/web/scrapers/openlibrary.ts +6 -6
  396. package/src/web/scrapers/orcid.ts +0 -1
  397. package/src/web/scrapers/osv.ts +2 -2
  398. package/src/web/scrapers/packagist.ts +1 -1
  399. package/src/web/scrapers/pubmed.ts +1 -2
  400. package/src/web/scrapers/rawg.ts +2 -2
  401. package/src/web/scrapers/readthedocs.ts +1 -2
  402. package/src/web/scrapers/repology.ts +2 -2
  403. package/src/web/scrapers/rfc.ts +1 -1
  404. package/src/web/scrapers/searchcode.ts +2 -2
  405. package/src/web/scrapers/semantic-scholar.ts +1 -1
  406. package/src/web/scrapers/snapcraft.ts +2 -2
  407. package/src/web/scrapers/sourcegraph.ts +1 -1
  408. package/src/web/scrapers/spdx.ts +3 -3
  409. package/src/web/scrapers/spotify.ts +0 -1
  410. package/src/web/scrapers/twitter.ts +1 -1
  411. package/src/web/scrapers/types.ts +1 -2
  412. package/src/web/scrapers/utils.ts +5 -5
  413. package/src/web/scrapers/wikidata.ts +3 -3
  414. package/src/web/scrapers/youtube.ts +9 -14
  415. package/src/web/search/auth.ts +5 -10
  416. package/src/web/search/index.ts +11 -21
  417. package/src/web/search/providers/anthropic.ts +3 -9
  418. package/src/web/search/providers/exa.ts +6 -10
  419. package/src/web/search/providers/perplexity.ts +5 -5
  420. package/src/web/search/render.ts +129 -175
  421. package/tsconfig.json +0 -42
@@ -0,0 +1,194 @@
1
+ import { Type } from "@sinclair/typebox";
2
+ import type { CommitAgentState } from "../../../commit/agentic/state";
3
+ import type { ControlledGit } from "../../../commit/git";
4
+ import type { CustomTool } from "../../../extensibility/custom-tools/types";
5
+
6
+ const TARGET_TOKENS = 30000;
7
+ const CHARS_PER_TOKEN = 4;
8
+ const MAX_CHARS = TARGET_TOKENS * CHARS_PER_TOKEN;
9
+ const TRUNCATE_THRESHOLD_LINES = 30;
10
+ const KEEP_HEAD_LINES = 15;
11
+ const KEEP_TAIL_LINES = 10;
12
+
13
+ const HIGH_PRIORITY_EXTENSIONS = new Set([
14
+ ".rs",
15
+ ".go",
16
+ ".py",
17
+ ".js",
18
+ ".ts",
19
+ ".tsx",
20
+ ".jsx",
21
+ ".java",
22
+ ".c",
23
+ ".cpp",
24
+ ".h",
25
+ ".hpp",
26
+ ]);
27
+ const SHELL_SQL_EXTENSIONS = new Set([".sh", ".bash", ".zsh", ".sql"]);
28
+ const MANIFEST_FILES = new Set([
29
+ "Cargo.toml",
30
+ "package.json",
31
+ "go.mod",
32
+ "pyproject.toml",
33
+ "requirements.txt",
34
+ "Gemfile",
35
+ "build.gradle",
36
+ "pom.xml",
37
+ ]);
38
+ const LOW_PRIORITY_EXTENSIONS = new Set([".md", ".txt", ".json", ".yaml", ".yml", ".toml", ".xml", ".csv"]);
39
+ const BINARY_EXTENSIONS = new Set([
40
+ ".png",
41
+ ".jpg",
42
+ ".jpeg",
43
+ ".gif",
44
+ ".ico",
45
+ ".woff",
46
+ ".woff2",
47
+ ".ttf",
48
+ ".eot",
49
+ ".pdf",
50
+ ".zip",
51
+ ".tar",
52
+ ".gz",
53
+ ".exe",
54
+ ".dll",
55
+ ".so",
56
+ ".dylib",
57
+ ]);
58
+ const TEST_PATTERNS = ["/test/", "/tests/", "/__tests__/", "_test.", ".test.", ".spec.", "_spec."];
59
+
60
+ export function getFilePriority(filename: string): number {
61
+ const basename = filename.split("/").pop() ?? filename;
62
+ const ext = basename.includes(".") ? `.${basename.split(".").pop()}` : "";
63
+
64
+ if (BINARY_EXTENSIONS.has(ext)) return -100;
65
+
66
+ const lowerPath = filename.toLowerCase();
67
+ for (const pattern of TEST_PATTERNS) {
68
+ if (lowerPath.includes(pattern)) return 10;
69
+ }
70
+
71
+ if (LOW_PRIORITY_EXTENSIONS.has(ext) && !MANIFEST_FILES.has(basename)) return 20;
72
+ if (MANIFEST_FILES.has(basename)) return 70;
73
+ if (SHELL_SQL_EXTENSIONS.has(ext)) return 80;
74
+ if (HIGH_PRIORITY_EXTENSIONS.has(ext)) return 100;
75
+
76
+ return 50;
77
+ }
78
+
79
+ function truncateDiffContent(diff: string): { content: string; truncated: boolean } {
80
+ const lines = diff.split("\n");
81
+ if (lines.length <= TRUNCATE_THRESHOLD_LINES) {
82
+ return { content: diff, truncated: false };
83
+ }
84
+
85
+ const head = lines.slice(0, KEEP_HEAD_LINES);
86
+ const tail = lines.slice(-KEEP_TAIL_LINES);
87
+ const truncatedCount = lines.length - KEEP_HEAD_LINES - KEEP_TAIL_LINES;
88
+
89
+ return {
90
+ content: [...head, `\n... (truncated ${truncatedCount} lines) ...\n`, ...tail].join("\n"),
91
+ truncated: true,
92
+ };
93
+ }
94
+
95
+ function processDiffs(files: string[], diffs: Map<string, string>): { result: string; truncatedFiles: string[] } {
96
+ const sortedFiles = [...files].sort((a, b) => getFilePriority(b) - getFilePriority(a));
97
+
98
+ const truncatedFiles: string[] = [];
99
+ const parts: string[] = [];
100
+ let totalChars = 0;
101
+
102
+ for (const file of sortedFiles) {
103
+ const diff = diffs.get(file);
104
+ if (!diff) continue;
105
+
106
+ const remaining = MAX_CHARS - totalChars;
107
+ if (remaining <= 0) {
108
+ truncatedFiles.push(file);
109
+ continue;
110
+ }
111
+
112
+ let content = diff;
113
+ if (content.length > remaining || content.split("\n").length > TRUNCATE_THRESHOLD_LINES) {
114
+ const { content: truncated, truncated: wasTruncated } = truncateDiffContent(content);
115
+ if (wasTruncated) {
116
+ truncatedFiles.push(file);
117
+ }
118
+ content = truncated;
119
+ if (content.length > remaining) {
120
+ content = `${content.slice(0, remaining)}\n... (diff truncated due to size) ...`;
121
+ if (!truncatedFiles.includes(file)) {
122
+ truncatedFiles.push(file);
123
+ }
124
+ }
125
+ }
126
+
127
+ parts.push(`=== ${file} ===\n${content}`);
128
+ totalChars += content.length;
129
+ }
130
+
131
+ return { result: parts.join("\n\n"), truncatedFiles };
132
+ }
133
+
134
+ const gitFileDiffSchema = Type.Object({
135
+ files: Type.Array(Type.String({ description: "Files to diff" }), { minItems: 1, maxItems: 10 }),
136
+ staged: Type.Optional(Type.Boolean({ description: "Use staged changes (default: true)" })),
137
+ });
138
+
139
+ export function createGitFileDiffTool(
140
+ git: ControlledGit,
141
+ state: CommitAgentState,
142
+ ): CustomTool<typeof gitFileDiffSchema> {
143
+ return {
144
+ name: "git_file_diff",
145
+ label: "Git File Diff",
146
+ description: "Return the diff for specific files.",
147
+ parameters: gitFileDiffSchema,
148
+ async execute(_toolCallId, params) {
149
+ const staged = params.staged ?? true;
150
+ const cacheKey = (file: string) => `${file}:${staged}`;
151
+
152
+ if (!state.diffCache) {
153
+ state.diffCache = new Map();
154
+ }
155
+
156
+ const diffs = new Map<string, string>();
157
+ const uncachedFiles: string[] = [];
158
+
159
+ for (const file of params.files) {
160
+ const cached = state.diffCache.get(cacheKey(file));
161
+ if (cached !== undefined) {
162
+ diffs.set(file, cached);
163
+ } else {
164
+ uncachedFiles.push(file);
165
+ }
166
+ }
167
+
168
+ if (uncachedFiles.length > 0) {
169
+ for (const file of uncachedFiles) {
170
+ const diff = await git.getDiffForFiles([file], staged);
171
+ if (diff) {
172
+ diffs.set(file, diff);
173
+ state.diffCache.set(cacheKey(file), diff);
174
+ } else {
175
+ state.diffCache.set(cacheKey(file), "");
176
+ }
177
+ }
178
+ }
179
+
180
+ const { result, truncatedFiles } = processDiffs(params.files, diffs);
181
+ const output = result || "(no diff)";
182
+
183
+ return {
184
+ content: [{ type: "text", text: output }],
185
+ details: {
186
+ files: params.files,
187
+ staged,
188
+ truncatedFiles: truncatedFiles.length > 0 ? truncatedFiles : undefined,
189
+ cacheHits: params.files.length - uncachedFiles.length,
190
+ },
191
+ };
192
+ },
193
+ };
194
+ }
@@ -0,0 +1,50 @@
1
+ import { Type } from "@sinclair/typebox";
2
+ import type { ControlledGit } from "../../../commit/git";
3
+ import type { DiffHunk, FileHunks } from "../../../commit/types";
4
+ import type { CustomTool } from "../../../extensibility/custom-tools/types";
5
+
6
+ const gitHunkSchema = Type.Object({
7
+ file: Type.String({ description: "File path" }),
8
+ hunks: Type.Optional(Type.Array(Type.Number({ description: "1-based hunk indices" }), { minItems: 1 })),
9
+ staged: Type.Optional(Type.Boolean({ description: "Use staged changes (default: true)" })),
10
+ });
11
+
12
+ function selectHunks(fileHunks: FileHunks, requested?: number[]): DiffHunk[] {
13
+ if (!requested || requested.length === 0) return fileHunks.hunks;
14
+ const wanted = new Set(requested.map(value => Math.max(1, Math.floor(value))));
15
+ return fileHunks.hunks.filter(hunk => wanted.has(hunk.index + 1));
16
+ }
17
+
18
+ export function createGitHunkTool(git: ControlledGit): CustomTool<typeof gitHunkSchema> {
19
+ return {
20
+ name: "git_hunk",
21
+ label: "Git Hunk",
22
+ description: "Return specific hunks from a file diff.",
23
+ parameters: gitHunkSchema,
24
+ async execute(_toolCallId, params) {
25
+ const staged = params.staged ?? true;
26
+ const hunks = await git.getHunks([params.file], staged);
27
+ const fileHunks = hunks.find(entry => entry.filename === params.file) ?? {
28
+ filename: params.file,
29
+ isBinary: false,
30
+ hunks: [],
31
+ };
32
+ if (fileHunks.isBinary) {
33
+ return {
34
+ content: [{ type: "text", text: "Binary file diff; no hunks available." }],
35
+ details: { file: params.file, staged, hunks: [] },
36
+ };
37
+ }
38
+ const selected = selectHunks(fileHunks, params.hunks);
39
+ const text = selected.length ? selected.map(hunk => hunk.content).join("\n\n") : "(no matching hunks)";
40
+ return {
41
+ content: [{ type: "text", text }],
42
+ details: {
43
+ file: params.file,
44
+ staged,
45
+ hunks: selected,
46
+ },
47
+ };
48
+ },
49
+ };
50
+ }
@@ -0,0 +1,84 @@
1
+ import { Type } from "@sinclair/typebox";
2
+ import type { CommitAgentState, GitOverviewSnapshot } from "../../../commit/agentic/state";
3
+ import { extractScopeCandidates } from "../../../commit/analysis/scope";
4
+ import type { ControlledGit } from "../../../commit/git";
5
+ import type { CustomTool } from "../../../extensibility/custom-tools/types";
6
+
7
+ const EXCLUDED_LOCK_FILES = new Set([
8
+ "Cargo.lock",
9
+ "package-lock.json",
10
+ "yarn.lock",
11
+ "pnpm-lock.yaml",
12
+ "bun.lock",
13
+ "bun.lockb",
14
+ "go.sum",
15
+ "poetry.lock",
16
+ "Pipfile.lock",
17
+ "uv.lock",
18
+ "composer.lock",
19
+ "Gemfile.lock",
20
+ "flake.lock",
21
+ "pubspec.lock",
22
+ "Podfile.lock",
23
+ "mix.lock",
24
+ "gradle.lockfile",
25
+ ]);
26
+
27
+ function isExcludedFile(path: string): boolean {
28
+ const basename = path.split("/").pop() ?? path;
29
+ return EXCLUDED_LOCK_FILES.has(basename);
30
+ }
31
+
32
+ function filterExcludedFiles(files: string[]): { filtered: string[]; excluded: string[] } {
33
+ const filtered: string[] = [];
34
+ const excluded: string[] = [];
35
+ for (const file of files) {
36
+ if (isExcludedFile(file)) {
37
+ excluded.push(file);
38
+ } else {
39
+ filtered.push(file);
40
+ }
41
+ }
42
+ return { filtered, excluded };
43
+ }
44
+
45
+ const gitOverviewSchema = Type.Object({
46
+ staged: Type.Optional(Type.Boolean({ description: "Use staged changes (default: true)" })),
47
+ include_untracked: Type.Optional(Type.Boolean({ description: "Include untracked files when staged=false" })),
48
+ });
49
+
50
+ export function createGitOverviewTool(
51
+ git: ControlledGit,
52
+ state: CommitAgentState,
53
+ ): CustomTool<typeof gitOverviewSchema> {
54
+ return {
55
+ name: "git_overview",
56
+ label: "Git Overview",
57
+ description: "Return staged files, diff stat summary, and numstat entries.",
58
+ parameters: gitOverviewSchema,
59
+ async execute(_toolCallId, params) {
60
+ const staged = params.staged ?? true;
61
+ const allFiles = staged ? await git.getStagedFiles() : await git.getChangedFiles(false);
62
+ const { filtered: files, excluded } = filterExcludedFiles(allFiles);
63
+ const stat = await git.getStat(staged);
64
+ const allNumstat = await git.getNumstat(staged);
65
+ const numstat = allNumstat.filter(entry => !isExcludedFile(entry.path));
66
+ const scopeResult = extractScopeCandidates(numstat);
67
+ const untrackedFiles = !staged && params.include_untracked ? await git.getUntrackedFiles() : undefined;
68
+ const snapshot: GitOverviewSnapshot = {
69
+ files,
70
+ stat,
71
+ numstat,
72
+ scopeCandidates: scopeResult.scopeCandidates,
73
+ isWideScope: scopeResult.isWide,
74
+ untrackedFiles,
75
+ excludedFiles: excluded.length > 0 ? excluded : undefined,
76
+ };
77
+ state.overview = snapshot;
78
+ return {
79
+ content: [{ type: "text", text: JSON.stringify(snapshot, null, 2) }],
80
+ details: snapshot,
81
+ };
82
+ },
83
+ };
84
+ }
@@ -0,0 +1,56 @@
1
+ import type { CommitAgentState } from "../../../commit/agentic/state";
2
+ import type { ControlledGit } from "../../../commit/git";
3
+ import type { ModelRegistry } from "../../../config/model-registry";
4
+ import type { SettingsManager } from "../../../config/settings-manager";
5
+ import type { CustomTool } from "../../../extensibility/custom-tools/types";
6
+ import type { AuthStorage } from "../../../session/auth-storage";
7
+ import { createAnalyzeFileTool } from "./analyze-file";
8
+ import { createGitFileDiffTool } from "./git-file-diff";
9
+ import { createGitHunkTool } from "./git-hunk";
10
+ import { createGitOverviewTool } from "./git-overview";
11
+ import { createProposeChangelogTool } from "./propose-changelog";
12
+ import { createProposeCommitTool } from "./propose-commit";
13
+ import { createRecentCommitsTool } from "./recent-commits";
14
+ import { createSplitCommitTool } from "./split-commit";
15
+
16
+ export interface CommitToolOptions {
17
+ cwd: string;
18
+ git: ControlledGit;
19
+ authStorage: AuthStorage;
20
+ modelRegistry: ModelRegistry;
21
+ settingsManager: SettingsManager;
22
+ spawns: string;
23
+ state: CommitAgentState;
24
+ changelogTargets: string[];
25
+ enableAnalyzeFiles?: boolean;
26
+ }
27
+
28
+ export function createCommitTools(options: CommitToolOptions): Array<CustomTool<any, any>> {
29
+ const tools: Array<CustomTool<any, any>> = [
30
+ createGitOverviewTool(options.git, options.state),
31
+ createGitFileDiffTool(options.git, options.state),
32
+ createGitHunkTool(options.git),
33
+ createRecentCommitsTool(options.git),
34
+ ];
35
+
36
+ if (options.enableAnalyzeFiles ?? true) {
37
+ tools.push(
38
+ createAnalyzeFileTool({
39
+ cwd: options.cwd,
40
+ authStorage: options.authStorage,
41
+ modelRegistry: options.modelRegistry,
42
+ settingsManager: options.settingsManager,
43
+ spawns: options.spawns,
44
+ state: options.state,
45
+ }),
46
+ );
47
+ }
48
+
49
+ tools.push(
50
+ createProposeChangelogTool(options.state, options.changelogTargets),
51
+ createProposeCommitTool(options.git, options.state),
52
+ createSplitCommitTool(options.git, options.state, options.changelogTargets),
53
+ );
54
+
55
+ return tools;
56
+ }
@@ -0,0 +1,128 @@
1
+ import { Type } from "@sinclair/typebox";
2
+ import type { CommitAgentState } from "../../../commit/agentic/state";
3
+ import type { ChangelogCategory } from "../../../commit/types";
4
+ import type { CustomTool } from "../../../extensibility/custom-tools/types";
5
+
6
+ const changelogEntrySchema = Type.Object({
7
+ path: Type.String(),
8
+ entries: Type.Record(Type.String(), Type.Array(Type.String())),
9
+ deletions: Type.Optional(
10
+ Type.Record(Type.String(), Type.Array(Type.String()), {
11
+ description: "Entries to remove from existing changelog sections (case-insensitive match)",
12
+ }),
13
+ ),
14
+ });
15
+
16
+ const proposeChangelogSchema = Type.Object({
17
+ entries: Type.Array(changelogEntrySchema),
18
+ });
19
+
20
+ interface ChangelogResponse {
21
+ valid: boolean;
22
+ errors: string[];
23
+ warnings: string[];
24
+ }
25
+
26
+ const allowedCategories = new Set<ChangelogCategory>([
27
+ "Breaking Changes",
28
+ "Added",
29
+ "Changed",
30
+ "Deprecated",
31
+ "Removed",
32
+ "Fixed",
33
+ "Security",
34
+ ]);
35
+
36
+ export function createProposeChangelogTool(
37
+ state: CommitAgentState,
38
+ changelogTargets: string[],
39
+ ): CustomTool<typeof proposeChangelogSchema> {
40
+ return {
41
+ name: "propose_changelog",
42
+ label: "Propose Changelog",
43
+ description: "Provide changelog entries for targeted CHANGELOG.md files.",
44
+ parameters: proposeChangelogSchema,
45
+ async execute(_toolCallId, params) {
46
+ const errors: string[] = [];
47
+ const warnings: string[] = [];
48
+ const targets = new Set(changelogTargets);
49
+ const seen = new Set<string>();
50
+
51
+ const normalized = params.entries.map(entry => {
52
+ const cleaned: Record<string, string[]> = {};
53
+ for (const [category, values] of Object.entries(entry.entries ?? {})) {
54
+ if (!allowedCategories.has(category as ChangelogCategory)) {
55
+ errors.push(`Unknown changelog category for ${entry.path}: ${category}`);
56
+ continue;
57
+ }
58
+ const items = values.map(value => value.trim().replace(/\.$/, "")).filter(value => value.length > 0);
59
+ if (items.length > 0) {
60
+ cleaned[category] = Array.from(new Set(items));
61
+ }
62
+ }
63
+
64
+ let cleanedDeletions: Record<string, string[]> | undefined;
65
+ if (entry.deletions) {
66
+ cleanedDeletions = {};
67
+ for (const [category, values] of Object.entries(entry.deletions)) {
68
+ if (!allowedCategories.has(category as ChangelogCategory)) {
69
+ errors.push(`Unknown deletion category for ${entry.path}: ${category}`);
70
+ continue;
71
+ }
72
+ const items = values.map(value => value.trim()).filter(value => value.length > 0);
73
+ if (items.length > 0) {
74
+ cleanedDeletions[category] = Array.from(new Set(items));
75
+ }
76
+ }
77
+ if (Object.keys(cleanedDeletions).length === 0) {
78
+ cleanedDeletions = undefined;
79
+ }
80
+ }
81
+
82
+ if (Object.keys(cleaned).length === 0 && !cleanedDeletions) {
83
+ warnings.push(`No changelog entries provided for ${entry.path}.`);
84
+ }
85
+ return {
86
+ path: entry.path,
87
+ entries: cleaned,
88
+ deletions: cleanedDeletions,
89
+ };
90
+ });
91
+
92
+ for (const entry of normalized) {
93
+ if (targets.size > 0 && !targets.has(entry.path)) {
94
+ errors.push(`Changelog not expected: ${entry.path}`);
95
+ continue;
96
+ }
97
+ if (seen.has(entry.path)) {
98
+ errors.push(`Duplicate changelog entry for ${entry.path}`);
99
+ continue;
100
+ }
101
+ seen.add(entry.path);
102
+ }
103
+
104
+ if (targets.size > 0) {
105
+ for (const target of targets) {
106
+ if (!seen.has(target)) {
107
+ errors.push(`Missing changelog entries for ${target}`);
108
+ }
109
+ }
110
+ }
111
+
112
+ const response: ChangelogResponse = {
113
+ valid: errors.length === 0,
114
+ errors,
115
+ warnings,
116
+ };
117
+
118
+ if (response.valid) {
119
+ state.changelogProposal = { entries: normalized };
120
+ }
121
+
122
+ return {
123
+ content: [{ type: "text", text: JSON.stringify(response, null, 2) }],
124
+ details: response,
125
+ };
126
+ },
127
+ };
128
+ }
@@ -0,0 +1,154 @@
1
+ import { Type } from "@sinclair/typebox";
2
+ import type { CommitAgentState } from "../../../commit/agentic/state";
3
+ import {
4
+ capDetails,
5
+ MAX_DETAIL_ITEMS,
6
+ normalizeSummary,
7
+ SUMMARY_MAX_CHARS,
8
+ validateSummaryRules,
9
+ validateTypeConsistency,
10
+ } from "../../../commit/agentic/validation";
11
+ import { validateAnalysis } from "../../../commit/analysis/validation";
12
+ import type { ControlledGit } from "../../../commit/git";
13
+ import type { CommitType, ConventionalAnalysis, ConventionalDetail } from "../../../commit/types";
14
+ import type { CustomTool } from "../../../extensibility/custom-tools/types";
15
+
16
+ const commitTypeSchema = Type.Union([
17
+ Type.Literal("feat"),
18
+ Type.Literal("fix"),
19
+ Type.Literal("refactor"),
20
+ Type.Literal("perf"),
21
+ Type.Literal("docs"),
22
+ Type.Literal("test"),
23
+ Type.Literal("build"),
24
+ Type.Literal("ci"),
25
+ Type.Literal("chore"),
26
+ Type.Literal("style"),
27
+ Type.Literal("revert"),
28
+ ]);
29
+
30
+ const detailSchema = Type.Object({
31
+ text: Type.String(),
32
+ changelog_category: Type.Optional(
33
+ Type.Union([
34
+ Type.Literal("Added"),
35
+ Type.Literal("Changed"),
36
+ Type.Literal("Fixed"),
37
+ Type.Literal("Deprecated"),
38
+ Type.Literal("Removed"),
39
+ Type.Literal("Security"),
40
+ Type.Literal("Breaking Changes"),
41
+ ]),
42
+ ),
43
+ user_visible: Type.Optional(Type.Boolean()),
44
+ });
45
+
46
+ const proposeCommitSchema = Type.Object({
47
+ type: commitTypeSchema,
48
+ scope: Type.Union([Type.String(), Type.Null()]),
49
+ summary: Type.String(),
50
+ details: Type.Array(detailSchema),
51
+ issue_refs: Type.Array(Type.String()),
52
+ });
53
+
54
+ interface ProposalResponse {
55
+ valid: boolean;
56
+ errors: string[];
57
+ warnings: string[];
58
+ proposal?: {
59
+ type: CommitType;
60
+ scope: string | null;
61
+ summary: string;
62
+ details: ConventionalDetail[];
63
+ issue_refs: string[];
64
+ };
65
+ }
66
+
67
+ function normalizeDetails(
68
+ details: Array<{
69
+ text: string;
70
+ changelog_category?: ConventionalDetail["changelogCategory"];
71
+ user_visible?: boolean;
72
+ }>,
73
+ ): ConventionalDetail[] {
74
+ return details.map(detail => ({
75
+ text: detail.text.trim(),
76
+ changelogCategory: detail.user_visible ? detail.changelog_category : undefined,
77
+ userVisible: detail.user_visible ?? false,
78
+ }));
79
+ }
80
+
81
+ export function createProposeCommitTool(
82
+ git: ControlledGit,
83
+ state: CommitAgentState,
84
+ ): CustomTool<typeof proposeCommitSchema> {
85
+ return {
86
+ name: "propose_commit",
87
+ label: "Propose Commit",
88
+ description: "Submit the final conventional commit proposal.",
89
+ parameters: proposeCommitSchema,
90
+ async execute(_toolCallId, params) {
91
+ const scope = params.scope?.trim() || null;
92
+ const summary = normalizeSummary(params.summary, params.type, scope);
93
+ const details = normalizeDetails(params.details);
94
+ const { details: cappedDetails, warnings: detailWarnings } = capDetails(details);
95
+ const analysis: ConventionalAnalysis = {
96
+ type: params.type,
97
+ scope,
98
+ details: cappedDetails,
99
+ issueRefs: params.issue_refs ?? [],
100
+ };
101
+
102
+ const summaryValidation = validateSummaryRules(summary);
103
+ const analysisValidation = validateAnalysis(analysis);
104
+ const stagedFiles = state.overview?.files ?? (await git.getStagedFiles());
105
+ const diffText = state.diffText ?? (await git.getDiff(true));
106
+ const typeValidation = validateTypeConsistency(params.type, stagedFiles, {
107
+ diffText,
108
+ summary,
109
+ details: cappedDetails,
110
+ });
111
+
112
+ const errors = [...summaryValidation.errors, ...analysisValidation.errors, ...typeValidation.errors];
113
+ const warnings = [...summaryValidation.warnings, ...detailWarnings, ...typeValidation.warnings];
114
+
115
+ const response: ProposalResponse = {
116
+ valid: errors.length === 0,
117
+ errors,
118
+ warnings,
119
+ };
120
+
121
+ if (response.valid) {
122
+ response.proposal = {
123
+ type: analysis.type,
124
+ scope: analysis.scope,
125
+ summary,
126
+ details: analysis.details,
127
+ issue_refs: analysis.issueRefs,
128
+ };
129
+ state.proposal = {
130
+ analysis,
131
+ summary,
132
+ warnings,
133
+ };
134
+ }
135
+
136
+ const text = JSON.stringify(
137
+ {
138
+ ...response,
139
+ constraints: {
140
+ maxSummaryChars: SUMMARY_MAX_CHARS,
141
+ maxDetailItems: MAX_DETAIL_ITEMS,
142
+ },
143
+ },
144
+ null,
145
+ 2,
146
+ );
147
+
148
+ return {
149
+ content: [{ type: "text", text }],
150
+ details: response,
151
+ };
152
+ },
153
+ };
154
+ }