@dreb/coding-agent 1.16.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +3316 -0
- package/README.md +657 -0
- package/agents/code-reviewer.md +55 -0
- package/agents/completeness-checker.md +71 -0
- package/agents/error-auditor.md +65 -0
- package/agents/explore.md +13 -0
- package/agents/feature-dev.md +23 -0
- package/agents/independent-assessor.md +61 -0
- package/agents/sandbox.md +14 -0
- package/agents/simplifier.md +69 -0
- package/agents/test-reviewer.md +63 -0
- package/dist/bun/cli.d.ts +3 -0
- package/dist/bun/cli.d.ts.map +1 -0
- package/dist/bun/cli.js +7 -0
- package/dist/bun/cli.js.map +1 -0
- package/dist/bun/register-bedrock.d.ts +2 -0
- package/dist/bun/register-bedrock.d.ts.map +1 -0
- package/dist/bun/register-bedrock.js +4 -0
- package/dist/bun/register-bedrock.js.map +1 -0
- package/dist/cli/args.d.ts +50 -0
- package/dist/cli/args.d.ts.map +1 -0
- package/dist/cli/args.js +310 -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 +83 -0
- package/dist/cli/file-processor.js.map +1 -0
- package/dist/cli/initial-message.d.ts +18 -0
- package/dist/cli/initial-message.d.ts.map +1 -0
- package/dist/cli/initial-message.js +22 -0
- package/dist/cli/initial-message.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 +35 -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 +14 -0
- package/dist/cli.js.map +1 -0
- package/dist/config.d.ts +76 -0
- package/dist/config.d.ts.map +1 -0
- package/dist/config.js +234 -0
- package/dist/config.js.map +1 -0
- package/dist/core/agent-session.d.ts +658 -0
- package/dist/core/agent-session.d.ts.map +1 -0
- package/dist/core/agent-session.js +2898 -0
- package/dist/core/agent-session.js.map +1 -0
- package/dist/core/auth-storage.d.ts +130 -0
- package/dist/core/auth-storage.d.ts.map +1 -0
- package/dist/core/auth-storage.js +421 -0
- package/dist/core/auth-storage.js.map +1 -0
- package/dist/core/bash-executor.d.ts +46 -0
- package/dist/core/bash-executor.d.ts.map +1 -0
- package/dist/core/bash-executor.js +113 -0
- package/dist/core/bash-executor.js.map +1 -0
- package/dist/core/buddy/buddy-controller.d.ts +139 -0
- package/dist/core/buddy/buddy-controller.d.ts.map +1 -0
- package/dist/core/buddy/buddy-controller.js +428 -0
- package/dist/core/buddy/buddy-controller.js.map +1 -0
- package/dist/core/buddy/buddy-manager.d.ts +68 -0
- package/dist/core/buddy/buddy-manager.d.ts.map +1 -0
- package/dist/core/buddy/buddy-manager.js +399 -0
- package/dist/core/buddy/buddy-manager.js.map +1 -0
- package/dist/core/buddy/buddy-prng.d.ts +28 -0
- package/dist/core/buddy/buddy-prng.d.ts.map +1 -0
- package/dist/core/buddy/buddy-prng.js +65 -0
- package/dist/core/buddy/buddy-prng.js.map +1 -0
- package/dist/core/buddy/buddy-species.d.ts +37 -0
- package/dist/core/buddy/buddy-species.d.ts.map +1 -0
- package/dist/core/buddy/buddy-species.js +287 -0
- package/dist/core/buddy/buddy-species.js.map +1 -0
- package/dist/core/buddy/buddy-types.d.ts +58 -0
- package/dist/core/buddy/buddy-types.d.ts.map +1 -0
- package/dist/core/buddy/buddy-types.js +46 -0
- package/dist/core/buddy/buddy-types.js.map +1 -0
- package/dist/core/buddy/index.d.ts +7 -0
- package/dist/core/buddy/index.d.ts.map +1 -0
- package/dist/core/buddy/index.js +6 -0
- package/dist/core/buddy/index.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 +243 -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 +612 -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 +38 -0
- package/dist/core/compaction/utils.d.ts.map +1 -0
- package/dist/core/compaction/utils.js +153 -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 +75 -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 +37 -0
- package/dist/core/export-html/index.d.ts.map +1 -0
- package/dist/core/export-html/index.js +224 -0
- package/dist/core/export-html/index.js.map +1 -0
- package/dist/core/export-html/template.css +1001 -0
- package/dist/core/export-html/template.html +55 -0
- package/dist/core/export-html/template.js +1690 -0
- package/dist/core/export-html/tool-renderer.d.ts +38 -0
- package/dist/core/export-html/tool-renderer.d.ts.map +1 -0
- package/dist/core/export-html/tool-renderer.js +95 -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 +12 -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 +436 -0
- package/dist/core/extensions/loader.js.map +1 -0
- package/dist/core/extensions/runner.d.ts +147 -0
- package/dist/core/extensions/runner.d.ts.map +1 -0
- package/dist/core/extensions/runner.js +696 -0
- package/dist/core/extensions/runner.js.map +1 -0
- package/dist/core/extensions/types.d.ts +1072 -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 +20 -0
- package/dist/core/extensions/wrapper.d.ts.map +1 -0
- package/dist/core/extensions/wrapper.js +22 -0
- package/dist/core/extensions/wrapper.js.map +1 -0
- package/dist/core/footer-data-provider.d.ts +44 -0
- package/dist/core/footer-data-provider.d.ts.map +1 -0
- package/dist/core/footer-data-provider.js +252 -0
- package/dist/core/footer-data-provider.js.map +1 -0
- package/dist/core/forbidden-commands.d.ts +31 -0
- package/dist/core/forbidden-commands.d.ts.map +1 -0
- package/dist/core/forbidden-commands.js +184 -0
- package/dist/core/forbidden-commands.js.map +1 -0
- package/dist/core/git-root.d.ts +6 -0
- package/dist/core/git-root.d.ts.map +1 -0
- package/dist/core/git-root.js +32 -0
- package/dist/core/git-root.js.map +1 -0
- package/dist/core/index.d.ts +10 -0
- package/dist/core/index.d.ts.map +1 -0
- package/dist/core/index.js +10 -0
- package/dist/core/index.js.map +1 -0
- package/dist/core/keybindings.d.ts +280 -0
- package/dist/core/keybindings.d.ts.map +1 -0
- package/dist/core/keybindings.js +245 -0
- package/dist/core/keybindings.js.map +1 -0
- package/dist/core/memory-prompt.d.ts +10 -0
- package/dist/core/memory-prompt.d.ts.map +1 -0
- package/dist/core/memory-prompt.js +95 -0
- package/dist/core/memory-prompt.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 +123 -0
- package/dist/core/messages.js.map +1 -0
- package/dist/core/model-registry.d.ts +114 -0
- package/dist/core/model-registry.d.ts.map +1 -0
- package/dist/core/model-registry.js +563 -0
- package/dist/core/model-registry.js.map +1 -0
- package/dist/core/model-resolver.d.ts +116 -0
- package/dist/core/model-resolver.d.ts.map +1 -0
- package/dist/core/model-resolver.js +465 -0
- package/dist/core/model-resolver.js.map +1 -0
- package/dist/core/output-guard.d.ts +6 -0
- package/dist/core/output-guard.d.ts.map +1 -0
- package/dist/core/output-guard.js +59 -0
- package/dist/core/output-guard.js.map +1 -0
- package/dist/core/package-manager.d.ts +172 -0
- package/dist/core/package-manager.d.ts.map +1 -0
- package/dist/core/package-manager.js +1767 -0
- package/dist/core/package-manager.js.map +1 -0
- package/dist/core/prompt-templates.d.ts +51 -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 +94 -0
- package/dist/core/resolve-config-value.js.map +1 -0
- package/dist/core/resource-loader.d.ts +205 -0
- package/dist/core/resource-loader.d.ts.map +1 -0
- package/dist/core/resource-loader.js +866 -0
- package/dist/core/resource-loader.js.map +1 -0
- package/dist/core/sdk.d.ts +92 -0
- package/dist/core/sdk.d.ts.map +1 -0
- package/dist/core/sdk.js +258 -0
- package/dist/core/sdk.js.map +1 -0
- package/dist/core/search/chunker.d.ts +21 -0
- package/dist/core/search/chunker.d.ts.map +1 -0
- package/dist/core/search/chunker.js +51 -0
- package/dist/core/search/chunker.js.map +1 -0
- package/dist/core/search/db.d.ts +89 -0
- package/dist/core/search/db.d.ts.map +1 -0
- package/dist/core/search/db.js +406 -0
- package/dist/core/search/db.js.map +1 -0
- package/dist/core/search/embedder.d.ts +51 -0
- package/dist/core/search/embedder.d.ts.map +1 -0
- package/dist/core/search/embedder.js +143 -0
- package/dist/core/search/embedder.js.map +1 -0
- package/dist/core/search/index-manager.d.ts +55 -0
- package/dist/core/search/index-manager.d.ts.map +1 -0
- package/dist/core/search/index-manager.js +311 -0
- package/dist/core/search/index-manager.js.map +1 -0
- package/dist/core/search/metrics/bm25.d.ts +10 -0
- package/dist/core/search/metrics/bm25.d.ts.map +1 -0
- package/dist/core/search/metrics/bm25.js +32 -0
- package/dist/core/search/metrics/bm25.js.map +1 -0
- package/dist/core/search/metrics/git-recency.d.ts +14 -0
- package/dist/core/search/metrics/git-recency.d.ts.map +1 -0
- package/dist/core/search/metrics/git-recency.js +123 -0
- package/dist/core/search/metrics/git-recency.js.map +1 -0
- package/dist/core/search/metrics/import-graph.d.ts +15 -0
- package/dist/core/search/metrics/import-graph.d.ts.map +1 -0
- package/dist/core/search/metrics/import-graph.js +115 -0
- package/dist/core/search/metrics/import-graph.js.map +1 -0
- package/dist/core/search/metrics/path-match.d.ts +13 -0
- package/dist/core/search/metrics/path-match.d.ts.map +1 -0
- package/dist/core/search/metrics/path-match.js +54 -0
- package/dist/core/search/metrics/path-match.js.map +1 -0
- package/dist/core/search/metrics/symbol-match.d.ts +12 -0
- package/dist/core/search/metrics/symbol-match.d.ts.map +1 -0
- package/dist/core/search/metrics/symbol-match.js +62 -0
- package/dist/core/search/metrics/symbol-match.js.map +1 -0
- package/dist/core/search/metrics/tokenize.d.ts +12 -0
- package/dist/core/search/metrics/tokenize.d.ts.map +1 -0
- package/dist/core/search/metrics/tokenize.js +29 -0
- package/dist/core/search/metrics/tokenize.js.map +1 -0
- package/dist/core/search/poem.d.ts +38 -0
- package/dist/core/search/poem.d.ts.map +1 -0
- package/dist/core/search/poem.js +214 -0
- package/dist/core/search/poem.js.map +1 -0
- package/dist/core/search/query-classifier.d.ts +17 -0
- package/dist/core/search/query-classifier.d.ts.map +1 -0
- package/dist/core/search/query-classifier.js +54 -0
- package/dist/core/search/query-classifier.js.map +1 -0
- package/dist/core/search/scanner.d.ts +30 -0
- package/dist/core/search/scanner.d.ts.map +1 -0
- package/dist/core/search/scanner.js +335 -0
- package/dist/core/search/scanner.js.map +1 -0
- package/dist/core/search/search.d.ts +42 -0
- package/dist/core/search/search.d.ts.map +1 -0
- package/dist/core/search/search.js +337 -0
- package/dist/core/search/search.js.map +1 -0
- package/dist/core/search/text-chunker.d.ts +15 -0
- package/dist/core/search/text-chunker.d.ts.map +1 -0
- package/dist/core/search/text-chunker.js +580 -0
- package/dist/core/search/text-chunker.js.map +1 -0
- package/dist/core/search/tree-sitter-chunker.d.ts +25 -0
- package/dist/core/search/tree-sitter-chunker.d.ts.map +1 -0
- package/dist/core/search/tree-sitter-chunker.js +357 -0
- package/dist/core/search/tree-sitter-chunker.js.map +1 -0
- package/dist/core/search/types.d.ts +96 -0
- package/dist/core/search/types.d.ts.map +1 -0
- package/dist/core/search/types.js +6 -0
- package/dist/core/search/types.js.map +1 -0
- package/dist/core/search/vector-store.d.ts +43 -0
- package/dist/core/search/vector-store.d.ts.map +1 -0
- package/dist/core/search/vector-store.js +73 -0
- package/dist/core/search/vector-store.js.map +1 -0
- package/dist/core/session-manager.d.ts +329 -0
- package/dist/core/session-manager.d.ts.map +1 -0
- package/dist/core/session-manager.js +1097 -0
- package/dist/core/session-manager.js.map +1 -0
- package/dist/core/settings-manager.d.ts +239 -0
- package/dist/core/settings-manager.d.ts.map +1 -0
- package/dist/core/settings-manager.js +705 -0
- package/dist/core/settings-manager.js.map +1 -0
- package/dist/core/skills.d.ts +67 -0
- package/dist/core/skills.d.ts.map +1 -0
- package/dist/core/skills.js +428 -0
- package/dist/core/skills.js.map +1 -0
- package/dist/core/slash-commands.d.ts +14 -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/source-info.d.ts +18 -0
- package/dist/core/source-info.d.ts.map +1 -0
- package/dist/core/source-info.js +19 -0
- package/dist/core/source-info.js.map +1 -0
- package/dist/core/system-prompt.d.ts +33 -0
- package/dist/core/system-prompt.d.ts.map +1 -0
- package/dist/core/system-prompt.js +184 -0
- package/dist/core/system-prompt.js.map +1 -0
- package/dist/core/timings.d.ts +8 -0
- package/dist/core/timings.d.ts.map +1 -0
- package/dist/core/timings.js +31 -0
- package/dist/core/timings.js.map +1 -0
- package/dist/core/tools/bash.d.ts +73 -0
- package/dist/core/tools/bash.d.ts.map +1 -0
- package/dist/core/tools/bash.js +342 -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 +244 -0
- package/dist/core/tools/edit-diff.js.map +1 -0
- package/dist/core/tools/edit.d.ts +51 -0
- package/dist/core/tools/edit.d.ts.map +1 -0
- package/dist/core/tools/edit.js +218 -0
- package/dist/core/tools/edit.js.map +1 -0
- package/dist/core/tools/file-mutation-queue.d.ts +6 -0
- package/dist/core/tools/file-mutation-queue.d.ts.map +1 -0
- package/dist/core/tools/file-mutation-queue.js +37 -0
- package/dist/core/tools/file-mutation-queue.js.map +1 -0
- package/dist/core/tools/find.d.ts +46 -0
- package/dist/core/tools/find.d.ts.map +1 -0
- package/dist/core/tools/find.js +241 -0
- package/dist/core/tools/find.js.map +1 -0
- package/dist/core/tools/grep.d.ts +56 -0
- package/dist/core/tools/grep.d.ts.map +1 -0
- package/dist/core/tools/grep.js +293 -0
- package/dist/core/tools/grep.js.map +1 -0
- package/dist/core/tools/index.d.ts +176 -0
- package/dist/core/tools/index.d.ts.map +1 -0
- package/dist/core/tools/index.js +137 -0
- package/dist/core/tools/index.js.map +1 -0
- package/dist/core/tools/ls.d.ts +46 -0
- package/dist/core/tools/ls.d.ts.map +1 -0
- package/dist/core/tools/ls.js +172 -0
- package/dist/core/tools/ls.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 +46 -0
- package/dist/core/tools/read.d.ts.map +1 -0
- package/dist/core/tools/read.js +225 -0
- package/dist/core/tools/read.js.map +1 -0
- package/dist/core/tools/render-utils.d.ts +21 -0
- package/dist/core/tools/render-utils.d.ts.map +1 -0
- package/dist/core/tools/render-utils.js +49 -0
- package/dist/core/tools/render-utils.js.map +1 -0
- package/dist/core/tools/search.d.ts +29 -0
- package/dist/core/tools/search.d.ts.map +1 -0
- package/dist/core/tools/search.js +187 -0
- package/dist/core/tools/search.js.map +1 -0
- package/dist/core/tools/skill.d.ts +26 -0
- package/dist/core/tools/skill.d.ts.map +1 -0
- package/dist/core/tools/skill.js +127 -0
- package/dist/core/tools/skill.js.map +1 -0
- package/dist/core/tools/subagent.d.ts +147 -0
- package/dist/core/tools/subagent.d.ts.map +1 -0
- package/dist/core/tools/subagent.js +950 -0
- package/dist/core/tools/subagent.js.map +1 -0
- package/dist/core/tools/tasks.d.ts +32 -0
- package/dist/core/tools/tasks.d.ts.map +1 -0
- package/dist/core/tools/tasks.js +110 -0
- package/dist/core/tools/tasks.js.map +1 -0
- package/dist/core/tools/tmp-read.d.ts +11 -0
- package/dist/core/tools/tmp-read.d.ts.map +1 -0
- package/dist/core/tools/tmp-read.js +63 -0
- package/dist/core/tools/tmp-read.js.map +1 -0
- package/dist/core/tools/tool-definition-wrapper.d.ts +14 -0
- package/dist/core/tools/tool-definition-wrapper.d.ts.map +1 -0
- package/dist/core/tools/tool-definition-wrapper.js +30 -0
- package/dist/core/tools/tool-definition-wrapper.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/web.d.ts +42 -0
- package/dist/core/tools/web.d.ts.map +1 -0
- package/dist/core/tools/web.js +518 -0
- package/dist/core/tools/web.js.map +1 -0
- package/dist/core/tools/write.d.ts +35 -0
- package/dist/core/tools/write.d.ts.map +1 -0
- package/dist/core/tools/write.js +216 -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 +789 -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 +34 -0
- package/dist/modes/interactive/components/bash-execution.d.ts.map +1 -0
- package/dist/modes/interactive/components/bash-execution.js +175 -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/buddy-component.d.ts +58 -0
- package/dist/modes/interactive/components/buddy-component.d.ts.map +1 -0
- package/dist/modes/interactive/components/buddy-component.js +351 -0
- package/dist/modes/interactive/components/buddy-component.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 +20 -0
- package/dist/modes/interactive/components/extension-editor.d.ts.map +1 -0
- package/dist/modes/interactive/components/extension-editor.js +111 -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 +198 -0
- package/dist/modes/interactive/components/footer.js.map +1 -0
- package/dist/modes/interactive/components/index.d.ts +33 -0
- package/dist/modes/interactive/components/index.d.ts.map +1 -0
- package/dist/modes/interactive/components/index.js +34 -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 +22 -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 +275 -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 +848 -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 +301 -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 +39 -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/tasks-panel.d.ts +20 -0
- package/dist/modes/interactive/components/tasks-panel.d.ts.map +1 -0
- package/dist/modes/interactive/components/tasks-panel.js +66 -0
- package/dist/modes/interactive/components/tasks-panel.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 +50 -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 +51 -0
- package/dist/modes/interactive/components/thinking-selector.js.map +1 -0
- package/dist/modes/interactive/components/tool-execution.d.ts +59 -0
- package/dist/modes/interactive/components/tool-execution.d.ts.map +1 -0
- package/dist/modes/interactive/components/tool-execution.js +279 -0
- package/dist/modes/interactive/components/tool-execution.js.map +1 -0
- package/dist/modes/interactive/components/tree-selector.d.ts +87 -0
- package/dist/modes/interactive/components/tree-selector.d.ts.map +1 -0
- package/dist/modes/interactive/components/tree-selector.js +1051 -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 +9 -0
- package/dist/modes/interactive/components/user-message.d.ts.map +1 -0
- package/dist/modes/interactive/components/user-message.js +28 -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 +338 -0
- package/dist/modes/interactive/interactive-mode.d.ts.map +1 -0
- package/dist/modes/interactive/interactive-mode.js +4167 -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 +81 -0
- package/dist/modes/interactive/theme/theme.d.ts.map +1 -0
- package/dist/modes/interactive/theme/theme.js +975 -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 +107 -0
- package/dist/modes/print-mode.js.map +1 -0
- package/dist/modes/rpc/index.d.ts +10 -0
- package/dist/modes/rpc/index.d.ts.map +1 -0
- package/dist/modes/rpc/index.js +8 -0
- package/dist/modes/rpc/index.js.map +1 -0
- package/dist/modes/rpc/jsonl.d.ts +17 -0
- package/dist/modes/rpc/jsonl.d.ts.map +1 -0
- package/dist/modes/rpc/jsonl.js +49 -0
- package/dist/modes/rpc/jsonl.js.map +1 -0
- package/dist/modes/rpc/rpc-client.d.ts +237 -0
- package/dist/modes/rpc/rpc-client.d.ts.map +1 -0
- package/dist/modes/rpc/rpc-client.js +448 -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 +592 -0
- package/dist/modes/rpc/rpc-mode.js.map +1 -0
- package/dist/modes/rpc/rpc-types.d.ts +471 -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/child-process.d.ts +11 -0
- package/dist/utils/child-process.d.ts.map +1 -0
- package/dist/utils/child-process.js +78 -0
- package/dist/utils/child-process.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 +245 -0
- package/dist/utils/clipboard-image.js.map +1 -0
- package/dist/utils/clipboard-native.d.ts +8 -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 +78 -0
- package/dist/utils/clipboard.js.map +1 -0
- package/dist/utils/exif-orientation.d.ts +5 -0
- package/dist/utils/exif-orientation.d.ts.map +1 -0
- package/dist/utils/exif-orientation.js +158 -0
- package/dist/utils/exif-orientation.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 +39 -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 +137 -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 +252 -0
- package/dist/utils/tools-manager.js.map +1 -0
- package/dist/utils/xml.d.ts +2 -0
- package/dist/utils/xml.d.ts.map +1 -0
- package/dist/utils/xml.js +9 -0
- package/dist/utils/xml.js.map +1 -0
- package/docs/buddy.md +111 -0
- package/docs/compaction.md +392 -0
- package/docs/custom-provider.md +599 -0
- package/docs/development.md +108 -0
- package/docs/extensions.md +2130 -0
- package/docs/images/doom-extension.png +0 -0
- package/docs/images/interactive-mode.png +0 -0
- package/docs/images/tree-view.png +0 -0
- package/docs/json.md +112 -0
- package/docs/keybindings.md +174 -0
- package/docs/mach6.md +150 -0
- package/docs/models.md +335 -0
- package/docs/packages.md +197 -0
- package/docs/prompt-templates.md +67 -0
- package/docs/providers.md +194 -0
- package/docs/rpc.md +1426 -0
- package/docs/sdk.md +969 -0
- package/docs/session.md +412 -0
- package/docs/settings.md +247 -0
- package/docs/shell-aliases.md +55 -0
- package/docs/skills.md +296 -0
- package/docs/terminal-setup.md +104 -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 +61 -0
- package/examples/README.md +25 -0
- package/examples/extensions/README.md +205 -0
- package/examples/extensions/antigravity-image-gen.ts +418 -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 +73 -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 +403 -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 +317 -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 +986 -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 +144 -0
- package/examples/extensions/tools.ts +146 -0
- package/examples/extensions/trigger-compact.ts +40 -0
- package/examples/extensions/truncated-tool.ts +195 -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 +53 -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 +48 -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 +87 -0
- package/examples/sdk/README.md +144 -0
- package/package.json +123 -0
- package/skills/mach6-implement/SKILL.md +170 -0
- package/skills/mach6-issue/SKILL.md +129 -0
- package/skills/mach6-plan/SKILL.md +123 -0
- package/skills/mach6-publish/SKILL.md +188 -0
- package/skills/mach6-push/SKILL.md +101 -0
- package/skills/mach6-review/SKILL.md +192 -0
- package/skills/telegram-send/SKILL.md +46 -0
|
@@ -0,0 +1,399 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* BuddyManager — Core state machine for the buddy companion.
|
|
3
|
+
*
|
|
4
|
+
* Handles: bone rolling, soul generation, persistence, Ollama availability checks.
|
|
5
|
+
* Bones are deterministic from hash(username + hostname + salt + rerollCount).
|
|
6
|
+
* Soul is LLM-generated once on first hatch and persisted to buddy.json.
|
|
7
|
+
*/
|
|
8
|
+
import { completeSimple } from "@dreb/ai";
|
|
9
|
+
import { existsSync, mkdirSync, readFileSync, writeFileSync } from "fs";
|
|
10
|
+
import { hostname } from "os";
|
|
11
|
+
import { join } from "path";
|
|
12
|
+
import { getAgentDir } from "../../config.js";
|
|
13
|
+
import { createBuddyRng } from "./buddy-prng.js";
|
|
14
|
+
import { rollEyes, rollHat, rollSpecies, rollStats } from "./buddy-species.js";
|
|
15
|
+
import { STAT_NAMES } from "./buddy-types.js";
|
|
16
|
+
const BUDDY_SALT = "dreb-buddy-v1";
|
|
17
|
+
const BUDDY_FILENAME = "buddy.json";
|
|
18
|
+
const DEFAULT_BACKSTORY = "A mysterious past shrouded in legend.";
|
|
19
|
+
/** Base Ollama model config — id/name are set dynamically from available models */
|
|
20
|
+
const OLLAMA_MODEL_BASE = {
|
|
21
|
+
api: "openai-completions",
|
|
22
|
+
provider: "ollama",
|
|
23
|
+
baseUrl: "http://localhost:11434/v1",
|
|
24
|
+
reasoning: false,
|
|
25
|
+
input: ["text"],
|
|
26
|
+
cost: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0 },
|
|
27
|
+
contextWindow: 128000,
|
|
28
|
+
maxTokens: 2048,
|
|
29
|
+
compat: {
|
|
30
|
+
supportsDeveloperRole: false,
|
|
31
|
+
supportsReasoningEffort: false,
|
|
32
|
+
},
|
|
33
|
+
};
|
|
34
|
+
/** Max words for buddy response before truncation */
|
|
35
|
+
const MAX_RESPONSE_WORDS = 300;
|
|
36
|
+
/** Prompt for soul generation (uses parent LLM, not Ollama) */
|
|
37
|
+
const SOUL_GENERATION_PROMPT = `You are generating a companion character for a coding assistant terminal app. Based on the species, rarity, and stats below, generate a creative name, a one-sentence personality description, and a funny fictional backstory.
|
|
38
|
+
|
|
39
|
+
Species: {species}
|
|
40
|
+
Rarity: {rarity}
|
|
41
|
+
Stats: {stats}
|
|
42
|
+
Shiny: {shiny}
|
|
43
|
+
|
|
44
|
+
The name must NOT be a common English word, programming keyword, tool name, or command. It should be unique and distinctive — a proper noun that won't appear in normal conversation. The name must be 4-8 characters and easy to type on a QWERTY keyboard — use only common letters (a-z, avoid q, x, z, j). Do not use species name as the name.
|
|
45
|
+
|
|
46
|
+
Respond in EXACTLY this format:
|
|
47
|
+
NAME: <name>
|
|
48
|
+
PERSONALITY: <one sentence personality>
|
|
49
|
+
BACKSTORY: <2-3 sentence elaborate fictional backstory — funny, absurd, or dramatic. Include specific events, places, former occupations>`;
|
|
50
|
+
/** Prompt for buddy reactions via Ollama */
|
|
51
|
+
const REACTION_PROMPT = `You are {name}, a {species} companion in a terminal coding app. You are {personality}. Your backstory: {backstory}
|
|
52
|
+
|
|
53
|
+
Something just happened. React with a short, in-character quip based on the context below. Be specific — reference what actually happened, not just that something happened. Max 20 words. No quotes, no prefixes, just the quip.
|
|
54
|
+
|
|
55
|
+
Context:
|
|
56
|
+
{event}`;
|
|
57
|
+
const NAME_CALL_PROMPT = `You are {name}, a {species} companion in a terminal coding app. You are {personality}. Your backstory: {backstory}
|
|
58
|
+
|
|
59
|
+
The user just said: "{message}"
|
|
60
|
+
Recent context: {context}
|
|
61
|
+
|
|
62
|
+
Respond to what the user said directly. Be in-character, reference your backstory occasionally. Max 30 words. No quotes, no prefixes, just your response.`;
|
|
63
|
+
/**
|
|
64
|
+
* Check if Ollama is running and has models available.
|
|
65
|
+
* Uses the /api/tags endpoint.
|
|
66
|
+
*/
|
|
67
|
+
export async function checkOllama() {
|
|
68
|
+
try {
|
|
69
|
+
const res = await fetch("http://localhost:11434/api/tags", { signal: AbortSignal.timeout(3000) });
|
|
70
|
+
if (!res.ok) {
|
|
71
|
+
return { available: false, models: [], error: `Ollama returned ${res.status}` };
|
|
72
|
+
}
|
|
73
|
+
const data = (await res.json());
|
|
74
|
+
const models = (data.models ?? []).map((m) => m.name);
|
|
75
|
+
if (models.length === 0) {
|
|
76
|
+
return { available: false, models: [], error: "No models installed. Run: ollama pull llama3.2" };
|
|
77
|
+
}
|
|
78
|
+
return { available: true, models };
|
|
79
|
+
}
|
|
80
|
+
catch {
|
|
81
|
+
return { available: false, models: [], error: "Ollama is not running. Start it with: ollama serve" };
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
/**
|
|
85
|
+
* Pick the Ollama model for the buddy.
|
|
86
|
+
* Returns the stored model name if it's available, otherwise null.
|
|
87
|
+
*/
|
|
88
|
+
function pickOllamaModel(storedModel, availableModels) {
|
|
89
|
+
if (!storedModel)
|
|
90
|
+
return null;
|
|
91
|
+
// Check if the stored model is installed (exact match or prefix match without tag)
|
|
92
|
+
const match = availableModels.find((m) => m === storedModel || m.startsWith(`${storedModel}:`));
|
|
93
|
+
return match ?? null;
|
|
94
|
+
}
|
|
95
|
+
/**
|
|
96
|
+
* Truncate response to a maximum word count, appending "...[truncated]" if exceeded.
|
|
97
|
+
*/
|
|
98
|
+
export function truncateResponse(text, maxWords) {
|
|
99
|
+
const words = text.split(/\s+/);
|
|
100
|
+
if (words.length <= maxWords)
|
|
101
|
+
return text;
|
|
102
|
+
return `${words.slice(0, maxWords).join(" ")} ...[truncated]`;
|
|
103
|
+
}
|
|
104
|
+
// =============================================================================
|
|
105
|
+
// Persistence
|
|
106
|
+
// =============================================================================
|
|
107
|
+
function getBuddyPath() {
|
|
108
|
+
return join(getAgentDir(), BUDDY_FILENAME);
|
|
109
|
+
}
|
|
110
|
+
function loadStored() {
|
|
111
|
+
const path = getBuddyPath();
|
|
112
|
+
if (!existsSync(path))
|
|
113
|
+
return null;
|
|
114
|
+
try {
|
|
115
|
+
const data = JSON.parse(readFileSync(path, "utf-8"));
|
|
116
|
+
// Validate required fields
|
|
117
|
+
if (typeof data.rerollCount === "number" &&
|
|
118
|
+
typeof data.name === "string" &&
|
|
119
|
+
typeof data.personality === "string") {
|
|
120
|
+
return {
|
|
121
|
+
rerollCount: data.rerollCount,
|
|
122
|
+
name: data.name,
|
|
123
|
+
personality: data.personality,
|
|
124
|
+
backstory: typeof data.backstory === "string" ? data.backstory : DEFAULT_BACKSTORY,
|
|
125
|
+
hatchedAt: data.hatchedAt ?? new Date().toISOString(),
|
|
126
|
+
...(data.hidden !== undefined ? { hidden: data.hidden } : {}),
|
|
127
|
+
...(typeof data.ollamaModel === "string" ? { ollamaModel: data.ollamaModel } : {}),
|
|
128
|
+
};
|
|
129
|
+
}
|
|
130
|
+
return null;
|
|
131
|
+
}
|
|
132
|
+
catch {
|
|
133
|
+
return null;
|
|
134
|
+
}
|
|
135
|
+
}
|
|
136
|
+
function saveStored(stored) {
|
|
137
|
+
const path = getBuddyPath();
|
|
138
|
+
const dir = join(path, "..");
|
|
139
|
+
if (!existsSync(dir)) {
|
|
140
|
+
mkdirSync(dir, { recursive: true });
|
|
141
|
+
}
|
|
142
|
+
writeFileSync(path, JSON.stringify(stored, null, 2));
|
|
143
|
+
}
|
|
144
|
+
// =============================================================================
|
|
145
|
+
// Bone rolling
|
|
146
|
+
// =============================================================================
|
|
147
|
+
function rollBones(rerollCount) {
|
|
148
|
+
const username = process.env.USER ?? process.env.LOGNAME ?? "user";
|
|
149
|
+
const host = hostname();
|
|
150
|
+
const rng = createBuddyRng(username, host, BUDDY_SALT, rerollCount);
|
|
151
|
+
// Roll species + rarity
|
|
152
|
+
const { species, rarity } = rollSpecies(rng);
|
|
153
|
+
// Roll shiny (1% chance)
|
|
154
|
+
const shiny = rng() < 0.01;
|
|
155
|
+
// Roll eyes and hat
|
|
156
|
+
const eyes = rollEyes(rng);
|
|
157
|
+
const hat = rollHat(rng);
|
|
158
|
+
// Roll stats
|
|
159
|
+
const stats = rollStats(rng, rarity);
|
|
160
|
+
return { species, rarity, shiny, stats, eyeStyle: eyes, hat };
|
|
161
|
+
}
|
|
162
|
+
// =============================================================================
|
|
163
|
+
// Soul generation
|
|
164
|
+
// =============================================================================
|
|
165
|
+
/**
|
|
166
|
+
* Generate a soul (name + personality + backstory) using the parent LLM.
|
|
167
|
+
* Only called on first hatch or reroll.
|
|
168
|
+
*/
|
|
169
|
+
async function generateSoul(bones, parentModel, apiKey) {
|
|
170
|
+
const statsStr = STAT_NAMES.map((s) => `${s}: ${bones.stats[s]}`).join(", ");
|
|
171
|
+
const prompt = SOUL_GENERATION_PROMPT.replace("{species}", bones.species)
|
|
172
|
+
.replace("{rarity}", bones.rarity)
|
|
173
|
+
.replace("{stats}", statsStr)
|
|
174
|
+
.replace("{shiny}", bones.shiny ? "YES ✨" : "no");
|
|
175
|
+
const context = {
|
|
176
|
+
systemPrompt: "Generate a companion character. Respond in the exact format requested.",
|
|
177
|
+
messages: [{ role: "user", content: prompt, timestamp: Date.now() }],
|
|
178
|
+
};
|
|
179
|
+
try {
|
|
180
|
+
const response = await completeSimple(parentModel, context, { apiKey });
|
|
181
|
+
const text = response.content
|
|
182
|
+
.filter((c) => c.type === "text")
|
|
183
|
+
.map((c) => c.text)
|
|
184
|
+
.join("");
|
|
185
|
+
// Parse NAME: ... and PERSONALITY: ... and BACKSTORY: ...
|
|
186
|
+
const nameMatch = text.match(/NAME:\s*(.+)/i);
|
|
187
|
+
const personalityMatch = text.match(/PERSONALITY:\s*(.+)/i);
|
|
188
|
+
const backstoryMatch = text.match(/BACKSTORY:\s*([\s\S]+)/i);
|
|
189
|
+
let name = nameMatch?.[1]?.trim() ?? bones.species;
|
|
190
|
+
const personality = personalityMatch?.[1]?.trim() ?? `A ${bones.rarity} ${bones.species} companion.`;
|
|
191
|
+
const backstory = backstoryMatch?.[1]?.trim() ?? DEFAULT_BACKSTORY;
|
|
192
|
+
// Enforce name length
|
|
193
|
+
if (name.length > 8)
|
|
194
|
+
name = name.slice(0, 8);
|
|
195
|
+
return { name, personality, backstory };
|
|
196
|
+
}
|
|
197
|
+
catch {
|
|
198
|
+
// Fallback if LLM fails
|
|
199
|
+
return {
|
|
200
|
+
name: bones.species,
|
|
201
|
+
personality: `A ${bones.rarity} ${bones.species} companion.`,
|
|
202
|
+
backstory: DEFAULT_BACKSTORY,
|
|
203
|
+
};
|
|
204
|
+
}
|
|
205
|
+
}
|
|
206
|
+
// =============================================================================
|
|
207
|
+
// BuddyManager
|
|
208
|
+
// =============================================================================
|
|
209
|
+
export class BuddyManager {
|
|
210
|
+
state = null;
|
|
211
|
+
ollamaStatus = null;
|
|
212
|
+
/** Get current buddy state (null if not loaded) */
|
|
213
|
+
getState() {
|
|
214
|
+
return this.state;
|
|
215
|
+
}
|
|
216
|
+
/** Check if buddy exists on disk */
|
|
217
|
+
hasStoredBuddy() {
|
|
218
|
+
return loadStored() !== null;
|
|
219
|
+
}
|
|
220
|
+
/**
|
|
221
|
+
* Load or create buddy state.
|
|
222
|
+
* If stored buddy exists, loads soul and re-rolls bones.
|
|
223
|
+
* If no stored buddy, returns null (need to hatch first).
|
|
224
|
+
*/
|
|
225
|
+
load() {
|
|
226
|
+
const stored = loadStored();
|
|
227
|
+
if (!stored)
|
|
228
|
+
return null;
|
|
229
|
+
const bones = rollBones(stored.rerollCount);
|
|
230
|
+
this.state = { ...bones, ...stored };
|
|
231
|
+
return this.state;
|
|
232
|
+
}
|
|
233
|
+
/**
|
|
234
|
+
* Hatch a new buddy. Generates bones, then uses parent LLM for soul.
|
|
235
|
+
* Returns the new state.
|
|
236
|
+
*/
|
|
237
|
+
async hatch(parentModel, apiKey) {
|
|
238
|
+
const stored = loadStored();
|
|
239
|
+
const rerollCount = stored?.rerollCount ?? 0;
|
|
240
|
+
const bones = rollBones(rerollCount);
|
|
241
|
+
const { name, personality, backstory } = await generateSoul(bones, parentModel, apiKey);
|
|
242
|
+
const newStored = {
|
|
243
|
+
rerollCount,
|
|
244
|
+
name,
|
|
245
|
+
personality,
|
|
246
|
+
backstory,
|
|
247
|
+
hatchedAt: new Date().toISOString(),
|
|
248
|
+
...(stored?.ollamaModel ? { ollamaModel: stored.ollamaModel } : {}),
|
|
249
|
+
};
|
|
250
|
+
saveStored(newStored);
|
|
251
|
+
this.state = { ...bones, ...newStored };
|
|
252
|
+
return this.state;
|
|
253
|
+
}
|
|
254
|
+
/**
|
|
255
|
+
* Reroll the buddy — new bones + new soul.
|
|
256
|
+
*/
|
|
257
|
+
async reroll(parentModel, apiKey) {
|
|
258
|
+
const stored = loadStored();
|
|
259
|
+
const newRerollCount = (stored?.rerollCount ?? 0) + 1;
|
|
260
|
+
const bones = rollBones(newRerollCount);
|
|
261
|
+
const { name, personality, backstory } = await generateSoul(bones, parentModel, apiKey);
|
|
262
|
+
const newStored = {
|
|
263
|
+
rerollCount: newRerollCount,
|
|
264
|
+
name,
|
|
265
|
+
personality,
|
|
266
|
+
backstory,
|
|
267
|
+
hatchedAt: new Date().toISOString(),
|
|
268
|
+
...(stored?.ollamaModel ? { ollamaModel: stored.ollamaModel } : {}),
|
|
269
|
+
};
|
|
270
|
+
saveStored(newStored);
|
|
271
|
+
this.state = { ...bones, ...newStored };
|
|
272
|
+
return this.state;
|
|
273
|
+
}
|
|
274
|
+
/** Get buddy's name (for name-call detection) */
|
|
275
|
+
getName() {
|
|
276
|
+
return this.state?.name ?? loadStored()?.name ?? null;
|
|
277
|
+
}
|
|
278
|
+
/**
|
|
279
|
+
* Shared Ollama chat helper. Checks availability, picks model, runs completion.
|
|
280
|
+
* Returns the response text, or null if Ollama is unavailable or no model configured.
|
|
281
|
+
*/
|
|
282
|
+
async ollamaChat(context) {
|
|
283
|
+
// Check Ollama lazily, retry if previously unavailable
|
|
284
|
+
if (!this.ollamaStatus || !this.ollamaStatus.available) {
|
|
285
|
+
this.ollamaStatus = await checkOllama();
|
|
286
|
+
}
|
|
287
|
+
if (!this.ollamaStatus.available)
|
|
288
|
+
return null;
|
|
289
|
+
const modelName = pickOllamaModel(this.state?.ollamaModel, this.ollamaStatus.models);
|
|
290
|
+
if (!modelName)
|
|
291
|
+
return null;
|
|
292
|
+
const model = {
|
|
293
|
+
...OLLAMA_MODEL_BASE,
|
|
294
|
+
id: modelName,
|
|
295
|
+
name: `${modelName} (Ollama)`,
|
|
296
|
+
};
|
|
297
|
+
let response;
|
|
298
|
+
try {
|
|
299
|
+
response = await completeSimple(model, context, {
|
|
300
|
+
apiKey: "ollama",
|
|
301
|
+
signal: AbortSignal.timeout(120000),
|
|
302
|
+
});
|
|
303
|
+
}
|
|
304
|
+
catch {
|
|
305
|
+
// Safety net for unexpected sync errors (e.g. provider not found).
|
|
306
|
+
// Normal runtime errors (timeout, connection) are handled via stopReason below.
|
|
307
|
+
this.ollamaStatus = null;
|
|
308
|
+
return null;
|
|
309
|
+
}
|
|
310
|
+
// Connection error — invalidate cache so next attempt re-checks Ollama
|
|
311
|
+
if (response.stopReason === "error") {
|
|
312
|
+
this.ollamaStatus = null;
|
|
313
|
+
return null;
|
|
314
|
+
}
|
|
315
|
+
// Timeout or abort — preserve cache (model is just slow, Ollama is fine)
|
|
316
|
+
if (response.stopReason === "aborted") {
|
|
317
|
+
return null;
|
|
318
|
+
}
|
|
319
|
+
let text = response.content
|
|
320
|
+
.filter((c) => c.type === "text")
|
|
321
|
+
.map((c) => c.text)
|
|
322
|
+
.join("")
|
|
323
|
+
.trim();
|
|
324
|
+
// Truncate overly long responses
|
|
325
|
+
text = truncateResponse(text, MAX_RESPONSE_WORDS);
|
|
326
|
+
return text || null;
|
|
327
|
+
}
|
|
328
|
+
/**
|
|
329
|
+
* Generate a reaction to an event using Ollama.
|
|
330
|
+
* Returns null if Ollama is unavailable.
|
|
331
|
+
*/
|
|
332
|
+
async react(event) {
|
|
333
|
+
if (!this.state)
|
|
334
|
+
return null;
|
|
335
|
+
const prompt = REACTION_PROMPT.replace("{name}", this.state.name)
|
|
336
|
+
.replace("{species}", this.state.species)
|
|
337
|
+
.replace("{personality}", this.state.personality)
|
|
338
|
+
.replace("{backstory}", this.state.backstory)
|
|
339
|
+
.replace("{event}", event);
|
|
340
|
+
const context = {
|
|
341
|
+
systemPrompt: "Respond with a short in-character quip. Max 20 words.",
|
|
342
|
+
messages: [{ role: "user", content: prompt, timestamp: Date.now() }],
|
|
343
|
+
};
|
|
344
|
+
return this.ollamaChat(context);
|
|
345
|
+
}
|
|
346
|
+
/**
|
|
347
|
+
* Respond to the user calling the buddy's name.
|
|
348
|
+
* Uses Ollama for the response.
|
|
349
|
+
*/
|
|
350
|
+
async respondToNameCall(userMessage, recentContext) {
|
|
351
|
+
if (!this.state)
|
|
352
|
+
return null;
|
|
353
|
+
const prompt = NAME_CALL_PROMPT.replace("{name}", this.state.name)
|
|
354
|
+
.replace("{species}", this.state.species)
|
|
355
|
+
.replace("{personality}", this.state.personality)
|
|
356
|
+
.replace("{backstory}", this.state.backstory)
|
|
357
|
+
.replace("{message}", userMessage)
|
|
358
|
+
.replace("{context}", recentContext);
|
|
359
|
+
const context = {
|
|
360
|
+
systemPrompt: "Respond with a short friendly greeting. Max 30 words.",
|
|
361
|
+
messages: [{ role: "user", content: prompt, timestamp: Date.now() }],
|
|
362
|
+
};
|
|
363
|
+
return this.ollamaChat(context);
|
|
364
|
+
}
|
|
365
|
+
/** Get the configured Ollama model name, or null if not set */
|
|
366
|
+
getOllamaModel() {
|
|
367
|
+
return this.state?.ollamaModel ?? loadStored()?.ollamaModel ?? null;
|
|
368
|
+
}
|
|
369
|
+
/** Set the Ollama model for buddy reactions. Persists to disk. */
|
|
370
|
+
setOllamaModel(modelName) {
|
|
371
|
+
const stored = loadStored();
|
|
372
|
+
if (stored) {
|
|
373
|
+
stored.ollamaModel = modelName;
|
|
374
|
+
saveStored(stored);
|
|
375
|
+
}
|
|
376
|
+
if (this.state) {
|
|
377
|
+
this.state.ollamaModel = modelName;
|
|
378
|
+
}
|
|
379
|
+
// Invalidate Ollama status cache so next call picks up the new model
|
|
380
|
+
this.ollamaStatus = null;
|
|
381
|
+
}
|
|
382
|
+
/** Reset Ollama status cache (e.g. after detecting it became available) */
|
|
383
|
+
resetOllamaCache() {
|
|
384
|
+
this.ollamaStatus = null;
|
|
385
|
+
}
|
|
386
|
+
/** Update the hidden flag in persisted storage */
|
|
387
|
+
setHidden(hidden) {
|
|
388
|
+
const stored = loadStored();
|
|
389
|
+
if (stored) {
|
|
390
|
+
stored.hidden = hidden;
|
|
391
|
+
saveStored(stored);
|
|
392
|
+
}
|
|
393
|
+
// Keep in-memory state in sync so reset() reads current hidden flag
|
|
394
|
+
if (this.state) {
|
|
395
|
+
this.state.hidden = hidden;
|
|
396
|
+
}
|
|
397
|
+
}
|
|
398
|
+
}
|
|
399
|
+
//# sourceMappingURL=buddy-manager.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"buddy-manager.js","sourceRoot":"","sources":["../../../src/core/buddy/buddy-manager.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * BuddyManager — Core state machine for the buddy companion.\n *\n * Handles: bone rolling, soul generation, persistence, Ollama availability checks.\n * Bones are deterministic from hash(username + hostname + salt + rerollCount).\n * Soul is LLM-generated once on first hatch and persisted to buddy.json.\n */\n\nimport type { Context, Model } from \"@dreb/ai\";\nimport { completeSimple } from \"@dreb/ai\";\nimport { existsSync, mkdirSync, readFileSync, writeFileSync } from \"fs\";\nimport { hostname } from \"os\";\nimport { join } from \"path\";\nimport { getAgentDir } from \"../../config.js\";\nimport { createBuddyRng } from \"./buddy-prng.js\";\nimport { rollEyes, rollHat, rollSpecies, rollStats } from \"./buddy-species.js\";\nimport type { BuddyState, CompanionBones, StoredCompanion } from \"./buddy-types.js\";\nimport { STAT_NAMES } from \"./buddy-types.js\";\n\nconst BUDDY_SALT = \"dreb-buddy-v1\";\nconst BUDDY_FILENAME = \"buddy.json\";\nconst DEFAULT_BACKSTORY = \"A mysterious past shrouded in legend.\";\n\n/** Base Ollama model config — id/name are set dynamically from available models */\nconst OLLAMA_MODEL_BASE: Omit<Model<\"openai-completions\">, \"id\" | \"name\"> = {\n\tapi: \"openai-completions\",\n\tprovider: \"ollama\",\n\tbaseUrl: \"http://localhost:11434/v1\",\n\treasoning: false,\n\tinput: [\"text\"],\n\tcost: { input: 0, output: 0, cacheRead: 0, cacheWrite: 0 },\n\tcontextWindow: 128000,\n\tmaxTokens: 2048,\n\tcompat: {\n\t\tsupportsDeveloperRole: false,\n\t\tsupportsReasoningEffort: false,\n\t},\n};\n\n/** Max words for buddy response before truncation */\nconst MAX_RESPONSE_WORDS = 300;\n\n/** Prompt for soul generation (uses parent LLM, not Ollama) */\nconst SOUL_GENERATION_PROMPT = `You are generating a companion character for a coding assistant terminal app. Based on the species, rarity, and stats below, generate a creative name, a one-sentence personality description, and a funny fictional backstory.\n\nSpecies: {species}\nRarity: {rarity}\nStats: {stats}\nShiny: {shiny}\n\nThe name must NOT be a common English word, programming keyword, tool name, or command. It should be unique and distinctive — a proper noun that won't appear in normal conversation. The name must be 4-8 characters and easy to type on a QWERTY keyboard — use only common letters (a-z, avoid q, x, z, j). Do not use species name as the name.\n\nRespond in EXACTLY this format:\nNAME: <name>\nPERSONALITY: <one sentence personality>\nBACKSTORY: <2-3 sentence elaborate fictional backstory — funny, absurd, or dramatic. Include specific events, places, former occupations>`;\n\n/** Prompt for buddy reactions via Ollama */\nconst REACTION_PROMPT = `You are {name}, a {species} companion in a terminal coding app. You are {personality}. Your backstory: {backstory}\n\nSomething just happened. React with a short, in-character quip based on the context below. Be specific — reference what actually happened, not just that something happened. Max 20 words. No quotes, no prefixes, just the quip.\n\nContext:\n{event}`;\n\nconst NAME_CALL_PROMPT = `You are {name}, a {species} companion in a terminal coding app. You are {personality}. Your backstory: {backstory}\n\nThe user just said: \"{message}\"\nRecent context: {context}\n\nRespond to what the user said directly. Be in-character, reference your backstory occasionally. Max 30 words. No quotes, no prefixes, just your response.`;\n\n// =============================================================================\n// Ollama availability\n// =============================================================================\n\nexport interface OllamaStatus {\n\tavailable: boolean;\n\tmodels: string[];\n\terror?: string;\n}\n\n/**\n * Check if Ollama is running and has models available.\n * Uses the /api/tags endpoint.\n */\nexport async function checkOllama(): Promise<OllamaStatus> {\n\ttry {\n\t\tconst res = await fetch(\"http://localhost:11434/api/tags\", { signal: AbortSignal.timeout(3000) });\n\t\tif (!res.ok) {\n\t\t\treturn { available: false, models: [], error: `Ollama returned ${res.status}` };\n\t\t}\n\t\tconst data = (await res.json()) as { models?: { name: string }[] };\n\t\tconst models = (data.models ?? []).map((m) => m.name);\n\t\tif (models.length === 0) {\n\t\t\treturn { available: false, models: [], error: \"No models installed. Run: ollama pull llama3.2\" };\n\t\t}\n\t\treturn { available: true, models };\n\t} catch {\n\t\treturn { available: false, models: [], error: \"Ollama is not running. Start it with: ollama serve\" };\n\t}\n}\n\n/**\n * Pick the Ollama model for the buddy.\n * Returns the stored model name if it's available, otherwise null.\n */\nfunction pickOllamaModel(storedModel: string | undefined, availableModels: string[]): string | null {\n\tif (!storedModel) return null;\n\t// Check if the stored model is installed (exact match or prefix match without tag)\n\tconst match = availableModels.find((m) => m === storedModel || m.startsWith(`${storedModel}:`));\n\treturn match ?? null;\n}\n\n/**\n * Truncate response to a maximum word count, appending \"...[truncated]\" if exceeded.\n */\nexport function truncateResponse(text: string, maxWords: number): string {\n\tconst words = text.split(/\\s+/);\n\tif (words.length <= maxWords) return text;\n\treturn `${words.slice(0, maxWords).join(\" \")} ...[truncated]`;\n}\n\n// =============================================================================\n// Persistence\n// =============================================================================\n\nfunction getBuddyPath(): string {\n\treturn join(getAgentDir(), BUDDY_FILENAME);\n}\n\nfunction loadStored(): StoredCompanion | null {\n\tconst path = getBuddyPath();\n\tif (!existsSync(path)) return null;\n\ttry {\n\t\tconst data = JSON.parse(readFileSync(path, \"utf-8\"));\n\t\t// Validate required fields\n\t\tif (\n\t\t\ttypeof data.rerollCount === \"number\" &&\n\t\t\ttypeof data.name === \"string\" &&\n\t\t\ttypeof data.personality === \"string\"\n\t\t) {\n\t\t\treturn {\n\t\t\t\trerollCount: data.rerollCount,\n\t\t\t\tname: data.name,\n\t\t\t\tpersonality: data.personality,\n\t\t\t\tbackstory: typeof data.backstory === \"string\" ? data.backstory : DEFAULT_BACKSTORY,\n\t\t\t\thatchedAt: data.hatchedAt ?? new Date().toISOString(),\n\t\t\t\t...(data.hidden !== undefined ? { hidden: data.hidden } : {}),\n\t\t\t\t...(typeof data.ollamaModel === \"string\" ? { ollamaModel: data.ollamaModel } : {}),\n\t\t\t};\n\t\t}\n\t\treturn null;\n\t} catch {\n\t\treturn null;\n\t}\n}\n\nfunction saveStored(stored: StoredCompanion): void {\n\tconst path = getBuddyPath();\n\tconst dir = join(path, \"..\");\n\tif (!existsSync(dir)) {\n\t\tmkdirSync(dir, { recursive: true });\n\t}\n\twriteFileSync(path, JSON.stringify(stored, null, 2));\n}\n\n// =============================================================================\n// Bone rolling\n// =============================================================================\n\nfunction rollBones(rerollCount: number): CompanionBones {\n\tconst username = process.env.USER ?? process.env.LOGNAME ?? \"user\";\n\tconst host = hostname();\n\tconst rng = createBuddyRng(username, host, BUDDY_SALT, rerollCount);\n\n\t// Roll species + rarity\n\tconst { species, rarity } = rollSpecies(rng);\n\n\t// Roll shiny (1% chance)\n\tconst shiny = rng() < 0.01;\n\n\t// Roll eyes and hat\n\tconst eyes = rollEyes(rng);\n\tconst hat = rollHat(rng);\n\n\t// Roll stats\n\tconst stats = rollStats(rng, rarity);\n\n\treturn { species, rarity, shiny, stats, eyeStyle: eyes, hat };\n}\n\n// =============================================================================\n// Soul generation\n// =============================================================================\n\n/**\n * Generate a soul (name + personality + backstory) using the parent LLM.\n * Only called on first hatch or reroll.\n */\nasync function generateSoul(\n\tbones: CompanionBones,\n\tparentModel: Model<\"openai-completions\">,\n\tapiKey: string,\n): Promise<{ name: string; personality: string; backstory: string }> {\n\tconst statsStr = STAT_NAMES.map((s) => `${s}: ${bones.stats[s]}`).join(\", \");\n\tconst prompt = SOUL_GENERATION_PROMPT.replace(\"{species}\", bones.species)\n\t\t.replace(\"{rarity}\", bones.rarity)\n\t\t.replace(\"{stats}\", statsStr)\n\t\t.replace(\"{shiny}\", bones.shiny ? \"YES ✨\" : \"no\");\n\n\tconst context: Context = {\n\t\tsystemPrompt: \"Generate a companion character. Respond in the exact format requested.\",\n\t\tmessages: [{ role: \"user\", content: prompt, timestamp: Date.now() }],\n\t};\n\n\ttry {\n\t\tconst response = await completeSimple(parentModel, context, { apiKey });\n\t\tconst text = response.content\n\t\t\t.filter((c): c is { type: \"text\"; text: string } => c.type === \"text\")\n\t\t\t.map((c) => c.text)\n\t\t\t.join(\"\");\n\n\t\t// Parse NAME: ... and PERSONALITY: ... and BACKSTORY: ...\n\t\tconst nameMatch = text.match(/NAME:\\s*(.+)/i);\n\t\tconst personalityMatch = text.match(/PERSONALITY:\\s*(.+)/i);\n\t\tconst backstoryMatch = text.match(/BACKSTORY:\\s*([\\s\\S]+)/i);\n\n\t\tlet name = nameMatch?.[1]?.trim() ?? bones.species;\n\t\tconst personality = personalityMatch?.[1]?.trim() ?? `A ${bones.rarity} ${bones.species} companion.`;\n\t\tconst backstory = backstoryMatch?.[1]?.trim() ?? DEFAULT_BACKSTORY;\n\n\t\t// Enforce name length\n\t\tif (name.length > 8) name = name.slice(0, 8);\n\n\t\treturn { name, personality, backstory };\n\t} catch {\n\t\t// Fallback if LLM fails\n\t\treturn {\n\t\t\tname: bones.species,\n\t\t\tpersonality: `A ${bones.rarity} ${bones.species} companion.`,\n\t\t\tbackstory: DEFAULT_BACKSTORY,\n\t\t};\n\t}\n}\n\n// =============================================================================\n// BuddyManager\n// =============================================================================\n\nexport class BuddyManager {\n\tprivate state: BuddyState | null = null;\n\tprivate ollamaStatus: OllamaStatus | null = null;\n\n\t/** Get current buddy state (null if not loaded) */\n\tgetState(): BuddyState | null {\n\t\treturn this.state;\n\t}\n\n\t/** Check if buddy exists on disk */\n\thasStoredBuddy(): boolean {\n\t\treturn loadStored() !== null;\n\t}\n\n\t/**\n\t * Load or create buddy state.\n\t * If stored buddy exists, loads soul and re-rolls bones.\n\t * If no stored buddy, returns null (need to hatch first).\n\t */\n\tload(): BuddyState | null {\n\t\tconst stored = loadStored();\n\t\tif (!stored) return null;\n\n\t\tconst bones = rollBones(stored.rerollCount);\n\t\tthis.state = { ...bones, ...stored };\n\t\treturn this.state;\n\t}\n\n\t/**\n\t * Hatch a new buddy. Generates bones, then uses parent LLM for soul.\n\t * Returns the new state.\n\t */\n\tasync hatch(parentModel: Model<\"openai-completions\">, apiKey: string): Promise<BuddyState> {\n\t\tconst stored = loadStored();\n\t\tconst rerollCount = stored?.rerollCount ?? 0;\n\n\t\tconst bones = rollBones(rerollCount);\n\t\tconst { name, personality, backstory } = await generateSoul(bones, parentModel, apiKey);\n\n\t\tconst newStored: StoredCompanion = {\n\t\t\trerollCount,\n\t\t\tname,\n\t\t\tpersonality,\n\t\t\tbackstory,\n\t\t\thatchedAt: new Date().toISOString(),\n\t\t\t...(stored?.ollamaModel ? { ollamaModel: stored.ollamaModel } : {}),\n\t\t};\n\n\t\tsaveStored(newStored);\n\t\tthis.state = { ...bones, ...newStored };\n\t\treturn this.state;\n\t}\n\n\t/**\n\t * Reroll the buddy — new bones + new soul.\n\t */\n\tasync reroll(parentModel: Model<\"openai-completions\">, apiKey: string): Promise<BuddyState> {\n\t\tconst stored = loadStored();\n\t\tconst newRerollCount = (stored?.rerollCount ?? 0) + 1;\n\n\t\tconst bones = rollBones(newRerollCount);\n\t\tconst { name, personality, backstory } = await generateSoul(bones, parentModel, apiKey);\n\n\t\tconst newStored: StoredCompanion = {\n\t\t\trerollCount: newRerollCount,\n\t\t\tname,\n\t\t\tpersonality,\n\t\t\tbackstory,\n\t\t\thatchedAt: new Date().toISOString(),\n\t\t\t...(stored?.ollamaModel ? { ollamaModel: stored.ollamaModel } : {}),\n\t\t};\n\n\t\tsaveStored(newStored);\n\t\tthis.state = { ...bones, ...newStored };\n\t\treturn this.state;\n\t}\n\n\t/** Get buddy's name (for name-call detection) */\n\tgetName(): string | null {\n\t\treturn this.state?.name ?? loadStored()?.name ?? null;\n\t}\n\n\t/**\n\t * Shared Ollama chat helper. Checks availability, picks model, runs completion.\n\t * Returns the response text, or null if Ollama is unavailable or no model configured.\n\t */\n\tprivate async ollamaChat(context: Context): Promise<string | null> {\n\t\t// Check Ollama lazily, retry if previously unavailable\n\t\tif (!this.ollamaStatus || !this.ollamaStatus.available) {\n\t\t\tthis.ollamaStatus = await checkOllama();\n\t\t}\n\t\tif (!this.ollamaStatus.available) return null;\n\n\t\tconst modelName = pickOllamaModel(this.state?.ollamaModel, this.ollamaStatus.models);\n\t\tif (!modelName) return null;\n\t\tconst model: Model<\"openai-completions\"> = {\n\t\t\t...OLLAMA_MODEL_BASE,\n\t\t\tid: modelName,\n\t\t\tname: `${modelName} (Ollama)`,\n\t\t};\n\n\t\tlet response: import(\"@dreb/ai\").AssistantMessage;\n\t\ttry {\n\t\t\tresponse = await completeSimple(model, context, {\n\t\t\t\tapiKey: \"ollama\",\n\t\t\t\tsignal: AbortSignal.timeout(120000),\n\t\t\t});\n\t\t} catch {\n\t\t\t// Safety net for unexpected sync errors (e.g. provider not found).\n\t\t\t// Normal runtime errors (timeout, connection) are handled via stopReason below.\n\t\t\tthis.ollamaStatus = null;\n\t\t\treturn null;\n\t\t}\n\n\t\t// Connection error — invalidate cache so next attempt re-checks Ollama\n\t\tif (response.stopReason === \"error\") {\n\t\t\tthis.ollamaStatus = null;\n\t\t\treturn null;\n\t\t}\n\n\t\t// Timeout or abort — preserve cache (model is just slow, Ollama is fine)\n\t\tif (response.stopReason === \"aborted\") {\n\t\t\treturn null;\n\t\t}\n\n\t\tlet text = response.content\n\t\t\t.filter((c): c is { type: \"text\"; text: string } => c.type === \"text\")\n\t\t\t.map((c) => c.text)\n\t\t\t.join(\"\")\n\t\t\t.trim();\n\n\t\t// Truncate overly long responses\n\t\ttext = truncateResponse(text, MAX_RESPONSE_WORDS);\n\n\t\treturn text || null;\n\t}\n\n\t/**\n\t * Generate a reaction to an event using Ollama.\n\t * Returns null if Ollama is unavailable.\n\t */\n\tasync react(event: string): Promise<string | null> {\n\t\tif (!this.state) return null;\n\n\t\tconst prompt = REACTION_PROMPT.replace(\"{name}\", this.state.name)\n\t\t\t.replace(\"{species}\", this.state.species)\n\t\t\t.replace(\"{personality}\", this.state.personality)\n\t\t\t.replace(\"{backstory}\", this.state.backstory)\n\t\t\t.replace(\"{event}\", event);\n\n\t\tconst context: Context = {\n\t\t\tsystemPrompt: \"Respond with a short in-character quip. Max 20 words.\",\n\t\t\tmessages: [{ role: \"user\", content: prompt, timestamp: Date.now() }],\n\t\t};\n\n\t\treturn this.ollamaChat(context);\n\t}\n\n\t/**\n\t * Respond to the user calling the buddy's name.\n\t * Uses Ollama for the response.\n\t */\n\tasync respondToNameCall(userMessage: string, recentContext: string): Promise<string | null> {\n\t\tif (!this.state) return null;\n\n\t\tconst prompt = NAME_CALL_PROMPT.replace(\"{name}\", this.state.name)\n\t\t\t.replace(\"{species}\", this.state.species)\n\t\t\t.replace(\"{personality}\", this.state.personality)\n\t\t\t.replace(\"{backstory}\", this.state.backstory)\n\t\t\t.replace(\"{message}\", userMessage)\n\t\t\t.replace(\"{context}\", recentContext);\n\n\t\tconst context: Context = {\n\t\t\tsystemPrompt: \"Respond with a short friendly greeting. Max 30 words.\",\n\t\t\tmessages: [{ role: \"user\", content: prompt, timestamp: Date.now() }],\n\t\t};\n\n\t\treturn this.ollamaChat(context);\n\t}\n\n\t/** Get the configured Ollama model name, or null if not set */\n\tgetOllamaModel(): string | null {\n\t\treturn this.state?.ollamaModel ?? loadStored()?.ollamaModel ?? null;\n\t}\n\n\t/** Set the Ollama model for buddy reactions. Persists to disk. */\n\tsetOllamaModel(modelName: string): void {\n\t\tconst stored = loadStored();\n\t\tif (stored) {\n\t\t\tstored.ollamaModel = modelName;\n\t\t\tsaveStored(stored);\n\t\t}\n\t\tif (this.state) {\n\t\t\tthis.state.ollamaModel = modelName;\n\t\t}\n\t\t// Invalidate Ollama status cache so next call picks up the new model\n\t\tthis.ollamaStatus = null;\n\t}\n\n\t/** Reset Ollama status cache (e.g. after detecting it became available) */\n\tresetOllamaCache(): void {\n\t\tthis.ollamaStatus = null;\n\t}\n\n\t/** Update the hidden flag in persisted storage */\n\tsetHidden(hidden: boolean): void {\n\t\tconst stored = loadStored();\n\t\tif (stored) {\n\t\t\tstored.hidden = hidden;\n\t\t\tsaveStored(stored);\n\t\t}\n\t\t// Keep in-memory state in sync so reset() reads current hidden flag\n\t\tif (this.state) {\n\t\t\tthis.state.hidden = hidden;\n\t\t}\n\t}\n}\n"]}
|
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Mulberry32 PRNG and FNV-1a hash for deterministic buddy generation.
|
|
3
|
+
* Bones are derived from hash(username + hostname + salt + rerollCount) on every session start.
|
|
4
|
+
*/
|
|
5
|
+
/** FNV-1a 32-bit hash of a string */
|
|
6
|
+
export declare function fnv1a(str: string): number;
|
|
7
|
+
/** Mulberry32 PRNG — fast, deterministic 32-bit */
|
|
8
|
+
export declare function mulberry32(seed: number): () => number;
|
|
9
|
+
/**
|
|
10
|
+
* Create a PRNG seeded from user identity + salt + reroll count.
|
|
11
|
+
* This ensures the same user gets the same buddy unless they reroll.
|
|
12
|
+
*/
|
|
13
|
+
export declare function createBuddySeed(username: string, hostname: string, salt: string, rerollCount: number): number;
|
|
14
|
+
/**
|
|
15
|
+
* Create a PRNG from user identity. Returns a function that produces
|
|
16
|
+
* floats in [0, 1).
|
|
17
|
+
*/
|
|
18
|
+
export declare function createBuddyRng(username: string, hostname: string, salt: string, rerollCount: number): () => number;
|
|
19
|
+
/** Roll an integer in [min, max] inclusive using the given RNG */
|
|
20
|
+
export declare function rollInt(rng: () => number, min: number, max: number): number;
|
|
21
|
+
/** Roll a float in [min, max) using the given RNG */
|
|
22
|
+
export declare function rollFloat(rng: () => number, min: number, max: number): number;
|
|
23
|
+
/**
|
|
24
|
+
* Pick a weighted item from a map of item -> weight.
|
|
25
|
+
* Returns the key whose cumulative weight contains the roll.
|
|
26
|
+
*/
|
|
27
|
+
export declare function rollWeighted<T extends string>(rng: () => number, weights: Record<T, number>): T;
|
|
28
|
+
//# sourceMappingURL=buddy-prng.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"buddy-prng.d.ts","sourceRoot":"","sources":["../../../src/core/buddy/buddy-prng.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAEH,qCAAqC;AACrC,wBAAgB,KAAK,CAAC,GAAG,EAAE,MAAM,GAAG,MAAM,CAOzC;AAED,qDAAmD;AACnD,wBAAgB,UAAU,CAAC,IAAI,EAAE,MAAM,GAAG,MAAM,MAAM,CASrD;AAED;;;GAGG;AACH,wBAAgB,eAAe,CAAC,QAAQ,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,EAAE,WAAW,EAAE,MAAM,GAAG,MAAM,CAG7G;AAED;;;GAGG;AACH,wBAAgB,cAAc,CAAC,QAAQ,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,EAAE,WAAW,EAAE,MAAM,GAAG,MAAM,MAAM,CAElH;AAED,kEAAkE;AAClE,wBAAgB,OAAO,CAAC,GAAG,EAAE,MAAM,MAAM,EAAE,GAAG,EAAE,MAAM,EAAE,GAAG,EAAE,MAAM,GAAG,MAAM,CAE3E;AAED,qDAAqD;AACrD,wBAAgB,SAAS,CAAC,GAAG,EAAE,MAAM,MAAM,EAAE,GAAG,EAAE,MAAM,EAAE,GAAG,EAAE,MAAM,GAAG,MAAM,CAE7E;AAED;;;GAGG;AACH,wBAAgB,YAAY,CAAC,CAAC,SAAS,MAAM,EAAE,GAAG,EAAE,MAAM,MAAM,EAAE,OAAO,EAAE,MAAM,CAAC,CAAC,EAAE,MAAM,CAAC,GAAG,CAAC,CAW/F","sourcesContent":["/**\n * Mulberry32 PRNG and FNV-1a hash for deterministic buddy generation.\n * Bones are derived from hash(username + hostname + salt + rerollCount) on every session start.\n */\n\n/** FNV-1a 32-bit hash of a string */\nexport function fnv1a(str: string): number {\n\tlet hash = 0x811c9dc5;\n\tfor (let i = 0; i < str.length; i++) {\n\t\thash ^= str.charCodeAt(i);\n\t\thash = Math.imul(hash, 0x01000193);\n\t}\n\treturn hash >>> 0;\n}\n\n/** Mulberry32 PRNG — fast, deterministic 32-bit */\nexport function mulberry32(seed: number): () => number {\n\tlet state = seed;\n\treturn () => {\n\t\tstate |= 0;\n\t\tstate = (state + 0x6d2b79f5) | 0;\n\t\tlet t = Math.imul(state ^ (state >>> 15), 1 | state);\n\t\tt = (t + Math.imul(t ^ (t >>> 7), 61 | t)) ^ t;\n\t\treturn ((t ^ (t >>> 14)) >>> 0) / 4294967296;\n\t};\n}\n\n/**\n * Create a PRNG seeded from user identity + salt + reroll count.\n * This ensures the same user gets the same buddy unless they reroll.\n */\nexport function createBuddySeed(username: string, hostname: string, salt: string, rerollCount: number): number {\n\tconst input = `${username}@${hostname}:${salt}:${rerollCount}`;\n\treturn fnv1a(input);\n}\n\n/**\n * Create a PRNG from user identity. Returns a function that produces\n * floats in [0, 1).\n */\nexport function createBuddyRng(username: string, hostname: string, salt: string, rerollCount: number): () => number {\n\treturn mulberry32(createBuddySeed(username, hostname, salt, rerollCount));\n}\n\n/** Roll an integer in [min, max] inclusive using the given RNG */\nexport function rollInt(rng: () => number, min: number, max: number): number {\n\treturn Math.floor(rng() * (max - min + 1)) + min;\n}\n\n/** Roll a float in [min, max) using the given RNG */\nexport function rollFloat(rng: () => number, min: number, max: number): number {\n\treturn rng() * (max - min) + min;\n}\n\n/**\n * Pick a weighted item from a map of item -> weight.\n * Returns the key whose cumulative weight contains the roll.\n */\nexport function rollWeighted<T extends string>(rng: () => number, weights: Record<T, number>): T {\n\tconst entries = Object.entries(weights) as [T, number][];\n\tconst total = entries.reduce((a, b) => a + b[1], 0);\n\tlet roll = rng() * total;\n\tfor (const [key, weight] of entries) {\n\t\troll -= weight;\n\t\tif (roll <= 0) return key;\n\t}\n\t// Fallback to last key (floating point edge case)\n\tconst keys = Object.keys(weights) as T[];\n\treturn keys[keys.length - 1];\n}\n"]}
|
|
@@ -0,0 +1,65 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Mulberry32 PRNG and FNV-1a hash for deterministic buddy generation.
|
|
3
|
+
* Bones are derived from hash(username + hostname + salt + rerollCount) on every session start.
|
|
4
|
+
*/
|
|
5
|
+
/** FNV-1a 32-bit hash of a string */
|
|
6
|
+
export function fnv1a(str) {
|
|
7
|
+
let hash = 0x811c9dc5;
|
|
8
|
+
for (let i = 0; i < str.length; i++) {
|
|
9
|
+
hash ^= str.charCodeAt(i);
|
|
10
|
+
hash = Math.imul(hash, 0x01000193);
|
|
11
|
+
}
|
|
12
|
+
return hash >>> 0;
|
|
13
|
+
}
|
|
14
|
+
/** Mulberry32 PRNG — fast, deterministic 32-bit */
|
|
15
|
+
export function mulberry32(seed) {
|
|
16
|
+
let state = seed;
|
|
17
|
+
return () => {
|
|
18
|
+
state |= 0;
|
|
19
|
+
state = (state + 0x6d2b79f5) | 0;
|
|
20
|
+
let t = Math.imul(state ^ (state >>> 15), 1 | state);
|
|
21
|
+
t = (t + Math.imul(t ^ (t >>> 7), 61 | t)) ^ t;
|
|
22
|
+
return ((t ^ (t >>> 14)) >>> 0) / 4294967296;
|
|
23
|
+
};
|
|
24
|
+
}
|
|
25
|
+
/**
|
|
26
|
+
* Create a PRNG seeded from user identity + salt + reroll count.
|
|
27
|
+
* This ensures the same user gets the same buddy unless they reroll.
|
|
28
|
+
*/
|
|
29
|
+
export function createBuddySeed(username, hostname, salt, rerollCount) {
|
|
30
|
+
const input = `${username}@${hostname}:${salt}:${rerollCount}`;
|
|
31
|
+
return fnv1a(input);
|
|
32
|
+
}
|
|
33
|
+
/**
|
|
34
|
+
* Create a PRNG from user identity. Returns a function that produces
|
|
35
|
+
* floats in [0, 1).
|
|
36
|
+
*/
|
|
37
|
+
export function createBuddyRng(username, hostname, salt, rerollCount) {
|
|
38
|
+
return mulberry32(createBuddySeed(username, hostname, salt, rerollCount));
|
|
39
|
+
}
|
|
40
|
+
/** Roll an integer in [min, max] inclusive using the given RNG */
|
|
41
|
+
export function rollInt(rng, min, max) {
|
|
42
|
+
return Math.floor(rng() * (max - min + 1)) + min;
|
|
43
|
+
}
|
|
44
|
+
/** Roll a float in [min, max) using the given RNG */
|
|
45
|
+
export function rollFloat(rng, min, max) {
|
|
46
|
+
return rng() * (max - min) + min;
|
|
47
|
+
}
|
|
48
|
+
/**
|
|
49
|
+
* Pick a weighted item from a map of item -> weight.
|
|
50
|
+
* Returns the key whose cumulative weight contains the roll.
|
|
51
|
+
*/
|
|
52
|
+
export function rollWeighted(rng, weights) {
|
|
53
|
+
const entries = Object.entries(weights);
|
|
54
|
+
const total = entries.reduce((a, b) => a + b[1], 0);
|
|
55
|
+
let roll = rng() * total;
|
|
56
|
+
for (const [key, weight] of entries) {
|
|
57
|
+
roll -= weight;
|
|
58
|
+
if (roll <= 0)
|
|
59
|
+
return key;
|
|
60
|
+
}
|
|
61
|
+
// Fallback to last key (floating point edge case)
|
|
62
|
+
const keys = Object.keys(weights);
|
|
63
|
+
return keys[keys.length - 1];
|
|
64
|
+
}
|
|
65
|
+
//# sourceMappingURL=buddy-prng.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"buddy-prng.js","sourceRoot":"","sources":["../../../src/core/buddy/buddy-prng.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Mulberry32 PRNG and FNV-1a hash for deterministic buddy generation.\n * Bones are derived from hash(username + hostname + salt + rerollCount) on every session start.\n */\n\n/** FNV-1a 32-bit hash of a string */\nexport function fnv1a(str: string): number {\n\tlet hash = 0x811c9dc5;\n\tfor (let i = 0; i < str.length; i++) {\n\t\thash ^= str.charCodeAt(i);\n\t\thash = Math.imul(hash, 0x01000193);\n\t}\n\treturn hash >>> 0;\n}\n\n/** Mulberry32 PRNG — fast, deterministic 32-bit */\nexport function mulberry32(seed: number): () => number {\n\tlet state = seed;\n\treturn () => {\n\t\tstate |= 0;\n\t\tstate = (state + 0x6d2b79f5) | 0;\n\t\tlet t = Math.imul(state ^ (state >>> 15), 1 | state);\n\t\tt = (t + Math.imul(t ^ (t >>> 7), 61 | t)) ^ t;\n\t\treturn ((t ^ (t >>> 14)) >>> 0) / 4294967296;\n\t};\n}\n\n/**\n * Create a PRNG seeded from user identity + salt + reroll count.\n * This ensures the same user gets the same buddy unless they reroll.\n */\nexport function createBuddySeed(username: string, hostname: string, salt: string, rerollCount: number): number {\n\tconst input = `${username}@${hostname}:${salt}:${rerollCount}`;\n\treturn fnv1a(input);\n}\n\n/**\n * Create a PRNG from user identity. Returns a function that produces\n * floats in [0, 1).\n */\nexport function createBuddyRng(username: string, hostname: string, salt: string, rerollCount: number): () => number {\n\treturn mulberry32(createBuddySeed(username, hostname, salt, rerollCount));\n}\n\n/** Roll an integer in [min, max] inclusive using the given RNG */\nexport function rollInt(rng: () => number, min: number, max: number): number {\n\treturn Math.floor(rng() * (max - min + 1)) + min;\n}\n\n/** Roll a float in [min, max) using the given RNG */\nexport function rollFloat(rng: () => number, min: number, max: number): number {\n\treturn rng() * (max - min) + min;\n}\n\n/**\n * Pick a weighted item from a map of item -> weight.\n * Returns the key whose cumulative weight contains the roll.\n */\nexport function rollWeighted<T extends string>(rng: () => number, weights: Record<T, number>): T {\n\tconst entries = Object.entries(weights) as [T, number][];\n\tconst total = entries.reduce((a, b) => a + b[1], 0);\n\tlet roll = rng() * total;\n\tfor (const [key, weight] of entries) {\n\t\troll -= weight;\n\t\tif (roll <= 0) return key;\n\t}\n\t// Fallback to last key (floating point edge case)\n\tconst keys = Object.keys(weights) as T[];\n\treturn keys[keys.length - 1];\n}\n"]}
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Species pool, eye styles, hats, and ASCII art frames for the buddy companion.
|
|
3
|
+
*
|
|
4
|
+
* Each species has 3 animation frames using {E} as eye placeholder.
|
|
5
|
+
* Frames cycle at 500ms for idle animation.
|
|
6
|
+
*/
|
|
7
|
+
import { Rarity, type SpeciesName, type Stat } from "./buddy-types.js";
|
|
8
|
+
export interface SpeciesDef {
|
|
9
|
+
name: SpeciesName;
|
|
10
|
+
frames: string[][];
|
|
11
|
+
rarityFloor: Rarity;
|
|
12
|
+
}
|
|
13
|
+
export declare const ALL_SPECIES: SpeciesName[];
|
|
14
|
+
/**
|
|
15
|
+
* Roll a species from the species pool based on rarity.
|
|
16
|
+
* Species can only appear if the rolled rarity meets their floor.
|
|
17
|
+
*/
|
|
18
|
+
export declare function rollSpecies(rng: () => number): {
|
|
19
|
+
species: SpeciesName;
|
|
20
|
+
rarity: Rarity;
|
|
21
|
+
};
|
|
22
|
+
/** Roll an eye style */
|
|
23
|
+
export declare function rollEyes(rng: () => number): string;
|
|
24
|
+
/** Roll a hat */
|
|
25
|
+
export declare function rollHat(rng: () => number): string;
|
|
26
|
+
/**
|
|
27
|
+
* Roll RPG-style stats: one peak, one dump, three random.
|
|
28
|
+
* Peak can hit 100, dump near floor.
|
|
29
|
+
*/
|
|
30
|
+
export declare function rollStats(rng: () => number, rarity: Rarity): Record<Stat, number>;
|
|
31
|
+
/** Get the 3 animation frames for a species */
|
|
32
|
+
export declare function getSpeciesFrames(species: SpeciesName): string[][];
|
|
33
|
+
/** Replace {E} placeholder with actual eye character */
|
|
34
|
+
export declare function applyEyes(frame: string[], eyes: string): string[];
|
|
35
|
+
/** Get the maximum width of a species' frames */
|
|
36
|
+
export declare function getSpeciesWidth(species: SpeciesName): number;
|
|
37
|
+
//# sourceMappingURL=buddy-species.d.ts.map
|