goatcode-sh 0.0.1 → 0.1.3
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/.github/workflows/ci.yml +85 -0
- package/.github/workflows/release.yml +107 -0
- package/.opencode/plugins/goatcode.js +1 -0
- package/AGENTS.md +59 -0
- package/CONTRIBUTING.md +110 -0
- package/LICENSE +21 -0
- package/README.md +106 -12
- package/bun.lock +2081 -0
- package/bunfig.toml +3 -0
- package/dist/agents/advisor/config.d.ts +4 -0
- package/dist/agents/advisor/index.d.ts +2 -0
- package/dist/agents/advisor/plugin.d.ts +1 -0
- package/dist/agents/advisor/prompt-meta.d.ts +2 -0
- package/dist/agents/advisor/prompt.d.ts +1 -0
- package/dist/agents/agent-builder.d.ts +10 -0
- package/dist/agents/agent-registry.d.ts +8 -0
- package/dist/agents/builtin-agents.d.ts +2 -0
- package/dist/agents/deep-worker/config.d.ts +4 -0
- package/dist/agents/deep-worker/index.d.ts +2 -0
- package/dist/agents/deep-worker/plugin.d.ts +1 -0
- package/dist/agents/deep-worker/prompt-meta.d.ts +2 -0
- package/dist/agents/deep-worker/prompt.d.ts +1 -0
- package/dist/agents/explorer/config.d.ts +4 -0
- package/dist/agents/explorer/index.d.ts +2 -0
- package/dist/agents/explorer/plugin.d.ts +1 -0
- package/dist/agents/explorer/prompt-meta.d.ts +2 -0
- package/dist/agents/explorer/prompt.d.ts +1 -0
- package/dist/agents/fallback-chains.d.ts +2 -0
- package/dist/agents/index.d.ts +14 -0
- package/dist/agents/orchestrator/config.d.ts +4 -0
- package/dist/agents/orchestrator/index.d.ts +2 -0
- package/dist/agents/orchestrator/plugin.d.ts +1 -0
- package/dist/agents/orchestrator/prompt-meta.d.ts +2 -0
- package/dist/agents/orchestrator/prompt.d.ts +1 -0
- package/dist/agents/planner/config.d.ts +4 -0
- package/dist/agents/planner/index.d.ts +2 -0
- package/dist/agents/planner/plugin.d.ts +1 -0
- package/dist/agents/planner/prompt-meta.d.ts +2 -0
- package/dist/agents/planner/prompt.d.ts +1 -0
- package/dist/agents/prompt-meta.d.ts +11 -0
- package/dist/agents/prompt-registry.d.ts +4 -0
- package/dist/agents/researcher/config.d.ts +4 -0
- package/dist/agents/researcher/index.d.ts +2 -0
- package/dist/agents/researcher/plugin.d.ts +1 -0
- package/dist/agents/researcher/prompt-meta.d.ts +2 -0
- package/dist/agents/researcher/prompt.d.ts +1 -0
- package/dist/agents/tool-restrictions.d.ts +7 -0
- package/dist/agents/worker/config.d.ts +4 -0
- package/dist/agents/worker/index.d.ts +2 -0
- package/dist/agents/worker/plugin.d.ts +1 -0
- package/dist/agents/worker/prompt-meta.d.ts +2 -0
- package/dist/agents/worker/prompt.d.ts +1 -0
- package/dist/bootstrap.d.ts +3 -0
- package/dist/cli/cli.d.ts +3 -0
- package/dist/cli/commands/install.d.ts +12 -0
- package/dist/cli/commands/update.d.ts +1 -0
- package/dist/cli/config-generator.d.ts +26 -0
- package/dist/cli/index.d.ts +2 -0
- package/dist/config/defaults.d.ts +3 -0
- package/dist/config/define-config.d.ts +27 -0
- package/dist/config/index.d.ts +6 -0
- package/dist/config/loader.d.ts +3 -0
- package/dist/config/paths.d.ts +4 -0
- package/dist/config/schema.d.ts +273 -0
- package/dist/config/validator.d.ts +9 -0
- package/dist/features/auto-update/index.d.ts +3 -0
- package/dist/features/auto-update/plugin.d.ts +2 -0
- package/dist/features/auto-update/update-checker.d.ts +7 -0
- package/dist/features/background-agent/concurrency.d.ts +10 -0
- package/dist/features/background-agent/index.d.ts +6 -0
- package/dist/features/background-agent/manager.d.ts +18 -0
- package/dist/features/background-agent/poller.d.ts +17 -0
- package/dist/features/background-agent/singleton.d.ts +7 -0
- package/dist/features/background-agent/spawner.d.ts +6 -0
- package/dist/features/background-agent/types.d.ts +18 -0
- package/dist/features/builtin-features.d.ts +2 -0
- package/dist/features/categories/category-config.d.ts +9 -0
- package/dist/features/categories/category-resolver.d.ts +6 -0
- package/dist/features/categories/index.d.ts +3 -0
- package/dist/features/categories/prompt-appends.d.ts +10 -0
- package/dist/features/loops/file-store.d.ts +22 -0
- package/dist/features/loops/handler.d.ts +9 -0
- package/dist/features/loops/index.d.ts +5 -0
- package/dist/features/loops/memory-store.d.ts +11 -0
- package/dist/features/loops/plugin.d.ts +34 -0
- package/dist/features/loops/shared/event-utils.d.ts +7 -0
- package/dist/features/loops/state.d.ts +22 -0
- package/dist/features/prompt-builder/agent-table-builder.d.ts +6 -0
- package/dist/features/prompt-builder/category-section-builder.d.ts +2 -0
- package/dist/features/prompt-builder/dynamic-prompt-builder.d.ts +9 -0
- package/dist/features/prompt-builder/index.d.ts +7 -0
- package/dist/features/prompt-builder/skill-section-builder.d.ts +5 -0
- package/dist/features/session-state/index.d.ts +5 -0
- package/dist/features/session-state/session-cursor.d.ts +13 -0
- package/dist/features/session-state/session-store.d.ts +12 -0
- package/dist/features/session-state/session-tools-store.d.ts +4 -0
- package/dist/features/skills/builtin/git-master.d.ts +2 -0
- package/dist/features/skills/index.d.ts +10 -0
- package/dist/features/skills/skill-loader.d.ts +22 -0
- package/dist/features/skills/skill-merger.d.ts +2 -0
- package/dist/features/slash-commands/command-registry.d.ts +3 -0
- package/dist/features/slash-commands/commands/cancel-loop.d.ts +2 -0
- package/dist/features/slash-commands/commands/handoff.d.ts +2 -0
- package/dist/features/slash-commands/commands/init-deep.d.ts +2 -0
- package/dist/features/slash-commands/commands/loop.d.ts +2 -0
- package/dist/features/slash-commands/commands/start-work.d.ts +2 -0
- package/dist/features/slash-commands/commands/stop-continuation.d.ts +2 -0
- package/dist/features/slash-commands/index.d.ts +2 -0
- package/dist/features/slash-commands/types.d.ts +5 -0
- package/dist/hooks/anthropic-effort/handler.d.ts +5 -0
- package/dist/hooks/anthropic-effort/index.d.ts +3 -0
- package/dist/hooks/anthropic-effort/plugin.d.ts +1 -0
- package/dist/hooks/builtin-hooks.d.ts +2 -0
- package/dist/hooks/comment-checker/handler.d.ts +5 -0
- package/dist/hooks/comment-checker/index.d.ts +2 -0
- package/dist/hooks/comment-checker/plugin.d.ts +1 -0
- package/dist/hooks/compaction-context/handler.d.ts +2 -0
- package/dist/hooks/compaction-context/index.d.ts +2 -0
- package/dist/hooks/compaction-context/plugin.d.ts +1 -0
- package/dist/hooks/compaction-todo-preserver/handler.d.ts +4 -0
- package/dist/hooks/compaction-todo-preserver/index.d.ts +2 -0
- package/dist/hooks/compaction-todo-preserver/plugin.d.ts +1 -0
- package/dist/hooks/context-injector/handlers/agents.d.ts +1 -0
- package/dist/hooks/context-injector/handlers/readme.d.ts +1 -0
- package/dist/hooks/context-injector/handlers/rules.d.ts +1 -0
- package/dist/hooks/context-injector/index.d.ts +4 -0
- package/dist/hooks/context-injector/plugin.d.ts +1 -0
- package/dist/hooks/context-window-limit/handler.d.ts +6 -0
- package/dist/hooks/context-window-limit/index.d.ts +2 -0
- package/dist/hooks/context-window-limit/plugin.d.ts +1 -0
- package/dist/hooks/delegate-retry/handler.d.ts +16 -0
- package/dist/hooks/delegate-retry/index.d.ts +3 -0
- package/dist/hooks/delegate-retry/plugin.d.ts +1 -0
- package/dist/hooks/edit-error/handler.d.ts +6 -0
- package/dist/hooks/edit-error/index.d.ts +2 -0
- package/dist/hooks/edit-error/plugin.d.ts +1 -0
- package/dist/hooks/empty-response-detector/handler.d.ts +5 -0
- package/dist/hooks/empty-response-detector/index.d.ts +2 -0
- package/dist/hooks/empty-response-detector/plugin.d.ts +1 -0
- package/dist/hooks/error-diagnostics/handler.d.ts +6 -0
- package/dist/hooks/error-diagnostics/patterns.d.ts +3 -0
- package/dist/hooks/error-diagnostics/plugin.d.ts +1 -0
- package/dist/hooks/error-diagnostics/types.d.ts +14 -0
- package/dist/hooks/foreground-fallback/handler.d.ts +22 -0
- package/dist/hooks/foreground-fallback/index.d.ts +2 -0
- package/dist/hooks/foreground-fallback/plugin.d.ts +1 -0
- package/dist/hooks/hashline-diff-enhancer/handler.d.ts +28 -0
- package/dist/hooks/hashline-diff-enhancer/index.d.ts +2 -0
- package/dist/hooks/hashline-diff-enhancer/plugin.d.ts +1 -0
- package/dist/hooks/hashline-read-enhancer/handler.d.ts +4 -0
- package/dist/hooks/hashline-read-enhancer/index.d.ts +2 -0
- package/dist/hooks/hashline-read-enhancer/plugin.d.ts +1 -0
- package/dist/hooks/hook-composer.d.ts +4 -0
- package/dist/hooks/hook-ordering.d.ts +9 -0
- package/dist/hooks/hook-types.d.ts +3 -0
- package/dist/hooks/index.d.ts +6 -0
- package/dist/hooks/json-error/handler.d.ts +6 -0
- package/dist/hooks/json-error/index.d.ts +2 -0
- package/dist/hooks/json-error/plugin.d.ts +1 -0
- package/dist/hooks/keyword-detector/handler.d.ts +8 -0
- package/dist/hooks/keyword-detector/index.d.ts +3 -0
- package/dist/hooks/keyword-detector/plugin.d.ts +1 -0
- package/dist/hooks/model-fallback/handler.d.ts +21 -0
- package/dist/hooks/model-fallback/index.d.ts +2 -0
- package/dist/hooks/model-fallback/plugin.d.ts +1 -0
- package/dist/hooks/phase-reminder/handler.d.ts +5 -0
- package/dist/hooks/phase-reminder/index.d.ts +2 -0
- package/dist/hooks/phase-reminder/plugin.d.ts +1 -0
- package/dist/hooks/post-read-nudge/handler.d.ts +6 -0
- package/dist/hooks/post-read-nudge/index.d.ts +2 -0
- package/dist/hooks/post-read-nudge/plugin.d.ts +1 -0
- package/dist/hooks/preemptive-compaction/handler.d.ts +31 -0
- package/dist/hooks/preemptive-compaction/index.d.ts +2 -0
- package/dist/hooks/preemptive-compaction/plugin.d.ts +1 -0
- package/dist/hooks/runtime-fallback/handler.d.ts +21 -0
- package/dist/hooks/runtime-fallback/index.d.ts +2 -0
- package/dist/hooks/runtime-fallback/plugin.d.ts +1 -0
- package/dist/hooks/safe-hook-wrapper.d.ts +2 -0
- package/dist/hooks/session-recovery/handler.d.ts +6 -0
- package/dist/hooks/session-recovery/index.d.ts +2 -0
- package/dist/hooks/session-recovery/plugin.d.ts +1 -0
- package/dist/hooks/skill-discovery/plugin.d.ts +1 -0
- package/dist/hooks/stop-guard/handler.d.ts +4 -0
- package/dist/hooks/stop-guard/index.d.ts +2 -0
- package/dist/hooks/stop-guard/plugin.d.ts +1 -0
- package/dist/hooks/task-resume-info/handler.d.ts +4 -0
- package/dist/hooks/task-resume-info/index.d.ts +2 -0
- package/dist/hooks/task-resume-info/plugin.d.ts +1 -0
- package/dist/hooks/think-mode/handler.d.ts +4 -0
- package/dist/hooks/think-mode/index.d.ts +2 -0
- package/dist/hooks/think-mode/plugin.d.ts +1 -0
- package/dist/hooks/thinking-block-validator/handler.d.ts +4 -0
- package/dist/hooks/thinking-block-validator/index.d.ts +2 -0
- package/dist/hooks/thinking-block-validator/plugin.d.ts +1 -0
- package/dist/hooks/todo-enforcer/handler.d.ts +4 -0
- package/dist/hooks/todo-enforcer/index.d.ts +2 -0
- package/dist/hooks/todo-enforcer/plugin.d.ts +1 -0
- package/dist/hooks/todowrite-disabler/handler.d.ts +6 -0
- package/dist/hooks/todowrite-disabler/index.d.ts +2 -0
- package/dist/hooks/todowrite-disabler/plugin.d.ts +1 -0
- package/dist/hooks/tool-output-truncator/handler.d.ts +4 -0
- package/dist/hooks/tool-output-truncator/index.d.ts +2 -0
- package/dist/hooks/tool-output-truncator/plugin.d.ts +1 -0
- package/dist/hooks/write-file-guard/handler.d.ts +7 -0
- package/dist/hooks/write-file-guard/index.d.ts +2 -0
- package/dist/hooks/write-file-guard/plugin.d.ts +1 -0
- package/dist/index.d.ts +5 -0
- package/dist/index.js +8623 -0
- package/dist/plugin/compositor.d.ts +9 -0
- package/dist/plugin/index.d.ts +1 -0
- package/dist/plugin-api/define-plugin.d.ts +33 -0
- package/dist/plugin-api/index.d.ts +3 -0
- package/dist/plugin-api/types.d.ts +10 -0
- package/dist/registry/agent-aggregator.d.ts +3 -0
- package/dist/registry/contribution-aggregator.d.ts +10 -0
- package/dist/registry/dependency-resolver.d.ts +3 -0
- package/dist/registry/hook-aggregator.d.ts +2 -0
- package/dist/registry/index.d.ts +6 -0
- package/dist/registry/plugin-registry.d.ts +16 -0
- package/dist/registry/tool-aggregator.d.ts +3 -0
- package/dist/registry/types.d.ts +9 -0
- package/dist/runtime/index.d.ts +7 -0
- package/dist/shared/data-path.d.ts +4 -0
- package/dist/shared/deep-merge.d.ts +13 -0
- package/dist/shared/fallback-chain.d.ts +1 -0
- package/dist/shared/index.d.ts +19 -0
- package/dist/shared/logger.d.ts +4 -0
- package/dist/shared/model-availability.d.ts +1 -0
- package/dist/shared/model-normalization.d.ts +6 -0
- package/dist/shared/model-prefix-map.d.ts +11 -0
- package/dist/shared/model-resolution-pipeline.d.ts +14 -0
- package/dist/shared/models-dev.d.ts +29 -0
- package/dist/shared/provider-discovery.d.ts +28 -0
- package/dist/shared/provider-registry.d.ts +16 -0
- package/dist/shared/safe-create-hook.d.ts +3 -0
- package/dist/shared/snake-case.d.ts +1 -0
- package/dist/shared/truncate-description.d.ts +1 -0
- package/dist/test-utils/index.d.ts +6 -0
- package/dist/test-utils/mock-agent-config.d.ts +5 -0
- package/dist/test-utils/mock-hook-inputs.d.ts +20 -0
- package/dist/test-utils/mock-hook-outputs.d.ts +20 -0
- package/dist/test-utils/mock-plugin-context.d.ts +6 -0
- package/dist/test-utils/mock-sdk-client.d.ts +41 -0
- package/dist/test-utils/mock-tool-context.d.ts +6 -0
- package/dist/tools/ast-grep/index.d.ts +4 -0
- package/dist/tools/ast-grep/replace/handler.d.ts +7 -0
- package/dist/tools/ast-grep/replace/plugin.d.ts +1 -0
- package/dist/tools/ast-grep/replace/types.d.ts +37 -0
- package/dist/tools/ast-grep/search/handler.d.ts +14 -0
- package/dist/tools/ast-grep/search/plugin.d.ts +1 -0
- package/dist/tools/ast-grep/search/types.d.ts +37 -0
- package/dist/tools/background-task/cancel/handler.d.ts +4 -0
- package/dist/tools/background-task/cancel/plugin.d.ts +2 -0
- package/dist/tools/background-task/cancel/types.d.ts +4 -0
- package/dist/tools/background-task/index.d.ts +2 -0
- package/dist/tools/background-task/output/handler.d.ts +3 -0
- package/dist/tools/background-task/output/plugin.d.ts +2 -0
- package/dist/tools/background-task/output/types.d.ts +11 -0
- package/dist/tools/bridge.d.ts +7 -0
- package/dist/tools/builtin-tools.d.ts +2 -0
- package/dist/tools/delegate-task/category-resolver.d.ts +4 -0
- package/dist/tools/delegate-task/constants.d.ts +3 -0
- package/dist/tools/delegate-task/executor.d.ts +10 -0
- package/dist/tools/delegate-task/handler.d.ts +4 -0
- package/dist/tools/delegate-task/index.d.ts +7 -0
- package/dist/tools/delegate-task/plugin.d.ts +4 -0
- package/dist/tools/delegate-task/types.d.ts +17 -0
- package/dist/tools/glob/handler.d.ts +7 -0
- package/dist/tools/glob/index.d.ts +2 -0
- package/dist/tools/glob/plugin.d.ts +1 -0
- package/dist/tools/glob/types.d.ts +7 -0
- package/dist/tools/grep/handler.d.ts +13 -0
- package/dist/tools/grep/index.d.ts +2 -0
- package/dist/tools/grep/plugin.d.ts +1 -0
- package/dist/tools/grep/types.d.ts +15 -0
- package/dist/tools/hashline-edit/constants.d.ts +4 -0
- package/dist/tools/hashline-edit/edit-operations.d.ts +4 -0
- package/dist/tools/hashline-edit/handler.d.ts +5 -0
- package/dist/tools/hashline-edit/hash-computation.d.ts +2 -0
- package/dist/tools/hashline-edit/index.d.ts +5 -0
- package/dist/tools/hashline-edit/plugin.d.ts +3 -0
- package/dist/tools/hashline-edit/types.d.ts +21 -0
- package/dist/tools/index.d.ts +4 -0
- package/dist/tools/look-at/handler.d.ts +6 -0
- package/dist/tools/look-at/index.d.ts +3 -0
- package/dist/tools/look-at/plugin.d.ts +1 -0
- package/dist/tools/look-at/types.d.ts +9 -0
- package/dist/tools/lsp/client.d.ts +7 -0
- package/dist/tools/lsp/diagnostics/handler.d.ts +2 -0
- package/dist/tools/lsp/diagnostics/plugin.d.ts +1 -0
- package/dist/tools/lsp/diagnostics/types.d.ts +13 -0
- package/dist/tools/lsp/find-references/handler.d.ts +2 -0
- package/dist/tools/lsp/find-references/plugin.d.ts +1 -0
- package/dist/tools/lsp/find-references/types.d.ts +8 -0
- package/dist/tools/lsp/goto-definition/handler.d.ts +2 -0
- package/dist/tools/lsp/goto-definition/plugin.d.ts +1 -0
- package/dist/tools/lsp/goto-definition/types.d.ts +7 -0
- package/dist/tools/lsp/index.d.ts +6 -0
- package/dist/tools/lsp/prepare-rename/handler.d.ts +2 -0
- package/dist/tools/lsp/prepare-rename/plugin.d.ts +1 -0
- package/dist/tools/lsp/prepare-rename/types.d.ts +7 -0
- package/dist/tools/lsp/rename/handler.d.ts +2 -0
- package/dist/tools/lsp/rename/plugin.d.ts +1 -0
- package/dist/tools/lsp/rename/types.d.ts +8 -0
- package/dist/tools/lsp/symbols/handler.d.ts +2 -0
- package/dist/tools/lsp/symbols/plugin.d.ts +1 -0
- package/dist/tools/lsp/symbols/types.d.ts +11 -0
- package/dist/tools/session-manager/client-context.d.ts +3 -0
- package/dist/tools/session-manager/index.d.ts +5 -0
- package/dist/tools/session-manager/info/handler.d.ts +3 -0
- package/dist/tools/session-manager/info/plugin.d.ts +1 -0
- package/dist/tools/session-manager/info/types.d.ts +3 -0
- package/dist/tools/session-manager/list/handler.d.ts +3 -0
- package/dist/tools/session-manager/list/plugin.d.ts +1 -0
- package/dist/tools/session-manager/list/types.d.ts +6 -0
- package/dist/tools/session-manager/read/handler.d.ts +3 -0
- package/dist/tools/session-manager/read/plugin.d.ts +1 -0
- package/dist/tools/session-manager/read/types.d.ts +6 -0
- package/dist/tools/session-manager/search/handler.d.ts +3 -0
- package/dist/tools/session-manager/search/plugin.d.ts +1 -0
- package/dist/tools/session-manager/search/types.d.ts +6 -0
- package/dist/tools/session-manager/session-formatter.d.ts +29 -0
- package/dist/tools/session-manager/types.d.ts +36 -0
- package/dist/tools/shared/constants.d.ts +3 -0
- package/dist/tools/skill/handler.d.ts +4 -0
- package/dist/tools/skill/index.d.ts +3 -0
- package/dist/tools/skill/plugin.d.ts +1 -0
- package/dist/tools/skill/types.d.ts +12 -0
- package/dist/tools/skill-mcp/handler.d.ts +5 -0
- package/dist/tools/skill-mcp/index.d.ts +3 -0
- package/dist/tools/skill-mcp/plugin.d.ts +1 -0
- package/dist/tools/skill-mcp/types.d.ts +13 -0
- package/dist/tools/task/create/handler.d.ts +2 -0
- package/dist/tools/task/create/plugin.d.ts +1 -0
- package/dist/tools/task/format-task.d.ts +2 -0
- package/dist/tools/task/get/handler.d.ts +2 -0
- package/dist/tools/task/get/plugin.d.ts +1 -0
- package/dist/tools/task/index.d.ts +4 -0
- package/dist/tools/task/list/handler.d.ts +2 -0
- package/dist/tools/task/list/plugin.d.ts +1 -0
- package/dist/tools/task/storage.d.ts +7 -0
- package/dist/tools/task/types.d.ts +84 -0
- package/dist/tools/task/update/handler.d.ts +2 -0
- package/dist/tools/task/update/plugin.d.ts +1 -0
- package/dist/tools/tool-builder.d.ts +7 -0
- package/dist/tools/tool-registry-adapter.d.ts +3 -0
- package/dist/types/agent.d.ts +40 -0
- package/dist/types/category.d.ts +22 -0
- package/dist/types/config.d.ts +23 -0
- package/dist/types/hook.d.ts +18 -0
- package/dist/types/index.d.ts +12 -0
- package/dist/types/plugin.d.ts +41 -0
- package/dist/types/tool.d.ts +7 -0
- package/eval/README.md +160 -0
- package/eval/ablation-config.yaml +43 -0
- package/eval/assertions/ablation-scorer.ts +81 -0
- package/eval/assertions/hook-impact.ts +152 -0
- package/eval/assertions/task-completion.ts +65 -0
- package/eval/assertions/tool-accuracy.ts +56 -0
- package/eval/promptfooconfig.yaml +42 -0
- package/eval/providers/opencode-baseline.ts +63 -0
- package/eval/providers/opencode-client.ts +112 -0
- package/eval/providers/opencode-provider.ts +66 -0
- package/eval/spike/config.yaml +13 -0
- package/eval/spike/provider.ts +15 -0
- package/npm-reserve/README.md +19 -0
- package/npm-reserve/package.json +24 -0
- package/opencode.json +6 -0
- package/package.json +47 -8
- package/src/agents/advisor/config.ts +6 -0
- package/src/agents/advisor/index.ts +2 -0
- package/src/agents/advisor/plugin.test.ts +48 -0
- package/src/agents/advisor/plugin.ts +17 -0
- package/src/agents/advisor/prompt-meta.ts +14 -0
- package/src/agents/advisor/prompt.ts +93 -0
- package/src/agents/agent-builder.test.ts +66 -0
- package/src/agents/agent-builder.ts +97 -0
- package/src/agents/agent-plugins.test.ts +98 -0
- package/src/agents/agent-registry.ts +25 -0
- package/src/agents/builtin-agents.ts +18 -0
- package/src/agents/deep-worker/config.ts +6 -0
- package/src/agents/deep-worker/index.ts +2 -0
- package/src/agents/deep-worker/plugin.test.ts +31 -0
- package/src/agents/deep-worker/plugin.ts +16 -0
- package/src/agents/deep-worker/prompt-meta.ts +14 -0
- package/src/agents/deep-worker/prompt.ts +121 -0
- package/src/agents/disabled/analyst/config.ts +6 -0
- package/src/agents/disabled/analyst/index.ts +2 -0
- package/src/agents/disabled/analyst/plugin.ts +16 -0
- package/src/agents/disabled/analyst/prompt.ts +1 -0
- package/src/agents/disabled/executor/config.ts +9 -0
- package/src/agents/disabled/executor/index.ts +2 -0
- package/src/agents/disabled/executor/plugin.ts +16 -0
- package/src/agents/disabled/executor/prompt.ts +1 -0
- package/src/agents/disabled/inspector/config.ts +6 -0
- package/src/agents/disabled/inspector/index.ts +2 -0
- package/src/agents/disabled/inspector/plugin.ts +18 -0
- package/src/agents/disabled/inspector/prompt.ts +1 -0
- package/src/agents/disabled/reviewer/config.ts +6 -0
- package/src/agents/disabled/reviewer/index.ts +2 -0
- package/src/agents/disabled/reviewer/plugin.ts +18 -0
- package/src/agents/disabled/reviewer/prompt.ts +1 -0
- package/src/agents/explorer/config.ts +6 -0
- package/src/agents/explorer/index.ts +2 -0
- package/src/agents/explorer/plugin.test.ts +36 -0
- package/src/agents/explorer/plugin.ts +15 -0
- package/src/agents/explorer/prompt-meta.ts +14 -0
- package/src/agents/explorer/prompt.ts +96 -0
- package/src/agents/fallback-chains.ts +13 -0
- package/src/agents/index.ts +18 -0
- package/src/agents/model-resolution.test.ts +79 -0
- package/src/agents/orchestrator/config.ts +10 -0
- package/src/agents/orchestrator/index.ts +2 -0
- package/src/agents/orchestrator/plugin.test.ts +31 -0
- package/src/agents/orchestrator/plugin.ts +16 -0
- package/src/agents/orchestrator/prompt-meta.ts +14 -0
- package/src/agents/orchestrator/prompt.ts +166 -0
- package/src/agents/planner/config.ts +6 -0
- package/src/agents/planner/index.ts +2 -0
- package/src/agents/planner/plugin.test.ts +31 -0
- package/src/agents/planner/plugin.ts +16 -0
- package/src/agents/planner/prompt-meta.ts +14 -0
- package/src/agents/planner/prompt.ts +138 -0
- package/src/agents/prompt-meta.ts +12 -0
- package/src/agents/prompt-registry.test.ts +98 -0
- package/src/agents/prompt-registry.ts +22 -0
- package/src/agents/researcher/config.ts +6 -0
- package/src/agents/researcher/index.ts +2 -0
- package/src/agents/researcher/plugin.test.ts +31 -0
- package/src/agents/researcher/plugin.ts +16 -0
- package/src/agents/researcher/prompt-meta.ts +14 -0
- package/src/agents/researcher/prompt.ts +116 -0
- package/src/agents/tool-restrictions.ts +87 -0
- package/src/agents/worker/config.ts +6 -0
- package/src/agents/worker/index.ts +2 -0
- package/src/agents/worker/plugin.test.ts +31 -0
- package/src/agents/worker/plugin.ts +15 -0
- package/src/agents/worker/prompt-meta.ts +14 -0
- package/src/agents/worker/prompt.ts +83 -0
- package/src/bootstrap.integration.test.ts +168 -0
- package/src/bootstrap.ts +171 -0
- package/src/cli/cli.ts +42 -0
- package/src/cli/commands/install.test.ts +40 -0
- package/src/cli/commands/install.ts +125 -0
- package/src/cli/commands/update.test.ts +84 -0
- package/src/cli/commands/update.ts +45 -0
- package/src/cli/config-generator.test.ts +178 -0
- package/src/cli/config-generator.ts +119 -0
- package/src/cli/index.test.ts +34 -0
- package/src/cli/index.ts +4 -0
- package/src/config/defaults.ts +24 -0
- package/src/config/define-config.ts +38 -0
- package/src/config/index.ts +6 -0
- package/src/config/loader.test.ts +218 -0
- package/src/config/loader.ts +89 -0
- package/src/config/paths.ts +30 -0
- package/src/config/schema.test.ts +69 -0
- package/src/config/schema.ts +57 -0
- package/src/config/validator.ts +24 -0
- package/src/features/auto-update/auto-update.test.ts +105 -0
- package/src/features/auto-update/index.ts +4 -0
- package/src/features/auto-update/plugin.ts +45 -0
- package/src/features/auto-update/update-checker.ts +66 -0
- package/src/features/background-agent/concurrency.test.ts +65 -0
- package/src/features/background-agent/concurrency.ts +44 -0
- package/src/features/background-agent/index.ts +12 -0
- package/src/features/background-agent/manager.ts +214 -0
- package/src/features/background-agent/poller.test.ts +33 -0
- package/src/features/background-agent/poller.ts +75 -0
- package/src/features/background-agent/singleton.ts +26 -0
- package/src/features/background-agent/spawner.ts +51 -0
- package/src/features/background-agent/types.ts +20 -0
- package/src/features/builtin-features.ts +5 -0
- package/src/features/categories/categories.test.ts +68 -0
- package/src/features/categories/category-config.ts +70 -0
- package/src/features/categories/category-resolver.ts +36 -0
- package/src/features/categories/index.ts +8 -0
- package/src/features/categories/prompt-appends.ts +38 -0
- package/src/features/loops/file-store.ts +151 -0
- package/src/features/loops/handler.ts +89 -0
- package/src/features/loops/index.ts +28 -0
- package/src/features/loops/loops.test.ts +175 -0
- package/src/features/loops/memory-store.ts +53 -0
- package/src/features/loops/plugin.ts +107 -0
- package/src/features/loops/shared/event-utils.ts +50 -0
- package/src/features/loops/state.ts +44 -0
- package/src/features/prompt-builder/agent-table-builder.ts +23 -0
- package/src/features/prompt-builder/category-section-builder.ts +21 -0
- package/src/features/prompt-builder/dynamic-prompt-builder.ts +42 -0
- package/src/features/prompt-builder/index.ts +7 -0
- package/src/features/prompt-builder/prompt-builder.test.ts +244 -0
- package/src/features/prompt-builder/skill-section-builder.ts +25 -0
- package/src/features/session-state/index.ts +17 -0
- package/src/features/session-state/session-cursor.test.ts +137 -0
- package/src/features/session-state/session-cursor.ts +80 -0
- package/src/features/session-state/session-store.test.ts +82 -0
- package/src/features/session-state/session-store.ts +37 -0
- package/src/features/session-state/session-tools-store.ts +18 -0
- package/src/features/skills/builtin/git-master.ts +109 -0
- package/src/features/skills/index.ts +97 -0
- package/src/features/skills/skill-loader.ts +133 -0
- package/src/features/skills/skill-merger.ts +15 -0
- package/src/features/skills/skills.test.ts +120 -0
- package/src/features/slash-commands/command-registry.ts +36 -0
- package/src/features/slash-commands/commands/cancel-loop.ts +17 -0
- package/src/features/slash-commands/commands/handoff.ts +59 -0
- package/src/features/slash-commands/commands/init-deep.ts +40 -0
- package/src/features/slash-commands/commands/loop.ts +39 -0
- package/src/features/slash-commands/commands/start-work.ts +39 -0
- package/src/features/slash-commands/commands/stop-continuation.ts +21 -0
- package/src/features/slash-commands/index.ts +2 -0
- package/src/features/slash-commands/slash-commands.test.ts +68 -0
- package/src/features/slash-commands/types.ts +5 -0
- package/src/hooks/anthropic-effort/handler.test.ts +156 -0
- package/src/hooks/anthropic-effort/handler.ts +64 -0
- package/src/hooks/anthropic-effort/index.ts +3 -0
- package/src/hooks/anthropic-effort/plugin.ts +17 -0
- package/src/hooks/builtin-hooks.ts +64 -0
- package/src/hooks/comment-checker/handler.test.ts +65 -0
- package/src/hooks/comment-checker/handler.ts +60 -0
- package/src/hooks/comment-checker/index.ts +2 -0
- package/src/hooks/comment-checker/plugin.ts +15 -0
- package/src/hooks/compaction-context/handler.test.ts +160 -0
- package/src/hooks/compaction-context/handler.ts +179 -0
- package/src/hooks/compaction-context/index.ts +5 -0
- package/src/hooks/compaction-context/plugin.ts +40 -0
- package/src/hooks/compaction-todo-preserver/handler.test.ts +155 -0
- package/src/hooks/compaction-todo-preserver/handler.ts +129 -0
- package/src/hooks/compaction-todo-preserver/index.ts +2 -0
- package/src/hooks/compaction-todo-preserver/plugin.ts +18 -0
- package/src/hooks/context-injection.test.ts +124 -0
- package/src/hooks/context-injector/handlers/agents.test.ts +140 -0
- package/src/hooks/context-injector/handlers/agents.ts +101 -0
- package/src/hooks/context-injector/handlers/readme.ts +55 -0
- package/src/hooks/context-injector/handlers/rules.ts +62 -0
- package/src/hooks/context-injector/index.ts +4 -0
- package/src/hooks/context-injector/plugin.ts +56 -0
- package/src/hooks/context-window-limit/handler.test.ts +103 -0
- package/src/hooks/context-window-limit/handler.ts +128 -0
- package/src/hooks/context-window-limit/index.ts +6 -0
- package/src/hooks/context-window-limit/plugin.ts +15 -0
- package/src/hooks/continuation.test.ts +103 -0
- package/src/hooks/delegate-retry/handler.test.ts +212 -0
- package/src/hooks/delegate-retry/handler.ts +137 -0
- package/src/hooks/delegate-retry/index.ts +8 -0
- package/src/hooks/delegate-retry/plugin.ts +15 -0
- package/src/hooks/edit-error/handler.test.ts +82 -0
- package/src/hooks/edit-error/handler.ts +50 -0
- package/src/hooks/edit-error/index.ts +6 -0
- package/src/hooks/edit-error/plugin.ts +15 -0
- package/src/hooks/empty-response-detector/handler.test.ts +133 -0
- package/src/hooks/empty-response-detector/handler.ts +62 -0
- package/src/hooks/empty-response-detector/index.ts +2 -0
- package/src/hooks/empty-response-detector/plugin.ts +18 -0
- package/src/hooks/error-diagnostics/error-diagnostics.test.ts +116 -0
- package/src/hooks/error-diagnostics/handler.test.ts +147 -0
- package/src/hooks/error-diagnostics/handler.ts +135 -0
- package/src/hooks/error-diagnostics/patterns.ts +93 -0
- package/src/hooks/error-diagnostics/plugin.ts +11 -0
- package/src/hooks/error-diagnostics/types.ts +26 -0
- package/src/hooks/error-recovery.test.ts +85 -0
- package/src/hooks/foreground-fallback/handler.test.ts +229 -0
- package/src/hooks/foreground-fallback/handler.ts +294 -0
- package/src/hooks/foreground-fallback/index.ts +2 -0
- package/src/hooks/foreground-fallback/plugin.ts +18 -0
- package/src/hooks/hashline-diff-enhancer/handler.test.ts +166 -0
- package/src/hooks/hashline-diff-enhancer/handler.ts +186 -0
- package/src/hooks/hashline-diff-enhancer/index.ts +6 -0
- package/src/hooks/hashline-diff-enhancer/plugin.ts +24 -0
- package/src/hooks/hashline-read-enhancer/handler.test.ts +121 -0
- package/src/hooks/hashline-read-enhancer/handler.ts +165 -0
- package/src/hooks/hashline-read-enhancer/index.ts +2 -0
- package/src/hooks/hashline-read-enhancer/plugin.ts +18 -0
- package/src/hooks/hook-composer.test.ts +52 -0
- package/src/hooks/hook-composer.ts +17 -0
- package/src/hooks/hook-composition.integration.test.ts +274 -0
- package/src/hooks/hook-ordering.ts +41 -0
- package/src/hooks/hook-types.ts +22 -0
- package/src/hooks/index.ts +6 -0
- package/src/hooks/json-error/handler.test.ts +95 -0
- package/src/hooks/json-error/handler.ts +82 -0
- package/src/hooks/json-error/index.ts +6 -0
- package/src/hooks/json-error/plugin.ts +15 -0
- package/src/hooks/keyword-detector/handler.test.ts +113 -0
- package/src/hooks/keyword-detector/handler.ts +73 -0
- package/src/hooks/keyword-detector/index.ts +8 -0
- package/src/hooks/keyword-detector/plugin.ts +24 -0
- package/src/hooks/model-fallback/handler.test.ts +163 -0
- package/src/hooks/model-fallback/handler.ts +178 -0
- package/src/hooks/model-fallback/index.ts +2 -0
- package/src/hooks/model-fallback/plugin.ts +11 -0
- package/src/hooks/model-management.test.ts +121 -0
- package/src/hooks/phase-reminder/handler.test.ts +105 -0
- package/src/hooks/phase-reminder/handler.ts +54 -0
- package/src/hooks/phase-reminder/index.ts +2 -0
- package/src/hooks/phase-reminder/plugin.ts +18 -0
- package/src/hooks/post-read-nudge/handler.test.ts +159 -0
- package/src/hooks/post-read-nudge/handler.ts +64 -0
- package/src/hooks/post-read-nudge/index.ts +6 -0
- package/src/hooks/post-read-nudge/plugin.ts +18 -0
- package/src/hooks/preemptive-compaction/handler.test.ts +130 -0
- package/src/hooks/preemptive-compaction/handler.ts +84 -0
- package/src/hooks/preemptive-compaction/index.ts +2 -0
- package/src/hooks/preemptive-compaction/plugin.ts +15 -0
- package/src/hooks/productivity.test.ts +332 -0
- package/src/hooks/quality.test.ts +330 -0
- package/src/hooks/runtime-fallback/handler.test.ts +142 -0
- package/src/hooks/runtime-fallback/handler.ts +171 -0
- package/src/hooks/runtime-fallback/index.ts +2 -0
- package/src/hooks/runtime-fallback/plugin.ts +13 -0
- package/src/hooks/safe-hook-wrapper.test.ts +35 -0
- package/src/hooks/safe-hook-wrapper.ts +12 -0
- package/src/hooks/session-recovery/handler.test.ts +88 -0
- package/src/hooks/session-recovery/handler.ts +87 -0
- package/src/hooks/session-recovery/index.ts +6 -0
- package/src/hooks/session-recovery/plugin.ts +15 -0
- package/src/hooks/skill-discovery/plugin.ts +45 -0
- package/src/hooks/stop-guard/handler.test.ts +147 -0
- package/src/hooks/stop-guard/handler.ts +127 -0
- package/src/hooks/stop-guard/index.ts +2 -0
- package/src/hooks/stop-guard/plugin.ts +15 -0
- package/src/hooks/task-hooks.test.ts +324 -0
- package/src/hooks/task-resume-info/handler.test.ts +180 -0
- package/src/hooks/task-resume-info/handler.ts +61 -0
- package/src/hooks/task-resume-info/index.ts +2 -0
- package/src/hooks/task-resume-info/plugin.ts +15 -0
- package/src/hooks/think-mode/handler.test.ts +139 -0
- package/src/hooks/think-mode/handler.ts +50 -0
- package/src/hooks/think-mode/index.ts +2 -0
- package/src/hooks/think-mode/plugin.ts +15 -0
- package/src/hooks/thinking-block-validator/handler.test.ts +79 -0
- package/src/hooks/thinking-block-validator/handler.ts +93 -0
- package/src/hooks/thinking-block-validator/index.ts +2 -0
- package/src/hooks/thinking-block-validator/plugin.ts +18 -0
- package/src/hooks/todo-enforcer/handler.test.ts +153 -0
- package/src/hooks/todo-enforcer/handler.ts +100 -0
- package/src/hooks/todo-enforcer/index.ts +2 -0
- package/src/hooks/todo-enforcer/plugin.ts +15 -0
- package/src/hooks/todowrite-disabler/handler.test.ts +119 -0
- package/src/hooks/todowrite-disabler/handler.ts +50 -0
- package/src/hooks/todowrite-disabler/index.ts +6 -0
- package/src/hooks/todowrite-disabler/plugin.ts +46 -0
- package/src/hooks/tool-output-truncator/handler.test.ts +113 -0
- package/src/hooks/tool-output-truncator/handler.ts +83 -0
- package/src/hooks/tool-output-truncator/index.ts +2 -0
- package/src/hooks/tool-output-truncator/plugin.ts +18 -0
- package/src/hooks/tool-output.test.ts +238 -0
- package/src/hooks/workflow-reminders.test.ts +187 -0
- package/src/hooks/write-file-guard/handler.test.ts +107 -0
- package/src/hooks/write-file-guard/handler.ts +166 -0
- package/src/hooks/write-file-guard/index.ts +2 -0
- package/src/hooks/write-file-guard/plugin.ts +23 -0
- package/src/index.ts +8 -0
- package/src/plugin/compositor.ts +99 -0
- package/src/plugin/index.ts +1 -0
- package/src/plugin-api/define-plugin.test.ts +66 -0
- package/src/plugin-api/define-plugin.ts +36 -0
- package/src/plugin-api/index.ts +26 -0
- package/src/plugin-api/types.ts +28 -0
- package/src/registry/agent-aggregator.ts +13 -0
- package/src/registry/contribution-aggregator.test.ts +62 -0
- package/src/registry/contribution-aggregator.ts +114 -0
- package/src/registry/contribution-conflicts.integration.test.ts +186 -0
- package/src/registry/dependency-resolver.test.ts +35 -0
- package/src/registry/dependency-resolver.ts +64 -0
- package/src/registry/hook-aggregator.test.ts +78 -0
- package/src/registry/hook-aggregator.ts +63 -0
- package/src/registry/index.ts +6 -0
- package/src/registry/plugin-lifecycle.integration.test.ts +94 -0
- package/src/registry/plugin-overrides.integration.test.ts +140 -0
- package/src/registry/plugin-registry.test.ts +56 -0
- package/src/registry/plugin-registry.ts +82 -0
- package/src/registry/tool-aggregator.ts +13 -0
- package/src/registry/types.ts +11 -0
- package/src/runtime/index.ts +43 -0
- package/src/shared/data-path.ts +18 -0
- package/src/shared/deep-merge.test.ts +36 -0
- package/src/shared/deep-merge.ts +61 -0
- package/src/shared/fallback-chain.ts +8 -0
- package/src/shared/index.ts +59 -0
- package/src/shared/logger.ts +54 -0
- package/src/shared/model-availability.ts +18 -0
- package/src/shared/model-normalization.test.ts +75 -0
- package/src/shared/model-normalization.ts +28 -0
- package/src/shared/model-prefix-map.test.ts +75 -0
- package/src/shared/model-prefix-map.ts +58 -0
- package/src/shared/model-resolution-pipeline.test.ts +111 -0
- package/src/shared/model-resolution-pipeline.ts +55 -0
- package/src/shared/models-dev.test.ts +277 -0
- package/src/shared/models-dev.ts +176 -0
- package/src/shared/provider-discovery.test.ts +97 -0
- package/src/shared/provider-discovery.ts +73 -0
- package/src/shared/provider-registry.test.ts +212 -0
- package/src/shared/provider-registry.ts +157 -0
- package/src/shared/safe-create-hook.ts +15 -0
- package/src/shared/snake-case.ts +7 -0
- package/src/shared/truncate-description.ts +6 -0
- package/src/test-utils/index.ts +6 -0
- package/src/test-utils/mock-agent-config.ts +13 -0
- package/src/test-utils/mock-hook-inputs.ts +148 -0
- package/src/test-utils/mock-hook-outputs.ts +153 -0
- package/src/test-utils/mock-plugin-context.test.ts +32 -0
- package/src/test-utils/mock-plugin-context.ts +21 -0
- package/src/test-utils/mock-sdk-client.ts +52 -0
- package/src/test-utils/mock-tool-context.ts +24 -0
- package/src/tools/ast-grep/index.ts +4 -0
- package/src/tools/ast-grep/replace/handler.test.ts +93 -0
- package/src/tools/ast-grep/replace/handler.ts +89 -0
- package/src/tools/ast-grep/replace/plugin.ts +10 -0
- package/src/tools/ast-grep/replace/types.ts +14 -0
- package/src/tools/ast-grep/search/handler.test.ts +94 -0
- package/src/tools/ast-grep/search/handler.ts +93 -0
- package/src/tools/ast-grep/search/plugin.ts +10 -0
- package/src/tools/ast-grep/search/types.ts +42 -0
- package/src/tools/background-task/background-task.test.ts +185 -0
- package/src/tools/background-task/cancel/handler.test.ts +141 -0
- package/src/tools/background-task/cancel/handler.ts +52 -0
- package/src/tools/background-task/cancel/plugin.ts +30 -0
- package/src/tools/background-task/cancel/types.ts +4 -0
- package/src/tools/background-task/index.ts +2 -0
- package/src/tools/background-task/output/handler.test.ts +142 -0
- package/src/tools/background-task/output/handler.ts +93 -0
- package/src/tools/background-task/output/plugin.ts +60 -0
- package/src/tools/background-task/output/types.ts +11 -0
- package/src/tools/bridge.test.ts +55 -0
- package/src/tools/bridge.ts +13 -0
- package/src/tools/builtin-tools.ts +54 -0
- package/src/tools/code-search.test.ts +203 -0
- package/src/tools/delegate-task/category-resolver.ts +18 -0
- package/src/tools/delegate-task/constants.ts +43 -0
- package/src/tools/delegate-task/delegate-task.test.ts +245 -0
- package/src/tools/delegate-task/executor.ts +157 -0
- package/src/tools/delegate-task/handler.ts +107 -0
- package/src/tools/delegate-task/index.ts +7 -0
- package/src/tools/delegate-task/plugin.ts +30 -0
- package/src/tools/delegate-task/types.ts +19 -0
- package/src/tools/glob/handler.test.ts +80 -0
- package/src/tools/glob/handler.ts +74 -0
- package/src/tools/glob/index.ts +2 -0
- package/src/tools/glob/plugin.ts +10 -0
- package/src/tools/glob/types.ts +14 -0
- package/src/tools/grep/handler.test.ts +79 -0
- package/src/tools/grep/handler.ts +104 -0
- package/src/tools/grep/index.ts +2 -0
- package/src/tools/grep/plugin.ts +10 -0
- package/src/tools/grep/types.ts +28 -0
- package/src/tools/hashline-edit/constants.ts +10 -0
- package/src/tools/hashline-edit/edit-operations.ts +115 -0
- package/src/tools/hashline-edit/handler.test.ts +105 -0
- package/src/tools/hashline-edit/handler.ts +33 -0
- package/src/tools/hashline-edit/hash-computation.ts +20 -0
- package/src/tools/hashline-edit/hashline-edit.test.ts +62 -0
- package/src/tools/hashline-edit/index.ts +5 -0
- package/src/tools/hashline-edit/plugin.ts +28 -0
- package/src/tools/hashline-edit/types.ts +42 -0
- package/src/tools/index.ts +4 -0
- package/src/tools/look-at/handler.test.ts +189 -0
- package/src/tools/look-at/handler.ts +232 -0
- package/src/tools/look-at/index.ts +3 -0
- package/src/tools/look-at/look-at.test.ts +200 -0
- package/src/tools/look-at/plugin.ts +10 -0
- package/src/tools/look-at/types.ts +17 -0
- package/src/tools/lsp/client.ts +145 -0
- package/src/tools/lsp/diagnostics/handler.test.ts +94 -0
- package/src/tools/lsp/diagnostics/handler.ts +39 -0
- package/src/tools/lsp/diagnostics/plugin.ts +10 -0
- package/src/tools/lsp/diagnostics/types.ts +15 -0
- package/src/tools/lsp/find-references/handler.test.ts +79 -0
- package/src/tools/lsp/find-references/handler.ts +38 -0
- package/src/tools/lsp/find-references/plugin.ts +10 -0
- package/src/tools/lsp/find-references/types.ts +10 -0
- package/src/tools/lsp/goto-definition/handler.test.ts +80 -0
- package/src/tools/lsp/goto-definition/handler.ts +38 -0
- package/src/tools/lsp/goto-definition/plugin.ts +10 -0
- package/src/tools/lsp/goto-definition/types.ts +9 -0
- package/src/tools/lsp/index.ts +6 -0
- package/src/tools/lsp/lsp-tools.test.ts +150 -0
- package/src/tools/lsp/prepare-rename/handler.test.ts +81 -0
- package/src/tools/lsp/prepare-rename/handler.ts +34 -0
- package/src/tools/lsp/prepare-rename/plugin.ts +10 -0
- package/src/tools/lsp/prepare-rename/types.ts +9 -0
- package/src/tools/lsp/rename/handler.test.ts +87 -0
- package/src/tools/lsp/rename/handler.ts +34 -0
- package/src/tools/lsp/rename/plugin.ts +10 -0
- package/src/tools/lsp/rename/types.ts +10 -0
- package/src/tools/lsp/symbols/handler.test.ts +108 -0
- package/src/tools/lsp/symbols/handler.ts +43 -0
- package/src/tools/lsp/symbols/plugin.ts +10 -0
- package/src/tools/lsp/symbols/types.ts +13 -0
- package/src/tools/session-manager/client-context.ts +16 -0
- package/src/tools/session-manager/index.ts +5 -0
- package/src/tools/session-manager/info/handler.test.ts +100 -0
- package/src/tools/session-manager/info/handler.ts +27 -0
- package/src/tools/session-manager/info/plugin.ts +40 -0
- package/src/tools/session-manager/info/types.ts +3 -0
- package/src/tools/session-manager/list/handler.test.ts +122 -0
- package/src/tools/session-manager/list/handler.ts +56 -0
- package/src/tools/session-manager/list/plugin.ts +52 -0
- package/src/tools/session-manager/list/types.ts +6 -0
- package/src/tools/session-manager/read/handler.test.ts +114 -0
- package/src/tools/session-manager/read/handler.ts +36 -0
- package/src/tools/session-manager/read/plugin.ts +57 -0
- package/src/tools/session-manager/read/types.ts +6 -0
- package/src/tools/session-manager/search/handler.test.ts +115 -0
- package/src/tools/session-manager/search/handler.ts +72 -0
- package/src/tools/session-manager/search/plugin.ts +57 -0
- package/src/tools/session-manager/search/types.ts +6 -0
- package/src/tools/session-manager/session-formatter.ts +315 -0
- package/src/tools/session-manager/session-manager.test.ts +254 -0
- package/src/tools/session-manager/types.ts +41 -0
- package/src/tools/shared/constants.ts +3 -0
- package/src/tools/skill/handler.test.ts +57 -0
- package/src/tools/skill/handler.ts +27 -0
- package/src/tools/skill/index.ts +3 -0
- package/src/tools/skill/plugin.ts +41 -0
- package/src/tools/skill/types.ts +14 -0
- package/src/tools/skill-mcp/handler.test.ts +68 -0
- package/src/tools/skill-mcp/handler.ts +84 -0
- package/src/tools/skill-mcp/index.ts +3 -0
- package/src/tools/skill-mcp/plugin.ts +37 -0
- package/src/tools/skill-mcp/types.ts +15 -0
- package/src/tools/skill-tools.test.ts +172 -0
- package/src/tools/task/create/handler.test.ts +64 -0
- package/src/tools/task/create/handler.ts +43 -0
- package/src/tools/task/create/plugin.ts +10 -0
- package/src/tools/task/format-task.test.ts +37 -0
- package/src/tools/task/format-task.ts +19 -0
- package/src/tools/task/get/handler.test.ts +76 -0
- package/src/tools/task/get/handler.ts +35 -0
- package/src/tools/task/get/plugin.ts +10 -0
- package/src/tools/task/index.ts +4 -0
- package/src/tools/task/list/handler.test.ts +70 -0
- package/src/tools/task/list/handler.ts +48 -0
- package/src/tools/task/list/plugin.ts +10 -0
- package/src/tools/task/storage.ts +14 -0
- package/src/tools/task/task.test.ts +165 -0
- package/src/tools/task/types.ts +51 -0
- package/src/tools/task/update/handler.test.ts +86 -0
- package/src/tools/task/update/handler.ts +54 -0
- package/src/tools/task/update/plugin.ts +10 -0
- package/src/tools/tool-builder.test.ts +32 -0
- package/src/tools/tool-builder.ts +24 -0
- package/src/tools/tool-registry-adapter.test.ts +51 -0
- package/src/tools/tool-registry-adapter.ts +19 -0
- package/src/types/agent.ts +53 -0
- package/src/types/category.ts +32 -0
- package/src/types/config.ts +26 -0
- package/src/types/hook.ts +44 -0
- package/src/types/index.ts +12 -0
- package/src/types/plugin.ts +47 -0
- package/src/types/tool.ts +10 -0
- package/test-setup.ts +8 -0
- package/tsconfig.json +20 -0
- /package/{index.js → npm-reserve/index.js} +0 -0
|
@@ -0,0 +1,178 @@
|
|
|
1
|
+
import { buildFallbackChain } from "../../shared/fallback-chain";
|
|
2
|
+
import { resolveModel } from "../../shared/model-resolution-pipeline";
|
|
3
|
+
import { log } from "../../shared/logger";
|
|
4
|
+
|
|
5
|
+
type ModelSwitchReason = "rate-limit" | "service-unavailable";
|
|
6
|
+
|
|
7
|
+
type ModelFallbackErrorEvent = {
|
|
8
|
+
event: {
|
|
9
|
+
type: string;
|
|
10
|
+
properties?: unknown;
|
|
11
|
+
};
|
|
12
|
+
};
|
|
13
|
+
|
|
14
|
+
type ModelFallbackState = {
|
|
15
|
+
sessionModels: Map<string, string>;
|
|
16
|
+
};
|
|
17
|
+
|
|
18
|
+
type ModelFallbackDependencies = {
|
|
19
|
+
defaultFallbackChain?: string[];
|
|
20
|
+
getCurrentModel?: (sessionID: string) => string | undefined;
|
|
21
|
+
getAvailableModels?: (sessionID: string) => Set<string>;
|
|
22
|
+
setCurrentModel?: (sessionID: string, model: string) => void | Promise<void>;
|
|
23
|
+
onFallbackApplied?: (input: {
|
|
24
|
+
sessionID: string;
|
|
25
|
+
previousModel: string;
|
|
26
|
+
nextModel: string;
|
|
27
|
+
reason: ModelSwitchReason;
|
|
28
|
+
}) => void | Promise<void>;
|
|
29
|
+
};
|
|
30
|
+
|
|
31
|
+
function toRecord(value: unknown): Record<string, unknown> | undefined {
|
|
32
|
+
if (!value || typeof value !== "object") return undefined;
|
|
33
|
+
return value as Record<string, unknown>;
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
function toStatusCode(error: unknown): number | undefined {
|
|
37
|
+
const record = toRecord(error);
|
|
38
|
+
if (!record) return undefined;
|
|
39
|
+
|
|
40
|
+
const direct = record["statusCode"];
|
|
41
|
+
if (typeof direct === "number") return direct;
|
|
42
|
+
|
|
43
|
+
const status = record["status"];
|
|
44
|
+
if (typeof status === "number") return status;
|
|
45
|
+
|
|
46
|
+
const nested = toRecord(record["error"]);
|
|
47
|
+
if (nested) {
|
|
48
|
+
const nestedCode = nested["statusCode"];
|
|
49
|
+
if (typeof nestedCode === "number") return nestedCode;
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
return undefined;
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
function getErrorMessage(error: unknown): string {
|
|
56
|
+
if (typeof error === "string") return error.toLowerCase();
|
|
57
|
+
const record = toRecord(error);
|
|
58
|
+
const message = record?.["message"];
|
|
59
|
+
if (typeof message === "string") return message.toLowerCase();
|
|
60
|
+
return "";
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
function classifyModelFallbackReason(error: unknown): ModelSwitchReason | undefined {
|
|
64
|
+
const statusCode = toStatusCode(error);
|
|
65
|
+
const message = getErrorMessage(error);
|
|
66
|
+
|
|
67
|
+
if (
|
|
68
|
+
statusCode === 429 ||
|
|
69
|
+
message.includes("rate limit") ||
|
|
70
|
+
message.includes("too many requests")
|
|
71
|
+
) {
|
|
72
|
+
return "rate-limit";
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
if (
|
|
76
|
+
statusCode === 503 ||
|
|
77
|
+
message.includes("service unavailable") ||
|
|
78
|
+
message.includes("temporarily unavailable")
|
|
79
|
+
) {
|
|
80
|
+
return "service-unavailable";
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
return undefined;
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
function getSessionID(properties: Record<string, unknown>): string | undefined {
|
|
87
|
+
const sessionID = properties["sessionID"];
|
|
88
|
+
if (typeof sessionID === "string" && sessionID.length > 0) return sessionID;
|
|
89
|
+
|
|
90
|
+
const info = toRecord(properties["info"]);
|
|
91
|
+
const infoID = info?.["id"];
|
|
92
|
+
if (typeof infoID === "string" && infoID.length > 0) return infoID;
|
|
93
|
+
|
|
94
|
+
return undefined;
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
function getNextFallbackModel(input: {
|
|
98
|
+
currentModel: string;
|
|
99
|
+
configuredFallbacks: string | string[] | undefined;
|
|
100
|
+
defaultChain: string[];
|
|
101
|
+
availableModels: Set<string>;
|
|
102
|
+
}): string | undefined {
|
|
103
|
+
const fullChain = buildFallbackChain(input.configuredFallbacks, input.defaultChain);
|
|
104
|
+
const currentIndex = fullChain.findIndex(
|
|
105
|
+
(model) => model.toLowerCase() === input.currentModel.toLowerCase(),
|
|
106
|
+
);
|
|
107
|
+
const remaining = currentIndex >= 0 ? fullChain.slice(currentIndex + 1) : fullChain;
|
|
108
|
+
|
|
109
|
+
return resolveModel({
|
|
110
|
+
fallbackChain: remaining,
|
|
111
|
+
availableModels: input.availableModels,
|
|
112
|
+
})?.model;
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
export function createModelFallbackHandler(deps: ModelFallbackDependencies = {}) {
|
|
116
|
+
const state: ModelFallbackState = { sessionModels: new Map() };
|
|
117
|
+
const defaultSwitch = async (sessionID: string, model: string) => {
|
|
118
|
+
state.sessionModels.set(sessionID, model);
|
|
119
|
+
};
|
|
120
|
+
|
|
121
|
+
return async ({ event }: ModelFallbackErrorEvent): Promise<void> => {
|
|
122
|
+
if (event.type !== "session.error") return;
|
|
123
|
+
|
|
124
|
+
const properties = toRecord(event.properties);
|
|
125
|
+
if (!properties) return;
|
|
126
|
+
|
|
127
|
+
const reason = classifyModelFallbackReason(properties["error"]);
|
|
128
|
+
if (!reason) return;
|
|
129
|
+
|
|
130
|
+
const sessionID = getSessionID(properties);
|
|
131
|
+
if (!sessionID) return;
|
|
132
|
+
|
|
133
|
+
const configuredFallbacks = properties["fallbackChain"];
|
|
134
|
+
const fallbackInput =
|
|
135
|
+
typeof configuredFallbacks === "string" || Array.isArray(configuredFallbacks)
|
|
136
|
+
? configuredFallbacks
|
|
137
|
+
: undefined;
|
|
138
|
+
|
|
139
|
+
const eventModel = properties["model"];
|
|
140
|
+
const currentModel =
|
|
141
|
+
(typeof eventModel === "string" && eventModel.length > 0 ? eventModel : undefined) ??
|
|
142
|
+
deps.getCurrentModel?.(sessionID) ??
|
|
143
|
+
state.sessionModels.get(sessionID);
|
|
144
|
+
if (!currentModel) return;
|
|
145
|
+
|
|
146
|
+
const availableModels = deps.getAvailableModels?.(sessionID) ?? new Set<string>();
|
|
147
|
+
const nextModel = getNextFallbackModel({
|
|
148
|
+
currentModel,
|
|
149
|
+
configuredFallbacks: fallbackInput,
|
|
150
|
+
defaultChain: deps.defaultFallbackChain ?? [],
|
|
151
|
+
availableModels,
|
|
152
|
+
});
|
|
153
|
+
|
|
154
|
+
if (!nextModel || nextModel.toLowerCase() === currentModel.toLowerCase()) {
|
|
155
|
+
log("[model-fallback] no eligible fallback model", { sessionID, currentModel, reason });
|
|
156
|
+
return;
|
|
157
|
+
}
|
|
158
|
+
|
|
159
|
+
const applyModel = deps.setCurrentModel ?? defaultSwitch;
|
|
160
|
+
await Promise.resolve(applyModel(sessionID, nextModel));
|
|
161
|
+
|
|
162
|
+
await Promise.resolve(
|
|
163
|
+
deps.onFallbackApplied?.({
|
|
164
|
+
sessionID,
|
|
165
|
+
previousModel: currentModel,
|
|
166
|
+
nextModel,
|
|
167
|
+
reason,
|
|
168
|
+
}),
|
|
169
|
+
);
|
|
170
|
+
|
|
171
|
+
log("[model-fallback] switched model", {
|
|
172
|
+
sessionID,
|
|
173
|
+
from: currentModel,
|
|
174
|
+
to: nextModel,
|
|
175
|
+
reason,
|
|
176
|
+
});
|
|
177
|
+
};
|
|
178
|
+
}
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
import { definePlugin } from "../../plugin-api";
|
|
2
|
+
import { safeCreateHook } from "../../shared/safe-create-hook";
|
|
3
|
+
import { createModelFallbackHandler } from "./handler";
|
|
4
|
+
|
|
5
|
+
const modelFallbackEventHook = safeCreateHook("model-fallback", () => createModelFallbackHandler());
|
|
6
|
+
|
|
7
|
+
export const modelFallbackPlugin = definePlugin({
|
|
8
|
+
name: "model-fallback",
|
|
9
|
+
version: "0.1.0",
|
|
10
|
+
hooks: modelFallbackEventHook ? { event: modelFallbackEventHook } : undefined,
|
|
11
|
+
});
|
|
@@ -0,0 +1,121 @@
|
|
|
1
|
+
declare const require: (name: string) => any;
|
|
2
|
+
|
|
3
|
+
const { describe, expect, it, mock } = require("bun:test");
|
|
4
|
+
import { createModelFallbackHandler, modelFallbackPlugin } from "./model-fallback";
|
|
5
|
+
import { createRuntimeFallbackHandler, runtimeFallbackPlugin } from "./runtime-fallback";
|
|
6
|
+
import {
|
|
7
|
+
createPreemptiveCompactionHandler,
|
|
8
|
+
preemptiveCompactionPlugin,
|
|
9
|
+
} from "./preemptive-compaction";
|
|
10
|
+
|
|
11
|
+
describe("model management hooks", () => {
|
|
12
|
+
describe("#given model-fallback receives rate-limit errors", () => {
|
|
13
|
+
describe("#when session.error is emitted with status 429", () => {
|
|
14
|
+
it("#then switches to the next model in fallback chain", async () => {
|
|
15
|
+
const sessionModels = new Map<string, string>([["ses-model", "openai/gpt-5"]]);
|
|
16
|
+
const onFallbackApplied = mock(() => undefined);
|
|
17
|
+
|
|
18
|
+
const handler = createModelFallbackHandler({
|
|
19
|
+
getCurrentModel: (sessionID) => sessionModels.get(sessionID),
|
|
20
|
+
setCurrentModel: (sessionID, model) => {
|
|
21
|
+
sessionModels.set(sessionID, model);
|
|
22
|
+
},
|
|
23
|
+
onFallbackApplied,
|
|
24
|
+
});
|
|
25
|
+
|
|
26
|
+
await handler({
|
|
27
|
+
event: {
|
|
28
|
+
type: "session.error",
|
|
29
|
+
properties: {
|
|
30
|
+
sessionID: "ses-model",
|
|
31
|
+
model: "openai/gpt-5",
|
|
32
|
+
error: { statusCode: 429, message: "Too many requests" },
|
|
33
|
+
fallbackChain: ["openai/gpt-5", "openai/gpt-4.1-mini", "anthropic/claude-3.7-sonnet"],
|
|
34
|
+
},
|
|
35
|
+
},
|
|
36
|
+
});
|
|
37
|
+
|
|
38
|
+
expect(sessionModels.get("ses-model")).toBe("openai/gpt-4.1-mini");
|
|
39
|
+
expect(onFallbackApplied).toHaveBeenCalledTimes(1);
|
|
40
|
+
});
|
|
41
|
+
});
|
|
42
|
+
});
|
|
43
|
+
|
|
44
|
+
describe("#given runtime-fallback receives model/runtime errors", () => {
|
|
45
|
+
describe("#when session.error reports model not found", () => {
|
|
46
|
+
it("#then switches to a compatible fallback model", async () => {
|
|
47
|
+
const sessionModels = new Map<string, string>([["ses-runtime", "openai/gpt-5"]]);
|
|
48
|
+
|
|
49
|
+
const handler = createRuntimeFallbackHandler({
|
|
50
|
+
getCurrentModel: (sessionID) => sessionModels.get(sessionID),
|
|
51
|
+
setCurrentModel: (sessionID, model) => {
|
|
52
|
+
sessionModels.set(sessionID, model);
|
|
53
|
+
},
|
|
54
|
+
});
|
|
55
|
+
|
|
56
|
+
await handler({
|
|
57
|
+
event: {
|
|
58
|
+
type: "session.error",
|
|
59
|
+
properties: {
|
|
60
|
+
sessionID: "ses-runtime",
|
|
61
|
+
model: "openai/gpt-5",
|
|
62
|
+
error: { message: "Model not found for this provider" },
|
|
63
|
+
fallbackChain: [
|
|
64
|
+
"anthropic/claude-3.7-sonnet",
|
|
65
|
+
"openai/gpt-4.1-mini",
|
|
66
|
+
"google/gemini-2.5-pro",
|
|
67
|
+
],
|
|
68
|
+
},
|
|
69
|
+
},
|
|
70
|
+
});
|
|
71
|
+
|
|
72
|
+
expect(sessionModels.get("ses-runtime")).toBe("openai/gpt-4.1-mini");
|
|
73
|
+
});
|
|
74
|
+
});
|
|
75
|
+
});
|
|
76
|
+
|
|
77
|
+
describe("#given preemptive-compaction monitors token usage", () => {
|
|
78
|
+
describe("#when usage crosses and recrosses the 80% threshold", () => {
|
|
79
|
+
it("#then triggers compaction once per high-usage window", async () => {
|
|
80
|
+
const compactSession = mock(() => Promise.resolve());
|
|
81
|
+
const handler = createPreemptiveCompactionHandler({ compactSession });
|
|
82
|
+
|
|
83
|
+
await handler({
|
|
84
|
+
sessionID: "ses-compact",
|
|
85
|
+
usage: { inputTokens: 170_000, cacheReadTokens: 0 },
|
|
86
|
+
contextLimit: 200_000,
|
|
87
|
+
});
|
|
88
|
+
|
|
89
|
+
await handler({
|
|
90
|
+
sessionID: "ses-compact",
|
|
91
|
+
usage: { inputTokens: 175_000, cacheReadTokens: 0 },
|
|
92
|
+
contextLimit: 200_000,
|
|
93
|
+
});
|
|
94
|
+
|
|
95
|
+
await handler({
|
|
96
|
+
sessionID: "ses-compact",
|
|
97
|
+
usage: { inputTokens: 100_000, cacheReadTokens: 0 },
|
|
98
|
+
contextLimit: 200_000,
|
|
99
|
+
});
|
|
100
|
+
|
|
101
|
+
await handler({
|
|
102
|
+
sessionID: "ses-compact",
|
|
103
|
+
usage: { inputTokens: 180_000, cacheReadTokens: 0 },
|
|
104
|
+
contextLimit: 200_000,
|
|
105
|
+
});
|
|
106
|
+
|
|
107
|
+
expect(compactSession).toHaveBeenCalledTimes(2);
|
|
108
|
+
});
|
|
109
|
+
});
|
|
110
|
+
});
|
|
111
|
+
|
|
112
|
+
describe("#given hook micro-plugins are created", () => {
|
|
113
|
+
describe("#when each plugin registers hooks", () => {
|
|
114
|
+
it("#then wires event/message handlers through definePlugin", () => {
|
|
115
|
+
expect(typeof modelFallbackPlugin.hooks?.event).toBe("function");
|
|
116
|
+
expect(typeof runtimeFallbackPlugin.hooks?.event).toBe("function");
|
|
117
|
+
expect(typeof preemptiveCompactionPlugin.hooks?.["chat.message"]).toBe("function");
|
|
118
|
+
});
|
|
119
|
+
});
|
|
120
|
+
});
|
|
121
|
+
});
|
|
@@ -0,0 +1,105 @@
|
|
|
1
|
+
import { describe, expect, it } from "bun:test";
|
|
2
|
+
import type { HookHandler } from "../../types/hook";
|
|
3
|
+
import { createPhaseReminderHandler, PHASE_REMINDER } from "./handler";
|
|
4
|
+
|
|
5
|
+
const createHandler = (): HookHandler => createPhaseReminderHandler() as HookHandler;
|
|
6
|
+
|
|
7
|
+
describe("createPhaseReminderHandler", () => {
|
|
8
|
+
describe("#given a user message from the orchestrator agent", () => {
|
|
9
|
+
describe("#when the handler processes messages", () => {
|
|
10
|
+
it("#then it prepends the phase reminder to the text", async () => {
|
|
11
|
+
const handler = createHandler();
|
|
12
|
+
const output = {
|
|
13
|
+
messages: [
|
|
14
|
+
{
|
|
15
|
+
info: { role: "user" },
|
|
16
|
+
parts: [{ type: "text", text: "Do the task" }],
|
|
17
|
+
},
|
|
18
|
+
],
|
|
19
|
+
};
|
|
20
|
+
|
|
21
|
+
await handler({}, output);
|
|
22
|
+
|
|
23
|
+
expect(output.messages[0].parts[0].text).toContain(PHASE_REMINDER);
|
|
24
|
+
expect(output.messages[0].parts[0].text).toContain("Do the task");
|
|
25
|
+
});
|
|
26
|
+
});
|
|
27
|
+
});
|
|
28
|
+
|
|
29
|
+
describe("#given a user message with an explicit orchestrator agent", () => {
|
|
30
|
+
describe("#when the handler processes messages", () => {
|
|
31
|
+
it("#then it prepends the phase reminder", async () => {
|
|
32
|
+
const handler = createHandler();
|
|
33
|
+
const output = {
|
|
34
|
+
messages: [
|
|
35
|
+
{
|
|
36
|
+
info: { role: "user", agent: "orchestrator" },
|
|
37
|
+
parts: [{ type: "text", text: "Plan the work" }],
|
|
38
|
+
},
|
|
39
|
+
],
|
|
40
|
+
};
|
|
41
|
+
|
|
42
|
+
await handler({}, output);
|
|
43
|
+
|
|
44
|
+
expect(output.messages[0].parts[0].text).toContain(PHASE_REMINDER);
|
|
45
|
+
expect(output.messages[0].parts[0].text).toContain("Plan the work");
|
|
46
|
+
});
|
|
47
|
+
});
|
|
48
|
+
});
|
|
49
|
+
|
|
50
|
+
describe("#given a user message from a non-orchestrator agent", () => {
|
|
51
|
+
describe("#when the handler processes messages", () => {
|
|
52
|
+
it("#then it does not modify the text", async () => {
|
|
53
|
+
const handler = createHandler();
|
|
54
|
+
const output = {
|
|
55
|
+
messages: [
|
|
56
|
+
{
|
|
57
|
+
info: { role: "user", agent: "worker" },
|
|
58
|
+
parts: [{ type: "text", text: "Do the task" }],
|
|
59
|
+
},
|
|
60
|
+
],
|
|
61
|
+
};
|
|
62
|
+
|
|
63
|
+
await handler({}, output);
|
|
64
|
+
|
|
65
|
+
expect(output.messages[0].parts[0].text).toBe("Do the task");
|
|
66
|
+
});
|
|
67
|
+
});
|
|
68
|
+
});
|
|
69
|
+
|
|
70
|
+
describe("#given a message that already contains the reminder", () => {
|
|
71
|
+
describe("#when the handler processes messages", () => {
|
|
72
|
+
it("#then it does not double-prepend the reminder", async () => {
|
|
73
|
+
const handler = createHandler();
|
|
74
|
+
const existingText = `${PHASE_REMINDER}\n\n---\n\nDo the task`;
|
|
75
|
+
const output = {
|
|
76
|
+
messages: [
|
|
77
|
+
{
|
|
78
|
+
info: { role: "user" },
|
|
79
|
+
parts: [{ type: "text", text: existingText }],
|
|
80
|
+
},
|
|
81
|
+
],
|
|
82
|
+
};
|
|
83
|
+
|
|
84
|
+
await handler({}, output);
|
|
85
|
+
|
|
86
|
+
const occurrences =
|
|
87
|
+
output.messages[0].parts[0].text.split("<reminder>Recall Workflow Rules:").length - 1;
|
|
88
|
+
expect(occurrences).toBe(1);
|
|
89
|
+
});
|
|
90
|
+
});
|
|
91
|
+
});
|
|
92
|
+
|
|
93
|
+
describe("#given an empty messages array", () => {
|
|
94
|
+
describe("#when the handler processes messages", () => {
|
|
95
|
+
it("#then it is a no-op", async () => {
|
|
96
|
+
const handler = createHandler();
|
|
97
|
+
const output = { messages: [] as unknown[] };
|
|
98
|
+
|
|
99
|
+
await handler({}, output);
|
|
100
|
+
|
|
101
|
+
expect(output.messages).toEqual([]);
|
|
102
|
+
});
|
|
103
|
+
});
|
|
104
|
+
});
|
|
105
|
+
});
|
|
@@ -0,0 +1,54 @@
|
|
|
1
|
+
import type { PluginHookContributions } from "../../types/hook";
|
|
2
|
+
|
|
3
|
+
type MessagesTransformHook = NonNullable<
|
|
4
|
+
PluginHookContributions["experimental.chat.messages.transform"]
|
|
5
|
+
>;
|
|
6
|
+
|
|
7
|
+
export const PHASE_REMINDER = `<reminder>Recall Workflow Rules:
|
|
8
|
+
Understand → choose best path (delegate by rules and parallelize independent work) → execute → verify.
|
|
9
|
+
If mentioning a specialist, launch it in the same turn.</reminder>`;
|
|
10
|
+
|
|
11
|
+
const REMINDER_PREFIX = "<reminder>Recall Workflow Rules:";
|
|
12
|
+
|
|
13
|
+
function isRecord(value: unknown): value is Record<string, unknown> {
|
|
14
|
+
return typeof value === "object" && value !== null;
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
function isOrchestratorAgent(agent: unknown): boolean {
|
|
18
|
+
if (typeof agent !== "string") return true;
|
|
19
|
+
return agent.toLowerCase() === "orchestrator";
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
function prependReminder(text: string): string {
|
|
23
|
+
if (text.includes(REMINDER_PREFIX)) {
|
|
24
|
+
return text;
|
|
25
|
+
}
|
|
26
|
+
return `${PHASE_REMINDER}\n\n---\n\n${text}`;
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
export function createPhaseReminderHandler(): MessagesTransformHook {
|
|
30
|
+
return async (_input: unknown, output: unknown) => {
|
|
31
|
+
if (!isRecord(output)) return;
|
|
32
|
+
|
|
33
|
+
const messages = output.messages;
|
|
34
|
+
if (!Array.isArray(messages) || messages.length === 0) return;
|
|
35
|
+
|
|
36
|
+
for (let index = messages.length - 1; index >= 0; index -= 1) {
|
|
37
|
+
const message = messages[index];
|
|
38
|
+
if (!isRecord(message)) continue;
|
|
39
|
+
|
|
40
|
+
const info = message.info;
|
|
41
|
+
if (!isRecord(info) || info.role !== "user") continue;
|
|
42
|
+
if (!isOrchestratorAgent(info.agent)) return;
|
|
43
|
+
|
|
44
|
+
const parts = message.parts;
|
|
45
|
+
if (!Array.isArray(parts)) return;
|
|
46
|
+
|
|
47
|
+
const textPart = parts.find((part) => part?.type === "text" && typeof part.text === "string");
|
|
48
|
+
if (!textPart || typeof textPart.text !== "string") return;
|
|
49
|
+
|
|
50
|
+
textPart.text = prependReminder(textPart.text);
|
|
51
|
+
return;
|
|
52
|
+
}
|
|
53
|
+
};
|
|
54
|
+
}
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
import { definePlugin } from "../../plugin-api";
|
|
2
|
+
import { safeCreateHook } from "../../shared/safe-create-hook";
|
|
3
|
+
import { createPhaseReminderHandler } from "./handler";
|
|
4
|
+
|
|
5
|
+
const phaseReminderMessagesTransformHook = safeCreateHook(
|
|
6
|
+
"phase-reminder",
|
|
7
|
+
createPhaseReminderHandler,
|
|
8
|
+
);
|
|
9
|
+
|
|
10
|
+
export const phaseReminderPlugin = definePlugin({
|
|
11
|
+
name: "phase-reminder",
|
|
12
|
+
version: "0.1.0",
|
|
13
|
+
hooks: phaseReminderMessagesTransformHook
|
|
14
|
+
? {
|
|
15
|
+
"experimental.chat.messages.transform": phaseReminderMessagesTransformHook,
|
|
16
|
+
}
|
|
17
|
+
: {},
|
|
18
|
+
});
|
|
@@ -0,0 +1,159 @@
|
|
|
1
|
+
import { describe, expect, it } from "bun:test";
|
|
2
|
+
|
|
3
|
+
import { createPostReadNudgeHandler, POST_READ_NUDGE } from "./handler";
|
|
4
|
+
|
|
5
|
+
describe("createPostReadNudgeHandler", () => {
|
|
6
|
+
describe("#given a Read tool call", () => {
|
|
7
|
+
describe("#when tool.execute.after runs on Read", () => {
|
|
8
|
+
it("#then appends the workflow nudge to the output", async () => {
|
|
9
|
+
const handler = createPostReadNudgeHandler();
|
|
10
|
+
const input = { tool: "Read", sessionID: "s1", callID: "c1", args: {} };
|
|
11
|
+
const output = { title: "read", output: "file content here", metadata: {} };
|
|
12
|
+
|
|
13
|
+
await handler(input, output);
|
|
14
|
+
|
|
15
|
+
expect(output.output.endsWith(POST_READ_NUDGE)).toBe(true);
|
|
16
|
+
expect(output.output).toContain("file content here");
|
|
17
|
+
expect(output.output).toContain("Workflow Reminder");
|
|
18
|
+
});
|
|
19
|
+
});
|
|
20
|
+
|
|
21
|
+
describe("#when the Read tool name is lowercase", () => {
|
|
22
|
+
it("#then still appends the nudge due to case-insensitive matching", async () => {
|
|
23
|
+
const handler = createPostReadNudgeHandler();
|
|
24
|
+
const input = { tool: "read", sessionID: "s1", callID: "c1", args: {} };
|
|
25
|
+
const output = { title: "read", output: "some content", metadata: {} };
|
|
26
|
+
|
|
27
|
+
await handler(input, output);
|
|
28
|
+
|
|
29
|
+
expect(output.output.endsWith(POST_READ_NUDGE)).toBe(true);
|
|
30
|
+
});
|
|
31
|
+
});
|
|
32
|
+
|
|
33
|
+
describe("#when the Read tool name is READ in uppercase", () => {
|
|
34
|
+
it("#then still appends the nudge", async () => {
|
|
35
|
+
const handler = createPostReadNudgeHandler();
|
|
36
|
+
const input = { tool: "READ", sessionID: "s1", callID: "c1", args: {} };
|
|
37
|
+
const output = { title: "read", output: "uppercase read content", metadata: {} };
|
|
38
|
+
|
|
39
|
+
await handler(input, output);
|
|
40
|
+
|
|
41
|
+
expect(output.output.endsWith(POST_READ_NUDGE)).toBe(true);
|
|
42
|
+
});
|
|
43
|
+
});
|
|
44
|
+
});
|
|
45
|
+
|
|
46
|
+
describe("#given a non-Read tool call", () => {
|
|
47
|
+
describe("#when the tool is Bash", () => {
|
|
48
|
+
it("#then leaves the output unchanged", async () => {
|
|
49
|
+
const handler = createPostReadNudgeHandler();
|
|
50
|
+
const input = { tool: "Bash", sessionID: "s1", callID: "c1", args: {} };
|
|
51
|
+
const original = "command output";
|
|
52
|
+
const output = { title: "bash", output: original, metadata: {} };
|
|
53
|
+
|
|
54
|
+
await handler(input, output);
|
|
55
|
+
|
|
56
|
+
expect(output.output).toBe(original);
|
|
57
|
+
});
|
|
58
|
+
});
|
|
59
|
+
|
|
60
|
+
describe("#when the tool is Edit", () => {
|
|
61
|
+
it("#then leaves the output unchanged", async () => {
|
|
62
|
+
const handler = createPostReadNudgeHandler();
|
|
63
|
+
const input = { tool: "Edit", sessionID: "s1", callID: "c1", args: {} };
|
|
64
|
+
const original = "edit successful";
|
|
65
|
+
const output = { title: "edit", output: original, metadata: {} };
|
|
66
|
+
|
|
67
|
+
await handler(input, output);
|
|
68
|
+
|
|
69
|
+
expect(output.output).toBe(original);
|
|
70
|
+
});
|
|
71
|
+
});
|
|
72
|
+
});
|
|
73
|
+
|
|
74
|
+
describe("#given a Read tool output that already contains the nudge", () => {
|
|
75
|
+
describe("#when the output already has the nudge text", () => {
|
|
76
|
+
it("#then does not double-append the nudge", async () => {
|
|
77
|
+
const handler = createPostReadNudgeHandler();
|
|
78
|
+
const input = { tool: "Read", sessionID: "s1", callID: "c1", args: {} };
|
|
79
|
+
const output = {
|
|
80
|
+
title: "read",
|
|
81
|
+
output: `file content${POST_READ_NUDGE}`,
|
|
82
|
+
metadata: {},
|
|
83
|
+
};
|
|
84
|
+
|
|
85
|
+
await handler(input, output);
|
|
86
|
+
|
|
87
|
+
const nudgeCount = output.output.split("Workflow Reminder").length - 1;
|
|
88
|
+
expect(nudgeCount).toBe(1);
|
|
89
|
+
});
|
|
90
|
+
});
|
|
91
|
+
});
|
|
92
|
+
|
|
93
|
+
describe("#given a Read tool with non-string output", () => {
|
|
94
|
+
describe("#when output.output is undefined", () => {
|
|
95
|
+
it("#then does not modify the output", async () => {
|
|
96
|
+
const handler = createPostReadNudgeHandler();
|
|
97
|
+
const input = { tool: "Read", sessionID: "s1", callID: "c1", args: {} };
|
|
98
|
+
const output = { title: "read", output: undefined as unknown as string, metadata: {} };
|
|
99
|
+
|
|
100
|
+
await handler(input, output);
|
|
101
|
+
|
|
102
|
+
expect(output.output).toBeUndefined();
|
|
103
|
+
});
|
|
104
|
+
});
|
|
105
|
+
});
|
|
106
|
+
|
|
107
|
+
describe("#given multiple exploration tool calls exceeding threshold", () => {
|
|
108
|
+
describe("#when 3+ exploration calls are made (read, grep, glob mix)", () => {
|
|
109
|
+
it("#then escalates to a delegation nudge on subsequent reads", async () => {
|
|
110
|
+
const handler = createPostReadNudgeHandler();
|
|
111
|
+
|
|
112
|
+
// First two reads get normal nudge
|
|
113
|
+
const input1 = { tool: "read", sessionID: "s1", callID: "c1", args: {} };
|
|
114
|
+
const output1 = { title: "read", output: "content1", metadata: {} };
|
|
115
|
+
await handler(input1, output1);
|
|
116
|
+
expect(output1.output).toContain("Workflow Reminder");
|
|
117
|
+
expect(output1.output).not.toContain("DELEGATION NUDGE");
|
|
118
|
+
|
|
119
|
+
// Second read
|
|
120
|
+
const input2 = { tool: "read", sessionID: "s1", callID: "c2", args: {} };
|
|
121
|
+
const output2 = { title: "read", output: "content2", metadata: {} };
|
|
122
|
+
await handler(input2, output2);
|
|
123
|
+
expect(output2.output).toContain("Workflow Reminder");
|
|
124
|
+
expect(output2.output).not.toContain("DELEGATION NUDGE");
|
|
125
|
+
|
|
126
|
+
// Third read triggers escalation
|
|
127
|
+
const input3 = { tool: "read", sessionID: "s1", callID: "c3", args: {} };
|
|
128
|
+
const output3 = { title: "read", output: "content3", metadata: {} };
|
|
129
|
+
await handler(input3, output3);
|
|
130
|
+
expect(output3.output).toContain("DELEGATION NUDGE");
|
|
131
|
+
expect(output3.output).toContain("delegate to a specialist agent NOW");
|
|
132
|
+
});
|
|
133
|
+
});
|
|
134
|
+
|
|
135
|
+
describe("#when grep and glob calls contribute to the count", () => {
|
|
136
|
+
it("#then counts all exploration tools toward the threshold", async () => {
|
|
137
|
+
const handler = createPostReadNudgeHandler();
|
|
138
|
+
|
|
139
|
+
// Grep call — doesn't append nudge to grep output
|
|
140
|
+
const grepInput = { tool: "grep", sessionID: "s1", callID: "c1", args: {} };
|
|
141
|
+
const grepOutput = { title: "grep", output: "grep results", metadata: {} };
|
|
142
|
+
await handler(grepInput, grepOutput);
|
|
143
|
+
expect(grepOutput.output).toBe("grep results");
|
|
144
|
+
|
|
145
|
+
// Glob call — doesn't append nudge to glob output
|
|
146
|
+
const globInput = { tool: "glob", sessionID: "s1", callID: "c2", args: {} };
|
|
147
|
+
const globOutput = { title: "glob", output: "glob results", metadata: {} };
|
|
148
|
+
await handler(globInput, globOutput);
|
|
149
|
+
expect(globOutput.output).toBe("glob results");
|
|
150
|
+
|
|
151
|
+
// Third exploration call is a read — should trigger escalation
|
|
152
|
+
const readInput = { tool: "read", sessionID: "s1", callID: "c3", args: {} };
|
|
153
|
+
const readOutput = { title: "read", output: "file content", metadata: {} };
|
|
154
|
+
await handler(readInput, readOutput);
|
|
155
|
+
expect(readOutput.output).toContain("DELEGATION NUDGE");
|
|
156
|
+
});
|
|
157
|
+
});
|
|
158
|
+
});
|
|
159
|
+
});
|