@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.
- package/CHANGELOG.md +125 -0
- package/docs/session.md +111 -46
- package/examples/custom-tools/hello/index.ts +1 -1
- package/examples/custom-tools/todo/index.ts +3 -4
- package/examples/extensions/api-demo.ts +0 -1
- package/examples/extensions/chalk-logger.ts +2 -3
- package/examples/extensions/hello.ts +0 -1
- package/examples/extensions/pirate.ts +0 -1
- package/examples/extensions/plan-mode.ts +15 -16
- package/examples/extensions/todo.ts +3 -4
- package/examples/extensions/tools.ts +1 -2
- package/examples/extensions/with-deps/index.ts +0 -1
- package/examples/hooks/auto-commit-on-exit.ts +1 -2
- package/examples/hooks/confirm-destructive.ts +0 -1
- package/examples/hooks/custom-compaction.ts +1 -2
- package/examples/hooks/dirty-repo-guard.ts +0 -1
- package/examples/hooks/file-trigger.ts +3 -4
- package/examples/hooks/git-checkpoint.ts +0 -1
- package/examples/hooks/handoff.ts +3 -4
- package/examples/hooks/permission-gate.ts +1 -2
- package/examples/hooks/protected-paths.ts +1 -2
- package/examples/hooks/qna.ts +2 -3
- package/examples/hooks/snake.ts +4 -5
- package/examples/hooks/status-line.ts +0 -1
- package/examples/sdk/01-minimal.ts +2 -3
- package/examples/sdk/02-custom-model.ts +2 -3
- package/examples/sdk/03-custom-prompt.ts +3 -4
- package/examples/sdk/04-skills.ts +2 -3
- package/examples/sdk/06-extensions.ts +1 -2
- package/examples/sdk/06-hooks.ts +6 -7
- package/examples/sdk/07-context-files.ts +0 -1
- package/examples/sdk/08-prompt-templates.ts +0 -1
- package/examples/sdk/08-slash-commands.ts +0 -1
- package/examples/sdk/09-api-keys-and-oauth.ts +0 -1
- package/examples/sdk/10-settings.ts +0 -1
- package/examples/sdk/11-sessions.ts +0 -1
- package/package.json +54 -23
- package/scripts/format-prompts.ts +0 -1
- package/src/capability/context-file.ts +3 -4
- package/src/capability/extension-module.ts +3 -4
- package/src/capability/extension.ts +3 -4
- package/src/capability/fs.ts +20 -21
- package/src/capability/hook.ts +3 -4
- package/src/capability/index.ts +15 -16
- package/src/capability/instruction.ts +3 -4
- package/src/capability/mcp.ts +3 -4
- package/src/capability/prompt.ts +3 -4
- package/src/capability/rule.ts +3 -4
- package/src/capability/settings.ts +2 -3
- package/src/capability/skill.ts +3 -4
- package/src/capability/slash-command.ts +3 -4
- package/src/capability/ssh.ts +3 -4
- package/src/capability/system-prompt.ts +3 -4
- package/src/capability/tool.ts +3 -4
- package/src/cli/args.ts +5 -6
- package/src/cli/config-cli.ts +6 -7
- package/src/cli/file-processor.ts +19 -17
- package/src/cli/jupyter-cli.ts +105 -0
- package/src/cli/list-models.ts +10 -11
- package/src/cli/plugin-cli.ts +20 -25
- package/src/cli/session-picker.ts +2 -3
- package/src/cli/setup-cli.ts +2 -3
- package/src/cli/stats-cli.ts +2 -3
- package/src/cli/update-cli.ts +25 -22
- package/src/commit/agentic/agent.ts +307 -0
- package/src/commit/agentic/fallback.ts +96 -0
- package/src/commit/agentic/index.ts +351 -0
- package/src/commit/agentic/prompts/analyze-file.md +22 -0
- package/src/commit/agentic/prompts/session-user.md +26 -0
- package/src/commit/agentic/prompts/split-confirm.md +1 -0
- package/src/commit/agentic/prompts/system.md +40 -0
- package/src/commit/agentic/state.ts +69 -0
- package/src/commit/agentic/tools/analyze-file.ts +131 -0
- package/src/commit/agentic/tools/git-file-diff.ts +194 -0
- package/src/commit/agentic/tools/git-hunk.ts +50 -0
- package/src/commit/agentic/tools/git-overview.ts +84 -0
- package/src/commit/agentic/tools/index.ts +56 -0
- package/src/commit/agentic/tools/propose-changelog.ts +128 -0
- package/src/commit/agentic/tools/propose-commit.ts +154 -0
- package/src/commit/agentic/tools/recent-commits.ts +81 -0
- package/src/commit/agentic/tools/split-commit.ts +280 -0
- package/src/commit/agentic/topo-sort.ts +44 -0
- package/src/commit/agentic/trivial.ts +51 -0
- package/src/commit/agentic/validation.ts +200 -0
- package/src/commit/analysis/conventional.ts +165 -0
- package/src/commit/analysis/index.ts +4 -0
- package/src/commit/analysis/scope.ts +242 -0
- package/src/commit/analysis/summary.ts +112 -0
- package/src/commit/analysis/validation.ts +66 -0
- package/src/commit/changelog/detect.ts +36 -0
- package/src/commit/changelog/generate.ts +110 -0
- package/src/commit/changelog/index.ts +233 -0
- package/src/commit/changelog/parse.ts +44 -0
- package/src/commit/cli.ts +93 -0
- package/src/commit/git/diff.ts +148 -0
- package/src/commit/git/errors.ts +11 -0
- package/src/commit/git/index.ts +212 -0
- package/src/commit/git/operations.ts +53 -0
- package/src/commit/index.ts +5 -0
- package/src/commit/map-reduce/index.ts +63 -0
- package/src/commit/map-reduce/map-phase.ts +178 -0
- package/src/commit/map-reduce/reduce-phase.ts +145 -0
- package/src/commit/map-reduce/utils.ts +9 -0
- package/src/commit/message.ts +11 -0
- package/src/commit/model-selection.ts +80 -0
- package/src/commit/pipeline.ts +240 -0
- package/src/commit/prompts/analysis-system.md +155 -0
- package/src/commit/prompts/analysis-user.md +41 -0
- package/src/commit/prompts/changelog-system.md +56 -0
- package/src/commit/prompts/changelog-user.md +19 -0
- package/src/commit/prompts/file-observer-system.md +26 -0
- package/src/commit/prompts/file-observer-user.md +9 -0
- package/src/commit/prompts/reduce-system.md +60 -0
- package/src/commit/prompts/reduce-user.md +17 -0
- package/src/commit/prompts/summary-retry.md +4 -0
- package/src/commit/prompts/summary-system.md +52 -0
- package/src/commit/prompts/summary-user.md +13 -0
- package/src/commit/prompts/types-description.md +2 -0
- package/src/commit/types.ts +109 -0
- package/src/commit/utils/exclusions.ts +42 -0
- package/src/config/file-lock.ts +121 -0
- package/src/config/keybindings.ts +6 -8
- package/src/config/model-registry.ts +65 -38
- package/src/config/model-resolver.ts +18 -19
- package/src/config/prompt-templates.ts +11 -11
- package/src/config/settings-manager.ts +141 -50
- package/src/config.ts +64 -66
- package/src/cursor.ts +11 -9
- package/src/discovery/agents-md.ts +11 -12
- package/src/discovery/builtin.ts +68 -73
- package/src/discovery/claude.ts +41 -42
- package/src/discovery/cline.ts +11 -12
- package/src/discovery/codex.ts +52 -53
- package/src/discovery/cursor.ts +9 -10
- package/src/discovery/gemini.ts +17 -22
- package/src/discovery/github.ts +13 -14
- package/src/discovery/helpers.ts +35 -34
- package/src/discovery/index.ts +22 -24
- package/src/discovery/mcp-json.ts +8 -9
- package/src/discovery/ssh.ts +8 -9
- package/src/discovery/vscode.ts +4 -5
- package/src/discovery/windsurf.ts +6 -7
- package/src/exa/company.ts +1 -2
- package/src/exa/index.ts +2 -3
- package/src/exa/linkedin.ts +1 -2
- package/src/exa/mcp-client.ts +14 -16
- package/src/exa/render.ts +10 -11
- package/src/exa/researcher.ts +1 -2
- package/src/exa/search.ts +1 -2
- package/src/exa/types.ts +0 -1
- package/src/exa/websets.ts +1 -2
- package/src/exec/bash-executor.ts +3 -4
- package/src/exec/exec.ts +0 -1
- package/src/export/custom-share.ts +5 -6
- package/src/export/html/index.ts +24 -21
- package/src/export/ttsr.ts +2 -3
- package/src/extensibility/custom-commands/bundled/review/index.ts +7 -8
- package/src/extensibility/custom-commands/loader.ts +18 -15
- package/src/extensibility/custom-commands/types.ts +2 -3
- package/src/extensibility/custom-tools/loader.ts +11 -12
- package/src/extensibility/custom-tools/types.ts +7 -8
- package/src/extensibility/custom-tools/wrapper.ts +2 -3
- package/src/extensibility/extensions/loader.ts +76 -54
- package/src/extensibility/extensions/runner.ts +11 -12
- package/src/extensibility/extensions/types.ts +20 -27
- package/src/extensibility/extensions/wrapper.ts +3 -4
- package/src/extensibility/hooks/index.ts +1 -1
- package/src/extensibility/hooks/loader.ts +9 -10
- package/src/extensibility/hooks/runner.ts +7 -8
- package/src/extensibility/hooks/tool-wrapper.ts +0 -1
- package/src/extensibility/hooks/types.ts +11 -18
- package/src/extensibility/plugins/doctor.ts +3 -3
- package/src/extensibility/plugins/installer.ts +27 -27
- package/src/extensibility/plugins/loader.ts +59 -56
- package/src/extensibility/plugins/manager.ts +211 -171
- package/src/extensibility/plugins/parser.ts +1 -1
- package/src/extensibility/plugins/paths.ts +8 -8
- package/src/extensibility/skills.ts +63 -60
- package/src/extensibility/slash-commands.ts +10 -10
- package/src/index.ts +54 -54
- package/src/internal-urls/agent-protocol.ts +21 -11
- package/src/internal-urls/artifact-protocol.ts +17 -13
- package/src/internal-urls/router.ts +1 -2
- package/src/internal-urls/rule-protocol.ts +3 -4
- package/src/internal-urls/skill-protocol.ts +3 -4
- package/src/ipy/executor.ts +109 -9
- package/src/ipy/gateway-coordinator.ts +79 -90
- package/src/ipy/kernel.ts +32 -30
- package/src/ipy/modules.ts +13 -13
- package/src/lsp/client.ts +21 -10
- package/src/lsp/clients/biome-client.ts +1 -2
- package/src/lsp/clients/index.ts +3 -3
- package/src/lsp/clients/lsp-linter-client.ts +4 -5
- package/src/lsp/config.ts +15 -15
- package/src/lsp/edits.ts +4 -5
- package/src/lsp/index.ts +43 -44
- package/src/lsp/lspmux.ts +8 -8
- package/src/lsp/render.ts +99 -61
- package/src/lsp/utils.ts +3 -3
- package/src/main.ts +71 -37
- package/src/mcp/client.ts +2 -3
- package/src/mcp/config.ts +5 -6
- package/src/mcp/json-rpc.ts +0 -1
- package/src/mcp/loader.ts +6 -7
- package/src/mcp/manager.ts +17 -18
- package/src/mcp/tool-bridge.ts +4 -9
- package/src/mcp/tool-cache.ts +2 -3
- package/src/mcp/transports/http.ts +2 -4
- package/src/mcp/transports/stdio.ts +1 -2
- package/src/migrations.ts +63 -52
- package/src/modes/components/armin.ts +4 -5
- package/src/modes/components/assistant-message.ts +33 -5
- package/src/modes/components/bash-execution.ts +7 -8
- package/src/modes/components/bordered-loader.ts +3 -3
- package/src/modes/components/branch-summary-message.ts +3 -3
- package/src/modes/components/compaction-summary-message.ts +3 -3
- package/src/modes/components/countdown-timer.ts +0 -1
- package/src/modes/components/custom-message.ts +5 -5
- package/src/modes/components/diff.ts +1 -1
- package/src/modes/components/dynamic-border.ts +2 -2
- package/src/modes/components/extensions/extension-dashboard.ts +6 -7
- package/src/modes/components/extensions/extension-list.ts +2 -3
- package/src/modes/components/extensions/inspector-panel.ts +3 -4
- package/src/modes/components/extensions/state-manager.ts +25 -26
- package/src/modes/components/extensions/types.ts +1 -2
- package/src/modes/components/footer.ts +47 -43
- package/src/modes/components/history-search.ts +2 -2
- package/src/modes/components/hook-editor.ts +3 -4
- package/src/modes/components/hook-input.ts +2 -3
- package/src/modes/components/hook-message.ts +5 -5
- package/src/modes/components/hook-selector.ts +2 -3
- package/src/modes/components/keybinding-hints.ts +2 -3
- package/src/modes/components/login-dialog.ts +2 -2
- package/src/modes/components/model-selector.ts +12 -12
- package/src/modes/components/oauth-selector.ts +2 -2
- package/src/modes/components/plugin-settings.ts +20 -20
- package/src/modes/components/python-execution.ts +7 -8
- package/src/modes/components/queue-mode-selector.ts +3 -3
- package/src/modes/components/read-tool-group.ts +2 -2
- package/src/modes/components/session-selector.ts +4 -4
- package/src/modes/components/settings-defs.ts +77 -69
- package/src/modes/components/settings-selector.ts +16 -16
- package/src/modes/components/show-images-selector.ts +2 -2
- package/src/modes/components/status-line/segments.ts +4 -4
- package/src/modes/components/status-line/separators.ts +1 -1
- package/src/modes/components/status-line/types.ts +2 -2
- package/src/modes/components/status-line-segment-editor.ts +7 -8
- package/src/modes/components/status-line.ts +12 -12
- package/src/modes/components/theme-selector.ts +8 -7
- package/src/modes/components/thinking-selector.ts +4 -4
- package/src/modes/components/todo-display.ts +2 -2
- package/src/modes/components/todo-reminder.ts +4 -4
- package/src/modes/components/tool-execution.ts +16 -19
- package/src/modes/components/tree-selector.ts +12 -12
- package/src/modes/components/ttsr-notification.ts +5 -5
- package/src/modes/components/user-message-selector.ts +1 -1
- package/src/modes/components/user-message.ts +1 -1
- package/src/modes/components/visual-truncate.ts +0 -1
- package/src/modes/components/welcome.ts +4 -4
- package/src/modes/controllers/command-controller.ts +46 -47
- package/src/modes/controllers/event-controller.ts +16 -20
- package/src/modes/controllers/extension-ui-controller.ts +40 -46
- package/src/modes/controllers/input-controller.ts +17 -18
- package/src/modes/controllers/selector-controller.ts +103 -91
- package/src/modes/index.ts +3 -3
- package/src/modes/interactive-mode.ts +31 -31
- package/src/modes/print-mode.ts +12 -13
- package/src/modes/rpc/rpc-client.ts +7 -8
- package/src/modes/rpc/rpc-mode.ts +24 -28
- package/src/modes/rpc/rpc-types.ts +3 -4
- package/src/modes/theme/mermaid-cache.ts +89 -0
- package/src/modes/theme/theme.ts +130 -53
- package/src/modes/types.ts +10 -10
- package/src/modes/utils/ui-helpers.ts +17 -17
- package/src/patch/applicator.ts +18 -19
- package/src/patch/diff.ts +1 -2
- package/src/patch/fuzzy.ts +1 -2
- package/src/patch/index.ts +11 -18
- package/src/patch/normalize.ts +4 -4
- package/src/patch/normative.ts +1 -2
- package/src/patch/parser.ts +8 -9
- package/src/patch/shared.ts +43 -16
- package/src/prompts/tools/task.md +2 -0
- package/src/sdk.ts +100 -65
- package/src/session/agent-session.ts +84 -85
- package/src/session/agent-storage.ts +43 -39
- package/src/session/artifacts.ts +32 -10
- package/src/session/auth-storage.ts +50 -39
- package/src/session/compaction/branch-summarization.ts +7 -10
- package/src/session/compaction/compaction.ts +8 -19
- package/src/session/compaction/utils.ts +6 -9
- package/src/session/history-storage.ts +10 -10
- package/src/session/messages.ts +4 -5
- package/src/session/session-manager.ts +76 -65
- package/src/session/session-storage.ts +57 -69
- package/src/session/storage-migration.ts +14 -56
- package/src/session/streaming-output.ts +2 -2
- package/src/ssh/connection-manager.ts +43 -50
- package/src/ssh/ssh-executor.ts +2 -2
- package/src/ssh/sshfs-mount.ts +11 -18
- package/src/system-prompt.ts +28 -35
- package/src/task/agents.ts +45 -30
- package/src/task/commands.ts +6 -7
- package/src/task/discovery.ts +39 -76
- package/src/task/executor.ts +14 -15
- package/src/task/index.ts +40 -34
- package/src/task/output-manager.ts +93 -0
- package/src/task/parallel.ts +0 -1
- package/src/task/render.ts +24 -30
- package/src/task/subprocess-tool-registry.ts +1 -2
- package/src/task/worker-protocol.ts +3 -3
- package/src/task/worker.ts +33 -39
- package/src/task/worktree.ts +19 -19
- package/src/tools/ask.ts +41 -20
- package/src/tools/bash-interceptor.ts +1 -5
- package/src/tools/bash.ts +91 -97
- package/src/tools/calculator.ts +49 -47
- package/src/tools/complete.ts +4 -5
- package/src/tools/context.ts +2 -2
- package/src/tools/fetch.ts +84 -124
- package/src/tools/find.ts +94 -98
- package/src/tools/gemini-image.ts +14 -14
- package/src/tools/grep.ts +100 -116
- package/src/tools/index.ts +80 -55
- package/src/tools/list-limit.ts +1 -1
- package/src/tools/ls.ts +44 -70
- package/src/tools/notebook.ts +51 -67
- package/src/tools/output-meta.ts +3 -4
- package/src/tools/output-utils.ts +2 -2
- package/src/tools/path-utils.ts +5 -5
- package/src/tools/python.ts +104 -217
- package/src/tools/read.ts +92 -33
- package/src/tools/render-utils.ts +8 -23
- package/src/tools/renderers.ts +6 -7
- package/src/tools/review.ts +8 -11
- package/src/tools/ssh.ts +69 -49
- package/src/tools/todo-write.ts +37 -25
- package/src/tools/tool-errors.ts +3 -3
- package/src/tools/tool-result.ts +3 -8
- package/src/tools/write.ts +99 -75
- package/src/tui/code-cell.ts +109 -0
- package/src/tui/file-list.ts +47 -0
- package/src/tui/index.ts +11 -0
- package/src/tui/output-block.ts +72 -0
- package/src/tui/status-line.ts +39 -0
- package/src/tui/tree-list.ts +55 -0
- package/src/tui/types.ts +16 -0
- package/src/tui/utils.ts +48 -0
- package/src/utils/changelog.ts +9 -10
- package/src/utils/clipboard.ts +11 -11
- package/src/utils/file-mentions.ts +4 -10
- package/src/utils/frontmatter.ts +6 -3
- package/src/utils/fuzzy.ts +2 -2
- package/src/utils/image-convert.ts +1 -1
- package/src/utils/image-resize.ts +1 -1
- package/src/utils/mime.ts +2 -2
- package/src/utils/shell-snapshot.ts +11 -13
- package/src/utils/shell.ts +4 -5
- package/src/utils/title-generator.ts +8 -9
- package/src/utils/tools-manager.ts +23 -23
- package/src/vendor/photon/index.js +1099 -1059
- package/src/vendor/photon/photon_rs_bg.wasm +0 -0
- package/src/web/scrapers/artifacthub.ts +1 -1
- package/src/web/scrapers/arxiv.ts +2 -2
- package/src/web/scrapers/bluesky.ts +2 -2
- package/src/web/scrapers/cheatsh.ts +1 -1
- package/src/web/scrapers/chocolatey.ts +2 -2
- package/src/web/scrapers/choosealicense.ts +5 -5
- package/src/web/scrapers/cisa-kev.ts +1 -1
- package/src/web/scrapers/crossref.ts +2 -2
- package/src/web/scrapers/devto.ts +3 -3
- package/src/web/scrapers/discogs.ts +3 -4
- package/src/web/scrapers/discourse.ts +1 -1
- package/src/web/scrapers/dockerhub.ts +1 -1
- package/src/web/scrapers/fdroid.ts +2 -2
- package/src/web/scrapers/firefox-addons.ts +3 -3
- package/src/web/scrapers/flathub.ts +1 -1
- package/src/web/scrapers/github.ts +3 -3
- package/src/web/scrapers/gitlab.ts +4 -4
- package/src/web/scrapers/hackernews.ts +2 -2
- package/src/web/scrapers/huggingface.ts +1 -1
- package/src/web/scrapers/iacr.ts +2 -2
- package/src/web/scrapers/index.ts +0 -1
- package/src/web/scrapers/jetbrains-marketplace.ts +1 -1
- package/src/web/scrapers/lemmy.ts +2 -2
- package/src/web/scrapers/maven.ts +2 -2
- package/src/web/scrapers/mdn.ts +2 -4
- package/src/web/scrapers/metacpan.ts +2 -2
- package/src/web/scrapers/musicbrainz.ts +1 -2
- package/src/web/scrapers/npm.ts +1 -1
- package/src/web/scrapers/nuget.ts +2 -2
- package/src/web/scrapers/nvd.ts +3 -3
- package/src/web/scrapers/ollama.ts +7 -9
- package/src/web/scrapers/opencorporates.ts +2 -2
- package/src/web/scrapers/openlibrary.ts +6 -6
- package/src/web/scrapers/orcid.ts +0 -1
- package/src/web/scrapers/osv.ts +2 -2
- package/src/web/scrapers/packagist.ts +1 -1
- package/src/web/scrapers/pubmed.ts +1 -2
- package/src/web/scrapers/rawg.ts +2 -2
- package/src/web/scrapers/readthedocs.ts +1 -2
- package/src/web/scrapers/repology.ts +2 -2
- package/src/web/scrapers/rfc.ts +1 -1
- package/src/web/scrapers/searchcode.ts +2 -2
- package/src/web/scrapers/semantic-scholar.ts +1 -1
- package/src/web/scrapers/snapcraft.ts +2 -2
- package/src/web/scrapers/sourcegraph.ts +1 -1
- package/src/web/scrapers/spdx.ts +3 -3
- package/src/web/scrapers/spotify.ts +0 -1
- package/src/web/scrapers/twitter.ts +1 -1
- package/src/web/scrapers/types.ts +1 -2
- package/src/web/scrapers/utils.ts +5 -5
- package/src/web/scrapers/wikidata.ts +3 -3
- package/src/web/scrapers/youtube.ts +9 -14
- package/src/web/search/auth.ts +5 -10
- package/src/web/search/index.ts +11 -21
- package/src/web/search/providers/anthropic.ts +3 -9
- package/src/web/search/providers/exa.ts +6 -10
- package/src/web/search/providers/perplexity.ts +5 -5
- package/src/web/search/render.ts +129 -175
- 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
|
+
}
|