openvibe 0.57.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 +3041 -0
- package/README.md +569 -0
- package/dist/cli/args.d.ts +44 -0
- package/dist/cli/args.d.ts.map +1 -0
- package/dist/cli/args.js +272 -0
- package/dist/cli/args.js.map +1 -0
- package/dist/cli/config-selector.d.ts +10 -0
- package/dist/cli/config-selector.d.ts.map +1 -0
- package/dist/cli/config-selector.js +26 -0
- package/dist/cli/config-selector.js.map +1 -0
- package/dist/cli/file-processor.d.ts +10 -0
- package/dist/cli/file-processor.d.ts.map +1 -0
- package/dist/cli/file-processor.js +68 -0
- package/dist/cli/file-processor.js.map +1 -0
- package/dist/cli/list-models.d.ts +3 -0
- package/dist/cli/list-models.d.ts.map +1 -0
- package/dist/cli/list-models.js +30 -0
- package/dist/cli/list-models.js.map +1 -0
- package/dist/cli/session-picker.d.ts +5 -0
- package/dist/cli/session-picker.d.ts.map +1 -0
- package/dist/cli/session-picker.js +30 -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 +31 -0
- package/dist/cli.js.map +1 -0
- package/dist/config.d.ts +29 -0
- package/dist/config.d.ts.map +1 -0
- package/dist/config.js +145 -0
- package/dist/config.js.map +1 -0
- package/dist/core/accelerated-client.d.ts +42 -0
- package/dist/core/accelerated-client.d.ts.map +1 -0
- package/dist/core/accelerated-client.js +97 -0
- package/dist/core/accelerated-client.js.map +1 -0
- package/dist/core/accelerated-stream.d.ts +41 -0
- package/dist/core/accelerated-stream.d.ts.map +1 -0
- package/dist/core/accelerated-stream.js +133 -0
- package/dist/core/accelerated-stream.js.map +1 -0
- package/dist/core/agent-session.d.ts +566 -0
- package/dist/core/agent-session.d.ts.map +1 -0
- package/dist/core/agent-session.js +2361 -0
- package/dist/core/agent-session.js.map +1 -0
- package/dist/core/api-concurrency.d.ts +37 -0
- package/dist/core/api-concurrency.d.ts.map +1 -0
- package/dist/core/api-concurrency.js +179 -0
- package/dist/core/api-concurrency.js.map +1 -0
- package/dist/core/auth-storage.d.ts +66 -0
- package/dist/core/auth-storage.d.ts.map +1 -0
- package/dist/core/auth-storage.js +336 -0
- package/dist/core/auth-storage.js.map +1 -0
- package/dist/core/bash-executor.d.ts +15 -0
- package/dist/core/bash-executor.d.ts.map +1 -0
- package/dist/core/bash-executor.js +166 -0
- package/dist/core/bash-executor.js.map +1 -0
- package/dist/core/branded-ai.d.ts +17 -0
- package/dist/core/branded-ai.d.ts.map +1 -0
- package/dist/core/branded-ai.js +123 -0
- package/dist/core/branded-ai.js.map +1 -0
- package/dist/core/compaction/branch-summarization.d.ts +37 -0
- package/dist/core/compaction/branch-summarization.d.ts.map +1 -0
- package/dist/core/compaction/branch-summarization.js +158 -0
- package/dist/core/compaction/branch-summarization.js.map +1 -0
- package/dist/core/compaction/compaction.d.ts +56 -0
- package/dist/core/compaction/compaction.d.ts.map +1 -0
- package/dist/core/compaction/compaction.js +501 -0
- package/dist/core/compaction/compaction.js.map +1 -0
- package/dist/core/compaction/index.d.ts +4 -0
- package/dist/core/compaction/index.d.ts.map +1 -0
- package/dist/core/compaction/index.js +4 -0
- package/dist/core/compaction/index.js.map +1 -0
- package/dist/core/compaction/utils.d.ts +17 -0
- package/dist/core/compaction/utils.d.ts.map +1 -0
- package/dist/core/compaction/utils.js +120 -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 +13 -0
- package/dist/core/exec.d.ts.map +1 -0
- package/dist/core/exec.js +61 -0
- package/dist/core/exec.js.map +1 -0
- package/dist/core/export-html/ansi-to-html.d.ts +3 -0
- package/dist/core/export-html/ansi-to-html.d.ts.map +1 -0
- package/dist/core/export-html/ansi-to-html.js +193 -0
- package/dist/core/export-html/ansi-to-html.js.map +1 -0
- package/dist/core/export-html/index.d.ts +22 -0
- package/dist/core/export-html/index.d.ts.map +1 -0
- package/dist/core/export-html/index.js +193 -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 +1583 -0
- package/dist/core/export-html/tool-renderer.d.ts +21 -0
- package/dist/core/export-html/tool-renderer.d.ts.map +1 -0
- package/dist/core/export-html/tool-renderer.js +51 -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 +8 -0
- package/dist/core/extensions/index.d.ts.map +1 -0
- package/dist/core/extensions/index.js +5 -0
- package/dist/core/extensions/index.js.map +1 -0
- package/dist/core/extensions/loader.d.ts +7 -0
- package/dist/core/extensions/loader.d.ts.map +1 -0
- package/dist/core/extensions/loader.js +349 -0
- package/dist/core/extensions/loader.js.map +1 -0
- package/dist/core/extensions/runner.d.ts +124 -0
- package/dist/core/extensions/runner.d.ts.map +1 -0
- package/dist/core/extensions/runner.js +657 -0
- package/dist/core/extensions/runner.js.map +1 -0
- package/dist/core/extensions/types.d.ts +1032 -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 +8 -0
- package/dist/core/extensions/wrapper.d.ts.map +1 -0
- package/dist/core/extensions/wrapper.js +79 -0
- package/dist/core/extensions/wrapper.js.map +1 -0
- package/dist/core/footer-data-provider.d.ts +19 -0
- package/dist/core/footer-data-provider.d.ts.map +1 -0
- package/dist/core/footer-data-provider.js +113 -0
- package/dist/core/footer-data-provider.js.map +1 -0
- package/dist/core/index.d.ts +13 -0
- package/dist/core/index.d.ts.map +1 -0
- package/dist/core/index.js +11 -0
- package/dist/core/index.js.map +1 -0
- package/dist/core/keybindings.d.ts +22 -0
- package/dist/core/keybindings.d.ts.map +1 -0
- package/dist/core/keybindings.js +124 -0
- package/dist/core/keybindings.js.map +1 -0
- package/dist/core/messages.d.ts +51 -0
- package/dist/core/messages.d.ts.map +1 -0
- package/dist/core/messages.js +102 -0
- package/dist/core/messages.js.map +1 -0
- package/dist/core/model-registry.d.ts +26 -0
- package/dist/core/model-registry.d.ts.map +1 -0
- package/dist/core/model-registry.js +61 -0
- package/dist/core/model-registry.js.map +1 -0
- package/dist/core/model-resolver.d.ts +20 -0
- package/dist/core/model-resolver.d.ts.map +1 -0
- package/dist/core/model-resolver.js +47 -0
- package/dist/core/model-resolver.js.map +1 -0
- package/dist/core/multi-gpu-executor.d.ts +50 -0
- package/dist/core/multi-gpu-executor.d.ts.map +1 -0
- package/dist/core/multi-gpu-executor.js +201 -0
- package/dist/core/multi-gpu-executor.js.map +1 -0
- package/dist/core/onboarding.d.ts +3 -0
- package/dist/core/onboarding.d.ts.map +1 -0
- package/dist/core/onboarding.js +109 -0
- package/dist/core/onboarding.js.map +1 -0
- package/dist/core/package-manager.d.ts +136 -0
- package/dist/core/package-manager.d.ts.map +1 -0
- package/dist/core/package-manager.js +1375 -0
- package/dist/core/package-manager.js.map +1 -0
- package/dist/core/prompt-templates.d.ts +18 -0
- package/dist/core/prompt-templates.d.ts.map +1 -0
- package/dist/core/prompt-templates.js +204 -0
- package/dist/core/prompt-templates.js.map +1 -0
- package/dist/core/resolve-config-value.d.ts +4 -0
- package/dist/core/resolve-config-value.d.ts.map +1 -0
- package/dist/core/resolve-config-value.js +45 -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 +661 -0
- package/dist/core/resource-loader.js.map +1 -0
- package/dist/core/response-accelerator.d.ts +51 -0
- package/dist/core/response-accelerator.d.ts.map +1 -0
- package/dist/core/response-accelerator.js +149 -0
- package/dist/core/response-accelerator.js.map +1 -0
- package/dist/core/sdk.d.ts +39 -0
- package/dist/core/sdk.d.ts.map +1 -0
- package/dist/core/sdk.js +151 -0
- package/dist/core/sdk.js.map +1 -0
- package/dist/core/session-manager.d.ts +160 -0
- package/dist/core/session-manager.d.ts.map +1 -0
- package/dist/core/session-manager.js +899 -0
- package/dist/core/session-manager.js.map +1 -0
- package/dist/core/settings-manager.d.ts +220 -0
- package/dist/core/settings-manager.d.ts.map +1 -0
- package/dist/core/settings-manager.js +673 -0
- package/dist/core/settings-manager.js.map +1 -0
- package/dist/core/skills.d.ts +33 -0
- package/dist/core/skills.d.ts.map +1 -0
- package/dist/core/skills.js +326 -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 +19 -0
- package/dist/core/slash-commands.js.map +1 -0
- package/dist/core/system-prompt.d.ts +23 -0
- package/dist/core/system-prompt.d.ts.map +1 -0
- package/dist/core/system-prompt.js +154 -0
- package/dist/core/system-prompt.js.map +1 -0
- package/dist/core/timings.d.ts +3 -0
- package/dist/core/timings.d.ts.map +1 -0
- package/dist/core/timings.js +21 -0
- package/dist/core/timings.js.map +1 -0
- package/dist/core/tools/bash.d.ts +40 -0
- package/dist/core/tools/bash.d.ts.map +1 -0
- package/dist/core/tools/bash.js +212 -0
- package/dist/core/tools/bash.js.map +1 -0
- package/dist/core/tools/edit-diff.d.ts +29 -0
- package/dist/core/tools/edit-diff.d.ts.map +1 -0
- package/dist/core/tools/edit-diff.js +182 -0
- package/dist/core/tools/edit-diff.js.map +1 -0
- package/dist/core/tools/edit.d.ts +28 -0
- package/dist/core/tools/edit.d.ts.map +1 -0
- package/dist/core/tools/edit.js +129 -0
- package/dist/core/tools/edit.js.map +1 -0
- package/dist/core/tools/fast-executor.d.ts +33 -0
- package/dist/core/tools/fast-executor.d.ts.map +1 -0
- package/dist/core/tools/fast-executor.js +83 -0
- package/dist/core/tools/fast-executor.js.map +1 -0
- package/dist/core/tools/find.d.ts +28 -0
- package/dist/core/tools/find.d.ts.map +1 -0
- package/dist/core/tools/find.js +50 -0
- package/dist/core/tools/find.js.map +1 -0
- package/dist/core/tools/grep.d.ts +37 -0
- package/dist/core/tools/grep.d.ts.map +1 -0
- package/dist/core/tools/grep.js +233 -0
- package/dist/core/tools/grep.js.map +1 -0
- package/dist/core/tools/index.d.ts +63 -0
- package/dist/core/tools/index.d.ts.map +1 -0
- package/dist/core/tools/index.js +52 -0
- package/dist/core/tools/index.js.map +1 -0
- package/dist/core/tools/local-accelerator.d.ts +65 -0
- package/dist/core/tools/local-accelerator.d.ts.map +1 -0
- package/dist/core/tools/local-accelerator.js +198 -0
- package/dist/core/tools/local-accelerator.js.map +1 -0
- package/dist/core/tools/ls.d.ts +31 -0
- package/dist/core/tools/ls.d.ts.map +1 -0
- package/dist/core/tools/ls.js +109 -0
- package/dist/core/tools/ls.js.map +1 -0
- package/dist/core/tools/parallel-executor.d.ts +60 -0
- package/dist/core/tools/parallel-executor.d.ts.map +1 -0
- package/dist/core/tools/parallel-executor.js +257 -0
- package/dist/core/tools/parallel-executor.js.map +1 -0
- package/dist/core/tools/path-utils.d.ts +4 -0
- package/dist/core/tools/path-utils.d.ts.map +1 -0
- package/dist/core/tools/path-utils.js +70 -0
- package/dist/core/tools/path-utils.js.map +1 -0
- package/dist/core/tools/read.d.ts +29 -0
- package/dist/core/tools/read.d.ts.map +1 -0
- package/dist/core/tools/read.js +146 -0
- package/dist/core/tools/read.js.map +1 -0
- package/dist/core/tools/truncate.d.ts +28 -0
- package/dist/core/tools/truncate.d.ts.map +1 -0
- package/dist/core/tools/truncate.js +161 -0
- package/dist/core/tools/truncate.js.map +1 -0
- package/dist/core/tools/write.d.ts +21 -0
- package/dist/core/tools/write.d.ts.map +1 -0
- package/dist/core/tools/write.js +69 -0
- package/dist/core/tools/write.js.map +1 -0
- package/dist/core/user-config.d.ts +26 -0
- package/dist/core/user-config.d.ts.map +1 -0
- package/dist/core/user-config.js +75 -0
- package/dist/core/user-config.js.map +1 -0
- package/dist/index.d.ts +27 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +42 -0
- package/dist/index.js.map +1 -0
- package/dist/main.d.ts +2 -0
- package/dist/main.d.ts.map +1 -0
- package/dist/main.js +609 -0
- package/dist/main.js.map +1 -0
- package/dist/migrations.d.ts +8 -0
- package/dist/migrations.d.ts.map +1 -0
- package/dist/migrations.js +197 -0
- package/dist/migrations.js.map +1 -0
- package/dist/modes/index.d.ts +6 -0
- package/dist/modes/index.d.ts.map +1 -0
- package/dist/modes/index.js +5 -0
- package/dist/modes/index.js.map +1 -0
- package/dist/modes/interactive/components/armin.d.ts +31 -0
- package/dist/modes/interactive/components/armin.d.ts.map +1 -0
- package/dist/modes/interactive/components/armin.js +306 -0
- package/dist/modes/interactive/components/armin.js.map +1 -0
- package/dist/modes/interactive/components/assistant-message.d.ts +13 -0
- package/dist/modes/interactive/components/assistant-message.d.ts.map +1 -0
- package/dist/modes/interactive/components/assistant-message.js +82 -0
- package/dist/modes/interactive/components/assistant-message.js.map +1 -0
- package/dist/modes/interactive/components/bash-execution.d.ts +23 -0
- package/dist/modes/interactive/components/bash-execution.d.ts.map +1 -0
- package/dist/modes/interactive/components/bash-execution.js +126 -0
- package/dist/modes/interactive/components/bash-execution.js.map +1 -0
- package/dist/modes/interactive/components/bordered-loader.d.ts +15 -0
- package/dist/modes/interactive/components/bordered-loader.d.ts.map +1 -0
- package/dist/modes/interactive/components/bordered-loader.js +50 -0
- package/dist/modes/interactive/components/bordered-loader.js.map +1 -0
- package/dist/modes/interactive/components/branch-summary-message.d.ts +12 -0
- package/dist/modes/interactive/components/branch-summary-message.d.ts.map +1 -0
- package/dist/modes/interactive/components/branch-summary-message.js +40 -0
- package/dist/modes/interactive/components/branch-summary-message.js.map +1 -0
- package/dist/modes/interactive/components/compaction-summary-message.d.ts +12 -0
- package/dist/modes/interactive/components/compaction-summary-message.d.ts.map +1 -0
- package/dist/modes/interactive/components/compaction-summary-message.js +41 -0
- package/dist/modes/interactive/components/compaction-summary-message.js.map +1 -0
- package/dist/modes/interactive/components/config-selector.d.ts +68 -0
- package/dist/modes/interactive/components/config-selector.d.ts.map +1 -0
- package/dist/modes/interactive/components/config-selector.js +451 -0
- package/dist/modes/interactive/components/config-selector.js.map +1 -0
- package/dist/modes/interactive/components/countdown-timer.d.ts +11 -0
- package/dist/modes/interactive/components/countdown-timer.d.ts.map +1 -0
- package/dist/modes/interactive/components/countdown-timer.js +30 -0
- package/dist/modes/interactive/components/countdown-timer.js.map +1 -0
- package/dist/modes/interactive/components/custom-editor.d.ts +14 -0
- package/dist/modes/interactive/components/custom-editor.d.ts.map +1 -0
- package/dist/modes/interactive/components/custom-editor.js +52 -0
- package/dist/modes/interactive/components/custom-editor.js.map +1 -0
- package/dist/modes/interactive/components/custom-message.d.ts +16 -0
- package/dist/modes/interactive/components/custom-message.d.ts.map +1 -0
- package/dist/modes/interactive/components/custom-message.js +66 -0
- package/dist/modes/interactive/components/custom-message.js.map +1 -0
- package/dist/modes/interactive/components/daxnuts.d.ts +18 -0
- package/dist/modes/interactive/components/daxnuts.d.ts.map +1 -0
- package/dist/modes/interactive/components/daxnuts.js +130 -0
- package/dist/modes/interactive/components/daxnuts.js.map +1 -0
- package/dist/modes/interactive/components/diff.d.ts +5 -0
- package/dist/modes/interactive/components/diff.d.ts.map +1 -0
- package/dist/modes/interactive/components/diff.js +106 -0
- package/dist/modes/interactive/components/diff.js.map +1 -0
- package/dist/modes/interactive/components/dynamic-border.d.ts +8 -0
- package/dist/modes/interactive/components/dynamic-border.d.ts.map +1 -0
- package/dist/modes/interactive/components/dynamic-border.js +12 -0
- package/dist/modes/interactive/components/dynamic-border.js.map +1 -0
- package/dist/modes/interactive/components/extension-editor.d.ts +16 -0
- package/dist/modes/interactive/components/extension-editor.d.ts.map +1 -0
- package/dist/modes/interactive/components/extension-editor.js +95 -0
- package/dist/modes/interactive/components/extension-editor.js.map +1 -0
- package/dist/modes/interactive/components/extension-input.d.ts +20 -0
- package/dist/modes/interactive/components/extension-input.d.ts.map +1 -0
- package/dist/modes/interactive/components/extension-input.js +57 -0
- package/dist/modes/interactive/components/extension-input.js.map +1 -0
- package/dist/modes/interactive/components/extension-selector.d.ts +20 -0
- package/dist/modes/interactive/components/extension-selector.d.ts.map +1 -0
- package/dist/modes/interactive/components/extension-selector.js +74 -0
- package/dist/modes/interactive/components/extension-selector.js.map +1 -0
- package/dist/modes/interactive/components/footer.d.ts +14 -0
- package/dist/modes/interactive/components/footer.d.ts.map +1 -0
- package/dist/modes/interactive/components/footer.js +78 -0
- package/dist/modes/interactive/components/footer.js.map +1 -0
- package/dist/modes/interactive/components/index.d.ts +28 -0
- package/dist/modes/interactive/components/index.d.ts.map +1 -0
- package/dist/modes/interactive/components/index.js +28 -0
- package/dist/modes/interactive/components/index.js.map +1 -0
- package/dist/modes/interactive/components/keybinding-hints.d.ts +8 -0
- package/dist/modes/interactive/components/keybinding-hints.d.ts.map +1 -0
- package/dist/modes/interactive/components/keybinding-hints.js +25 -0
- package/dist/modes/interactive/components/keybinding-hints.js.map +1 -0
- package/dist/modes/interactive/components/onboarding-wizard.d.ts +23 -0
- package/dist/modes/interactive/components/onboarding-wizard.d.ts.map +1 -0
- package/dist/modes/interactive/components/onboarding-wizard.js +250 -0
- package/dist/modes/interactive/components/onboarding-wizard.js.map +1 -0
- package/dist/modes/interactive/components/session-selector-search.d.ts +21 -0
- package/dist/modes/interactive/components/session-selector-search.d.ts.map +1 -0
- package/dist/modes/interactive/components/session-selector-search.js +149 -0
- package/dist/modes/interactive/components/session-selector-search.js.map +1 -0
- package/dist/modes/interactive/components/session-selector.d.ts +89 -0
- package/dist/modes/interactive/components/session-selector.d.ts.map +1 -0
- package/dist/modes/interactive/components/session-selector.js +786 -0
- package/dist/modes/interactive/components/session-selector.js.map +1 -0
- package/dist/modes/interactive/components/settings-selector.d.ts +55 -0
- package/dist/modes/interactive/components/settings-selector.d.ts.map +1 -0
- package/dist/modes/interactive/components/settings-selector.js +273 -0
- package/dist/modes/interactive/components/settings-selector.js.map +1 -0
- package/dist/modes/interactive/components/show-images-selector.d.ts +7 -0
- package/dist/modes/interactive/components/show-images-selector.d.ts.map +1 -0
- package/dist/modes/interactive/components/show-images-selector.js +28 -0
- package/dist/modes/interactive/components/show-images-selector.js.map +1 -0
- package/dist/modes/interactive/components/skill-invocation-message.d.ts +12 -0
- package/dist/modes/interactive/components/skill-invocation-message.d.ts.map +1 -0
- package/dist/modes/interactive/components/skill-invocation-message.js +40 -0
- package/dist/modes/interactive/components/skill-invocation-message.js.map +1 -0
- package/dist/modes/interactive/components/theme-selector.d.ts +8 -0
- package/dist/modes/interactive/components/theme-selector.d.ts.map +1 -0
- package/dist/modes/interactive/components/theme-selector.js +38 -0
- package/dist/modes/interactive/components/theme-selector.js.map +1 -0
- package/dist/modes/interactive/components/thinking-selector.d.ts +8 -0
- package/dist/modes/interactive/components/thinking-selector.d.ts.map +1 -0
- package/dist/modes/interactive/components/thinking-selector.js +40 -0
- package/dist/modes/interactive/components/thinking-selector.js.map +1 -0
- package/dist/modes/interactive/components/tool-execution.d.ts +54 -0
- package/dist/modes/interactive/components/tool-execution.d.ts.map +1 -0
- package/dist/modes/interactive/components/tool-execution.js +704 -0
- package/dist/modes/interactive/components/tool-execution.js.map +1 -0
- package/dist/modes/interactive/components/tree-selector.d.ts +59 -0
- package/dist/modes/interactive/components/tree-selector.d.ts.map +1 -0
- package/dist/modes/interactive/components/tree-selector.js +929 -0
- package/dist/modes/interactive/components/tree-selector.js.map +1 -0
- package/dist/modes/interactive/components/user-message-selector.d.ts +24 -0
- package/dist/modes/interactive/components/user-message-selector.d.ts.map +1 -0
- package/dist/modes/interactive/components/user-message-selector.js +89 -0
- package/dist/modes/interactive/components/user-message-selector.js.map +1 -0
- package/dist/modes/interactive/components/user-message.d.ts +6 -0
- package/dist/modes/interactive/components/user-message.d.ts.map +1 -0
- package/dist/modes/interactive/components/user-message.js +24 -0
- package/dist/modes/interactive/components/user-message.js.map +1 -0
- package/dist/modes/interactive/components/visual-truncate.d.ts +6 -0
- package/dist/modes/interactive/components/visual-truncate.d.ts.map +1 -0
- package/dist/modes/interactive/components/visual-truncate.js +15 -0
- package/dist/modes/interactive/components/visual-truncate.js.map +1 -0
- package/dist/modes/interactive/interactive-mode.d.ts +177 -0
- package/dist/modes/interactive/interactive-mode.d.ts.map +1 -0
- package/dist/modes/interactive/interactive-mode.js +3037 -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 +60 -0
- package/dist/modes/interactive/theme/theme.d.ts.map +1 -0
- package/dist/modes/interactive/theme/theme.js +852 -0
- package/dist/modes/interactive/theme/theme.js.map +1 -0
- package/dist/modes/print-mode.d.ts +10 -0
- package/dist/modes/print-mode.d.ts.map +1 -0
- package/dist/modes/print-mode.js +80 -0
- package/dist/modes/print-mode.js.map +1 -0
- package/dist/modes/rpc/jsonl.d.ts +4 -0
- package/dist/modes/rpc/jsonl.d.ts.map +1 -0
- package/dist/modes/rpc/jsonl.js +36 -0
- package/dist/modes/rpc/jsonl.js.map +1 -0
- package/dist/modes/rpc/rpc-client.d.ts +94 -0
- package/dist/modes/rpc/rpc-client.d.ts.map +1 -0
- package/dist/modes/rpc/rpc-client.js +262 -0
- package/dist/modes/rpc/rpc-client.js.map +1 -0
- package/dist/modes/rpc/rpc-mode.d.ts +3 -0
- package/dist/modes/rpc/rpc-mode.d.ts.map +1 -0
- package/dist/modes/rpc/rpc-mode.js +227 -0
- package/dist/modes/rpc/rpc-mode.js.map +1 -0
- package/dist/modes/rpc/rpc-types.d.ts +395 -0
- package/dist/modes/rpc/rpc-types.d.ts.map +1 -0
- package/dist/modes/rpc/rpc-types.js +2 -0
- package/dist/modes/rpc/rpc-types.js.map +1 -0
- package/dist/utils/changelog.d.ts +11 -0
- package/dist/utils/changelog.d.ts.map +1 -0
- package/dist/utils/changelog.js +69 -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 +157 -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 +55 -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 +10 -0
- package/dist/utils/git.d.ts.map +1 -0
- package/dist/utils/git.js +156 -0
- package/dist/utils/git.js.map +1 -0
- package/dist/utils/image-convert.d.ts +5 -0
- package/dist/utils/image-convert.d.ts.map +1 -0
- package/dist/utils/image-convert.js +28 -0
- package/dist/utils/image-convert.js.map +1 -0
- package/dist/utils/image-resize.d.ts +19 -0
- package/dist/utils/image-resize.d.ts.map +1 -0
- package/dist/utils/image-resize.js +152 -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 +3 -0
- package/dist/utils/photon.d.ts.map +1 -0
- package/dist/utils/photon.js +102 -0
- package/dist/utils/photon.js.map +1 -0
- package/dist/utils/shell.d.ts +8 -0
- package/dist/utils/shell.d.ts.map +1 -0
- package/dist/utils/shell.js +135 -0
- package/dist/utils/shell.js.map +1 -0
- package/dist/utils/sleep.d.ts +2 -0
- package/dist/utils/sleep.d.ts.map +1 -0
- package/dist/utils/sleep.js +14 -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 +227 -0
- package/dist/utils/tools-manager.js.map +1 -0
- package/docs/compaction.md +392 -0
- package/docs/custom-provider.md +592 -0
- package/docs/development.md +69 -0
- package/docs/extensions.md +2023 -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 +182 -0
- package/docs/models.md +297 -0
- package/docs/packages.md +209 -0
- package/docs/prompt-templates.md +67 -0
- package/docs/providers.md +188 -0
- package/docs/rpc.md +1354 -0
- package/docs/sdk.md +968 -0
- package/docs/session.md +412 -0
- package/docs/settings.md +225 -0
- package/docs/shell-aliases.md +13 -0
- package/docs/skills.md +231 -0
- package/docs/terminal-setup.md +87 -0
- package/docs/termux.md +127 -0
- package/docs/themes.md +295 -0
- package/docs/tmux.md +61 -0
- package/docs/tree.md +228 -0
- package/docs/tui.md +887 -0
- package/docs/windows.md +17 -0
- package/examples/README.md +25 -0
- package/examples/extensions/README.md +205 -0
- package/examples/extensions/antigravity-image-gen.ts +415 -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/built-in-tool-renderer.ts +246 -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/dynamic-tools.ts +74 -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 +1348 -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/provider-payload.ts +14 -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 +126 -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 +32 -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 +100 -0
|
@@ -0,0 +1,179 @@
|
|
|
1
|
+
class ConcurrencyManager {
|
|
2
|
+
maxConcurrent;
|
|
3
|
+
activeRequests = new Set();
|
|
4
|
+
queue = [];
|
|
5
|
+
requestStartTimes = new Map();
|
|
6
|
+
latencies = [];
|
|
7
|
+
constructor(maxConcurrent = 4) {
|
|
8
|
+
this.maxConcurrent = maxConcurrent;
|
|
9
|
+
}
|
|
10
|
+
updateConcurrency(newConcurrency) {
|
|
11
|
+
const old = this.maxConcurrent;
|
|
12
|
+
this.maxConcurrent = Math.max(1, newConcurrency);
|
|
13
|
+
this.processQueue();
|
|
14
|
+
}
|
|
15
|
+
async execute(requestId, task, signal) {
|
|
16
|
+
if (this.activeRequests.size >= this.maxConcurrent) {
|
|
17
|
+
await this.waitForSlot(requestId, signal);
|
|
18
|
+
}
|
|
19
|
+
if (signal?.aborted) {
|
|
20
|
+
throw new Error("Request aborted");
|
|
21
|
+
}
|
|
22
|
+
this.activeRequests.add(requestId);
|
|
23
|
+
this.requestStartTimes.set(requestId, Date.now());
|
|
24
|
+
try {
|
|
25
|
+
const result = await task();
|
|
26
|
+
return result;
|
|
27
|
+
}
|
|
28
|
+
finally {
|
|
29
|
+
const startTime = this.requestStartTimes.get(requestId);
|
|
30
|
+
if (startTime) {
|
|
31
|
+
const latency = Date.now() - startTime;
|
|
32
|
+
this.latencies.push(latency);
|
|
33
|
+
if (this.latencies.length > 100) {
|
|
34
|
+
this.latencies.shift();
|
|
35
|
+
}
|
|
36
|
+
this.requestStartTimes.delete(requestId);
|
|
37
|
+
}
|
|
38
|
+
this.activeRequests.delete(requestId);
|
|
39
|
+
this.processQueue();
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
async waitForSlot(requestId, signal) {
|
|
43
|
+
return new Promise((resolve, reject) => {
|
|
44
|
+
const checkQueue = () => {
|
|
45
|
+
if (this.activeRequests.size < this.maxConcurrent) {
|
|
46
|
+
resolve();
|
|
47
|
+
return true;
|
|
48
|
+
}
|
|
49
|
+
if (signal?.aborted) {
|
|
50
|
+
reject(new Error("Request aborted"));
|
|
51
|
+
return true;
|
|
52
|
+
}
|
|
53
|
+
return false;
|
|
54
|
+
};
|
|
55
|
+
if (!checkQueue()) {
|
|
56
|
+
const interval = setInterval(() => {
|
|
57
|
+
if (checkQueue()) {
|
|
58
|
+
clearInterval(interval);
|
|
59
|
+
}
|
|
60
|
+
}, 10);
|
|
61
|
+
}
|
|
62
|
+
});
|
|
63
|
+
}
|
|
64
|
+
processQueue() {
|
|
65
|
+
while (this.queue.length > 0 && this.activeRequests.size < this.maxConcurrent) {
|
|
66
|
+
const next = this.queue.shift();
|
|
67
|
+
if (next) {
|
|
68
|
+
this.executeRequest(next);
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
async executeRequest(pending) {
|
|
73
|
+
try {
|
|
74
|
+
const stream = await import("@mariozechner/pi-ai");
|
|
75
|
+
const response = await stream.streamSimple(pending.model, pending.context, pending.options);
|
|
76
|
+
const result = await response.result();
|
|
77
|
+
pending.resolve(result);
|
|
78
|
+
}
|
|
79
|
+
catch (error) {
|
|
80
|
+
pending.reject(error instanceof Error ? error : new Error(String(error)));
|
|
81
|
+
}
|
|
82
|
+
}
|
|
83
|
+
getStats() {
|
|
84
|
+
const avgLatency = this.latencies.length > 0 ? this.latencies.reduce((a, b) => a + b, 0) / this.latencies.length : 0;
|
|
85
|
+
return {
|
|
86
|
+
activeRequests: this.activeRequests.size,
|
|
87
|
+
queuedRequests: this.queue.length,
|
|
88
|
+
totalCompleted: this.latencies.length,
|
|
89
|
+
totalErrors: 0,
|
|
90
|
+
averageLatency: avgLatency,
|
|
91
|
+
};
|
|
92
|
+
}
|
|
93
|
+
clear() {
|
|
94
|
+
this.queue.forEach((req) => {
|
|
95
|
+
req.reject(new Error("Queue cleared"));
|
|
96
|
+
});
|
|
97
|
+
this.queue = [];
|
|
98
|
+
}
|
|
99
|
+
}
|
|
100
|
+
function detectOptimalConcurrency() {
|
|
101
|
+
try {
|
|
102
|
+
const os = require("os");
|
|
103
|
+
const cpuCores = os.cpus().length;
|
|
104
|
+
try {
|
|
105
|
+
const result = require("child_process").execSync("wmic path win32_VideoController get name", {
|
|
106
|
+
encoding: "utf-8",
|
|
107
|
+
});
|
|
108
|
+
const gpuCount = (result.match(/Name/g) || []).length;
|
|
109
|
+
return Math.max(2, gpuCount || cpuCores);
|
|
110
|
+
}
|
|
111
|
+
catch {
|
|
112
|
+
return Math.max(2, Math.floor(cpuCores / 2));
|
|
113
|
+
}
|
|
114
|
+
}
|
|
115
|
+
catch {
|
|
116
|
+
return 4;
|
|
117
|
+
}
|
|
118
|
+
}
|
|
119
|
+
const globalConcurrencyManager = new ConcurrencyManager(detectOptimalConcurrency());
|
|
120
|
+
export class APIParallelExecutor {
|
|
121
|
+
concurrencyManager;
|
|
122
|
+
constructor(concurrency) {
|
|
123
|
+
this.concurrencyManager = concurrency ? new ConcurrencyManager(concurrency) : globalConcurrencyManager;
|
|
124
|
+
}
|
|
125
|
+
async batchRequest(requests, options) {
|
|
126
|
+
const promises = requests.map((req, index) => {
|
|
127
|
+
return this.concurrencyManager.execute(`req-${Date.now()}-${index}`, async () => {
|
|
128
|
+
const stream = await import("@mariozechner/pi-ai");
|
|
129
|
+
const response = await stream.streamSimple(req.model, req.context, req.options);
|
|
130
|
+
return response.result();
|
|
131
|
+
}, options?.signal);
|
|
132
|
+
});
|
|
133
|
+
return Promise.all(promises);
|
|
134
|
+
}
|
|
135
|
+
async *streamBatch(requests, options) {
|
|
136
|
+
const promises = requests.map((req, index) => {
|
|
137
|
+
return this.concurrencyManager
|
|
138
|
+
.execute(`stream-req-${Date.now()}-${index}`, async () => {
|
|
139
|
+
const stream = await import("@mariozechner/pi-ai");
|
|
140
|
+
const response = await stream.streamSimple(req.model, req.context, req.options);
|
|
141
|
+
return response.result();
|
|
142
|
+
}, options?.signal)
|
|
143
|
+
.then((result) => ({ index, result }));
|
|
144
|
+
});
|
|
145
|
+
const remaining = new Set(promises.map((p, i) => i));
|
|
146
|
+
while (remaining.size > 0) {
|
|
147
|
+
if (options?.signal?.aborted) {
|
|
148
|
+
throw new Error("Stream batch aborted");
|
|
149
|
+
}
|
|
150
|
+
for (const idx of remaining) {
|
|
151
|
+
const promise = promises[idx];
|
|
152
|
+
if (promise && (await Promise.race([promise, Promise.resolve(null)]))) {
|
|
153
|
+
const result = await promise;
|
|
154
|
+
yield result;
|
|
155
|
+
remaining.delete(idx);
|
|
156
|
+
break;
|
|
157
|
+
}
|
|
158
|
+
}
|
|
159
|
+
if (remaining.size > 0) {
|
|
160
|
+
await new Promise((resolve) => setTimeout(resolve, 5));
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
}
|
|
164
|
+
getStats() {
|
|
165
|
+
return this.concurrencyManager.getStats();
|
|
166
|
+
}
|
|
167
|
+
setConcurrency(concurrency) {
|
|
168
|
+
this.concurrencyManager.updateConcurrency(concurrency);
|
|
169
|
+
}
|
|
170
|
+
getOptimalConcurrency() {
|
|
171
|
+
return detectOptimalConcurrency();
|
|
172
|
+
}
|
|
173
|
+
clear() {
|
|
174
|
+
this.concurrencyManager.clear();
|
|
175
|
+
}
|
|
176
|
+
}
|
|
177
|
+
export const globalParallelExecutor = new APIParallelExecutor();
|
|
178
|
+
export { detectOptimalConcurrency };
|
|
179
|
+
//# sourceMappingURL=api-concurrency.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"api-concurrency.js","sourceRoot":"","sources":["../../src/core/api-concurrency.ts"],"names":[],"mappings":"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","sourcesContent":["import type { Api, AssistantMessage, Context, Model } from \"@mariozechner/pi-ai\";\n\ninterface PendingRequest {\n\tid: string;\n\tmodel: Model<any>;\n\tcontext: Context;\n\tresolve: (value: AssistantMessage) => void;\n\treject: (error: Error) => void;\n\tsignal?: AbortSignal;\n\toptions?: any;\n}\ninterface ConcurrencyStats {\n\tactiveRequests: number;\n\tqueuedRequests: number;\n\ttotalCompleted: number;\n\ttotalErrors: number;\n\taverageLatency: number;\n}\nclass ConcurrencyManager {\n\tprivate maxConcurrent: number;\n\tprivate activeRequests: Set<string> = new Set();\n\tprivate queue: PendingRequest[] = [];\n\tprivate requestStartTimes: Map<string, number> = new Map();\n\tprivate latencies: number[] = [];\n\tconstructor(maxConcurrent: number = 4) {\n\t\tthis.maxConcurrent = maxConcurrent;\n\t}\n\tupdateConcurrency(newConcurrency: number): void {\n\t\tconst old = this.maxConcurrent;\n\t\tthis.maxConcurrent = Math.max(1, newConcurrency);\n\t\tthis.processQueue();\n\t}\n\tasync execute<T>(requestId: string, task: () => Promise<T>, signal?: AbortSignal): Promise<T> {\n\t\tif (this.activeRequests.size >= this.maxConcurrent) {\n\t\t\tawait this.waitForSlot(requestId, signal);\n\t\t}\n\t\tif (signal?.aborted) {\n\t\t\tthrow new Error(\"Request aborted\");\n\t\t}\n\t\tthis.activeRequests.add(requestId);\n\t\tthis.requestStartTimes.set(requestId, Date.now());\n\t\ttry {\n\t\t\tconst result = await task();\n\t\t\treturn result;\n\t\t} finally {\n\t\t\tconst startTime = this.requestStartTimes.get(requestId);\n\t\t\tif (startTime) {\n\t\t\t\tconst latency = Date.now() - startTime;\n\t\t\t\tthis.latencies.push(latency);\n\t\t\t\tif (this.latencies.length > 100) {\n\t\t\t\t\tthis.latencies.shift();\n\t\t\t\t}\n\t\t\t\tthis.requestStartTimes.delete(requestId);\n\t\t\t}\n\t\t\tthis.activeRequests.delete(requestId);\n\t\t\tthis.processQueue();\n\t\t}\n\t}\n\tprivate async waitForSlot(requestId: string, signal?: AbortSignal): Promise<void> {\n\t\treturn new Promise((resolve, reject) => {\n\t\t\tconst checkQueue = () => {\n\t\t\t\tif (this.activeRequests.size < this.maxConcurrent) {\n\t\t\t\t\tresolve();\n\t\t\t\t\treturn true;\n\t\t\t\t}\n\t\t\t\tif (signal?.aborted) {\n\t\t\t\t\treject(new Error(\"Request aborted\"));\n\t\t\t\t\treturn true;\n\t\t\t\t}\n\t\t\t\treturn false;\n\t\t\t};\n\t\t\tif (!checkQueue()) {\n\t\t\t\tconst interval = setInterval(() => {\n\t\t\t\t\tif (checkQueue()) {\n\t\t\t\t\t\tclearInterval(interval);\n\t\t\t\t\t}\n\t\t\t\t}, 10);\n\t\t\t}\n\t\t});\n\t}\n\tprivate processQueue(): void {\n\t\twhile (this.queue.length > 0 && this.activeRequests.size < this.maxConcurrent) {\n\t\t\tconst next = this.queue.shift();\n\t\t\tif (next) {\n\t\t\t\tthis.executeRequest(next);\n\t\t\t}\n\t\t}\n\t}\n\tprivate async executeRequest(pending: PendingRequest): Promise<void> {\n\t\ttry {\n\t\t\tconst stream = await import(\"@mariozechner/pi-ai\");\n\t\t\tconst response = await stream.streamSimple(pending.model, pending.context, pending.options);\n\t\t\tconst result = await response.result();\n\t\t\tpending.resolve(result);\n\t\t} catch (error) {\n\t\t\tpending.reject(error instanceof Error ? error : new Error(String(error)));\n\t\t}\n\t}\n\tgetStats(): ConcurrencyStats {\n\t\tconst avgLatency =\n\t\t\tthis.latencies.length > 0 ? this.latencies.reduce((a, b) => a + b, 0) / this.latencies.length : 0;\n\t\treturn {\n\t\t\tactiveRequests: this.activeRequests.size,\n\t\t\tqueuedRequests: this.queue.length,\n\t\t\ttotalCompleted: this.latencies.length,\n\t\t\ttotalErrors: 0,\n\t\t\taverageLatency: avgLatency,\n\t\t};\n\t}\n\tclear(): void {\n\t\tthis.queue.forEach((req) => {\n\t\t\treq.reject(new Error(\"Queue cleared\"));\n\t\t});\n\t\tthis.queue = [];\n\t}\n}\nfunction detectOptimalConcurrency(): number {\n\ttry {\n\t\tconst os = require(\"os\");\n\t\tconst cpuCores = os.cpus().length;\n\t\ttry {\n\t\t\tconst result = require(\"child_process\").execSync(\"wmic path win32_VideoController get name\", {\n\t\t\t\tencoding: \"utf-8\",\n\t\t\t});\n\t\t\tconst gpuCount = (result.match(/Name/g) || []).length;\n\t\t\treturn Math.max(2, gpuCount || cpuCores);\n\t\t} catch {\n\t\t\treturn Math.max(2, Math.floor(cpuCores / 2));\n\t\t}\n\t} catch {\n\t\treturn 4;\n\t}\n}\nconst globalConcurrencyManager = new ConcurrencyManager(detectOptimalConcurrency());\nexport class APIParallelExecutor {\n\tprivate concurrencyManager: ConcurrencyManager;\n\tconstructor(concurrency?: number) {\n\t\tthis.concurrencyManager = concurrency ? new ConcurrencyManager(concurrency) : globalConcurrencyManager;\n\t}\n\tasync batchRequest(\n\t\trequests: Array<{\n\t\t\tmodel: Model<any>;\n\t\t\tcontext: Context;\n\t\t\toptions?: any;\n\t\t}>,\n\t\toptions?: { signal?: AbortSignal },\n\t): Promise<AssistantMessage[]> {\n\t\tconst promises = requests.map((req, index) => {\n\t\t\treturn this.concurrencyManager.execute(\n\t\t\t\t`req-${Date.now()}-${index}`,\n\t\t\t\tasync () => {\n\t\t\t\t\tconst stream = await import(\"@mariozechner/pi-ai\");\n\t\t\t\t\tconst response = await stream.streamSimple(req.model, req.context, req.options);\n\t\t\t\t\treturn response.result();\n\t\t\t\t},\n\t\t\t\toptions?.signal,\n\t\t\t);\n\t\t});\n\t\treturn Promise.all(promises);\n\t}\n\tasync *streamBatch(\n\t\trequests: Array<{\n\t\t\tmodel: Model<any>;\n\t\t\tcontext: Context;\n\t\t\toptions?: any;\n\t\t}>,\n\t\toptions?: { signal?: AbortSignal },\n\t): AsyncGenerator<{ index: number; result: AssistantMessage }, void, void> {\n\t\tconst promises = requests.map((req, index) => {\n\t\t\treturn this.concurrencyManager\n\t\t\t\t.execute(\n\t\t\t\t\t`stream-req-${Date.now()}-${index}`,\n\t\t\t\t\tasync () => {\n\t\t\t\t\t\tconst stream = await import(\"@mariozechner/pi-ai\");\n\t\t\t\t\t\tconst response = await stream.streamSimple(req.model, req.context, req.options);\n\t\t\t\t\t\treturn response.result();\n\t\t\t\t\t},\n\t\t\t\t\toptions?.signal,\n\t\t\t\t)\n\t\t\t\t.then((result) => ({ index, result }));\n\t\t});\n\t\tconst remaining = new Set(promises.map((p, i) => i));\n\t\twhile (remaining.size > 0) {\n\t\t\tif (options?.signal?.aborted) {\n\t\t\t\tthrow new Error(\"Stream batch aborted\");\n\t\t\t}\n\t\t\tfor (const idx of remaining) {\n\t\t\t\tconst promise = promises[idx];\n\t\t\t\tif (promise && (await Promise.race([promise, Promise.resolve(null)]))) {\n\t\t\t\t\tconst result = await promise;\n\t\t\t\t\tyield result;\n\t\t\t\t\tremaining.delete(idx);\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\t\t\t}\n\t\t\tif (remaining.size > 0) {\n\t\t\t\tawait new Promise((resolve) => setTimeout(resolve, 5));\n\t\t\t}\n\t\t}\n\t}\n\tgetStats(): ConcurrencyStats {\n\t\treturn this.concurrencyManager.getStats();\n\t}\n\tsetConcurrency(concurrency: number): void {\n\t\tthis.concurrencyManager.updateConcurrency(concurrency);\n\t}\n\tgetOptimalConcurrency(): number {\n\t\treturn detectOptimalConcurrency();\n\t}\n\tclear(): void {\n\t\tthis.concurrencyManager.clear();\n\t}\n}\nexport const globalParallelExecutor = new APIParallelExecutor();\nexport { detectOptimalConcurrency, type ConcurrencyStats };\n"]}
|
|
@@ -0,0 +1,66 @@
|
|
|
1
|
+
import { type OAuthCredentials, type OAuthLoginCallbacks, type OAuthProviderId } from "@mariozechner/pi-ai";
|
|
2
|
+
export type ApiKeyCredential = {
|
|
3
|
+
type: "api_key";
|
|
4
|
+
key: string;
|
|
5
|
+
};
|
|
6
|
+
export type OAuthCredential = {
|
|
7
|
+
type: "oauth";
|
|
8
|
+
} & OAuthCredentials;
|
|
9
|
+
export type AuthCredential = ApiKeyCredential | OAuthCredential;
|
|
10
|
+
export type AuthStorageData = Record<string, AuthCredential>;
|
|
11
|
+
type LockResult<T> = {
|
|
12
|
+
result: T;
|
|
13
|
+
next?: string;
|
|
14
|
+
};
|
|
15
|
+
export interface AuthStorageBackend {
|
|
16
|
+
withLock<T>(fn: (current: string | undefined) => LockResult<T>): T;
|
|
17
|
+
withLockAsync<T>(fn: (current: string | undefined) => Promise<LockResult<T>>): Promise<T>;
|
|
18
|
+
}
|
|
19
|
+
export declare class FileAuthStorageBackend implements AuthStorageBackend {
|
|
20
|
+
private authPath;
|
|
21
|
+
constructor(authPath?: string);
|
|
22
|
+
private ensureParentDir;
|
|
23
|
+
private ensureFileExists;
|
|
24
|
+
private acquireLockSyncWithRetry;
|
|
25
|
+
withLock<T>(fn: (current: string | undefined) => LockResult<T>): T;
|
|
26
|
+
withLockAsync<T>(fn: (current: string | undefined) => Promise<LockResult<T>>): Promise<T>;
|
|
27
|
+
}
|
|
28
|
+
export declare class InMemoryAuthStorageBackend implements AuthStorageBackend {
|
|
29
|
+
private value;
|
|
30
|
+
withLock<T>(fn: (current: string | undefined) => LockResult<T>): T;
|
|
31
|
+
withLockAsync<T>(fn: (current: string | undefined) => Promise<LockResult<T>>): Promise<T>;
|
|
32
|
+
}
|
|
33
|
+
export declare class AuthStorage {
|
|
34
|
+
private storage;
|
|
35
|
+
private data;
|
|
36
|
+
private runtimeOverrides;
|
|
37
|
+
private fallbackResolver?;
|
|
38
|
+
private loadError;
|
|
39
|
+
private errors;
|
|
40
|
+
private constructor();
|
|
41
|
+
static create(authPath?: string): AuthStorage;
|
|
42
|
+
static fromStorage(storage: AuthStorageBackend): AuthStorage;
|
|
43
|
+
static inMemory(data?: AuthStorageData): AuthStorage;
|
|
44
|
+
setRuntimeApiKey(provider: string, apiKey: string): void;
|
|
45
|
+
removeRuntimeApiKey(provider: string): void;
|
|
46
|
+
setFallbackResolver(resolver: (provider: string) => string | undefined): void;
|
|
47
|
+
private recordError;
|
|
48
|
+
private parseStorageData;
|
|
49
|
+
reload(): void;
|
|
50
|
+
private persistProviderChange;
|
|
51
|
+
get(provider: string): AuthCredential | undefined;
|
|
52
|
+
set(provider: string, credential: AuthCredential): void;
|
|
53
|
+
remove(provider: string): void;
|
|
54
|
+
list(): string[];
|
|
55
|
+
has(provider: string): boolean;
|
|
56
|
+
hasAuth(provider: string): boolean;
|
|
57
|
+
getAll(): AuthStorageData;
|
|
58
|
+
drainErrors(): Error[];
|
|
59
|
+
login(providerId: OAuthProviderId, callbacks: OAuthLoginCallbacks): Promise<void>;
|
|
60
|
+
logout(provider: string): void;
|
|
61
|
+
private refreshOAuthTokenWithLock;
|
|
62
|
+
getApiKey(providerId: string): Promise<string | undefined>;
|
|
63
|
+
getOAuthProviders(): import("@mariozechner/pi-ai").OAuthProviderInterface[];
|
|
64
|
+
}
|
|
65
|
+
export {};
|
|
66
|
+
//# sourceMappingURL=auth-storage.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"auth-storage.d.ts","sourceRoot":"","sources":["../../src/core/auth-storage.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n\tgetEnvApiKey,\n\ttype OAuthCredentials,\n\ttype OAuthLoginCallbacks,\n\ttype OAuthProviderId,\n} from \"@mariozechner/pi-ai\";\nimport { getOAuthApiKey, getOAuthProvider, getOAuthProviders } from \"@mariozechner/pi-ai/oauth\";\nimport { chmodSync, existsSync, mkdirSync, readFileSync, writeFileSync } from \"fs\";\nimport { dirname, join } from \"path\";\nimport lockfile from \"proper-lockfile\";\nimport { getAgentDir } from \"../config.js\";\nimport { resolveConfigValue } from \"./resolve-config-value.js\";\nexport type ApiKeyCredential = {\n\ttype: \"api_key\";\n\tkey: string;\n};\nexport type OAuthCredential = {\n\ttype: \"oauth\";\n} & OAuthCredentials;\nexport type AuthCredential = ApiKeyCredential | OAuthCredential;\nexport type AuthStorageData = Record<string, AuthCredential>;\ntype LockResult<T> = {\n\tresult: T;\n\tnext?: string;\n};\nexport interface AuthStorageBackend {\n\twithLock<T>(fn: (current: string | undefined) => LockResult<T>): T;\n\twithLockAsync<T>(fn: (current: string | undefined) => Promise<LockResult<T>>): Promise<T>;\n}\nexport class FileAuthStorageBackend implements AuthStorageBackend {\n\tconstructor(private authPath: string = join(getAgentDir(), \"auth.json\")) {}\n\tprivate ensureParentDir(): void {\n\t\tconst dir = dirname(this.authPath);\n\t\tif (!existsSync(dir)) {\n\t\t\tmkdirSync(dir, { recursive: true, mode: 0o700 });\n\t\t}\n\t}\n\tprivate ensureFileExists(): void {\n\t\tif (!existsSync(this.authPath)) {\n\t\t\twriteFileSync(this.authPath, \"{}\", \"utf-8\");\n\t\t\tchmodSync(this.authPath, 0o600);\n\t\t}\n\t}\n\tprivate acquireLockSyncWithRetry(path: string): () => void {\n\t\tconst maxAttempts = 10;\n\t\tconst delayMs = 20;\n\t\tlet lastError: unknown;\n\t\tfor (let attempt = 1; attempt <= maxAttempts; attempt++) {\n\t\t\ttry {\n\t\t\t\treturn lockfile.lockSync(path, { realpath: false });\n\t\t\t} catch (error) {\n\t\t\t\tconst code =\n\t\t\t\t\ttypeof error === \"object\" && error !== null && \"code\" in error\n\t\t\t\t\t\t? String((error as { code?: unknown }).code)\n\t\t\t\t\t\t: undefined;\n\t\t\t\tif (code !== \"ELOCKED\" || attempt === maxAttempts) {\n\t\t\t\t\tthrow error;\n\t\t\t\t}\n\t\t\t\tlastError = error;\n\t\t\t\tconst start = Date.now();\n\t\t\t\twhile (Date.now() - start < delayMs) {}\n\t\t\t}\n\t\t}\n\t\tthrow (lastError as Error) ?? new Error(\"Failed to acquire auth storage lock\");\n\t}\n\twithLock<T>(fn: (current: string | undefined) => LockResult<T>): T {\n\t\tthis.ensureParentDir();\n\t\tthis.ensureFileExists();\n\t\tlet release: (() => void) | undefined;\n\t\ttry {\n\t\t\trelease = this.acquireLockSyncWithRetry(this.authPath);\n\t\t\tconst current = existsSync(this.authPath) ? readFileSync(this.authPath, \"utf-8\") : undefined;\n\t\t\tconst { result, next } = fn(current);\n\t\t\tif (next !== undefined) {\n\t\t\t\twriteFileSync(this.authPath, next, \"utf-8\");\n\t\t\t\tchmodSync(this.authPath, 0o600);\n\t\t\t}\n\t\t\treturn result;\n\t\t} finally {\n\t\t\tif (release) {\n\t\t\t\trelease();\n\t\t\t}\n\t\t}\n\t}\n\tasync withLockAsync<T>(fn: (current: string | undefined) => Promise<LockResult<T>>): Promise<T> {\n\t\tthis.ensureParentDir();\n\t\tthis.ensureFileExists();\n\t\tlet release: (() => Promise<void>) | undefined;\n\t\tlet lockCompromised = false;\n\t\tlet lockCompromisedError: Error | undefined;\n\t\tconst throwIfCompromised = () => {\n\t\t\tif (lockCompromised) {\n\t\t\t\tthrow lockCompromisedError ?? new Error(\"Auth storage lock was compromised\");\n\t\t\t}\n\t\t};\n\t\ttry {\n\t\t\trelease = await lockfile.lock(this.authPath, {\n\t\t\t\tretries: {\n\t\t\t\t\tretries: 10,\n\t\t\t\t\tfactor: 2,\n\t\t\t\t\tminTimeout: 100,\n\t\t\t\t\tmaxTimeout: 10000,\n\t\t\t\t\trandomize: true,\n\t\t\t\t},\n\t\t\t\tstale: 30000,\n\t\t\t\tonCompromised: (err) => {\n\t\t\t\t\tlockCompromised = true;\n\t\t\t\t\tlockCompromisedError = err;\n\t\t\t\t},\n\t\t\t});\n\t\t\tthrowIfCompromised();\n\t\t\tconst current = existsSync(this.authPath) ? readFileSync(this.authPath, \"utf-8\") : undefined;\n\t\t\tconst { result, next } = await fn(current);\n\t\t\tthrowIfCompromised();\n\t\t\tif (next !== undefined) {\n\t\t\t\twriteFileSync(this.authPath, next, \"utf-8\");\n\t\t\t\tchmodSync(this.authPath, 0o600);\n\t\t\t}\n\t\t\tthrowIfCompromised();\n\t\t\treturn result;\n\t\t} finally {\n\t\t\tif (release) {\n\t\t\t\ttry {\n\t\t\t\t\tawait release();\n\t\t\t\t} catch {}\n\t\t\t}\n\t\t}\n\t}\n}\nexport class InMemoryAuthStorageBackend implements AuthStorageBackend {\n\tprivate value: string | undefined;\n\twithLock<T>(fn: (current: string | undefined) => LockResult<T>): T {\n\t\tconst { result, next } = fn(this.value);\n\t\tif (next !== undefined) {\n\t\t\tthis.value = next;\n\t\t}\n\t\treturn result;\n\t}\n\tasync withLockAsync<T>(fn: (current: string | undefined) => Promise<LockResult<T>>): Promise<T> {\n\t\tconst { result, next } = await fn(this.value);\n\t\tif (next !== undefined) {\n\t\t\tthis.value = next;\n\t\t}\n\t\treturn result;\n\t}\n}\nexport class AuthStorage {\n\tprivate data: AuthStorageData = {};\n\tprivate runtimeOverrides: Map<string, string> = new Map();\n\tprivate fallbackResolver?: (provider: string) => string | undefined;\n\tprivate loadError: Error | null = null;\n\tprivate errors: Error[] = [];\n\tprivate constructor(private storage: AuthStorageBackend) {\n\t\tthis.reload();\n\t}\n\tstatic create(authPath?: string): AuthStorage {\n\t\treturn new AuthStorage(new FileAuthStorageBackend(authPath ?? join(getAgentDir(), \"auth.json\")));\n\t}\n\tstatic fromStorage(storage: AuthStorageBackend): AuthStorage {\n\t\treturn new AuthStorage(storage);\n\t}\n\tstatic inMemory(data: AuthStorageData = {}): AuthStorage {\n\t\tconst storage = new InMemoryAuthStorageBackend();\n\t\tstorage.withLock(() => ({ result: undefined, next: JSON.stringify(data, null, 2) }));\n\t\treturn AuthStorage.fromStorage(storage);\n\t}\n\tsetRuntimeApiKey(provider: string, apiKey: string): void {\n\t\tthis.runtimeOverrides.set(provider, apiKey);\n\t}\n\tremoveRuntimeApiKey(provider: string): void {\n\t\tthis.runtimeOverrides.delete(provider);\n\t}\n\tsetFallbackResolver(resolver: (provider: string) => string | undefined): void {\n\t\tthis.fallbackResolver = resolver;\n\t}\n\tprivate recordError(error: unknown): void {\n\t\tconst normalizedError = error instanceof Error ? error : new Error(String(error));\n\t\tthis.errors.push(normalizedError);\n\t}\n\tprivate parseStorageData(content: string | undefined): AuthStorageData {\n\t\tif (!content) {\n\t\t\treturn {};\n\t\t}\n\t\treturn JSON.parse(content) as AuthStorageData;\n\t}\n\treload(): void {\n\t\tlet content: string | undefined;\n\t\ttry {\n\t\t\tthis.storage.withLock((current) => {\n\t\t\t\tcontent = current;\n\t\t\t\treturn { result: undefined };\n\t\t\t});\n\t\t\tthis.data = this.parseStorageData(content);\n\t\t\tthis.loadError = null;\n\t\t} catch (error) {\n\t\t\tthis.loadError = error as Error;\n\t\t\tthis.recordError(error);\n\t\t}\n\t}\n\tprivate persistProviderChange(provider: string, credential: AuthCredential | undefined): void {\n\t\tif (this.loadError) {\n\t\t\treturn;\n\t\t}\n\t\ttry {\n\t\t\tthis.storage.withLock((current) => {\n\t\t\t\tconst currentData = this.parseStorageData(current);\n\t\t\t\tconst merged: AuthStorageData = { ...currentData };\n\t\t\t\tif (credential) {\n\t\t\t\t\tmerged[provider] = credential;\n\t\t\t\t} else {\n\t\t\t\t\tdelete merged[provider];\n\t\t\t\t}\n\t\t\t\treturn { result: undefined, next: JSON.stringify(merged, null, 2) };\n\t\t\t});\n\t\t} catch (error) {\n\t\t\tthis.recordError(error);\n\t\t}\n\t}\n\tget(provider: string): AuthCredential | undefined {\n\t\treturn this.data[provider] ?? undefined;\n\t}\n\tset(provider: string, credential: AuthCredential): void {\n\t\tthis.data[provider] = credential;\n\t\tthis.persistProviderChange(provider, credential);\n\t}\n\tremove(provider: string): void {\n\t\tdelete this.data[provider];\n\t\tthis.persistProviderChange(provider, undefined);\n\t}\n\tlist(): string[] {\n\t\treturn Object.keys(this.data);\n\t}\n\thas(provider: string): boolean {\n\t\treturn provider in this.data;\n\t}\n\thasAuth(provider: string): boolean {\n\t\tif (this.runtimeOverrides.has(provider)) return true;\n\t\tif (this.data[provider]) return true;\n\t\tif (getEnvApiKey(provider)) return true;\n\t\tif (this.fallbackResolver?.(provider)) return true;\n\t\treturn false;\n\t}\n\tgetAll(): AuthStorageData {\n\t\treturn { ...this.data };\n\t}\n\tdrainErrors(): Error[] {\n\t\tconst drained = [...this.errors];\n\t\tthis.errors = [];\n\t\treturn drained;\n\t}\n\tasync login(providerId: OAuthProviderId, callbacks: OAuthLoginCallbacks): Promise<void> {\n\t\tconst provider = getOAuthProvider(providerId);\n\t\tif (!provider) {\n\t\t\tthrow new Error(`Unknown OAuth provider: ${providerId}`);\n\t\t}\n\t\tconst credentials = await provider.login(callbacks);\n\t\tthis.set(providerId, { type: \"oauth\", ...credentials });\n\t}\n\tlogout(provider: string): void {\n\t\tthis.remove(provider);\n\t}\n\tprivate async refreshOAuthTokenWithLock(\n\t\tproviderId: OAuthProviderId,\n\t): Promise<{ apiKey: string; newCredentials: OAuthCredentials } | null> {\n\t\tconst provider = getOAuthProvider(providerId);\n\t\tif (!provider) {\n\t\t\treturn null;\n\t\t}\n\t\tconst result = await this.storage.withLockAsync(async (current) => {\n\t\t\tconst currentData = this.parseStorageData(current);\n\t\t\tthis.data = currentData;\n\t\t\tthis.loadError = null;\n\t\t\tconst cred = currentData[providerId];\n\t\t\tif (cred?.type !== \"oauth\") {\n\t\t\t\treturn { result: null };\n\t\t\t}\n\t\t\tif (Date.now() < cred.expires) {\n\t\t\t\treturn { result: { apiKey: provider.getApiKey(cred), newCredentials: cred } };\n\t\t\t}\n\t\t\tconst oauthCreds: Record<string, OAuthCredentials> = {};\n\t\t\tfor (const [key, value] of Object.entries(currentData)) {\n\t\t\t\tif (value.type === \"oauth\") {\n\t\t\t\t\toauthCreds[key] = value;\n\t\t\t\t}\n\t\t\t}\n\t\t\tconst refreshed = await getOAuthApiKey(providerId, oauthCreds);\n\t\t\tif (!refreshed) {\n\t\t\t\treturn { result: null };\n\t\t\t}\n\t\t\tconst merged: AuthStorageData = {\n\t\t\t\t...currentData,\n\t\t\t\t[providerId]: { type: \"oauth\", ...refreshed.newCredentials },\n\t\t\t};\n\t\t\tthis.data = merged;\n\t\t\tthis.loadError = null;\n\t\t\treturn { result: refreshed, next: JSON.stringify(merged, null, 2) };\n\t\t});\n\t\treturn result;\n\t}\n\tasync getApiKey(providerId: string): Promise<string | undefined> {\n\t\tconst runtimeKey = this.runtimeOverrides.get(providerId);\n\t\tif (runtimeKey) {\n\t\t\treturn runtimeKey;\n\t\t}\n\t\tconst cred = this.data[providerId];\n\t\tif (cred?.type === \"api_key\") {\n\t\t\treturn resolveConfigValue(cred.key);\n\t\t}\n\t\tif (cred?.type === \"oauth\") {\n\t\t\tconst provider = getOAuthProvider(providerId);\n\t\t\tif (!provider) {\n\t\t\t\treturn undefined;\n\t\t\t}\n\t\t\tconst needsRefresh = Date.now() >= cred.expires;\n\t\t\tif (needsRefresh) {\n\t\t\t\ttry {\n\t\t\t\t\tconst result = await this.refreshOAuthTokenWithLock(providerId);\n\t\t\t\t\tif (result) {\n\t\t\t\t\t\treturn result.apiKey;\n\t\t\t\t\t}\n\t\t\t\t} catch (error) {\n\t\t\t\t\tthis.recordError(error);\n\t\t\t\t\tthis.reload();\n\t\t\t\t\tconst updatedCred = this.data[providerId];\n\t\t\t\t\tif (updatedCred?.type === \"oauth\" && Date.now() < updatedCred.expires) {\n\t\t\t\t\t\treturn provider.getApiKey(updatedCred);\n\t\t\t\t\t}\n\t\t\t\t\treturn undefined;\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\treturn provider.getApiKey(cred);\n\t\t\t}\n\t\t}\n\t\tconst envKey = getEnvApiKey(providerId);\n\t\tif (envKey) return envKey;\n\t\treturn this.fallbackResolver?.(providerId) ?? undefined;\n\t}\n\tgetOAuthProviders() {\n\t\treturn getOAuthProviders();\n\t}\n}\n"]}
|
|
@@ -0,0 +1,336 @@
|
|
|
1
|
+
import { getEnvApiKey, } from "@mariozechner/pi-ai";
|
|
2
|
+
import { getOAuthApiKey, getOAuthProvider, getOAuthProviders } from "@mariozechner/pi-ai/oauth";
|
|
3
|
+
import { chmodSync, existsSync, mkdirSync, readFileSync, writeFileSync } from "fs";
|
|
4
|
+
import { dirname, join } from "path";
|
|
5
|
+
import lockfile from "proper-lockfile";
|
|
6
|
+
import { getAgentDir } from "../config.js";
|
|
7
|
+
import { resolveConfigValue } from "./resolve-config-value.js";
|
|
8
|
+
export class FileAuthStorageBackend {
|
|
9
|
+
authPath;
|
|
10
|
+
constructor(authPath = join(getAgentDir(), "auth.json")) {
|
|
11
|
+
this.authPath = authPath;
|
|
12
|
+
}
|
|
13
|
+
ensureParentDir() {
|
|
14
|
+
const dir = dirname(this.authPath);
|
|
15
|
+
if (!existsSync(dir)) {
|
|
16
|
+
mkdirSync(dir, { recursive: true, mode: 0o700 });
|
|
17
|
+
}
|
|
18
|
+
}
|
|
19
|
+
ensureFileExists() {
|
|
20
|
+
if (!existsSync(this.authPath)) {
|
|
21
|
+
writeFileSync(this.authPath, "{}", "utf-8");
|
|
22
|
+
chmodSync(this.authPath, 0o600);
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
acquireLockSyncWithRetry(path) {
|
|
26
|
+
const maxAttempts = 10;
|
|
27
|
+
const delayMs = 20;
|
|
28
|
+
let lastError;
|
|
29
|
+
for (let attempt = 1; attempt <= maxAttempts; attempt++) {
|
|
30
|
+
try {
|
|
31
|
+
return lockfile.lockSync(path, { realpath: false });
|
|
32
|
+
}
|
|
33
|
+
catch (error) {
|
|
34
|
+
const code = typeof error === "object" && error !== null && "code" in error
|
|
35
|
+
? String(error.code)
|
|
36
|
+
: undefined;
|
|
37
|
+
if (code !== "ELOCKED" || attempt === maxAttempts) {
|
|
38
|
+
throw error;
|
|
39
|
+
}
|
|
40
|
+
lastError = error;
|
|
41
|
+
const start = Date.now();
|
|
42
|
+
while (Date.now() - start < delayMs) { }
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
throw lastError ?? new Error("Failed to acquire auth storage lock");
|
|
46
|
+
}
|
|
47
|
+
withLock(fn) {
|
|
48
|
+
this.ensureParentDir();
|
|
49
|
+
this.ensureFileExists();
|
|
50
|
+
let release;
|
|
51
|
+
try {
|
|
52
|
+
release = this.acquireLockSyncWithRetry(this.authPath);
|
|
53
|
+
const current = existsSync(this.authPath) ? readFileSync(this.authPath, "utf-8") : undefined;
|
|
54
|
+
const { result, next } = fn(current);
|
|
55
|
+
if (next !== undefined) {
|
|
56
|
+
writeFileSync(this.authPath, next, "utf-8");
|
|
57
|
+
chmodSync(this.authPath, 0o600);
|
|
58
|
+
}
|
|
59
|
+
return result;
|
|
60
|
+
}
|
|
61
|
+
finally {
|
|
62
|
+
if (release) {
|
|
63
|
+
release();
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
}
|
|
67
|
+
async withLockAsync(fn) {
|
|
68
|
+
this.ensureParentDir();
|
|
69
|
+
this.ensureFileExists();
|
|
70
|
+
let release;
|
|
71
|
+
let lockCompromised = false;
|
|
72
|
+
let lockCompromisedError;
|
|
73
|
+
const throwIfCompromised = () => {
|
|
74
|
+
if (lockCompromised) {
|
|
75
|
+
throw lockCompromisedError ?? new Error("Auth storage lock was compromised");
|
|
76
|
+
}
|
|
77
|
+
};
|
|
78
|
+
try {
|
|
79
|
+
release = await lockfile.lock(this.authPath, {
|
|
80
|
+
retries: {
|
|
81
|
+
retries: 10,
|
|
82
|
+
factor: 2,
|
|
83
|
+
minTimeout: 100,
|
|
84
|
+
maxTimeout: 10000,
|
|
85
|
+
randomize: true,
|
|
86
|
+
},
|
|
87
|
+
stale: 30000,
|
|
88
|
+
onCompromised: (err) => {
|
|
89
|
+
lockCompromised = true;
|
|
90
|
+
lockCompromisedError = err;
|
|
91
|
+
},
|
|
92
|
+
});
|
|
93
|
+
throwIfCompromised();
|
|
94
|
+
const current = existsSync(this.authPath) ? readFileSync(this.authPath, "utf-8") : undefined;
|
|
95
|
+
const { result, next } = await fn(current);
|
|
96
|
+
throwIfCompromised();
|
|
97
|
+
if (next !== undefined) {
|
|
98
|
+
writeFileSync(this.authPath, next, "utf-8");
|
|
99
|
+
chmodSync(this.authPath, 0o600);
|
|
100
|
+
}
|
|
101
|
+
throwIfCompromised();
|
|
102
|
+
return result;
|
|
103
|
+
}
|
|
104
|
+
finally {
|
|
105
|
+
if (release) {
|
|
106
|
+
try {
|
|
107
|
+
await release();
|
|
108
|
+
}
|
|
109
|
+
catch { }
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
}
|
|
113
|
+
}
|
|
114
|
+
export class InMemoryAuthStorageBackend {
|
|
115
|
+
value;
|
|
116
|
+
withLock(fn) {
|
|
117
|
+
const { result, next } = fn(this.value);
|
|
118
|
+
if (next !== undefined) {
|
|
119
|
+
this.value = next;
|
|
120
|
+
}
|
|
121
|
+
return result;
|
|
122
|
+
}
|
|
123
|
+
async withLockAsync(fn) {
|
|
124
|
+
const { result, next } = await fn(this.value);
|
|
125
|
+
if (next !== undefined) {
|
|
126
|
+
this.value = next;
|
|
127
|
+
}
|
|
128
|
+
return result;
|
|
129
|
+
}
|
|
130
|
+
}
|
|
131
|
+
export class AuthStorage {
|
|
132
|
+
storage;
|
|
133
|
+
data = {};
|
|
134
|
+
runtimeOverrides = new Map();
|
|
135
|
+
fallbackResolver;
|
|
136
|
+
loadError = null;
|
|
137
|
+
errors = [];
|
|
138
|
+
constructor(storage) {
|
|
139
|
+
this.storage = storage;
|
|
140
|
+
this.reload();
|
|
141
|
+
}
|
|
142
|
+
static create(authPath) {
|
|
143
|
+
return new AuthStorage(new FileAuthStorageBackend(authPath ?? join(getAgentDir(), "auth.json")));
|
|
144
|
+
}
|
|
145
|
+
static fromStorage(storage) {
|
|
146
|
+
return new AuthStorage(storage);
|
|
147
|
+
}
|
|
148
|
+
static inMemory(data = {}) {
|
|
149
|
+
const storage = new InMemoryAuthStorageBackend();
|
|
150
|
+
storage.withLock(() => ({ result: undefined, next: JSON.stringify(data, null, 2) }));
|
|
151
|
+
return AuthStorage.fromStorage(storage);
|
|
152
|
+
}
|
|
153
|
+
setRuntimeApiKey(provider, apiKey) {
|
|
154
|
+
this.runtimeOverrides.set(provider, apiKey);
|
|
155
|
+
}
|
|
156
|
+
removeRuntimeApiKey(provider) {
|
|
157
|
+
this.runtimeOverrides.delete(provider);
|
|
158
|
+
}
|
|
159
|
+
setFallbackResolver(resolver) {
|
|
160
|
+
this.fallbackResolver = resolver;
|
|
161
|
+
}
|
|
162
|
+
recordError(error) {
|
|
163
|
+
const normalizedError = error instanceof Error ? error : new Error(String(error));
|
|
164
|
+
this.errors.push(normalizedError);
|
|
165
|
+
}
|
|
166
|
+
parseStorageData(content) {
|
|
167
|
+
if (!content) {
|
|
168
|
+
return {};
|
|
169
|
+
}
|
|
170
|
+
return JSON.parse(content);
|
|
171
|
+
}
|
|
172
|
+
reload() {
|
|
173
|
+
let content;
|
|
174
|
+
try {
|
|
175
|
+
this.storage.withLock((current) => {
|
|
176
|
+
content = current;
|
|
177
|
+
return { result: undefined };
|
|
178
|
+
});
|
|
179
|
+
this.data = this.parseStorageData(content);
|
|
180
|
+
this.loadError = null;
|
|
181
|
+
}
|
|
182
|
+
catch (error) {
|
|
183
|
+
this.loadError = error;
|
|
184
|
+
this.recordError(error);
|
|
185
|
+
}
|
|
186
|
+
}
|
|
187
|
+
persistProviderChange(provider, credential) {
|
|
188
|
+
if (this.loadError) {
|
|
189
|
+
return;
|
|
190
|
+
}
|
|
191
|
+
try {
|
|
192
|
+
this.storage.withLock((current) => {
|
|
193
|
+
const currentData = this.parseStorageData(current);
|
|
194
|
+
const merged = { ...currentData };
|
|
195
|
+
if (credential) {
|
|
196
|
+
merged[provider] = credential;
|
|
197
|
+
}
|
|
198
|
+
else {
|
|
199
|
+
delete merged[provider];
|
|
200
|
+
}
|
|
201
|
+
return { result: undefined, next: JSON.stringify(merged, null, 2) };
|
|
202
|
+
});
|
|
203
|
+
}
|
|
204
|
+
catch (error) {
|
|
205
|
+
this.recordError(error);
|
|
206
|
+
}
|
|
207
|
+
}
|
|
208
|
+
get(provider) {
|
|
209
|
+
return this.data[provider] ?? undefined;
|
|
210
|
+
}
|
|
211
|
+
set(provider, credential) {
|
|
212
|
+
this.data[provider] = credential;
|
|
213
|
+
this.persistProviderChange(provider, credential);
|
|
214
|
+
}
|
|
215
|
+
remove(provider) {
|
|
216
|
+
delete this.data[provider];
|
|
217
|
+
this.persistProviderChange(provider, undefined);
|
|
218
|
+
}
|
|
219
|
+
list() {
|
|
220
|
+
return Object.keys(this.data);
|
|
221
|
+
}
|
|
222
|
+
has(provider) {
|
|
223
|
+
return provider in this.data;
|
|
224
|
+
}
|
|
225
|
+
hasAuth(provider) {
|
|
226
|
+
if (this.runtimeOverrides.has(provider))
|
|
227
|
+
return true;
|
|
228
|
+
if (this.data[provider])
|
|
229
|
+
return true;
|
|
230
|
+
if (getEnvApiKey(provider))
|
|
231
|
+
return true;
|
|
232
|
+
if (this.fallbackResolver?.(provider))
|
|
233
|
+
return true;
|
|
234
|
+
return false;
|
|
235
|
+
}
|
|
236
|
+
getAll() {
|
|
237
|
+
return { ...this.data };
|
|
238
|
+
}
|
|
239
|
+
drainErrors() {
|
|
240
|
+
const drained = [...this.errors];
|
|
241
|
+
this.errors = [];
|
|
242
|
+
return drained;
|
|
243
|
+
}
|
|
244
|
+
async login(providerId, callbacks) {
|
|
245
|
+
const provider = getOAuthProvider(providerId);
|
|
246
|
+
if (!provider) {
|
|
247
|
+
throw new Error(`Unknown OAuth provider: ${providerId}`);
|
|
248
|
+
}
|
|
249
|
+
const credentials = await provider.login(callbacks);
|
|
250
|
+
this.set(providerId, { type: "oauth", ...credentials });
|
|
251
|
+
}
|
|
252
|
+
logout(provider) {
|
|
253
|
+
this.remove(provider);
|
|
254
|
+
}
|
|
255
|
+
async refreshOAuthTokenWithLock(providerId) {
|
|
256
|
+
const provider = getOAuthProvider(providerId);
|
|
257
|
+
if (!provider) {
|
|
258
|
+
return null;
|
|
259
|
+
}
|
|
260
|
+
const result = await this.storage.withLockAsync(async (current) => {
|
|
261
|
+
const currentData = this.parseStorageData(current);
|
|
262
|
+
this.data = currentData;
|
|
263
|
+
this.loadError = null;
|
|
264
|
+
const cred = currentData[providerId];
|
|
265
|
+
if (cred?.type !== "oauth") {
|
|
266
|
+
return { result: null };
|
|
267
|
+
}
|
|
268
|
+
if (Date.now() < cred.expires) {
|
|
269
|
+
return { result: { apiKey: provider.getApiKey(cred), newCredentials: cred } };
|
|
270
|
+
}
|
|
271
|
+
const oauthCreds = {};
|
|
272
|
+
for (const [key, value] of Object.entries(currentData)) {
|
|
273
|
+
if (value.type === "oauth") {
|
|
274
|
+
oauthCreds[key] = value;
|
|
275
|
+
}
|
|
276
|
+
}
|
|
277
|
+
const refreshed = await getOAuthApiKey(providerId, oauthCreds);
|
|
278
|
+
if (!refreshed) {
|
|
279
|
+
return { result: null };
|
|
280
|
+
}
|
|
281
|
+
const merged = {
|
|
282
|
+
...currentData,
|
|
283
|
+
[providerId]: { type: "oauth", ...refreshed.newCredentials },
|
|
284
|
+
};
|
|
285
|
+
this.data = merged;
|
|
286
|
+
this.loadError = null;
|
|
287
|
+
return { result: refreshed, next: JSON.stringify(merged, null, 2) };
|
|
288
|
+
});
|
|
289
|
+
return result;
|
|
290
|
+
}
|
|
291
|
+
async getApiKey(providerId) {
|
|
292
|
+
const runtimeKey = this.runtimeOverrides.get(providerId);
|
|
293
|
+
if (runtimeKey) {
|
|
294
|
+
return runtimeKey;
|
|
295
|
+
}
|
|
296
|
+
const cred = this.data[providerId];
|
|
297
|
+
if (cred?.type === "api_key") {
|
|
298
|
+
return resolveConfigValue(cred.key);
|
|
299
|
+
}
|
|
300
|
+
if (cred?.type === "oauth") {
|
|
301
|
+
const provider = getOAuthProvider(providerId);
|
|
302
|
+
if (!provider) {
|
|
303
|
+
return undefined;
|
|
304
|
+
}
|
|
305
|
+
const needsRefresh = Date.now() >= cred.expires;
|
|
306
|
+
if (needsRefresh) {
|
|
307
|
+
try {
|
|
308
|
+
const result = await this.refreshOAuthTokenWithLock(providerId);
|
|
309
|
+
if (result) {
|
|
310
|
+
return result.apiKey;
|
|
311
|
+
}
|
|
312
|
+
}
|
|
313
|
+
catch (error) {
|
|
314
|
+
this.recordError(error);
|
|
315
|
+
this.reload();
|
|
316
|
+
const updatedCred = this.data[providerId];
|
|
317
|
+
if (updatedCred?.type === "oauth" && Date.now() < updatedCred.expires) {
|
|
318
|
+
return provider.getApiKey(updatedCred);
|
|
319
|
+
}
|
|
320
|
+
return undefined;
|
|
321
|
+
}
|
|
322
|
+
}
|
|
323
|
+
else {
|
|
324
|
+
return provider.getApiKey(cred);
|
|
325
|
+
}
|
|
326
|
+
}
|
|
327
|
+
const envKey = getEnvApiKey(providerId);
|
|
328
|
+
if (envKey)
|
|
329
|
+
return envKey;
|
|
330
|
+
return this.fallbackResolver?.(providerId) ?? undefined;
|
|
331
|
+
}
|
|
332
|
+
getOAuthProviders() {
|
|
333
|
+
return getOAuthProviders();
|
|
334
|
+
}
|
|
335
|
+
}
|
|
336
|
+
//# sourceMappingURL=auth-storage.js.map
|