@imdigitalashish/zpi 0.1.1
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 +2801 -0
- package/README.md +95 -0
- package/dist/cli/args.d.ts +47 -0
- package/dist/cli/args.d.ts.map +1 -0
- package/dist/cli/args.js +293 -0
- package/dist/cli/args.js.map +1 -0
- package/dist/cli/config-selector.d.ts +14 -0
- package/dist/cli/config-selector.d.ts.map +1 -0
- package/dist/cli/config-selector.js +31 -0
- package/dist/cli/config-selector.js.map +1 -0
- package/dist/cli/file-processor.d.ts +15 -0
- package/dist/cli/file-processor.d.ts.map +1 -0
- package/dist/cli/file-processor.js +79 -0
- package/dist/cli/file-processor.js.map +1 -0
- package/dist/cli/list-models.d.ts +9 -0
- package/dist/cli/list-models.d.ts.map +1 -0
- package/dist/cli/list-models.js +92 -0
- package/dist/cli/list-models.js.map +1 -0
- package/dist/cli/session-picker.d.ts +9 -0
- package/dist/cli/session-picker.d.ts.map +1 -0
- package/dist/cli/session-picker.js +34 -0
- package/dist/cli/session-picker.js.map +1 -0
- package/dist/cli.d.ts +3 -0
- package/dist/cli.d.ts.map +1 -0
- package/dist/cli.js +11 -0
- package/dist/cli.js.map +1 -0
- package/dist/config.d.ts +68 -0
- package/dist/config.d.ts.map +1 -0
- package/dist/config.js +203 -0
- package/dist/config.js.map +1 -0
- package/dist/core/agent-session.d.ts +571 -0
- package/dist/core/agent-session.d.ts.map +1 -0
- package/dist/core/agent-session.js +2353 -0
- package/dist/core/agent-session.js.map +1 -0
- package/dist/core/auth-storage.d.ts +129 -0
- package/dist/core/auth-storage.d.ts.map +1 -0
- package/dist/core/auth-storage.js +394 -0
- package/dist/core/auth-storage.js.map +1 -0
- package/dist/core/bash-executor.d.ts +47 -0
- package/dist/core/bash-executor.d.ts.map +1 -0
- package/dist/core/bash-executor.js +212 -0
- package/dist/core/bash-executor.js.map +1 -0
- package/dist/core/compaction/branch-summarization.d.ts +86 -0
- package/dist/core/compaction/branch-summarization.d.ts.map +1 -0
- package/dist/core/compaction/branch-summarization.js +242 -0
- package/dist/core/compaction/branch-summarization.js.map +1 -0
- package/dist/core/compaction/compaction.d.ts +121 -0
- package/dist/core/compaction/compaction.d.ts.map +1 -0
- package/dist/core/compaction/compaction.js +607 -0
- package/dist/core/compaction/compaction.js.map +1 -0
- package/dist/core/compaction/index.d.ts +7 -0
- package/dist/core/compaction/index.d.ts.map +1 -0
- package/dist/core/compaction/index.js +7 -0
- package/dist/core/compaction/index.js.map +1 -0
- package/dist/core/compaction/utils.d.ts +35 -0
- package/dist/core/compaction/utils.d.ts.map +1 -0
- package/dist/core/compaction/utils.js +138 -0
- package/dist/core/compaction/utils.js.map +1 -0
- package/dist/core/defaults.d.ts +3 -0
- package/dist/core/defaults.d.ts.map +1 -0
- package/dist/core/defaults.js +2 -0
- package/dist/core/defaults.js.map +1 -0
- package/dist/core/diagnostics.d.ts +15 -0
- package/dist/core/diagnostics.d.ts.map +1 -0
- package/dist/core/diagnostics.js +2 -0
- package/dist/core/diagnostics.js.map +1 -0
- package/dist/core/event-bus.d.ts +9 -0
- package/dist/core/event-bus.d.ts.map +1 -0
- package/dist/core/event-bus.js +25 -0
- package/dist/core/event-bus.js.map +1 -0
- package/dist/core/exec.d.ts +29 -0
- package/dist/core/exec.d.ts.map +1 -0
- package/dist/core/exec.js +71 -0
- package/dist/core/exec.js.map +1 -0
- package/dist/core/export-html/ansi-to-html.d.ts +22 -0
- package/dist/core/export-html/ansi-to-html.d.ts.map +1 -0
- package/dist/core/export-html/ansi-to-html.js +249 -0
- package/dist/core/export-html/ansi-to-html.js.map +1 -0
- package/dist/core/export-html/index.d.ts +34 -0
- package/dist/core/export-html/index.d.ts.map +1 -0
- package/dist/core/export-html/index.js +222 -0
- package/dist/core/export-html/index.js.map +1 -0
- package/dist/core/export-html/template.css +971 -0
- package/dist/core/export-html/template.html +54 -0
- package/dist/core/export-html/template.js +1586 -0
- package/dist/core/export-html/tool-renderer.d.ts +35 -0
- package/dist/core/export-html/tool-renderer.d.ts.map +1 -0
- package/dist/core/export-html/tool-renderer.js +57 -0
- package/dist/core/export-html/tool-renderer.js.map +1 -0
- package/dist/core/export-html/vendor/highlight.min.js +1213 -0
- package/dist/core/export-html/vendor/marked.min.js +6 -0
- package/dist/core/extensions/index.d.ts +11 -0
- package/dist/core/extensions/index.d.ts.map +1 -0
- package/dist/core/extensions/index.js +9 -0
- package/dist/core/extensions/index.js.map +1 -0
- package/dist/core/extensions/loader.d.ts +25 -0
- package/dist/core/extensions/loader.d.ts.map +1 -0
- package/dist/core/extensions/loader.js +402 -0
- package/dist/core/extensions/loader.js.map +1 -0
- package/dist/core/extensions/runner.d.ts +146 -0
- package/dist/core/extensions/runner.d.ts.map +1 -0
- package/dist/core/extensions/runner.js +626 -0
- package/dist/core/extensions/runner.js.map +1 -0
- package/dist/core/extensions/types.d.ts +984 -0
- package/dist/core/extensions/types.d.ts.map +1 -0
- package/dist/core/extensions/types.js +35 -0
- package/dist/core/extensions/types.js.map +1 -0
- package/dist/core/extensions/wrapper.d.ts +27 -0
- package/dist/core/extensions/wrapper.d.ts.map +1 -0
- package/dist/core/extensions/wrapper.js +102 -0
- package/dist/core/extensions/wrapper.js.map +1 -0
- package/dist/core/footer-data-provider.d.ts +32 -0
- package/dist/core/footer-data-provider.d.ts.map +1 -0
- package/dist/core/footer-data-provider.js +134 -0
- package/dist/core/footer-data-provider.js.map +1 -0
- package/dist/core/index.d.ts +9 -0
- package/dist/core/index.d.ts.map +1 -0
- package/dist/core/index.js +9 -0
- package/dist/core/index.js.map +1 -0
- package/dist/core/keybindings.d.ts +55 -0
- package/dist/core/keybindings.d.ts.map +1 -0
- package/dist/core/keybindings.js +153 -0
- package/dist/core/keybindings.js.map +1 -0
- package/dist/core/memory.d.ts +64 -0
- package/dist/core/memory.d.ts.map +1 -0
- package/dist/core/memory.js +247 -0
- package/dist/core/memory.js.map +1 -0
- package/dist/core/messages.d.ts +77 -0
- package/dist/core/messages.d.ts.map +1 -0
- package/dist/core/messages.js +149 -0
- package/dist/core/messages.js.map +1 -0
- package/dist/core/model-registry.d.ts +102 -0
- package/dist/core/model-registry.d.ts.map +1 -0
- package/dist/core/model-registry.js +515 -0
- package/dist/core/model-registry.js.map +1 -0
- package/dist/core/model-resolver.d.ts +104 -0
- package/dist/core/model-resolver.d.ts.map +1 -0
- package/dist/core/model-resolver.js +403 -0
- package/dist/core/model-resolver.js.map +1 -0
- package/dist/core/package-manager.d.ts +151 -0
- package/dist/core/package-manager.d.ts.map +1 -0
- package/dist/core/package-manager.js +1426 -0
- package/dist/core/package-manager.js.map +1 -0
- package/dist/core/prompt-templates.d.ts +50 -0
- package/dist/core/prompt-templates.d.ts.map +1 -0
- package/dist/core/prompt-templates.js +251 -0
- package/dist/core/prompt-templates.js.map +1 -0
- package/dist/core/resolve-config-value.d.ts +17 -0
- package/dist/core/resolve-config-value.d.ts.map +1 -0
- package/dist/core/resolve-config-value.js +59 -0
- package/dist/core/resolve-config-value.js.map +1 -0
- package/dist/core/resource-loader.d.ts +184 -0
- package/dist/core/resource-loader.d.ts.map +1 -0
- package/dist/core/resource-loader.js +673 -0
- package/dist/core/resource-loader.js.map +1 -0
- package/dist/core/sdk.d.ts +90 -0
- package/dist/core/sdk.d.ts.map +1 -0
- package/dist/core/sdk.js +238 -0
- package/dist/core/sdk.js.map +1 -0
- package/dist/core/session-manager.d.ts +323 -0
- package/dist/core/session-manager.d.ts.map +1 -0
- package/dist/core/session-manager.js +1091 -0
- package/dist/core/session-manager.js.map +1 -0
- package/dist/core/settings-manager.d.ts +230 -0
- package/dist/core/settings-manager.d.ts.map +1 -0
- package/dist/core/settings-manager.js +656 -0
- package/dist/core/settings-manager.js.map +1 -0
- package/dist/core/skills.d.ts +58 -0
- package/dist/core/skills.d.ts.map +1 -0
- package/dist/core/skills.js +364 -0
- package/dist/core/skills.js.map +1 -0
- package/dist/core/slash-commands.d.ts +15 -0
- package/dist/core/slash-commands.d.ts.map +1 -0
- package/dist/core/slash-commands.js +23 -0
- package/dist/core/slash-commands.js.map +1 -0
- package/dist/core/system-prompt.d.ts +26 -0
- package/dist/core/system-prompt.d.ts.map +1 -0
- package/dist/core/system-prompt.js +150 -0
- package/dist/core/system-prompt.js.map +1 -0
- package/dist/core/timings.d.ts +7 -0
- package/dist/core/timings.d.ts.map +1 -0
- package/dist/core/timings.js +25 -0
- package/dist/core/timings.js.map +1 -0
- package/dist/core/tools/bash.d.ts +55 -0
- package/dist/core/tools/bash.d.ts.map +1 -0
- package/dist/core/tools/bash.js +242 -0
- package/dist/core/tools/bash.js.map +1 -0
- package/dist/core/tools/edit-diff.d.ts +63 -0
- package/dist/core/tools/edit-diff.d.ts.map +1 -0
- package/dist/core/tools/edit-diff.js +243 -0
- package/dist/core/tools/edit-diff.js.map +1 -0
- package/dist/core/tools/edit.d.ts +39 -0
- package/dist/core/tools/edit.d.ts.map +1 -0
- package/dist/core/tools/edit.js +146 -0
- package/dist/core/tools/edit.js.map +1 -0
- package/dist/core/tools/find.d.ts +39 -0
- package/dist/core/tools/find.d.ts.map +1 -0
- package/dist/core/tools/find.js +206 -0
- package/dist/core/tools/find.js.map +1 -0
- package/dist/core/tools/grep.d.ts +45 -0
- package/dist/core/tools/grep.d.ts.map +1 -0
- package/dist/core/tools/grep.js +239 -0
- package/dist/core/tools/grep.js.map +1 -0
- package/dist/core/tools/index.d.ts +102 -0
- package/dist/core/tools/index.d.ts.map +1 -0
- package/dist/core/tools/index.js +71 -0
- package/dist/core/tools/index.js.map +1 -0
- package/dist/core/tools/ls.d.ts +40 -0
- package/dist/core/tools/ls.d.ts.map +1 -0
- package/dist/core/tools/ls.js +118 -0
- package/dist/core/tools/ls.js.map +1 -0
- package/dist/core/tools/memory.d.ts +45 -0
- package/dist/core/tools/memory.d.ts.map +1 -0
- package/dist/core/tools/memory.js +346 -0
- package/dist/core/tools/memory.js.map +1 -0
- package/dist/core/tools/path-utils.d.ts +8 -0
- package/dist/core/tools/path-utils.d.ts.map +1 -0
- package/dist/core/tools/path-utils.js +81 -0
- package/dist/core/tools/path-utils.js.map +1 -0
- package/dist/core/tools/read.d.ts +39 -0
- package/dist/core/tools/read.d.ts.map +1 -0
- package/dist/core/tools/read.js +166 -0
- package/dist/core/tools/read.js.map +1 -0
- package/dist/core/tools/truncate.d.ts +70 -0
- package/dist/core/tools/truncate.d.ts.map +1 -0
- package/dist/core/tools/truncate.js +205 -0
- package/dist/core/tools/truncate.js.map +1 -0
- package/dist/core/tools/write.d.ts +29 -0
- package/dist/core/tools/write.d.ts.map +1 -0
- package/dist/core/tools/write.js +78 -0
- package/dist/core/tools/write.js.map +1 -0
- package/dist/index.d.ts +28 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +43 -0
- package/dist/index.js.map +1 -0
- package/dist/main.d.ts +8 -0
- package/dist/main.d.ts.map +1 -0
- package/dist/main.js +651 -0
- package/dist/main.js.map +1 -0
- package/dist/migrations.d.ts +33 -0
- package/dist/migrations.d.ts.map +1 -0
- package/dist/migrations.js +261 -0
- package/dist/migrations.js.map +1 -0
- package/dist/modes/index.d.ts +9 -0
- package/dist/modes/index.d.ts.map +1 -0
- package/dist/modes/index.js +8 -0
- package/dist/modes/index.js.map +1 -0
- package/dist/modes/interactive/components/armin.d.ts +34 -0
- package/dist/modes/interactive/components/armin.d.ts.map +1 -0
- package/dist/modes/interactive/components/armin.js +333 -0
- package/dist/modes/interactive/components/armin.js.map +1 -0
- package/dist/modes/interactive/components/assistant-message.d.ts +16 -0
- package/dist/modes/interactive/components/assistant-message.d.ts.map +1 -0
- package/dist/modes/interactive/components/assistant-message.js +96 -0
- package/dist/modes/interactive/components/assistant-message.js.map +1 -0
- package/dist/modes/interactive/components/bash-execution.d.ts +35 -0
- package/dist/modes/interactive/components/bash-execution.d.ts.map +1 -0
- package/dist/modes/interactive/components/bash-execution.js +162 -0
- package/dist/modes/interactive/components/bash-execution.js.map +1 -0
- package/dist/modes/interactive/components/bordered-loader.d.ts +16 -0
- package/dist/modes/interactive/components/bordered-loader.d.ts.map +1 -0
- package/dist/modes/interactive/components/bordered-loader.js +51 -0
- package/dist/modes/interactive/components/bordered-loader.js.map +1 -0
- package/dist/modes/interactive/components/branch-summary-message.d.ts +16 -0
- package/dist/modes/interactive/components/branch-summary-message.d.ts.map +1 -0
- package/dist/modes/interactive/components/branch-summary-message.js +44 -0
- package/dist/modes/interactive/components/branch-summary-message.js.map +1 -0
- package/dist/modes/interactive/components/compaction-summary-message.d.ts +16 -0
- package/dist/modes/interactive/components/compaction-summary-message.d.ts.map +1 -0
- package/dist/modes/interactive/components/compaction-summary-message.js +45 -0
- package/dist/modes/interactive/components/compaction-summary-message.js.map +1 -0
- package/dist/modes/interactive/components/config-selector.d.ts +71 -0
- package/dist/modes/interactive/components/config-selector.d.ts.map +1 -0
- package/dist/modes/interactive/components/config-selector.js +479 -0
- package/dist/modes/interactive/components/config-selector.js.map +1 -0
- package/dist/modes/interactive/components/countdown-timer.d.ts +14 -0
- package/dist/modes/interactive/components/countdown-timer.d.ts.map +1 -0
- package/dist/modes/interactive/components/countdown-timer.js +33 -0
- package/dist/modes/interactive/components/countdown-timer.js.map +1 -0
- package/dist/modes/interactive/components/custom-editor.d.ts +21 -0
- package/dist/modes/interactive/components/custom-editor.d.ts.map +1 -0
- package/dist/modes/interactive/components/custom-editor.js +70 -0
- package/dist/modes/interactive/components/custom-editor.js.map +1 -0
- package/dist/modes/interactive/components/custom-message.d.ts +20 -0
- package/dist/modes/interactive/components/custom-message.d.ts.map +1 -0
- package/dist/modes/interactive/components/custom-message.js +79 -0
- package/dist/modes/interactive/components/custom-message.js.map +1 -0
- package/dist/modes/interactive/components/daxnuts.d.ts +23 -0
- package/dist/modes/interactive/components/daxnuts.d.ts.map +1 -0
- package/dist/modes/interactive/components/daxnuts.js +140 -0
- package/dist/modes/interactive/components/daxnuts.js.map +1 -0
- package/dist/modes/interactive/components/diff.d.ts +12 -0
- package/dist/modes/interactive/components/diff.d.ts.map +1 -0
- package/dist/modes/interactive/components/diff.js +133 -0
- package/dist/modes/interactive/components/diff.js.map +1 -0
- package/dist/modes/interactive/components/dynamic-border.d.ts +15 -0
- package/dist/modes/interactive/components/dynamic-border.d.ts.map +1 -0
- package/dist/modes/interactive/components/dynamic-border.js +21 -0
- package/dist/modes/interactive/components/dynamic-border.js.map +1 -0
- package/dist/modes/interactive/components/extension-editor.d.ts +17 -0
- package/dist/modes/interactive/components/extension-editor.d.ts.map +1 -0
- package/dist/modes/interactive/components/extension-editor.js +102 -0
- package/dist/modes/interactive/components/extension-editor.js.map +1 -0
- package/dist/modes/interactive/components/extension-input.d.ts +23 -0
- package/dist/modes/interactive/components/extension-input.d.ts.map +1 -0
- package/dist/modes/interactive/components/extension-input.js +61 -0
- package/dist/modes/interactive/components/extension-input.js.map +1 -0
- package/dist/modes/interactive/components/extension-selector.d.ts +24 -0
- package/dist/modes/interactive/components/extension-selector.d.ts.map +1 -0
- package/dist/modes/interactive/components/extension-selector.js +78 -0
- package/dist/modes/interactive/components/extension-selector.js.map +1 -0
- package/dist/modes/interactive/components/footer.d.ts +26 -0
- package/dist/modes/interactive/components/footer.d.ts.map +1 -0
- package/dist/modes/interactive/components/footer.js +213 -0
- package/dist/modes/interactive/components/footer.js.map +1 -0
- package/dist/modes/interactive/components/index.d.ts +32 -0
- package/dist/modes/interactive/components/index.d.ts.map +1 -0
- package/dist/modes/interactive/components/index.js +33 -0
- package/dist/modes/interactive/components/index.js.map +1 -0
- package/dist/modes/interactive/components/keybinding-hints.d.ts +41 -0
- package/dist/modes/interactive/components/keybinding-hints.d.ts.map +1 -0
- package/dist/modes/interactive/components/keybinding-hints.js +61 -0
- package/dist/modes/interactive/components/keybinding-hints.js.map +1 -0
- package/dist/modes/interactive/components/login-dialog.d.ts +42 -0
- package/dist/modes/interactive/components/login-dialog.d.ts.map +1 -0
- package/dist/modes/interactive/components/login-dialog.js +145 -0
- package/dist/modes/interactive/components/login-dialog.js.map +1 -0
- package/dist/modes/interactive/components/model-selector.d.ts +47 -0
- package/dist/modes/interactive/components/model-selector.d.ts.map +1 -0
- package/dist/modes/interactive/components/model-selector.js +271 -0
- package/dist/modes/interactive/components/model-selector.js.map +1 -0
- package/dist/modes/interactive/components/oauth-selector.d.ts +19 -0
- package/dist/modes/interactive/components/oauth-selector.d.ts.map +1 -0
- package/dist/modes/interactive/components/oauth-selector.js +97 -0
- package/dist/modes/interactive/components/oauth-selector.js.map +1 -0
- package/dist/modes/interactive/components/scoped-models-selector.d.ts +49 -0
- package/dist/modes/interactive/components/scoped-models-selector.d.ts.map +1 -0
- package/dist/modes/interactive/components/scoped-models-selector.js +275 -0
- package/dist/modes/interactive/components/scoped-models-selector.js.map +1 -0
- package/dist/modes/interactive/components/session-selector-search.d.ts +23 -0
- package/dist/modes/interactive/components/session-selector-search.d.ts.map +1 -0
- package/dist/modes/interactive/components/session-selector-search.js +155 -0
- package/dist/modes/interactive/components/session-selector-search.js.map +1 -0
- package/dist/modes/interactive/components/session-selector.d.ts +95 -0
- package/dist/modes/interactive/components/session-selector.d.ts.map +1 -0
- package/dist/modes/interactive/components/session-selector.js +851 -0
- package/dist/modes/interactive/components/session-selector.js.map +1 -0
- package/dist/modes/interactive/components/settings-selector.d.ts +58 -0
- package/dist/modes/interactive/components/settings-selector.d.ts.map +1 -0
- package/dist/modes/interactive/components/settings-selector.js +299 -0
- package/dist/modes/interactive/components/settings-selector.js.map +1 -0
- package/dist/modes/interactive/components/show-images-selector.d.ts +10 -0
- package/dist/modes/interactive/components/show-images-selector.d.ts.map +1 -0
- package/dist/modes/interactive/components/show-images-selector.js +35 -0
- package/dist/modes/interactive/components/show-images-selector.js.map +1 -0
- package/dist/modes/interactive/components/skill-invocation-message.d.ts +17 -0
- package/dist/modes/interactive/components/skill-invocation-message.d.ts.map +1 -0
- package/dist/modes/interactive/components/skill-invocation-message.js +47 -0
- package/dist/modes/interactive/components/skill-invocation-message.js.map +1 -0
- package/dist/modes/interactive/components/theme-selector.d.ts +11 -0
- package/dist/modes/interactive/components/theme-selector.d.ts.map +1 -0
- package/dist/modes/interactive/components/theme-selector.js +46 -0
- package/dist/modes/interactive/components/theme-selector.js.map +1 -0
- package/dist/modes/interactive/components/thinking-selector.d.ts +11 -0
- package/dist/modes/interactive/components/thinking-selector.d.ts.map +1 -0
- package/dist/modes/interactive/components/thinking-selector.js +47 -0
- package/dist/modes/interactive/components/thinking-selector.js.map +1 -0
- package/dist/modes/interactive/components/tool-execution.d.ts +70 -0
- package/dist/modes/interactive/components/tool-execution.d.ts.map +1 -0
- package/dist/modes/interactive/components/tool-execution.js +636 -0
- package/dist/modes/interactive/components/tool-execution.js.map +1 -0
- package/dist/modes/interactive/components/tree-selector.d.ts +68 -0
- package/dist/modes/interactive/components/tree-selector.d.ts.map +1 -0
- package/dist/modes/interactive/components/tree-selector.js +934 -0
- package/dist/modes/interactive/components/tree-selector.js.map +1 -0
- package/dist/modes/interactive/components/user-message-selector.d.ts +30 -0
- package/dist/modes/interactive/components/user-message-selector.d.ts.map +1 -0
- package/dist/modes/interactive/components/user-message-selector.js +113 -0
- package/dist/modes/interactive/components/user-message-selector.js.map +1 -0
- package/dist/modes/interactive/components/user-message.d.ts +8 -0
- package/dist/modes/interactive/components/user-message.d.ts.map +1 -0
- package/dist/modes/interactive/components/user-message.js +16 -0
- package/dist/modes/interactive/components/user-message.js.map +1 -0
- package/dist/modes/interactive/components/visual-truncate.d.ts +24 -0
- package/dist/modes/interactive/components/visual-truncate.d.ts.map +1 -0
- package/dist/modes/interactive/components/visual-truncate.js +33 -0
- package/dist/modes/interactive/components/visual-truncate.js.map +1 -0
- package/dist/modes/interactive/interactive-mode.d.ts +316 -0
- package/dist/modes/interactive/interactive-mode.d.ts.map +1 -0
- package/dist/modes/interactive/interactive-mode.js +3848 -0
- package/dist/modes/interactive/interactive-mode.js.map +1 -0
- package/dist/modes/interactive/theme/dark.json +85 -0
- package/dist/modes/interactive/theme/light.json +84 -0
- package/dist/modes/interactive/theme/theme-schema.json +335 -0
- package/dist/modes/interactive/theme/theme.d.ts +78 -0
- package/dist/modes/interactive/theme/theme.d.ts.map +1 -0
- package/dist/modes/interactive/theme/theme.js +944 -0
- package/dist/modes/interactive/theme/theme.js.map +1 -0
- package/dist/modes/print-mode.d.ts +28 -0
- package/dist/modes/print-mode.d.ts.map +1 -0
- package/dist/modes/print-mode.js +101 -0
- package/dist/modes/print-mode.js.map +1 -0
- package/dist/modes/rpc/rpc-client.d.ts +217 -0
- package/dist/modes/rpc/rpc-client.d.ts.map +1 -0
- package/dist/modes/rpc/rpc-client.js +405 -0
- package/dist/modes/rpc/rpc-client.js.map +1 -0
- package/dist/modes/rpc/rpc-mode.d.ts +20 -0
- package/dist/modes/rpc/rpc-mode.d.ts.map +1 -0
- package/dist/modes/rpc/rpc-mode.js +511 -0
- package/dist/modes/rpc/rpc-mode.js.map +1 -0
- package/dist/modes/rpc/rpc-types.d.ts +409 -0
- package/dist/modes/rpc/rpc-types.d.ts.map +1 -0
- package/dist/modes/rpc/rpc-types.js +8 -0
- package/dist/modes/rpc/rpc-types.js.map +1 -0
- package/dist/utils/changelog.d.ts +21 -0
- package/dist/utils/changelog.d.ts.map +1 -0
- package/dist/utils/changelog.js +87 -0
- package/dist/utils/changelog.js.map +1 -0
- package/dist/utils/clipboard-image.d.ts +11 -0
- package/dist/utils/clipboard-image.d.ts.map +1 -0
- package/dist/utils/clipboard-image.js +162 -0
- package/dist/utils/clipboard-image.js.map +1 -0
- package/dist/utils/clipboard-native.d.ts +7 -0
- package/dist/utils/clipboard-native.d.ts.map +1 -0
- package/dist/utils/clipboard-native.js +14 -0
- package/dist/utils/clipboard-native.js.map +1 -0
- package/dist/utils/clipboard.d.ts +2 -0
- package/dist/utils/clipboard.d.ts.map +1 -0
- package/dist/utils/clipboard.js +67 -0
- package/dist/utils/clipboard.js.map +1 -0
- package/dist/utils/frontmatter.d.ts +8 -0
- package/dist/utils/frontmatter.d.ts.map +1 -0
- package/dist/utils/frontmatter.js +26 -0
- package/dist/utils/frontmatter.js.map +1 -0
- package/dist/utils/git.d.ts +26 -0
- package/dist/utils/git.d.ts.map +1 -0
- package/dist/utils/git.js +163 -0
- package/dist/utils/git.js.map +1 -0
- package/dist/utils/image-convert.d.ts +9 -0
- package/dist/utils/image-convert.d.ts.map +1 -0
- package/dist/utils/image-convert.js +35 -0
- package/dist/utils/image-convert.js.map +1 -0
- package/dist/utils/image-resize.d.ts +36 -0
- package/dist/utils/image-resize.d.ts.map +1 -0
- package/dist/utils/image-resize.js +181 -0
- package/dist/utils/image-resize.js.map +1 -0
- package/dist/utils/mime.d.ts +2 -0
- package/dist/utils/mime.d.ts.map +1 -0
- package/dist/utils/mime.js +26 -0
- package/dist/utils/mime.js.map +1 -0
- package/dist/utils/photon.d.ts +21 -0
- package/dist/utils/photon.d.ts.map +1 -0
- package/dist/utils/photon.js +121 -0
- package/dist/utils/photon.js.map +1 -0
- package/dist/utils/shell.d.ts +26 -0
- package/dist/utils/shell.d.ts.map +1 -0
- package/dist/utils/shell.js +186 -0
- package/dist/utils/shell.js.map +1 -0
- package/dist/utils/sleep.d.ts +5 -0
- package/dist/utils/sleep.d.ts.map +1 -0
- package/dist/utils/sleep.js +17 -0
- package/dist/utils/sleep.js.map +1 -0
- package/dist/utils/tools-manager.d.ts +3 -0
- package/dist/utils/tools-manager.d.ts.map +1 -0
- package/dist/utils/tools-manager.js +207 -0
- package/dist/utils/tools-manager.js.map +1 -0
- package/docs/compaction.md +390 -0
- package/docs/custom-provider.md +548 -0
- package/docs/development.md +69 -0
- package/docs/extensions.md +1935 -0
- package/docs/images/doom-extension.png +0 -0
- package/docs/images/exy.png +0 -0
- package/docs/images/interactive-mode.png +0 -0
- package/docs/images/tree-view.png +0 -0
- package/docs/json.md +79 -0
- package/docs/keybindings.md +174 -0
- package/docs/models.md +293 -0
- package/docs/packages.md +209 -0
- package/docs/prompt-templates.md +67 -0
- package/docs/providers.md +186 -0
- package/docs/rpc.md +1317 -0
- package/docs/sdk.md +968 -0
- package/docs/session.md +412 -0
- package/docs/settings.md +223 -0
- package/docs/shell-aliases.md +13 -0
- package/docs/skills.md +231 -0
- package/docs/terminal-setup.md +70 -0
- package/docs/termux.md +127 -0
- package/docs/themes.md +295 -0
- package/docs/tree.md +219 -0
- package/docs/tui.md +887 -0
- package/docs/windows.md +17 -0
- package/examples/README.md +25 -0
- package/examples/extensions/README.md +203 -0
- package/examples/extensions/antigravity-image-gen.ts +413 -0
- package/examples/extensions/auto-commit-on-exit.ts +49 -0
- package/examples/extensions/bash-spawn-hook.ts +30 -0
- package/examples/extensions/bookmark.ts +50 -0
- package/examples/extensions/claude-rules.ts +86 -0
- package/examples/extensions/commands.ts +72 -0
- package/examples/extensions/confirm-destructive.ts +59 -0
- package/examples/extensions/custom-compaction.ts +114 -0
- package/examples/extensions/custom-footer.ts +64 -0
- package/examples/extensions/custom-header.ts +73 -0
- package/examples/extensions/custom-provider-anthropic/index.ts +604 -0
- package/examples/extensions/custom-provider-anthropic/package-lock.json +24 -0
- package/examples/extensions/custom-provider-anthropic/package.json +19 -0
- package/examples/extensions/custom-provider-gitlab-duo/index.ts +349 -0
- package/examples/extensions/custom-provider-gitlab-duo/package.json +16 -0
- package/examples/extensions/custom-provider-gitlab-duo/test.ts +82 -0
- package/examples/extensions/custom-provider-qwen-cli/index.ts +345 -0
- package/examples/extensions/custom-provider-qwen-cli/package.json +16 -0
- package/examples/extensions/dirty-repo-guard.ts +56 -0
- package/examples/extensions/doom-overlay/README.md +46 -0
- package/examples/extensions/doom-overlay/doom/build/doom.js +21 -0
- package/examples/extensions/doom-overlay/doom/build/doom.wasm +0 -0
- package/examples/extensions/doom-overlay/doom/build.sh +152 -0
- package/examples/extensions/doom-overlay/doom/doomgeneric_pi.c +72 -0
- package/examples/extensions/doom-overlay/doom-component.ts +132 -0
- package/examples/extensions/doom-overlay/doom-engine.ts +173 -0
- package/examples/extensions/doom-overlay/doom-keys.ts +104 -0
- package/examples/extensions/doom-overlay/index.ts +74 -0
- package/examples/extensions/doom-overlay/wad-finder.ts +51 -0
- package/examples/extensions/dynamic-resources/SKILL.md +8 -0
- package/examples/extensions/dynamic-resources/dynamic.json +79 -0
- package/examples/extensions/dynamic-resources/dynamic.md +5 -0
- package/examples/extensions/dynamic-resources/index.ts +15 -0
- package/examples/extensions/event-bus.ts +43 -0
- package/examples/extensions/file-trigger.ts +41 -0
- package/examples/extensions/git-checkpoint.ts +53 -0
- package/examples/extensions/handoff.ts +150 -0
- package/examples/extensions/hello.ts +25 -0
- package/examples/extensions/inline-bash.ts +94 -0
- package/examples/extensions/input-transform.ts +43 -0
- package/examples/extensions/interactive-shell.ts +196 -0
- package/examples/extensions/mac-system-theme.ts +47 -0
- package/examples/extensions/message-renderer.ts +59 -0
- package/examples/extensions/minimal-mode.ts +426 -0
- package/examples/extensions/modal-editor.ts +85 -0
- package/examples/extensions/model-status.ts +31 -0
- package/examples/extensions/notify.ts +55 -0
- package/examples/extensions/overlay-qa-tests.ts +881 -0
- package/examples/extensions/overlay-test.ts +150 -0
- package/examples/extensions/permission-gate.ts +34 -0
- package/examples/extensions/pirate.ts +47 -0
- package/examples/extensions/plan-mode/README.md +65 -0
- package/examples/extensions/plan-mode/index.ts +340 -0
- package/examples/extensions/plan-mode/utils.ts +168 -0
- package/examples/extensions/preset.ts +398 -0
- package/examples/extensions/protected-paths.ts +30 -0
- package/examples/extensions/qna.ts +119 -0
- package/examples/extensions/question.ts +264 -0
- package/examples/extensions/questionnaire.ts +427 -0
- package/examples/extensions/rainbow-editor.ts +88 -0
- package/examples/extensions/reload-runtime.ts +37 -0
- package/examples/extensions/rpc-demo.ts +124 -0
- package/examples/extensions/sandbox/index.ts +318 -0
- package/examples/extensions/sandbox/package-lock.json +92 -0
- package/examples/extensions/sandbox/package.json +19 -0
- package/examples/extensions/send-user-message.ts +97 -0
- package/examples/extensions/session-name.ts +27 -0
- package/examples/extensions/shutdown-command.ts +63 -0
- package/examples/extensions/snake.ts +343 -0
- package/examples/extensions/space-invaders.ts +560 -0
- package/examples/extensions/ssh.ts +220 -0
- package/examples/extensions/status-line.ts +40 -0
- package/examples/extensions/subagent/README.md +172 -0
- package/examples/extensions/subagent/agents/planner.md +37 -0
- package/examples/extensions/subagent/agents/reviewer.md +35 -0
- package/examples/extensions/subagent/agents/scout.md +50 -0
- package/examples/extensions/subagent/agents/worker.md +24 -0
- package/examples/extensions/subagent/agents.ts +127 -0
- package/examples/extensions/subagent/index.ts +964 -0
- package/examples/extensions/subagent/prompts/implement-and-review.md +10 -0
- package/examples/extensions/subagent/prompts/implement.md +10 -0
- package/examples/extensions/subagent/prompts/scout-and-plan.md +9 -0
- package/examples/extensions/summarize.ts +195 -0
- package/examples/extensions/system-prompt-header.ts +17 -0
- package/examples/extensions/timed-confirm.ts +70 -0
- package/examples/extensions/titlebar-spinner.ts +58 -0
- package/examples/extensions/todo.ts +299 -0
- package/examples/extensions/tool-override.ts +143 -0
- package/examples/extensions/tools.ts +146 -0
- package/examples/extensions/trigger-compact.ts +40 -0
- package/examples/extensions/truncated-tool.ts +192 -0
- package/examples/extensions/widget-placement.ts +17 -0
- package/examples/extensions/with-deps/index.ts +36 -0
- package/examples/extensions/with-deps/package-lock.json +31 -0
- package/examples/extensions/with-deps/package.json +22 -0
- package/examples/rpc-extension-ui.ts +632 -0
- package/examples/sdk/01-minimal.ts +22 -0
- package/examples/sdk/02-custom-model.ts +49 -0
- package/examples/sdk/03-custom-prompt.ts +55 -0
- package/examples/sdk/04-skills.ts +46 -0
- package/examples/sdk/05-tools.ts +56 -0
- package/examples/sdk/06-extensions.ts +88 -0
- package/examples/sdk/07-context-files.ts +40 -0
- package/examples/sdk/08-prompt-templates.ts +47 -0
- package/examples/sdk/09-api-keys-and-oauth.ts +48 -0
- package/examples/sdk/10-settings.ts +51 -0
- package/examples/sdk/11-sessions.ts +48 -0
- package/examples/sdk/12-full-control.ts +82 -0
- package/examples/sdk/README.md +144 -0
- package/package.json +96 -0
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"settings-manager.js","sourceRoot":"","sources":["../../src/core/settings-manager.ts"],"names":[],"mappings":"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","sourcesContent":["import type { Transport } from \"@mariozechner/pi-ai\";\r\nimport { existsSync, mkdirSync, readFileSync, writeFileSync } from \"fs\";\r\nimport { dirname, join } from \"path\";\r\nimport lockfile from \"proper-lockfile\";\r\nimport { CONFIG_DIR_NAME, getAgentDir } from \"../config.js\";\r\n\r\nexport interface CompactionSettings {\r\n\tenabled?: boolean; // default: true\r\n\treserveTokens?: number; // default: 16384\r\n\tkeepRecentTokens?: number; // default: 20000\r\n}\r\n\r\nexport interface BranchSummarySettings {\r\n\treserveTokens?: number; // default: 16384 (tokens reserved for prompt + LLM response)\r\n}\r\n\r\nexport interface RetrySettings {\r\n\tenabled?: boolean; // default: true\r\n\tmaxRetries?: number; // default: 3\r\n\tbaseDelayMs?: number; // default: 2000 (exponential backoff: 2s, 4s, 8s)\r\n\tmaxDelayMs?: number; // default: 60000 (max server-requested delay before failing)\r\n}\r\n\r\nexport interface TerminalSettings {\r\n\tshowImages?: boolean; // default: true (only relevant if terminal supports images)\r\n\tclearOnShrink?: boolean; // default: false (clear empty rows when content shrinks)\r\n}\r\n\r\nexport interface ImageSettings {\r\n\tautoResize?: boolean; // default: true (resize images to 2000x2000 max for better model compatibility)\r\n\tblockImages?: boolean; // default: false - when true, prevents all images from being sent to LLM providers\r\n}\r\n\r\nexport interface ThinkingBudgetsSettings {\r\n\tminimal?: number;\r\n\tlow?: number;\r\n\tmedium?: number;\r\n\thigh?: number;\r\n}\r\n\r\nexport interface MarkdownSettings {\r\n\tcodeBlockIndent?: string; // default: \" \"\r\n}\r\n\r\nexport type TransportSetting = Transport;\r\n\r\n/**\r\n * Package source for npm/git packages.\r\n * - String form: load all resources from the package\r\n * - Object form: filter which resources to load\r\n */\r\nexport type PackageSource =\r\n\t| string\r\n\t| {\r\n\t\t\tsource: string;\r\n\t\t\textensions?: string[];\r\n\t\t\tskills?: string[];\r\n\t\t\tprompts?: string[];\r\n\t\t\tthemes?: string[];\r\n\t };\r\n\r\nexport interface Settings {\r\n\tlastChangelogVersion?: string;\r\n\tdefaultProvider?: string;\r\n\tdefaultModel?: string;\r\n\tdefaultThinkingLevel?: \"off\" | \"minimal\" | \"low\" | \"medium\" | \"high\" | \"xhigh\";\r\n\ttransport?: TransportSetting; // default: \"sse\"\r\n\tsteeringMode?: \"all\" | \"one-at-a-time\";\r\n\tfollowUpMode?: \"all\" | \"one-at-a-time\";\r\n\ttheme?: string;\r\n\tcompaction?: CompactionSettings;\r\n\tbranchSummary?: BranchSummarySettings;\r\n\tretry?: RetrySettings;\r\n\thideThinkingBlock?: boolean;\r\n\tshellPath?: string; // Custom shell path (e.g., for Cygwin users on Windows)\r\n\tquietStartup?: boolean;\r\n\tshellCommandPrefix?: string; // Prefix prepended to every bash command (e.g., \"shopt -s expand_aliases\" for alias support)\r\n\tcollapseChangelog?: boolean; // Show condensed changelog after update (use /changelog for full)\r\n\tpackages?: PackageSource[]; // Array of npm/git package sources (string or object with filtering)\r\n\textensions?: string[]; // Array of local extension file paths or directories\r\n\tskills?: string[]; // Array of local skill file paths or directories\r\n\tprompts?: string[]; // Array of local prompt template paths or directories\r\n\tthemes?: string[]; // Array of local theme file paths or directories\r\n\tenableSkillCommands?: boolean; // default: true - register skills as /skill:name commands\r\n\tterminal?: TerminalSettings;\r\n\timages?: ImageSettings;\r\n\tenabledModels?: string[]; // Model patterns for cycling (same format as --models CLI flag)\r\n\tdoubleEscapeAction?: \"fork\" | \"tree\" | \"none\"; // Action for double-escape with empty editor (default: \"tree\")\r\n\tthinkingBudgets?: ThinkingBudgetsSettings; // Custom token budgets for thinking levels\r\n\teditorPaddingX?: number; // Horizontal padding for input editor (default: 0)\r\n\tautocompleteMaxVisible?: number; // Max visible items in autocomplete dropdown (default: 5)\r\n\tshowHardwareCursor?: boolean; // Show terminal cursor while still positioning it for IME\r\n\tmarkdown?: MarkdownSettings;\r\n\tmemory?: { enabled?: boolean }; // default: true - persistent memory system (procedural, episodic, semantic)\r\n}\r\n\r\n/** Deep merge settings: project/overrides take precedence, nested objects merge recursively */\r\nfunction deepMergeSettings(base: Settings, overrides: Settings): Settings {\r\n\tconst result: Settings = { ...base };\r\n\r\n\tfor (const key of Object.keys(overrides) as (keyof Settings)[]) {\r\n\t\tconst overrideValue = overrides[key];\r\n\t\tconst baseValue = base[key];\r\n\r\n\t\tif (overrideValue === undefined) {\r\n\t\t\tcontinue;\r\n\t\t}\r\n\r\n\t\t// For nested objects, merge recursively\r\n\t\tif (\r\n\t\t\ttypeof overrideValue === \"object\" &&\r\n\t\t\toverrideValue !== null &&\r\n\t\t\t!Array.isArray(overrideValue) &&\r\n\t\t\ttypeof baseValue === \"object\" &&\r\n\t\t\tbaseValue !== null &&\r\n\t\t\t!Array.isArray(baseValue)\r\n\t\t) {\r\n\t\t\t(result as Record<string, unknown>)[key] = { ...baseValue, ...overrideValue };\r\n\t\t} else {\r\n\t\t\t// For primitives and arrays, override value wins\r\n\t\t\t(result as Record<string, unknown>)[key] = overrideValue;\r\n\t\t}\r\n\t}\r\n\r\n\treturn result;\r\n}\r\n\r\nexport type SettingsScope = \"global\" | \"project\";\r\n\r\nexport interface SettingsStorage {\r\n\twithLock(scope: SettingsScope, fn: (current: string | undefined) => string | undefined): void;\r\n}\r\n\r\nexport interface SettingsError {\r\n\tscope: SettingsScope;\r\n\terror: Error;\r\n}\r\n\r\nexport class FileSettingsStorage implements SettingsStorage {\r\n\tprivate globalSettingsPath: string;\r\n\tprivate projectSettingsPath: string;\r\n\r\n\tconstructor(cwd: string = process.cwd(), agentDir: string = getAgentDir()) {\r\n\t\tthis.globalSettingsPath = join(agentDir, \"settings.json\");\r\n\t\tthis.projectSettingsPath = join(cwd, CONFIG_DIR_NAME, \"settings.json\");\r\n\t}\r\n\r\n\twithLock(scope: SettingsScope, fn: (current: string | undefined) => string | undefined): void {\r\n\t\tconst path = scope === \"global\" ? this.globalSettingsPath : this.projectSettingsPath;\r\n\t\tconst dir = dirname(path);\r\n\t\tif (!existsSync(dir)) {\r\n\t\t\tmkdirSync(dir, { recursive: true });\r\n\t\t}\r\n\r\n\t\tlet release: (() => void) | undefined;\r\n\t\ttry {\r\n\t\t\trelease = lockfile.lockSync(path, { realpath: false });\r\n\t\t\tconst current = existsSync(path) ? readFileSync(path, \"utf-8\") : undefined;\r\n\t\t\tconst next = fn(current);\r\n\t\t\tif (next !== undefined) {\r\n\t\t\t\twriteFileSync(path, next, \"utf-8\");\r\n\t\t\t}\r\n\t\t} finally {\r\n\t\t\tif (release) {\r\n\t\t\t\trelease();\r\n\t\t\t}\r\n\t\t}\r\n\t}\r\n}\r\n\r\nexport class InMemorySettingsStorage implements SettingsStorage {\r\n\tprivate global: string | undefined;\r\n\tprivate project: string | undefined;\r\n\r\n\twithLock(scope: SettingsScope, fn: (current: string | undefined) => string | undefined): void {\r\n\t\tconst current = scope === \"global\" ? this.global : this.project;\r\n\t\tconst next = fn(current);\r\n\t\tif (next !== undefined) {\r\n\t\t\tif (scope === \"global\") {\r\n\t\t\t\tthis.global = next;\r\n\t\t\t} else {\r\n\t\t\t\tthis.project = next;\r\n\t\t\t}\r\n\t\t}\r\n\t}\r\n}\r\n\r\nexport class SettingsManager {\r\n\tprivate storage: SettingsStorage;\r\n\tprivate globalSettings: Settings;\r\n\tprivate projectSettings: Settings;\r\n\tprivate settings: Settings;\r\n\tprivate modifiedFields = new Set<keyof Settings>(); // Track global fields modified during session\r\n\tprivate modifiedNestedFields = new Map<keyof Settings, Set<string>>(); // Track global nested field modifications\r\n\tprivate modifiedProjectFields = new Set<keyof Settings>(); // Track project fields modified during session\r\n\tprivate modifiedProjectNestedFields = new Map<keyof Settings, Set<string>>(); // Track project nested field modifications\r\n\tprivate globalSettingsLoadError: Error | null = null; // Track if global settings file had parse errors\r\n\tprivate projectSettingsLoadError: Error | null = null; // Track if project settings file had parse errors\r\n\tprivate writeQueue: Promise<void> = Promise.resolve();\r\n\tprivate errors: SettingsError[];\r\n\r\n\tprivate constructor(\r\n\t\tstorage: SettingsStorage,\r\n\t\tinitialGlobal: Settings,\r\n\t\tinitialProject: Settings,\r\n\t\tglobalLoadError: Error | null = null,\r\n\t\tprojectLoadError: Error | null = null,\r\n\t\tinitialErrors: SettingsError[] = [],\r\n\t) {\r\n\t\tthis.storage = storage;\r\n\t\tthis.globalSettings = initialGlobal;\r\n\t\tthis.projectSettings = initialProject;\r\n\t\tthis.globalSettingsLoadError = globalLoadError;\r\n\t\tthis.projectSettingsLoadError = projectLoadError;\r\n\t\tthis.errors = [...initialErrors];\r\n\t\tthis.settings = deepMergeSettings(this.globalSettings, this.projectSettings);\r\n\t}\r\n\r\n\t/** Create a SettingsManager that loads from files */\r\n\tstatic create(cwd: string = process.cwd(), agentDir: string = getAgentDir()): SettingsManager {\r\n\t\tconst storage = new FileSettingsStorage(cwd, agentDir);\r\n\t\treturn SettingsManager.fromStorage(storage);\r\n\t}\r\n\r\n\t/** Create a SettingsManager from an arbitrary storage backend */\r\n\tstatic fromStorage(storage: SettingsStorage): SettingsManager {\r\n\t\tconst globalLoad = SettingsManager.tryLoadFromStorage(storage, \"global\");\r\n\t\tconst projectLoad = SettingsManager.tryLoadFromStorage(storage, \"project\");\r\n\t\tconst initialErrors: SettingsError[] = [];\r\n\t\tif (globalLoad.error) {\r\n\t\t\tinitialErrors.push({ scope: \"global\", error: globalLoad.error });\r\n\t\t}\r\n\t\tif (projectLoad.error) {\r\n\t\t\tinitialErrors.push({ scope: \"project\", error: projectLoad.error });\r\n\t\t}\r\n\r\n\t\treturn new SettingsManager(\r\n\t\t\tstorage,\r\n\t\t\tglobalLoad.settings,\r\n\t\t\tprojectLoad.settings,\r\n\t\t\tglobalLoad.error,\r\n\t\t\tprojectLoad.error,\r\n\t\t\tinitialErrors,\r\n\t\t);\r\n\t}\r\n\r\n\t/** Create an in-memory SettingsManager (no file I/O) */\r\n\tstatic inMemory(settings: Partial<Settings> = {}): SettingsManager {\r\n\t\tconst storage = new InMemorySettingsStorage();\r\n\t\treturn new SettingsManager(storage, settings, {});\r\n\t}\r\n\r\n\tprivate static loadFromStorage(storage: SettingsStorage, scope: SettingsScope): Settings {\r\n\t\tlet content: string | undefined;\r\n\t\tstorage.withLock(scope, (current) => {\r\n\t\t\tcontent = current;\r\n\t\t\treturn undefined;\r\n\t\t});\r\n\r\n\t\tif (!content) {\r\n\t\t\treturn {};\r\n\t\t}\r\n\t\tconst settings = JSON.parse(content);\r\n\t\treturn SettingsManager.migrateSettings(settings);\r\n\t}\r\n\r\n\tprivate static tryLoadFromStorage(\r\n\t\tstorage: SettingsStorage,\r\n\t\tscope: SettingsScope,\r\n\t): { settings: Settings; error: Error | null } {\r\n\t\ttry {\r\n\t\t\treturn { settings: SettingsManager.loadFromStorage(storage, scope), error: null };\r\n\t\t} catch (error) {\r\n\t\t\treturn { settings: {}, error: error as Error };\r\n\t\t}\r\n\t}\r\n\r\n\t/** Migrate old settings format to new format */\r\n\tprivate static migrateSettings(settings: Record<string, unknown>): Settings {\r\n\t\t// Migrate queueMode -> steeringMode\r\n\t\tif (\"queueMode\" in settings && !(\"steeringMode\" in settings)) {\r\n\t\t\tsettings.steeringMode = settings.queueMode;\r\n\t\t\tdelete settings.queueMode;\r\n\t\t}\r\n\r\n\t\t// Migrate legacy websockets boolean -> transport enum\r\n\t\tif (!(\"transport\" in settings) && typeof settings.websockets === \"boolean\") {\r\n\t\t\tsettings.transport = settings.websockets ? \"websocket\" : \"sse\";\r\n\t\t\tdelete settings.websockets;\r\n\t\t}\r\n\r\n\t\t// Migrate old skills object format to new array format\r\n\t\tif (\r\n\t\t\t\"skills\" in settings &&\r\n\t\t\ttypeof settings.skills === \"object\" &&\r\n\t\t\tsettings.skills !== null &&\r\n\t\t\t!Array.isArray(settings.skills)\r\n\t\t) {\r\n\t\t\tconst skillsSettings = settings.skills as {\r\n\t\t\t\tenableSkillCommands?: boolean;\r\n\t\t\t\tcustomDirectories?: unknown;\r\n\t\t\t};\r\n\t\t\tif (skillsSettings.enableSkillCommands !== undefined && settings.enableSkillCommands === undefined) {\r\n\t\t\t\tsettings.enableSkillCommands = skillsSettings.enableSkillCommands;\r\n\t\t\t}\r\n\t\t\tif (Array.isArray(skillsSettings.customDirectories) && skillsSettings.customDirectories.length > 0) {\r\n\t\t\t\tsettings.skills = skillsSettings.customDirectories;\r\n\t\t\t} else {\r\n\t\t\t\tdelete settings.skills;\r\n\t\t\t}\r\n\t\t}\r\n\r\n\t\treturn settings as Settings;\r\n\t}\r\n\r\n\tgetGlobalSettings(): Settings {\r\n\t\treturn structuredClone(this.globalSettings);\r\n\t}\r\n\r\n\tgetProjectSettings(): Settings {\r\n\t\treturn structuredClone(this.projectSettings);\r\n\t}\r\n\r\n\treload(): void {\r\n\t\tconst globalLoad = SettingsManager.tryLoadFromStorage(this.storage, \"global\");\r\n\t\tif (!globalLoad.error) {\r\n\t\t\tthis.globalSettings = globalLoad.settings;\r\n\t\t\tthis.globalSettingsLoadError = null;\r\n\t\t} else {\r\n\t\t\tthis.globalSettingsLoadError = globalLoad.error;\r\n\t\t\tthis.recordError(\"global\", globalLoad.error);\r\n\t\t}\r\n\r\n\t\tthis.modifiedFields.clear();\r\n\t\tthis.modifiedNestedFields.clear();\r\n\t\tthis.modifiedProjectFields.clear();\r\n\t\tthis.modifiedProjectNestedFields.clear();\r\n\r\n\t\tconst projectLoad = SettingsManager.tryLoadFromStorage(this.storage, \"project\");\r\n\t\tif (!projectLoad.error) {\r\n\t\t\tthis.projectSettings = projectLoad.settings;\r\n\t\t\tthis.projectSettingsLoadError = null;\r\n\t\t} else {\r\n\t\t\tthis.projectSettingsLoadError = projectLoad.error;\r\n\t\t\tthis.recordError(\"project\", projectLoad.error);\r\n\t\t}\r\n\r\n\t\tthis.settings = deepMergeSettings(this.globalSettings, this.projectSettings);\r\n\t}\r\n\r\n\t/** Apply additional overrides on top of current settings */\r\n\tapplyOverrides(overrides: Partial<Settings>): void {\r\n\t\tthis.settings = deepMergeSettings(this.settings, overrides);\r\n\t}\r\n\r\n\t/** Mark a global field as modified during this session */\r\n\tprivate markModified(field: keyof Settings, nestedKey?: string): void {\r\n\t\tthis.modifiedFields.add(field);\r\n\t\tif (nestedKey) {\r\n\t\t\tif (!this.modifiedNestedFields.has(field)) {\r\n\t\t\t\tthis.modifiedNestedFields.set(field, new Set());\r\n\t\t\t}\r\n\t\t\tthis.modifiedNestedFields.get(field)!.add(nestedKey);\r\n\t\t}\r\n\t}\r\n\r\n\t/** Mark a project field as modified during this session */\r\n\tprivate markProjectModified(field: keyof Settings, nestedKey?: string): void {\r\n\t\tthis.modifiedProjectFields.add(field);\r\n\t\tif (nestedKey) {\r\n\t\t\tif (!this.modifiedProjectNestedFields.has(field)) {\r\n\t\t\t\tthis.modifiedProjectNestedFields.set(field, new Set());\r\n\t\t\t}\r\n\t\t\tthis.modifiedProjectNestedFields.get(field)!.add(nestedKey);\r\n\t\t}\r\n\t}\r\n\r\n\tprivate recordError(scope: SettingsScope, error: unknown): void {\r\n\t\tconst normalizedError = error instanceof Error ? error : new Error(String(error));\r\n\t\tthis.errors.push({ scope, error: normalizedError });\r\n\t}\r\n\r\n\tprivate clearModifiedScope(scope: SettingsScope): void {\r\n\t\tif (scope === \"global\") {\r\n\t\t\tthis.modifiedFields.clear();\r\n\t\t\tthis.modifiedNestedFields.clear();\r\n\t\t\treturn;\r\n\t\t}\r\n\r\n\t\tthis.modifiedProjectFields.clear();\r\n\t\tthis.modifiedProjectNestedFields.clear();\r\n\t}\r\n\r\n\tprivate enqueueWrite(scope: SettingsScope, task: () => void): void {\r\n\t\tthis.writeQueue = this.writeQueue\r\n\t\t\t.then(() => {\r\n\t\t\t\ttask();\r\n\t\t\t\tthis.clearModifiedScope(scope);\r\n\t\t\t})\r\n\t\t\t.catch((error) => {\r\n\t\t\t\tthis.recordError(scope, error);\r\n\t\t\t});\r\n\t}\r\n\r\n\tprivate cloneModifiedNestedFields(source: Map<keyof Settings, Set<string>>): Map<keyof Settings, Set<string>> {\r\n\t\tconst snapshot = new Map<keyof Settings, Set<string>>();\r\n\t\tfor (const [key, value] of source.entries()) {\r\n\t\t\tsnapshot.set(key, new Set(value));\r\n\t\t}\r\n\t\treturn snapshot;\r\n\t}\r\n\r\n\tprivate persistScopedSettings(\r\n\t\tscope: SettingsScope,\r\n\t\tsnapshotSettings: Settings,\r\n\t\tmodifiedFields: Set<keyof Settings>,\r\n\t\tmodifiedNestedFields: Map<keyof Settings, Set<string>>,\r\n\t): void {\r\n\t\tthis.storage.withLock(scope, (current) => {\r\n\t\t\tconst currentFileSettings = current\r\n\t\t\t\t? SettingsManager.migrateSettings(JSON.parse(current) as Record<string, unknown>)\r\n\t\t\t\t: {};\r\n\t\t\tconst mergedSettings: Settings = { ...currentFileSettings };\r\n\t\t\tfor (const field of modifiedFields) {\r\n\t\t\t\tconst value = snapshotSettings[field];\r\n\t\t\t\tif (modifiedNestedFields.has(field) && typeof value === \"object\" && value !== null) {\r\n\t\t\t\t\tconst nestedModified = modifiedNestedFields.get(field)!;\r\n\t\t\t\t\tconst baseNested = (currentFileSettings[field] as Record<string, unknown>) ?? {};\r\n\t\t\t\t\tconst inMemoryNested = value as Record<string, unknown>;\r\n\t\t\t\t\tconst mergedNested = { ...baseNested };\r\n\t\t\t\t\tfor (const nestedKey of nestedModified) {\r\n\t\t\t\t\t\tmergedNested[nestedKey] = inMemoryNested[nestedKey];\r\n\t\t\t\t\t}\r\n\t\t\t\t\t(mergedSettings as Record<string, unknown>)[field] = mergedNested;\r\n\t\t\t\t} else {\r\n\t\t\t\t\t(mergedSettings as Record<string, unknown>)[field] = value;\r\n\t\t\t\t}\r\n\t\t\t}\r\n\r\n\t\t\treturn JSON.stringify(mergedSettings, null, 2);\r\n\t\t});\r\n\t}\r\n\r\n\tprivate save(): void {\r\n\t\tthis.settings = deepMergeSettings(this.globalSettings, this.projectSettings);\r\n\r\n\t\tif (this.globalSettingsLoadError) {\r\n\t\t\treturn;\r\n\t\t}\r\n\r\n\t\tconst snapshotGlobalSettings = structuredClone(this.globalSettings);\r\n\t\tconst modifiedFields = new Set(this.modifiedFields);\r\n\t\tconst modifiedNestedFields = this.cloneModifiedNestedFields(this.modifiedNestedFields);\r\n\r\n\t\tthis.enqueueWrite(\"global\", () => {\r\n\t\t\tthis.persistScopedSettings(\"global\", snapshotGlobalSettings, modifiedFields, modifiedNestedFields);\r\n\t\t});\r\n\t}\r\n\r\n\tprivate saveProjectSettings(settings: Settings): void {\r\n\t\tthis.projectSettings = structuredClone(settings);\r\n\t\tthis.settings = deepMergeSettings(this.globalSettings, this.projectSettings);\r\n\r\n\t\tif (this.projectSettingsLoadError) {\r\n\t\t\treturn;\r\n\t\t}\r\n\r\n\t\tconst snapshotProjectSettings = structuredClone(this.projectSettings);\r\n\t\tconst modifiedFields = new Set(this.modifiedProjectFields);\r\n\t\tconst modifiedNestedFields = this.cloneModifiedNestedFields(this.modifiedProjectNestedFields);\r\n\t\tthis.enqueueWrite(\"project\", () => {\r\n\t\t\tthis.persistScopedSettings(\"project\", snapshotProjectSettings, modifiedFields, modifiedNestedFields);\r\n\t\t});\r\n\t}\r\n\r\n\tasync flush(): Promise<void> {\r\n\t\tawait this.writeQueue;\r\n\t}\r\n\r\n\tdrainErrors(): SettingsError[] {\r\n\t\tconst drained = [...this.errors];\r\n\t\tthis.errors = [];\r\n\t\treturn drained;\r\n\t}\r\n\r\n\tgetLastChangelogVersion(): string | undefined {\r\n\t\treturn this.settings.lastChangelogVersion;\r\n\t}\r\n\r\n\tsetLastChangelogVersion(version: string): void {\r\n\t\tthis.globalSettings.lastChangelogVersion = version;\r\n\t\tthis.markModified(\"lastChangelogVersion\");\r\n\t\tthis.save();\r\n\t}\r\n\r\n\tgetDefaultProvider(): string | undefined {\r\n\t\treturn this.settings.defaultProvider;\r\n\t}\r\n\r\n\tgetDefaultModel(): string | undefined {\r\n\t\treturn this.settings.defaultModel;\r\n\t}\r\n\r\n\tsetDefaultProvider(provider: string): void {\r\n\t\tthis.globalSettings.defaultProvider = provider;\r\n\t\tthis.markModified(\"defaultProvider\");\r\n\t\tthis.save();\r\n\t}\r\n\r\n\tsetDefaultModel(modelId: string): void {\r\n\t\tthis.globalSettings.defaultModel = modelId;\r\n\t\tthis.markModified(\"defaultModel\");\r\n\t\tthis.save();\r\n\t}\r\n\r\n\tsetDefaultModelAndProvider(provider: string, modelId: string): void {\r\n\t\tthis.globalSettings.defaultProvider = provider;\r\n\t\tthis.globalSettings.defaultModel = modelId;\r\n\t\tthis.markModified(\"defaultProvider\");\r\n\t\tthis.markModified(\"defaultModel\");\r\n\t\tthis.save();\r\n\t}\r\n\r\n\tgetSteeringMode(): \"all\" | \"one-at-a-time\" {\r\n\t\treturn this.settings.steeringMode || \"one-at-a-time\";\r\n\t}\r\n\r\n\tsetSteeringMode(mode: \"all\" | \"one-at-a-time\"): void {\r\n\t\tthis.globalSettings.steeringMode = mode;\r\n\t\tthis.markModified(\"steeringMode\");\r\n\t\tthis.save();\r\n\t}\r\n\r\n\tgetFollowUpMode(): \"all\" | \"one-at-a-time\" {\r\n\t\treturn this.settings.followUpMode || \"one-at-a-time\";\r\n\t}\r\n\r\n\tsetFollowUpMode(mode: \"all\" | \"one-at-a-time\"): void {\r\n\t\tthis.globalSettings.followUpMode = mode;\r\n\t\tthis.markModified(\"followUpMode\");\r\n\t\tthis.save();\r\n\t}\r\n\r\n\tgetTheme(): string | undefined {\r\n\t\treturn this.settings.theme;\r\n\t}\r\n\r\n\tsetTheme(theme: string): void {\r\n\t\tthis.globalSettings.theme = theme;\r\n\t\tthis.markModified(\"theme\");\r\n\t\tthis.save();\r\n\t}\r\n\r\n\tgetDefaultThinkingLevel(): \"off\" | \"minimal\" | \"low\" | \"medium\" | \"high\" | \"xhigh\" | undefined {\r\n\t\treturn this.settings.defaultThinkingLevel;\r\n\t}\r\n\r\n\tsetDefaultThinkingLevel(level: \"off\" | \"minimal\" | \"low\" | \"medium\" | \"high\" | \"xhigh\"): void {\r\n\t\tthis.globalSettings.defaultThinkingLevel = level;\r\n\t\tthis.markModified(\"defaultThinkingLevel\");\r\n\t\tthis.save();\r\n\t}\r\n\r\n\tgetTransport(): TransportSetting {\r\n\t\treturn this.settings.transport ?? \"sse\";\r\n\t}\r\n\r\n\tsetTransport(transport: TransportSetting): void {\r\n\t\tthis.globalSettings.transport = transport;\r\n\t\tthis.markModified(\"transport\");\r\n\t\tthis.save();\r\n\t}\r\n\r\n\tgetCompactionEnabled(): boolean {\r\n\t\treturn this.settings.compaction?.enabled ?? true;\r\n\t}\r\n\r\n\tsetCompactionEnabled(enabled: boolean): void {\r\n\t\tif (!this.globalSettings.compaction) {\r\n\t\t\tthis.globalSettings.compaction = {};\r\n\t\t}\r\n\t\tthis.globalSettings.compaction.enabled = enabled;\r\n\t\tthis.markModified(\"compaction\", \"enabled\");\r\n\t\tthis.save();\r\n\t}\r\n\r\n\tgetCompactionReserveTokens(): number {\r\n\t\treturn this.settings.compaction?.reserveTokens ?? 16384;\r\n\t}\r\n\r\n\tgetCompactionKeepRecentTokens(): number {\r\n\t\treturn this.settings.compaction?.keepRecentTokens ?? 20000;\r\n\t}\r\n\r\n\tgetCompactionSettings(): { enabled: boolean; reserveTokens: number; keepRecentTokens: number } {\r\n\t\treturn {\r\n\t\t\tenabled: this.getCompactionEnabled(),\r\n\t\t\treserveTokens: this.getCompactionReserveTokens(),\r\n\t\t\tkeepRecentTokens: this.getCompactionKeepRecentTokens(),\r\n\t\t};\r\n\t}\r\n\r\n\tgetBranchSummarySettings(): { reserveTokens: number } {\r\n\t\treturn {\r\n\t\t\treserveTokens: this.settings.branchSummary?.reserveTokens ?? 16384,\r\n\t\t};\r\n\t}\r\n\r\n\tgetRetryEnabled(): boolean {\r\n\t\treturn this.settings.retry?.enabled ?? true;\r\n\t}\r\n\r\n\tsetRetryEnabled(enabled: boolean): void {\r\n\t\tif (!this.globalSettings.retry) {\r\n\t\t\tthis.globalSettings.retry = {};\r\n\t\t}\r\n\t\tthis.globalSettings.retry.enabled = enabled;\r\n\t\tthis.markModified(\"retry\", \"enabled\");\r\n\t\tthis.save();\r\n\t}\r\n\r\n\tgetRetrySettings(): { enabled: boolean; maxRetries: number; baseDelayMs: number; maxDelayMs: number } {\r\n\t\treturn {\r\n\t\t\tenabled: this.getRetryEnabled(),\r\n\t\t\tmaxRetries: this.settings.retry?.maxRetries ?? 3,\r\n\t\t\tbaseDelayMs: this.settings.retry?.baseDelayMs ?? 2000,\r\n\t\t\tmaxDelayMs: this.settings.retry?.maxDelayMs ?? 60000,\r\n\t\t};\r\n\t}\r\n\r\n\tgetHideThinkingBlock(): boolean {\r\n\t\treturn this.settings.hideThinkingBlock ?? false;\r\n\t}\r\n\r\n\tsetHideThinkingBlock(hide: boolean): void {\r\n\t\tthis.globalSettings.hideThinkingBlock = hide;\r\n\t\tthis.markModified(\"hideThinkingBlock\");\r\n\t\tthis.save();\r\n\t}\r\n\r\n\tgetShellPath(): string | undefined {\r\n\t\treturn this.settings.shellPath;\r\n\t}\r\n\r\n\tsetShellPath(path: string | undefined): void {\r\n\t\tthis.globalSettings.shellPath = path;\r\n\t\tthis.markModified(\"shellPath\");\r\n\t\tthis.save();\r\n\t}\r\n\r\n\tgetQuietStartup(): boolean {\r\n\t\treturn this.settings.quietStartup ?? false;\r\n\t}\r\n\r\n\tsetQuietStartup(quiet: boolean): void {\r\n\t\tthis.globalSettings.quietStartup = quiet;\r\n\t\tthis.markModified(\"quietStartup\");\r\n\t\tthis.save();\r\n\t}\r\n\r\n\tgetShellCommandPrefix(): string | undefined {\r\n\t\treturn this.settings.shellCommandPrefix;\r\n\t}\r\n\r\n\tsetShellCommandPrefix(prefix: string | undefined): void {\r\n\t\tthis.globalSettings.shellCommandPrefix = prefix;\r\n\t\tthis.markModified(\"shellCommandPrefix\");\r\n\t\tthis.save();\r\n\t}\r\n\r\n\tgetCollapseChangelog(): boolean {\r\n\t\treturn this.settings.collapseChangelog ?? false;\r\n\t}\r\n\r\n\tsetCollapseChangelog(collapse: boolean): void {\r\n\t\tthis.globalSettings.collapseChangelog = collapse;\r\n\t\tthis.markModified(\"collapseChangelog\");\r\n\t\tthis.save();\r\n\t}\r\n\r\n\tgetPackages(): PackageSource[] {\r\n\t\treturn [...(this.settings.packages ?? [])];\r\n\t}\r\n\r\n\tsetPackages(packages: PackageSource[]): void {\r\n\t\tthis.globalSettings.packages = packages;\r\n\t\tthis.markModified(\"packages\");\r\n\t\tthis.save();\r\n\t}\r\n\r\n\tsetProjectPackages(packages: PackageSource[]): void {\r\n\t\tconst projectSettings = structuredClone(this.projectSettings);\r\n\t\tprojectSettings.packages = packages;\r\n\t\tthis.markProjectModified(\"packages\");\r\n\t\tthis.saveProjectSettings(projectSettings);\r\n\t}\r\n\r\n\tgetExtensionPaths(): string[] {\r\n\t\treturn [...(this.settings.extensions ?? [])];\r\n\t}\r\n\r\n\tsetExtensionPaths(paths: string[]): void {\r\n\t\tthis.globalSettings.extensions = paths;\r\n\t\tthis.markModified(\"extensions\");\r\n\t\tthis.save();\r\n\t}\r\n\r\n\tsetProjectExtensionPaths(paths: string[]): void {\r\n\t\tconst projectSettings = structuredClone(this.projectSettings);\r\n\t\tprojectSettings.extensions = paths;\r\n\t\tthis.markProjectModified(\"extensions\");\r\n\t\tthis.saveProjectSettings(projectSettings);\r\n\t}\r\n\r\n\tgetSkillPaths(): string[] {\r\n\t\treturn [...(this.settings.skills ?? [])];\r\n\t}\r\n\r\n\tsetSkillPaths(paths: string[]): void {\r\n\t\tthis.globalSettings.skills = paths;\r\n\t\tthis.markModified(\"skills\");\r\n\t\tthis.save();\r\n\t}\r\n\r\n\tsetProjectSkillPaths(paths: string[]): void {\r\n\t\tconst projectSettings = structuredClone(this.projectSettings);\r\n\t\tprojectSettings.skills = paths;\r\n\t\tthis.markProjectModified(\"skills\");\r\n\t\tthis.saveProjectSettings(projectSettings);\r\n\t}\r\n\r\n\tgetPromptTemplatePaths(): string[] {\r\n\t\treturn [...(this.settings.prompts ?? [])];\r\n\t}\r\n\r\n\tsetPromptTemplatePaths(paths: string[]): void {\r\n\t\tthis.globalSettings.prompts = paths;\r\n\t\tthis.markModified(\"prompts\");\r\n\t\tthis.save();\r\n\t}\r\n\r\n\tsetProjectPromptTemplatePaths(paths: string[]): void {\r\n\t\tconst projectSettings = structuredClone(this.projectSettings);\r\n\t\tprojectSettings.prompts = paths;\r\n\t\tthis.markProjectModified(\"prompts\");\r\n\t\tthis.saveProjectSettings(projectSettings);\r\n\t}\r\n\r\n\tgetThemePaths(): string[] {\r\n\t\treturn [...(this.settings.themes ?? [])];\r\n\t}\r\n\r\n\tsetThemePaths(paths: string[]): void {\r\n\t\tthis.globalSettings.themes = paths;\r\n\t\tthis.markModified(\"themes\");\r\n\t\tthis.save();\r\n\t}\r\n\r\n\tsetProjectThemePaths(paths: string[]): void {\r\n\t\tconst projectSettings = structuredClone(this.projectSettings);\r\n\t\tprojectSettings.themes = paths;\r\n\t\tthis.markProjectModified(\"themes\");\r\n\t\tthis.saveProjectSettings(projectSettings);\r\n\t}\r\n\r\n\tgetEnableSkillCommands(): boolean {\r\n\t\treturn this.settings.enableSkillCommands ?? true;\r\n\t}\r\n\r\n\tsetEnableSkillCommands(enabled: boolean): void {\r\n\t\tthis.globalSettings.enableSkillCommands = enabled;\r\n\t\tthis.markModified(\"enableSkillCommands\");\r\n\t\tthis.save();\r\n\t}\r\n\r\n\tgetThinkingBudgets(): ThinkingBudgetsSettings | undefined {\r\n\t\treturn this.settings.thinkingBudgets;\r\n\t}\r\n\r\n\tgetShowImages(): boolean {\r\n\t\treturn this.settings.terminal?.showImages ?? true;\r\n\t}\r\n\r\n\tsetShowImages(show: boolean): void {\r\n\t\tif (!this.globalSettings.terminal) {\r\n\t\t\tthis.globalSettings.terminal = {};\r\n\t\t}\r\n\t\tthis.globalSettings.terminal.showImages = show;\r\n\t\tthis.markModified(\"terminal\", \"showImages\");\r\n\t\tthis.save();\r\n\t}\r\n\r\n\tgetClearOnShrink(): boolean {\r\n\t\t// Settings takes precedence, then env var, then default false\r\n\t\tif (this.settings.terminal?.clearOnShrink !== undefined) {\r\n\t\t\treturn this.settings.terminal.clearOnShrink;\r\n\t\t}\r\n\t\treturn process.env.PI_CLEAR_ON_SHRINK === \"1\";\r\n\t}\r\n\r\n\tsetClearOnShrink(enabled: boolean): void {\r\n\t\tif (!this.globalSettings.terminal) {\r\n\t\t\tthis.globalSettings.terminal = {};\r\n\t\t}\r\n\t\tthis.globalSettings.terminal.clearOnShrink = enabled;\r\n\t\tthis.markModified(\"terminal\", \"clearOnShrink\");\r\n\t\tthis.save();\r\n\t}\r\n\r\n\tgetImageAutoResize(): boolean {\r\n\t\treturn this.settings.images?.autoResize ?? true;\r\n\t}\r\n\r\n\tsetImageAutoResize(enabled: boolean): void {\r\n\t\tif (!this.globalSettings.images) {\r\n\t\t\tthis.globalSettings.images = {};\r\n\t\t}\r\n\t\tthis.globalSettings.images.autoResize = enabled;\r\n\t\tthis.markModified(\"images\", \"autoResize\");\r\n\t\tthis.save();\r\n\t}\r\n\r\n\tgetBlockImages(): boolean {\r\n\t\treturn this.settings.images?.blockImages ?? false;\r\n\t}\r\n\r\n\tsetBlockImages(blocked: boolean): void {\r\n\t\tif (!this.globalSettings.images) {\r\n\t\t\tthis.globalSettings.images = {};\r\n\t\t}\r\n\t\tthis.globalSettings.images.blockImages = blocked;\r\n\t\tthis.markModified(\"images\", \"blockImages\");\r\n\t\tthis.save();\r\n\t}\r\n\r\n\tgetEnabledModels(): string[] | undefined {\r\n\t\treturn this.settings.enabledModels;\r\n\t}\r\n\r\n\tsetEnabledModels(patterns: string[] | undefined): void {\r\n\t\tthis.globalSettings.enabledModels = patterns;\r\n\t\tthis.markModified(\"enabledModels\");\r\n\t\tthis.save();\r\n\t}\r\n\r\n\tgetDoubleEscapeAction(): \"fork\" | \"tree\" | \"none\" {\r\n\t\treturn this.settings.doubleEscapeAction ?? \"tree\";\r\n\t}\r\n\r\n\tsetDoubleEscapeAction(action: \"fork\" | \"tree\" | \"none\"): void {\r\n\t\tthis.globalSettings.doubleEscapeAction = action;\r\n\t\tthis.markModified(\"doubleEscapeAction\");\r\n\t\tthis.save();\r\n\t}\r\n\r\n\tgetShowHardwareCursor(): boolean {\r\n\t\treturn this.settings.showHardwareCursor ?? process.env.PI_HARDWARE_CURSOR === \"1\";\r\n\t}\r\n\r\n\tsetShowHardwareCursor(enabled: boolean): void {\r\n\t\tthis.globalSettings.showHardwareCursor = enabled;\r\n\t\tthis.markModified(\"showHardwareCursor\");\r\n\t\tthis.save();\r\n\t}\r\n\r\n\tgetEditorPaddingX(): number {\r\n\t\treturn this.settings.editorPaddingX ?? 0;\r\n\t}\r\n\r\n\tsetEditorPaddingX(padding: number): void {\r\n\t\tthis.globalSettings.editorPaddingX = Math.max(0, Math.min(3, Math.floor(padding)));\r\n\t\tthis.markModified(\"editorPaddingX\");\r\n\t\tthis.save();\r\n\t}\r\n\r\n\tgetAutocompleteMaxVisible(): number {\r\n\t\treturn this.settings.autocompleteMaxVisible ?? 5;\r\n\t}\r\n\r\n\tsetAutocompleteMaxVisible(maxVisible: number): void {\r\n\t\tthis.globalSettings.autocompleteMaxVisible = Math.max(3, Math.min(20, Math.floor(maxVisible)));\r\n\t\tthis.markModified(\"autocompleteMaxVisible\");\r\n\t\tthis.save();\r\n\t}\r\n\r\n\tgetCodeBlockIndent(): string {\r\n\t\treturn this.settings.markdown?.codeBlockIndent ?? \" \";\r\n\t}\r\n\r\n\tgetMemoryEnabled(): boolean {\r\n\t\treturn this.settings.memory?.enabled ?? true;\r\n\t}\r\n\r\n\tsetMemoryEnabled(enabled: boolean): void {\r\n\t\tif (!this.globalSettings.memory) {\r\n\t\t\tthis.globalSettings.memory = {};\r\n\t\t}\r\n\t\tthis.globalSettings.memory.enabled = enabled;\r\n\t\tthis.markModified(\"memory\", \"enabled\");\r\n\t\tthis.save();\r\n\t}\r\n}\r\n"]}
|
|
@@ -0,0 +1,58 @@
|
|
|
1
|
+
import type { ResourceDiagnostic } from "./diagnostics.js";
|
|
2
|
+
export interface SkillFrontmatter {
|
|
3
|
+
name?: string;
|
|
4
|
+
description?: string;
|
|
5
|
+
"disable-model-invocation"?: boolean;
|
|
6
|
+
[key: string]: unknown;
|
|
7
|
+
}
|
|
8
|
+
export interface Skill {
|
|
9
|
+
name: string;
|
|
10
|
+
description: string;
|
|
11
|
+
filePath: string;
|
|
12
|
+
baseDir: string;
|
|
13
|
+
source: string;
|
|
14
|
+
disableModelInvocation: boolean;
|
|
15
|
+
}
|
|
16
|
+
export interface LoadSkillsResult {
|
|
17
|
+
skills: Skill[];
|
|
18
|
+
diagnostics: ResourceDiagnostic[];
|
|
19
|
+
}
|
|
20
|
+
export interface LoadSkillsFromDirOptions {
|
|
21
|
+
/** Directory to scan for skills */
|
|
22
|
+
dir: string;
|
|
23
|
+
/** Source identifier for these skills */
|
|
24
|
+
source: string;
|
|
25
|
+
}
|
|
26
|
+
/**
|
|
27
|
+
* Load skills from a directory.
|
|
28
|
+
*
|
|
29
|
+
* Discovery rules:
|
|
30
|
+
* - direct .md children in the root
|
|
31
|
+
* - recursive SKILL.md under subdirectories
|
|
32
|
+
*/
|
|
33
|
+
export declare function loadSkillsFromDir(options: LoadSkillsFromDirOptions): LoadSkillsResult;
|
|
34
|
+
/**
|
|
35
|
+
* Format skills for inclusion in a system prompt.
|
|
36
|
+
* Uses XML format per Agent Skills standard.
|
|
37
|
+
* See: https://agentskills.io/integrate-skills
|
|
38
|
+
*
|
|
39
|
+
* Skills with disableModelInvocation=true are excluded from the prompt
|
|
40
|
+
* (they can only be invoked explicitly via /skill:name commands).
|
|
41
|
+
*/
|
|
42
|
+
export declare function formatSkillsForPrompt(skills: Skill[]): string;
|
|
43
|
+
export interface LoadSkillsOptions {
|
|
44
|
+
/** Working directory for project-local skills. Default: process.cwd() */
|
|
45
|
+
cwd?: string;
|
|
46
|
+
/** Agent config directory for global skills. Default: ~/.pi/agent */
|
|
47
|
+
agentDir?: string;
|
|
48
|
+
/** Explicit skill paths (files or directories) */
|
|
49
|
+
skillPaths?: string[];
|
|
50
|
+
/** Include default skills directories. Default: true */
|
|
51
|
+
includeDefaults?: boolean;
|
|
52
|
+
}
|
|
53
|
+
/**
|
|
54
|
+
* Load skills from all configured locations.
|
|
55
|
+
* Returns skills and any validation diagnostics.
|
|
56
|
+
*/
|
|
57
|
+
export declare function loadSkills(options?: LoadSkillsOptions): LoadSkillsResult;
|
|
58
|
+
//# sourceMappingURL=skills.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"skills.d.ts","sourceRoot":"","sources":["../../src/core/skills.ts"],"names":[],"mappings":"AAMA,OAAO,KAAK,EAAE,kBAAkB,EAAE,MAAM,kBAAkB,CAAC;AA2D3D,MAAM,WAAW,gBAAgB;IAChC,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,0BAA0B,CAAC,EAAE,OAAO,CAAC;IACrC,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAAC;CACvB;AAED,MAAM,WAAW,KAAK;IACrB,IAAI,EAAE,MAAM,CAAC;IACb,WAAW,EAAE,MAAM,CAAC;IACpB,QAAQ,EAAE,MAAM,CAAC;IACjB,OAAO,EAAE,MAAM,CAAC;IAChB,MAAM,EAAE,MAAM,CAAC;IACf,sBAAsB,EAAE,OAAO,CAAC;CAChC;AAED,MAAM,WAAW,gBAAgB;IAChC,MAAM,EAAE,KAAK,EAAE,CAAC;IAChB,WAAW,EAAE,kBAAkB,EAAE,CAAC;CAClC;AA+CD,MAAM,WAAW,wBAAwB;IACxC,mCAAmC;IACnC,GAAG,EAAE,MAAM,CAAC;IACZ,yCAAyC;IACzC,MAAM,EAAE,MAAM,CAAC;CACf;AAED;;;;;;GAMG;AACH,wBAAgB,iBAAiB,CAAC,OAAO,EAAE,wBAAwB,GAAG,gBAAgB,CAGrF;AAqID;;;;;;;GAOG;AACH,wBAAgB,qBAAqB,CAAC,MAAM,EAAE,KAAK,EAAE,GAAG,MAAM,CA0B7D;AAWD,MAAM,WAAW,iBAAiB;IACjC,yEAAyE;IACzE,GAAG,CAAC,EAAE,MAAM,CAAC;IACb,qEAAqE;IACrE,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,kDAAkD;IAClD,UAAU,CAAC,EAAE,MAAM,EAAE,CAAC;IACtB,wDAAwD;IACxD,eAAe,CAAC,EAAE,OAAO,CAAC;CAC1B;AAeD;;;GAGG;AACH,wBAAgB,UAAU,CAAC,OAAO,GAAE,iBAAsB,GAAG,gBAAgB,CAwG5E","sourcesContent":["import { existsSync, readdirSync, readFileSync, realpathSync, statSync } from \"fs\";\r\nimport ignore from \"ignore\";\r\nimport { homedir } from \"os\";\r\nimport { basename, dirname, isAbsolute, join, relative, resolve, sep } from \"path\";\r\nimport { CONFIG_DIR_NAME, getAgentDir } from \"../config.js\";\r\nimport { parseFrontmatter } from \"../utils/frontmatter.js\";\r\nimport type { ResourceDiagnostic } from \"./diagnostics.js\";\r\n\r\n/** Max name length per spec */\r\nconst MAX_NAME_LENGTH = 64;\r\n\r\n/** Max description length per spec */\r\nconst MAX_DESCRIPTION_LENGTH = 1024;\r\n\r\nconst IGNORE_FILE_NAMES = [\".gitignore\", \".ignore\", \".fdignore\"];\r\n\r\ntype IgnoreMatcher = ReturnType<typeof ignore>;\r\n\r\nfunction toPosixPath(p: string): string {\r\n\treturn p.split(sep).join(\"/\");\r\n}\r\n\r\nfunction prefixIgnorePattern(line: string, prefix: string): string | null {\r\n\tconst trimmed = line.trim();\r\n\tif (!trimmed) return null;\r\n\tif (trimmed.startsWith(\"#\") && !trimmed.startsWith(\"\\\\#\")) return null;\r\n\r\n\tlet pattern = line;\r\n\tlet negated = false;\r\n\r\n\tif (pattern.startsWith(\"!\")) {\r\n\t\tnegated = true;\r\n\t\tpattern = pattern.slice(1);\r\n\t} else if (pattern.startsWith(\"\\\\!\")) {\r\n\t\tpattern = pattern.slice(1);\r\n\t}\r\n\r\n\tif (pattern.startsWith(\"/\")) {\r\n\t\tpattern = pattern.slice(1);\r\n\t}\r\n\r\n\tconst prefixed = prefix ? `${prefix}${pattern}` : pattern;\r\n\treturn negated ? `!${prefixed}` : prefixed;\r\n}\r\n\r\nfunction addIgnoreRules(ig: IgnoreMatcher, dir: string, rootDir: string): void {\r\n\tconst relativeDir = relative(rootDir, dir);\r\n\tconst prefix = relativeDir ? `${toPosixPath(relativeDir)}/` : \"\";\r\n\r\n\tfor (const filename of IGNORE_FILE_NAMES) {\r\n\t\tconst ignorePath = join(dir, filename);\r\n\t\tif (!existsSync(ignorePath)) continue;\r\n\t\ttry {\r\n\t\t\tconst content = readFileSync(ignorePath, \"utf-8\");\r\n\t\t\tconst patterns = content\r\n\t\t\t\t.split(/\\r?\\n/)\r\n\t\t\t\t.map((line) => prefixIgnorePattern(line, prefix))\r\n\t\t\t\t.filter((line): line is string => Boolean(line));\r\n\t\t\tif (patterns.length > 0) {\r\n\t\t\t\tig.add(patterns);\r\n\t\t\t}\r\n\t\t} catch {}\r\n\t}\r\n}\r\n\r\nexport interface SkillFrontmatter {\r\n\tname?: string;\r\n\tdescription?: string;\r\n\t\"disable-model-invocation\"?: boolean;\r\n\t[key: string]: unknown;\r\n}\r\n\r\nexport interface Skill {\r\n\tname: string;\r\n\tdescription: string;\r\n\tfilePath: string;\r\n\tbaseDir: string;\r\n\tsource: string;\r\n\tdisableModelInvocation: boolean;\r\n}\r\n\r\nexport interface LoadSkillsResult {\r\n\tskills: Skill[];\r\n\tdiagnostics: ResourceDiagnostic[];\r\n}\r\n\r\n/**\r\n * Validate skill name per Agent Skills spec.\r\n * Returns array of validation error messages (empty if valid).\r\n */\r\nfunction validateName(name: string, parentDirName: string): string[] {\r\n\tconst errors: string[] = [];\r\n\r\n\tif (name !== parentDirName) {\r\n\t\terrors.push(`name \"${name}\" does not match parent directory \"${parentDirName}\"`);\r\n\t}\r\n\r\n\tif (name.length > MAX_NAME_LENGTH) {\r\n\t\terrors.push(`name exceeds ${MAX_NAME_LENGTH} characters (${name.length})`);\r\n\t}\r\n\r\n\tif (!/^[a-z0-9-]+$/.test(name)) {\r\n\t\terrors.push(`name contains invalid characters (must be lowercase a-z, 0-9, hyphens only)`);\r\n\t}\r\n\r\n\tif (name.startsWith(\"-\") || name.endsWith(\"-\")) {\r\n\t\terrors.push(`name must not start or end with a hyphen`);\r\n\t}\r\n\r\n\tif (name.includes(\"--\")) {\r\n\t\terrors.push(`name must not contain consecutive hyphens`);\r\n\t}\r\n\r\n\treturn errors;\r\n}\r\n\r\n/**\r\n * Validate description per Agent Skills spec.\r\n */\r\nfunction validateDescription(description: string | undefined): string[] {\r\n\tconst errors: string[] = [];\r\n\r\n\tif (!description || description.trim() === \"\") {\r\n\t\terrors.push(\"description is required\");\r\n\t} else if (description.length > MAX_DESCRIPTION_LENGTH) {\r\n\t\terrors.push(`description exceeds ${MAX_DESCRIPTION_LENGTH} characters (${description.length})`);\r\n\t}\r\n\r\n\treturn errors;\r\n}\r\n\r\nexport interface LoadSkillsFromDirOptions {\r\n\t/** Directory to scan for skills */\r\n\tdir: string;\r\n\t/** Source identifier for these skills */\r\n\tsource: string;\r\n}\r\n\r\n/**\r\n * Load skills from a directory.\r\n *\r\n * Discovery rules:\r\n * - direct .md children in the root\r\n * - recursive SKILL.md under subdirectories\r\n */\r\nexport function loadSkillsFromDir(options: LoadSkillsFromDirOptions): LoadSkillsResult {\r\n\tconst { dir, source } = options;\r\n\treturn loadSkillsFromDirInternal(dir, source, true);\r\n}\r\n\r\nfunction loadSkillsFromDirInternal(\r\n\tdir: string,\r\n\tsource: string,\r\n\tincludeRootFiles: boolean,\r\n\tignoreMatcher?: IgnoreMatcher,\r\n\trootDir?: string,\r\n): LoadSkillsResult {\r\n\tconst skills: Skill[] = [];\r\n\tconst diagnostics: ResourceDiagnostic[] = [];\r\n\r\n\tif (!existsSync(dir)) {\r\n\t\treturn { skills, diagnostics };\r\n\t}\r\n\r\n\tconst root = rootDir ?? dir;\r\n\tconst ig = ignoreMatcher ?? ignore();\r\n\taddIgnoreRules(ig, dir, root);\r\n\r\n\ttry {\r\n\t\tconst entries = readdirSync(dir, { withFileTypes: true });\r\n\r\n\t\tfor (const entry of entries) {\r\n\t\t\tif (entry.name.startsWith(\".\")) {\r\n\t\t\t\tcontinue;\r\n\t\t\t}\r\n\r\n\t\t\t// Skip node_modules to avoid scanning dependencies\r\n\t\t\tif (entry.name === \"node_modules\") {\r\n\t\t\t\tcontinue;\r\n\t\t\t}\r\n\r\n\t\t\tconst fullPath = join(dir, entry.name);\r\n\r\n\t\t\t// For symlinks, check if they point to a directory and follow them\r\n\t\t\tlet isDirectory = entry.isDirectory();\r\n\t\t\tlet isFile = entry.isFile();\r\n\t\t\tif (entry.isSymbolicLink()) {\r\n\t\t\t\ttry {\r\n\t\t\t\t\tconst stats = statSync(fullPath);\r\n\t\t\t\t\tisDirectory = stats.isDirectory();\r\n\t\t\t\t\tisFile = stats.isFile();\r\n\t\t\t\t} catch {\r\n\t\t\t\t\t// Broken symlink, skip it\r\n\t\t\t\t\tcontinue;\r\n\t\t\t\t}\r\n\t\t\t}\r\n\r\n\t\t\tconst relPath = toPosixPath(relative(root, fullPath));\r\n\t\t\tconst ignorePath = isDirectory ? `${relPath}/` : relPath;\r\n\t\t\tif (ig.ignores(ignorePath)) {\r\n\t\t\t\tcontinue;\r\n\t\t\t}\r\n\r\n\t\t\tif (isDirectory) {\r\n\t\t\t\tconst subResult = loadSkillsFromDirInternal(fullPath, source, false, ig, root);\r\n\t\t\t\tskills.push(...subResult.skills);\r\n\t\t\t\tdiagnostics.push(...subResult.diagnostics);\r\n\t\t\t\tcontinue;\r\n\t\t\t}\r\n\r\n\t\t\tif (!isFile) {\r\n\t\t\t\tcontinue;\r\n\t\t\t}\r\n\r\n\t\t\tconst isRootMd = includeRootFiles && entry.name.endsWith(\".md\");\r\n\t\t\tconst isSkillMd = !includeRootFiles && entry.name === \"SKILL.md\";\r\n\t\t\tif (!isRootMd && !isSkillMd) {\r\n\t\t\t\tcontinue;\r\n\t\t\t}\r\n\r\n\t\t\tconst result = loadSkillFromFile(fullPath, source);\r\n\t\t\tif (result.skill) {\r\n\t\t\t\tskills.push(result.skill);\r\n\t\t\t}\r\n\t\t\tdiagnostics.push(...result.diagnostics);\r\n\t\t}\r\n\t} catch {}\r\n\r\n\treturn { skills, diagnostics };\r\n}\r\n\r\nfunction loadSkillFromFile(\r\n\tfilePath: string,\r\n\tsource: string,\r\n): { skill: Skill | null; diagnostics: ResourceDiagnostic[] } {\r\n\tconst diagnostics: ResourceDiagnostic[] = [];\r\n\r\n\ttry {\r\n\t\tconst rawContent = readFileSync(filePath, \"utf-8\");\r\n\t\tconst { frontmatter } = parseFrontmatter<SkillFrontmatter>(rawContent);\r\n\t\tconst skillDir = dirname(filePath);\r\n\t\tconst parentDirName = basename(skillDir);\r\n\r\n\t\t// Validate description\r\n\t\tconst descErrors = validateDescription(frontmatter.description);\r\n\t\tfor (const error of descErrors) {\r\n\t\t\tdiagnostics.push({ type: \"warning\", message: error, path: filePath });\r\n\t\t}\r\n\r\n\t\t// Use name from frontmatter, or fall back to parent directory name\r\n\t\tconst name = frontmatter.name || parentDirName;\r\n\r\n\t\t// Validate name\r\n\t\tconst nameErrors = validateName(name, parentDirName);\r\n\t\tfor (const error of nameErrors) {\r\n\t\t\tdiagnostics.push({ type: \"warning\", message: error, path: filePath });\r\n\t\t}\r\n\r\n\t\t// Still load the skill even with warnings (unless description is completely missing)\r\n\t\tif (!frontmatter.description || frontmatter.description.trim() === \"\") {\r\n\t\t\treturn { skill: null, diagnostics };\r\n\t\t}\r\n\r\n\t\treturn {\r\n\t\t\tskill: {\r\n\t\t\t\tname,\r\n\t\t\t\tdescription: frontmatter.description,\r\n\t\t\t\tfilePath,\r\n\t\t\t\tbaseDir: skillDir,\r\n\t\t\t\tsource,\r\n\t\t\t\tdisableModelInvocation: frontmatter[\"disable-model-invocation\"] === true,\r\n\t\t\t},\r\n\t\t\tdiagnostics,\r\n\t\t};\r\n\t} catch (error) {\r\n\t\tconst message = error instanceof Error ? error.message : \"failed to parse skill file\";\r\n\t\tdiagnostics.push({ type: \"warning\", message, path: filePath });\r\n\t\treturn { skill: null, diagnostics };\r\n\t}\r\n}\r\n\r\n/**\r\n * Format skills for inclusion in a system prompt.\r\n * Uses XML format per Agent Skills standard.\r\n * See: https://agentskills.io/integrate-skills\r\n *\r\n * Skills with disableModelInvocation=true are excluded from the prompt\r\n * (they can only be invoked explicitly via /skill:name commands).\r\n */\r\nexport function formatSkillsForPrompt(skills: Skill[]): string {\r\n\tconst visibleSkills = skills.filter((s) => !s.disableModelInvocation);\r\n\r\n\tif (visibleSkills.length === 0) {\r\n\t\treturn \"\";\r\n\t}\r\n\r\n\tconst lines = [\r\n\t\t\"\\n\\nThe following skills provide specialized instructions for specific tasks.\",\r\n\t\t\"Use the read tool to load a skill's file when the task matches its description.\",\r\n\t\t\"When a skill file references a relative path, resolve it against the skill directory (parent of SKILL.md / dirname of the path) and use that absolute path in tool commands.\",\r\n\t\t\"\",\r\n\t\t\"<available_skills>\",\r\n\t];\r\n\r\n\tfor (const skill of visibleSkills) {\r\n\t\tlines.push(\" <skill>\");\r\n\t\tlines.push(` <name>${escapeXml(skill.name)}</name>`);\r\n\t\tlines.push(` <description>${escapeXml(skill.description)}</description>`);\r\n\t\tlines.push(` <location>${escapeXml(skill.filePath)}</location>`);\r\n\t\tlines.push(\" </skill>\");\r\n\t}\r\n\r\n\tlines.push(\"</available_skills>\");\r\n\r\n\treturn lines.join(\"\\n\");\r\n}\r\n\r\nfunction escapeXml(str: string): string {\r\n\treturn str\r\n\t\t.replace(/&/g, \"&\")\r\n\t\t.replace(/</g, \"<\")\r\n\t\t.replace(/>/g, \">\")\r\n\t\t.replace(/\"/g, \""\")\r\n\t\t.replace(/'/g, \"'\");\r\n}\r\n\r\nexport interface LoadSkillsOptions {\r\n\t/** Working directory for project-local skills. Default: process.cwd() */\r\n\tcwd?: string;\r\n\t/** Agent config directory for global skills. Default: ~/.pi/agent */\r\n\tagentDir?: string;\r\n\t/** Explicit skill paths (files or directories) */\r\n\tskillPaths?: string[];\r\n\t/** Include default skills directories. Default: true */\r\n\tincludeDefaults?: boolean;\r\n}\r\n\r\nfunction normalizePath(input: string): string {\r\n\tconst trimmed = input.trim();\r\n\tif (trimmed === \"~\") return homedir();\r\n\tif (trimmed.startsWith(\"~/\")) return join(homedir(), trimmed.slice(2));\r\n\tif (trimmed.startsWith(\"~\")) return join(homedir(), trimmed.slice(1));\r\n\treturn trimmed;\r\n}\r\n\r\nfunction resolveSkillPath(p: string, cwd: string): string {\r\n\tconst normalized = normalizePath(p);\r\n\treturn isAbsolute(normalized) ? normalized : resolve(cwd, normalized);\r\n}\r\n\r\n/**\r\n * Load skills from all configured locations.\r\n * Returns skills and any validation diagnostics.\r\n */\r\nexport function loadSkills(options: LoadSkillsOptions = {}): LoadSkillsResult {\r\n\tconst { cwd = process.cwd(), agentDir, skillPaths = [], includeDefaults = true } = options;\r\n\r\n\t// Resolve agentDir - if not provided, use default from config\r\n\tconst resolvedAgentDir = agentDir ?? getAgentDir();\r\n\r\n\tconst skillMap = new Map<string, Skill>();\r\n\tconst realPathSet = new Set<string>();\r\n\tconst allDiagnostics: ResourceDiagnostic[] = [];\r\n\tconst collisionDiagnostics: ResourceDiagnostic[] = [];\r\n\r\n\tfunction addSkills(result: LoadSkillsResult) {\r\n\t\tallDiagnostics.push(...result.diagnostics);\r\n\t\tfor (const skill of result.skills) {\r\n\t\t\t// Resolve symlinks to detect duplicate files\r\n\t\t\tlet realPath: string;\r\n\t\t\ttry {\r\n\t\t\t\trealPath = realpathSync(skill.filePath);\r\n\t\t\t} catch {\r\n\t\t\t\trealPath = skill.filePath;\r\n\t\t\t}\r\n\r\n\t\t\t// Skip silently if we've already loaded this exact file (via symlink)\r\n\t\t\tif (realPathSet.has(realPath)) {\r\n\t\t\t\tcontinue;\r\n\t\t\t}\r\n\r\n\t\t\tconst existing = skillMap.get(skill.name);\r\n\t\t\tif (existing) {\r\n\t\t\t\tcollisionDiagnostics.push({\r\n\t\t\t\t\ttype: \"collision\",\r\n\t\t\t\t\tmessage: `name \"${skill.name}\" collision`,\r\n\t\t\t\t\tpath: skill.filePath,\r\n\t\t\t\t\tcollision: {\r\n\t\t\t\t\t\tresourceType: \"skill\",\r\n\t\t\t\t\t\tname: skill.name,\r\n\t\t\t\t\t\twinnerPath: existing.filePath,\r\n\t\t\t\t\t\tloserPath: skill.filePath,\r\n\t\t\t\t\t},\r\n\t\t\t\t});\r\n\t\t\t} else {\r\n\t\t\t\tskillMap.set(skill.name, skill);\r\n\t\t\t\trealPathSet.add(realPath);\r\n\t\t\t}\r\n\t\t}\r\n\t}\r\n\r\n\tif (includeDefaults) {\r\n\t\taddSkills(loadSkillsFromDirInternal(join(resolvedAgentDir, \"skills\"), \"user\", true));\r\n\t\taddSkills(loadSkillsFromDirInternal(resolve(cwd, CONFIG_DIR_NAME, \"skills\"), \"project\", true));\r\n\t}\r\n\r\n\tconst userSkillsDir = join(resolvedAgentDir, \"skills\");\r\n\tconst projectSkillsDir = resolve(cwd, CONFIG_DIR_NAME, \"skills\");\r\n\r\n\tconst isUnderPath = (target: string, root: string): boolean => {\r\n\t\tconst normalizedRoot = resolve(root);\r\n\t\tif (target === normalizedRoot) {\r\n\t\t\treturn true;\r\n\t\t}\r\n\t\tconst prefix = normalizedRoot.endsWith(sep) ? normalizedRoot : `${normalizedRoot}${sep}`;\r\n\t\treturn target.startsWith(prefix);\r\n\t};\r\n\r\n\tconst getSource = (resolvedPath: string): \"user\" | \"project\" | \"path\" => {\r\n\t\tif (!includeDefaults) {\r\n\t\t\tif (isUnderPath(resolvedPath, userSkillsDir)) return \"user\";\r\n\t\t\tif (isUnderPath(resolvedPath, projectSkillsDir)) return \"project\";\r\n\t\t}\r\n\t\treturn \"path\";\r\n\t};\r\n\r\n\tfor (const rawPath of skillPaths) {\r\n\t\tconst resolvedPath = resolveSkillPath(rawPath, cwd);\r\n\t\tif (!existsSync(resolvedPath)) {\r\n\t\t\tallDiagnostics.push({ type: \"warning\", message: \"skill path does not exist\", path: resolvedPath });\r\n\t\t\tcontinue;\r\n\t\t}\r\n\r\n\t\ttry {\r\n\t\t\tconst stats = statSync(resolvedPath);\r\n\t\t\tconst source = getSource(resolvedPath);\r\n\t\t\tif (stats.isDirectory()) {\r\n\t\t\t\taddSkills(loadSkillsFromDirInternal(resolvedPath, source, true));\r\n\t\t\t} else if (stats.isFile() && resolvedPath.endsWith(\".md\")) {\r\n\t\t\t\tconst result = loadSkillFromFile(resolvedPath, source);\r\n\t\t\t\tif (result.skill) {\r\n\t\t\t\t\taddSkills({ skills: [result.skill], diagnostics: result.diagnostics });\r\n\t\t\t\t} else {\r\n\t\t\t\t\tallDiagnostics.push(...result.diagnostics);\r\n\t\t\t\t}\r\n\t\t\t} else {\r\n\t\t\t\tallDiagnostics.push({ type: \"warning\", message: \"skill path is not a markdown file\", path: resolvedPath });\r\n\t\t\t}\r\n\t\t} catch (error) {\r\n\t\t\tconst message = error instanceof Error ? error.message : \"failed to read skill path\";\r\n\t\t\tallDiagnostics.push({ type: \"warning\", message, path: resolvedPath });\r\n\t\t}\r\n\t}\r\n\r\n\treturn {\r\n\t\tskills: Array.from(skillMap.values()),\r\n\t\tdiagnostics: [...allDiagnostics, ...collisionDiagnostics],\r\n\t};\r\n}\r\n"]}
|
|
@@ -0,0 +1,364 @@
|
|
|
1
|
+
import { existsSync, readdirSync, readFileSync, realpathSync, statSync } from "fs";
|
|
2
|
+
import ignore from "ignore";
|
|
3
|
+
import { homedir } from "os";
|
|
4
|
+
import { basename, dirname, isAbsolute, join, relative, resolve, sep } from "path";
|
|
5
|
+
import { CONFIG_DIR_NAME, getAgentDir } from "../config.js";
|
|
6
|
+
import { parseFrontmatter } from "../utils/frontmatter.js";
|
|
7
|
+
/** Max name length per spec */
|
|
8
|
+
const MAX_NAME_LENGTH = 64;
|
|
9
|
+
/** Max description length per spec */
|
|
10
|
+
const MAX_DESCRIPTION_LENGTH = 1024;
|
|
11
|
+
const IGNORE_FILE_NAMES = [".gitignore", ".ignore", ".fdignore"];
|
|
12
|
+
function toPosixPath(p) {
|
|
13
|
+
return p.split(sep).join("/");
|
|
14
|
+
}
|
|
15
|
+
function prefixIgnorePattern(line, prefix) {
|
|
16
|
+
const trimmed = line.trim();
|
|
17
|
+
if (!trimmed)
|
|
18
|
+
return null;
|
|
19
|
+
if (trimmed.startsWith("#") && !trimmed.startsWith("\\#"))
|
|
20
|
+
return null;
|
|
21
|
+
let pattern = line;
|
|
22
|
+
let negated = false;
|
|
23
|
+
if (pattern.startsWith("!")) {
|
|
24
|
+
negated = true;
|
|
25
|
+
pattern = pattern.slice(1);
|
|
26
|
+
}
|
|
27
|
+
else if (pattern.startsWith("\\!")) {
|
|
28
|
+
pattern = pattern.slice(1);
|
|
29
|
+
}
|
|
30
|
+
if (pattern.startsWith("/")) {
|
|
31
|
+
pattern = pattern.slice(1);
|
|
32
|
+
}
|
|
33
|
+
const prefixed = prefix ? `${prefix}${pattern}` : pattern;
|
|
34
|
+
return negated ? `!${prefixed}` : prefixed;
|
|
35
|
+
}
|
|
36
|
+
function addIgnoreRules(ig, dir, rootDir) {
|
|
37
|
+
const relativeDir = relative(rootDir, dir);
|
|
38
|
+
const prefix = relativeDir ? `${toPosixPath(relativeDir)}/` : "";
|
|
39
|
+
for (const filename of IGNORE_FILE_NAMES) {
|
|
40
|
+
const ignorePath = join(dir, filename);
|
|
41
|
+
if (!existsSync(ignorePath))
|
|
42
|
+
continue;
|
|
43
|
+
try {
|
|
44
|
+
const content = readFileSync(ignorePath, "utf-8");
|
|
45
|
+
const patterns = content
|
|
46
|
+
.split(/\r?\n/)
|
|
47
|
+
.map((line) => prefixIgnorePattern(line, prefix))
|
|
48
|
+
.filter((line) => Boolean(line));
|
|
49
|
+
if (patterns.length > 0) {
|
|
50
|
+
ig.add(patterns);
|
|
51
|
+
}
|
|
52
|
+
}
|
|
53
|
+
catch { }
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
/**
|
|
57
|
+
* Validate skill name per Agent Skills spec.
|
|
58
|
+
* Returns array of validation error messages (empty if valid).
|
|
59
|
+
*/
|
|
60
|
+
function validateName(name, parentDirName) {
|
|
61
|
+
const errors = [];
|
|
62
|
+
if (name !== parentDirName) {
|
|
63
|
+
errors.push(`name "${name}" does not match parent directory "${parentDirName}"`);
|
|
64
|
+
}
|
|
65
|
+
if (name.length > MAX_NAME_LENGTH) {
|
|
66
|
+
errors.push(`name exceeds ${MAX_NAME_LENGTH} characters (${name.length})`);
|
|
67
|
+
}
|
|
68
|
+
if (!/^[a-z0-9-]+$/.test(name)) {
|
|
69
|
+
errors.push(`name contains invalid characters (must be lowercase a-z, 0-9, hyphens only)`);
|
|
70
|
+
}
|
|
71
|
+
if (name.startsWith("-") || name.endsWith("-")) {
|
|
72
|
+
errors.push(`name must not start or end with a hyphen`);
|
|
73
|
+
}
|
|
74
|
+
if (name.includes("--")) {
|
|
75
|
+
errors.push(`name must not contain consecutive hyphens`);
|
|
76
|
+
}
|
|
77
|
+
return errors;
|
|
78
|
+
}
|
|
79
|
+
/**
|
|
80
|
+
* Validate description per Agent Skills spec.
|
|
81
|
+
*/
|
|
82
|
+
function validateDescription(description) {
|
|
83
|
+
const errors = [];
|
|
84
|
+
if (!description || description.trim() === "") {
|
|
85
|
+
errors.push("description is required");
|
|
86
|
+
}
|
|
87
|
+
else if (description.length > MAX_DESCRIPTION_LENGTH) {
|
|
88
|
+
errors.push(`description exceeds ${MAX_DESCRIPTION_LENGTH} characters (${description.length})`);
|
|
89
|
+
}
|
|
90
|
+
return errors;
|
|
91
|
+
}
|
|
92
|
+
/**
|
|
93
|
+
* Load skills from a directory.
|
|
94
|
+
*
|
|
95
|
+
* Discovery rules:
|
|
96
|
+
* - direct .md children in the root
|
|
97
|
+
* - recursive SKILL.md under subdirectories
|
|
98
|
+
*/
|
|
99
|
+
export function loadSkillsFromDir(options) {
|
|
100
|
+
const { dir, source } = options;
|
|
101
|
+
return loadSkillsFromDirInternal(dir, source, true);
|
|
102
|
+
}
|
|
103
|
+
function loadSkillsFromDirInternal(dir, source, includeRootFiles, ignoreMatcher, rootDir) {
|
|
104
|
+
const skills = [];
|
|
105
|
+
const diagnostics = [];
|
|
106
|
+
if (!existsSync(dir)) {
|
|
107
|
+
return { skills, diagnostics };
|
|
108
|
+
}
|
|
109
|
+
const root = rootDir ?? dir;
|
|
110
|
+
const ig = ignoreMatcher ?? ignore();
|
|
111
|
+
addIgnoreRules(ig, dir, root);
|
|
112
|
+
try {
|
|
113
|
+
const entries = readdirSync(dir, { withFileTypes: true });
|
|
114
|
+
for (const entry of entries) {
|
|
115
|
+
if (entry.name.startsWith(".")) {
|
|
116
|
+
continue;
|
|
117
|
+
}
|
|
118
|
+
// Skip node_modules to avoid scanning dependencies
|
|
119
|
+
if (entry.name === "node_modules") {
|
|
120
|
+
continue;
|
|
121
|
+
}
|
|
122
|
+
const fullPath = join(dir, entry.name);
|
|
123
|
+
// For symlinks, check if they point to a directory and follow them
|
|
124
|
+
let isDirectory = entry.isDirectory();
|
|
125
|
+
let isFile = entry.isFile();
|
|
126
|
+
if (entry.isSymbolicLink()) {
|
|
127
|
+
try {
|
|
128
|
+
const stats = statSync(fullPath);
|
|
129
|
+
isDirectory = stats.isDirectory();
|
|
130
|
+
isFile = stats.isFile();
|
|
131
|
+
}
|
|
132
|
+
catch {
|
|
133
|
+
// Broken symlink, skip it
|
|
134
|
+
continue;
|
|
135
|
+
}
|
|
136
|
+
}
|
|
137
|
+
const relPath = toPosixPath(relative(root, fullPath));
|
|
138
|
+
const ignorePath = isDirectory ? `${relPath}/` : relPath;
|
|
139
|
+
if (ig.ignores(ignorePath)) {
|
|
140
|
+
continue;
|
|
141
|
+
}
|
|
142
|
+
if (isDirectory) {
|
|
143
|
+
const subResult = loadSkillsFromDirInternal(fullPath, source, false, ig, root);
|
|
144
|
+
skills.push(...subResult.skills);
|
|
145
|
+
diagnostics.push(...subResult.diagnostics);
|
|
146
|
+
continue;
|
|
147
|
+
}
|
|
148
|
+
if (!isFile) {
|
|
149
|
+
continue;
|
|
150
|
+
}
|
|
151
|
+
const isRootMd = includeRootFiles && entry.name.endsWith(".md");
|
|
152
|
+
const isSkillMd = !includeRootFiles && entry.name === "SKILL.md";
|
|
153
|
+
if (!isRootMd && !isSkillMd) {
|
|
154
|
+
continue;
|
|
155
|
+
}
|
|
156
|
+
const result = loadSkillFromFile(fullPath, source);
|
|
157
|
+
if (result.skill) {
|
|
158
|
+
skills.push(result.skill);
|
|
159
|
+
}
|
|
160
|
+
diagnostics.push(...result.diagnostics);
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
catch { }
|
|
164
|
+
return { skills, diagnostics };
|
|
165
|
+
}
|
|
166
|
+
function loadSkillFromFile(filePath, source) {
|
|
167
|
+
const diagnostics = [];
|
|
168
|
+
try {
|
|
169
|
+
const rawContent = readFileSync(filePath, "utf-8");
|
|
170
|
+
const { frontmatter } = parseFrontmatter(rawContent);
|
|
171
|
+
const skillDir = dirname(filePath);
|
|
172
|
+
const parentDirName = basename(skillDir);
|
|
173
|
+
// Validate description
|
|
174
|
+
const descErrors = validateDescription(frontmatter.description);
|
|
175
|
+
for (const error of descErrors) {
|
|
176
|
+
diagnostics.push({ type: "warning", message: error, path: filePath });
|
|
177
|
+
}
|
|
178
|
+
// Use name from frontmatter, or fall back to parent directory name
|
|
179
|
+
const name = frontmatter.name || parentDirName;
|
|
180
|
+
// Validate name
|
|
181
|
+
const nameErrors = validateName(name, parentDirName);
|
|
182
|
+
for (const error of nameErrors) {
|
|
183
|
+
diagnostics.push({ type: "warning", message: error, path: filePath });
|
|
184
|
+
}
|
|
185
|
+
// Still load the skill even with warnings (unless description is completely missing)
|
|
186
|
+
if (!frontmatter.description || frontmatter.description.trim() === "") {
|
|
187
|
+
return { skill: null, diagnostics };
|
|
188
|
+
}
|
|
189
|
+
return {
|
|
190
|
+
skill: {
|
|
191
|
+
name,
|
|
192
|
+
description: frontmatter.description,
|
|
193
|
+
filePath,
|
|
194
|
+
baseDir: skillDir,
|
|
195
|
+
source,
|
|
196
|
+
disableModelInvocation: frontmatter["disable-model-invocation"] === true,
|
|
197
|
+
},
|
|
198
|
+
diagnostics,
|
|
199
|
+
};
|
|
200
|
+
}
|
|
201
|
+
catch (error) {
|
|
202
|
+
const message = error instanceof Error ? error.message : "failed to parse skill file";
|
|
203
|
+
diagnostics.push({ type: "warning", message, path: filePath });
|
|
204
|
+
return { skill: null, diagnostics };
|
|
205
|
+
}
|
|
206
|
+
}
|
|
207
|
+
/**
|
|
208
|
+
* Format skills for inclusion in a system prompt.
|
|
209
|
+
* Uses XML format per Agent Skills standard.
|
|
210
|
+
* See: https://agentskills.io/integrate-skills
|
|
211
|
+
*
|
|
212
|
+
* Skills with disableModelInvocation=true are excluded from the prompt
|
|
213
|
+
* (they can only be invoked explicitly via /skill:name commands).
|
|
214
|
+
*/
|
|
215
|
+
export function formatSkillsForPrompt(skills) {
|
|
216
|
+
const visibleSkills = skills.filter((s) => !s.disableModelInvocation);
|
|
217
|
+
if (visibleSkills.length === 0) {
|
|
218
|
+
return "";
|
|
219
|
+
}
|
|
220
|
+
const lines = [
|
|
221
|
+
"\n\nThe following skills provide specialized instructions for specific tasks.",
|
|
222
|
+
"Use the read tool to load a skill's file when the task matches its description.",
|
|
223
|
+
"When a skill file references a relative path, resolve it against the skill directory (parent of SKILL.md / dirname of the path) and use that absolute path in tool commands.",
|
|
224
|
+
"",
|
|
225
|
+
"<available_skills>",
|
|
226
|
+
];
|
|
227
|
+
for (const skill of visibleSkills) {
|
|
228
|
+
lines.push(" <skill>");
|
|
229
|
+
lines.push(` <name>${escapeXml(skill.name)}</name>`);
|
|
230
|
+
lines.push(` <description>${escapeXml(skill.description)}</description>`);
|
|
231
|
+
lines.push(` <location>${escapeXml(skill.filePath)}</location>`);
|
|
232
|
+
lines.push(" </skill>");
|
|
233
|
+
}
|
|
234
|
+
lines.push("</available_skills>");
|
|
235
|
+
return lines.join("\n");
|
|
236
|
+
}
|
|
237
|
+
function escapeXml(str) {
|
|
238
|
+
return str
|
|
239
|
+
.replace(/&/g, "&")
|
|
240
|
+
.replace(/</g, "<")
|
|
241
|
+
.replace(/>/g, ">")
|
|
242
|
+
.replace(/"/g, """)
|
|
243
|
+
.replace(/'/g, "'");
|
|
244
|
+
}
|
|
245
|
+
function normalizePath(input) {
|
|
246
|
+
const trimmed = input.trim();
|
|
247
|
+
if (trimmed === "~")
|
|
248
|
+
return homedir();
|
|
249
|
+
if (trimmed.startsWith("~/"))
|
|
250
|
+
return join(homedir(), trimmed.slice(2));
|
|
251
|
+
if (trimmed.startsWith("~"))
|
|
252
|
+
return join(homedir(), trimmed.slice(1));
|
|
253
|
+
return trimmed;
|
|
254
|
+
}
|
|
255
|
+
function resolveSkillPath(p, cwd) {
|
|
256
|
+
const normalized = normalizePath(p);
|
|
257
|
+
return isAbsolute(normalized) ? normalized : resolve(cwd, normalized);
|
|
258
|
+
}
|
|
259
|
+
/**
|
|
260
|
+
* Load skills from all configured locations.
|
|
261
|
+
* Returns skills and any validation diagnostics.
|
|
262
|
+
*/
|
|
263
|
+
export function loadSkills(options = {}) {
|
|
264
|
+
const { cwd = process.cwd(), agentDir, skillPaths = [], includeDefaults = true } = options;
|
|
265
|
+
// Resolve agentDir - if not provided, use default from config
|
|
266
|
+
const resolvedAgentDir = agentDir ?? getAgentDir();
|
|
267
|
+
const skillMap = new Map();
|
|
268
|
+
const realPathSet = new Set();
|
|
269
|
+
const allDiagnostics = [];
|
|
270
|
+
const collisionDiagnostics = [];
|
|
271
|
+
function addSkills(result) {
|
|
272
|
+
allDiagnostics.push(...result.diagnostics);
|
|
273
|
+
for (const skill of result.skills) {
|
|
274
|
+
// Resolve symlinks to detect duplicate files
|
|
275
|
+
let realPath;
|
|
276
|
+
try {
|
|
277
|
+
realPath = realpathSync(skill.filePath);
|
|
278
|
+
}
|
|
279
|
+
catch {
|
|
280
|
+
realPath = skill.filePath;
|
|
281
|
+
}
|
|
282
|
+
// Skip silently if we've already loaded this exact file (via symlink)
|
|
283
|
+
if (realPathSet.has(realPath)) {
|
|
284
|
+
continue;
|
|
285
|
+
}
|
|
286
|
+
const existing = skillMap.get(skill.name);
|
|
287
|
+
if (existing) {
|
|
288
|
+
collisionDiagnostics.push({
|
|
289
|
+
type: "collision",
|
|
290
|
+
message: `name "${skill.name}" collision`,
|
|
291
|
+
path: skill.filePath,
|
|
292
|
+
collision: {
|
|
293
|
+
resourceType: "skill",
|
|
294
|
+
name: skill.name,
|
|
295
|
+
winnerPath: existing.filePath,
|
|
296
|
+
loserPath: skill.filePath,
|
|
297
|
+
},
|
|
298
|
+
});
|
|
299
|
+
}
|
|
300
|
+
else {
|
|
301
|
+
skillMap.set(skill.name, skill);
|
|
302
|
+
realPathSet.add(realPath);
|
|
303
|
+
}
|
|
304
|
+
}
|
|
305
|
+
}
|
|
306
|
+
if (includeDefaults) {
|
|
307
|
+
addSkills(loadSkillsFromDirInternal(join(resolvedAgentDir, "skills"), "user", true));
|
|
308
|
+
addSkills(loadSkillsFromDirInternal(resolve(cwd, CONFIG_DIR_NAME, "skills"), "project", true));
|
|
309
|
+
}
|
|
310
|
+
const userSkillsDir = join(resolvedAgentDir, "skills");
|
|
311
|
+
const projectSkillsDir = resolve(cwd, CONFIG_DIR_NAME, "skills");
|
|
312
|
+
const isUnderPath = (target, root) => {
|
|
313
|
+
const normalizedRoot = resolve(root);
|
|
314
|
+
if (target === normalizedRoot) {
|
|
315
|
+
return true;
|
|
316
|
+
}
|
|
317
|
+
const prefix = normalizedRoot.endsWith(sep) ? normalizedRoot : `${normalizedRoot}${sep}`;
|
|
318
|
+
return target.startsWith(prefix);
|
|
319
|
+
};
|
|
320
|
+
const getSource = (resolvedPath) => {
|
|
321
|
+
if (!includeDefaults) {
|
|
322
|
+
if (isUnderPath(resolvedPath, userSkillsDir))
|
|
323
|
+
return "user";
|
|
324
|
+
if (isUnderPath(resolvedPath, projectSkillsDir))
|
|
325
|
+
return "project";
|
|
326
|
+
}
|
|
327
|
+
return "path";
|
|
328
|
+
};
|
|
329
|
+
for (const rawPath of skillPaths) {
|
|
330
|
+
const resolvedPath = resolveSkillPath(rawPath, cwd);
|
|
331
|
+
if (!existsSync(resolvedPath)) {
|
|
332
|
+
allDiagnostics.push({ type: "warning", message: "skill path does not exist", path: resolvedPath });
|
|
333
|
+
continue;
|
|
334
|
+
}
|
|
335
|
+
try {
|
|
336
|
+
const stats = statSync(resolvedPath);
|
|
337
|
+
const source = getSource(resolvedPath);
|
|
338
|
+
if (stats.isDirectory()) {
|
|
339
|
+
addSkills(loadSkillsFromDirInternal(resolvedPath, source, true));
|
|
340
|
+
}
|
|
341
|
+
else if (stats.isFile() && resolvedPath.endsWith(".md")) {
|
|
342
|
+
const result = loadSkillFromFile(resolvedPath, source);
|
|
343
|
+
if (result.skill) {
|
|
344
|
+
addSkills({ skills: [result.skill], diagnostics: result.diagnostics });
|
|
345
|
+
}
|
|
346
|
+
else {
|
|
347
|
+
allDiagnostics.push(...result.diagnostics);
|
|
348
|
+
}
|
|
349
|
+
}
|
|
350
|
+
else {
|
|
351
|
+
allDiagnostics.push({ type: "warning", message: "skill path is not a markdown file", path: resolvedPath });
|
|
352
|
+
}
|
|
353
|
+
}
|
|
354
|
+
catch (error) {
|
|
355
|
+
const message = error instanceof Error ? error.message : "failed to read skill path";
|
|
356
|
+
allDiagnostics.push({ type: "warning", message, path: resolvedPath });
|
|
357
|
+
}
|
|
358
|
+
}
|
|
359
|
+
return {
|
|
360
|
+
skills: Array.from(skillMap.values()),
|
|
361
|
+
diagnostics: [...allDiagnostics, ...collisionDiagnostics],
|
|
362
|
+
};
|
|
363
|
+
}
|
|
364
|
+
//# sourceMappingURL=skills.js.map
|