@pellux/goodvibes-tui 0.18.4
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/.goodvibes/GOODVIBES.md +35 -0
- package/.goodvibes/agents/reviewer.md +89 -0
- package/.goodvibes/skills/add-provider/SKILL.md +199 -0
- package/CHANGELOG.md +1681 -0
- package/README.md +1607 -0
- package/bin/goodvibes +66 -0
- package/docs/README.md +32 -0
- package/docs/foundation-artifacts/README.md +16 -0
- package/docs/foundation-artifacts/knowledge-graphql.graphql +397 -0
- package/docs/foundation-artifacts/knowledge-store.sql +183 -0
- package/docs/foundation-artifacts/operator-contract.json +55157 -0
- package/docs/foundation-artifacts/peer-contract.json +2384 -0
- package/package.json +114 -0
- package/scripts/postinstall.mjs +203 -0
- package/src/acp/connection.ts +447 -0
- package/src/acp/index.ts +7 -0
- package/src/acp/manager.ts +133 -0
- package/src/adapters/bluebubbles/index.ts +127 -0
- package/src/adapters/discord/index.ts +297 -0
- package/src/adapters/github/index.ts +73 -0
- package/src/adapters/google-chat/index.ts +119 -0
- package/src/adapters/imessage/index.ts +92 -0
- package/src/adapters/index.ts +15 -0
- package/src/adapters/matrix/index.ts +116 -0
- package/src/adapters/mattermost/index.ts +151 -0
- package/src/adapters/msteams/index.ts +180 -0
- package/src/adapters/ntfy/index.ts +118 -0
- package/src/adapters/signal/index.ts +92 -0
- package/src/adapters/slack/index.ts +323 -0
- package/src/adapters/telegram/index.ts +160 -0
- package/src/adapters/types.ts +97 -0
- package/src/adapters/webhook/index.ts +178 -0
- package/src/adapters/whatsapp/index.ts +135 -0
- package/src/agents/message-bus-core.ts +312 -0
- package/src/agents/message-bus.ts +2 -0
- package/src/agents/orchestrator-prompts.ts +351 -0
- package/src/agents/orchestrator-runner.ts +668 -0
- package/src/agents/orchestrator.ts +437 -0
- package/src/agents/session.ts +108 -0
- package/src/agents/worktree.ts +153 -0
- package/src/agents/wrfc-config.ts +47 -0
- package/src/agents/wrfc-controller.ts +747 -0
- package/src/agents/wrfc-gate-runtime.ts +75 -0
- package/src/agents/wrfc-reporting.ts +284 -0
- package/src/agents/wrfc-runtime-events.ts +150 -0
- package/src/agents/wrfc-types.ts +67 -0
- package/src/automation/delivery-manager.ts +368 -0
- package/src/automation/index.ts +72 -0
- package/src/automation/manager-runtime-delivery.ts +139 -0
- package/src/automation/manager-runtime-events.ts +131 -0
- package/src/automation/manager-runtime-execution.ts +511 -0
- package/src/automation/manager-runtime-helpers.ts +433 -0
- package/src/automation/manager-runtime-job-mutations.ts +175 -0
- package/src/automation/manager-runtime-reconcile.ts +148 -0
- package/src/automation/manager-runtime-scheduling.ts +189 -0
- package/src/automation/manager-runtime-sync.ts +54 -0
- package/src/automation/manager-runtime.ts +721 -0
- package/src/automation/manager.ts +10 -0
- package/src/automation/service.ts +242 -0
- package/src/channels/builtin/account-actions.ts +490 -0
- package/src/channels/builtin/accounts.ts +433 -0
- package/src/channels/builtin/contracts.ts +405 -0
- package/src/channels/builtin/plugins.ts +308 -0
- package/src/channels/builtin/rendering.ts +174 -0
- package/src/channels/builtin/setup-schema.ts +504 -0
- package/src/channels/builtin/shared.ts +96 -0
- package/src/channels/builtin/surfaces.ts +57 -0
- package/src/channels/builtin/targets.ts +693 -0
- package/src/channels/builtin-runtime.ts +443 -0
- package/src/channels/delivery/shared.ts +199 -0
- package/src/channels/delivery/strategies-bridge.ts +246 -0
- package/src/channels/delivery/strategies-core.ts +299 -0
- package/src/channels/delivery/strategies-enterprise.ts +178 -0
- package/src/channels/delivery/types.ts +59 -0
- package/src/channels/delivery-router.ts +127 -0
- package/src/channels/index.ts +77 -0
- package/src/channels/plugin-registry.ts +551 -0
- package/src/channels/provider-runtime.ts +330 -0
- package/src/channels/reply-pipeline.ts +522 -0
- package/src/channels/route-manager.ts +340 -0
- package/src/channels/surface-registry.ts +186 -0
- package/src/config/helper-model.ts +233 -0
- package/src/config/index.ts +193 -0
- package/src/config/manager.ts +404 -0
- package/src/config/secrets.ts +547 -0
- package/src/config/service-registry.ts +329 -0
- package/src/config/subscription-auth.ts +31 -0
- package/src/config/subscription-providers.ts +127 -0
- package/src/config/tool-llm.ts +110 -0
- package/src/control-plane/approval-broker.ts +351 -0
- package/src/control-plane/gateway.ts +713 -0
- package/src/control-plane/index.ts +54 -0
- package/src/control-plane/media-contract-schemas.ts +208 -0
- package/src/control-plane/method-catalog-admin.ts +136 -0
- package/src/control-plane/method-catalog-channels.ts +591 -0
- package/src/control-plane/method-catalog-control-automation.ts +475 -0
- package/src/control-plane/method-catalog-control-core.ts +594 -0
- package/src/control-plane/method-catalog-control.ts +8 -0
- package/src/control-plane/method-catalog-events.ts +74 -0
- package/src/control-plane/method-catalog-knowledge.ts +531 -0
- package/src/control-plane/method-catalog-media.ts +279 -0
- package/src/control-plane/method-catalog-runtime.ts +304 -0
- package/src/control-plane/method-catalog-shared.ts +223 -0
- package/src/control-plane/method-catalog.ts +242 -0
- package/src/control-plane/operator-contract-schemas-admin.ts +639 -0
- package/src/control-plane/operator-contract-schemas-channels.ts +375 -0
- package/src/control-plane/operator-contract-schemas-control.ts +226 -0
- package/src/control-plane/operator-contract-schemas-domains.ts +4 -0
- package/src/control-plane/operator-contract-schemas-knowledge.ts +582 -0
- package/src/control-plane/operator-contract-schemas-media.ts +297 -0
- package/src/control-plane/operator-contract-schemas-permissions.ts +100 -0
- package/src/control-plane/operator-contract-schemas-remote.ts +38 -0
- package/src/control-plane/operator-contract-schemas-runtime.ts +563 -0
- package/src/control-plane/operator-contract-schemas-shared.ts +85 -0
- package/src/control-plane/operator-contract-schemas-telemetry.ts +349 -0
- package/src/control-plane/operator-contract-schemas.ts +6 -0
- package/src/control-plane/operator-contract.ts +163 -0
- package/src/control-plane/session-broker.ts +780 -0
- package/src/core/compaction-sections.ts +492 -0
- package/src/core/compaction-types.ts +147 -0
- package/src/core/composer-state.ts +59 -0
- package/src/core/context-compaction.ts +542 -0
- package/src/core/conversation-compaction.ts +68 -0
- package/src/core/conversation-diff.ts +55 -0
- package/src/core/conversation-rendering.ts +343 -0
- package/src/core/conversation-utils.ts +72 -0
- package/src/core/conversation.ts +775 -0
- package/src/core/event-replay.ts +287 -0
- package/src/core/orchestrator-context-runtime.ts +407 -0
- package/src/core/orchestrator-follow-up-runtime.ts +134 -0
- package/src/core/orchestrator-runtime.ts +132 -0
- package/src/core/orchestrator-tool-runtime.ts +468 -0
- package/src/core/orchestrator-turn-helpers.ts +355 -0
- package/src/core/orchestrator-turn-loop.ts +443 -0
- package/src/core/orchestrator.ts +733 -0
- package/src/core/plan-command-handler.ts +169 -0
- package/src/core/system-message-router.ts +210 -0
- package/src/core/transcript-events/classify.ts +95 -0
- package/src/core/transcript-events/index.ts +15 -0
- package/src/daemon/cli.ts +88 -0
- package/src/daemon/control-plane.ts +522 -0
- package/src/daemon/facade-composition.ts +397 -0
- package/src/daemon/facade.ts +638 -0
- package/src/daemon/helpers.ts +74 -0
- package/src/daemon/http/router-route-contexts.ts +370 -0
- package/src/daemon/http/router.ts +531 -0
- package/src/daemon/http-listener.ts +301 -0
- package/src/daemon/index.ts +3 -0
- package/src/daemon/server.ts +1 -0
- package/src/daemon/service-manager.ts +413 -0
- package/src/daemon/surface-actions.ts +183 -0
- package/src/daemon/surface-delivery.ts +530 -0
- package/src/daemon/surface-policy.ts +60 -0
- package/src/daemon/transport-events.ts +110 -0
- package/src/daemon/types.ts +191 -0
- package/src/export/markdown.ts +213 -0
- package/src/export/session-export.ts +633 -0
- package/src/git/index.ts +1 -0
- package/src/git/service.ts +414 -0
- package/src/hooks/chain-engine.ts +414 -0
- package/src/hooks/dispatcher.ts +414 -0
- package/src/hooks/hook-api.ts +170 -0
- package/src/hooks/index.ts +48 -0
- package/src/hooks/runners/agent.ts +93 -0
- package/src/hooks/runners/prompt.ts +69 -0
- package/src/hooks/workbench.ts +360 -0
- package/src/input/autocomplete.ts +96 -0
- package/src/input/bookmark-modal.ts +115 -0
- package/src/input/command-registry.ts +300 -0
- package/src/input/commands/branch-runtime.ts +72 -0
- package/src/input/commands/config.ts +515 -0
- package/src/input/commands/control-room-runtime.ts +255 -0
- package/src/input/commands/conversation-runtime.ts +207 -0
- package/src/input/commands/diff-runtime.ts +161 -0
- package/src/input/commands/discovery-runtime.ts +45 -0
- package/src/input/commands/eval.ts +204 -0
- package/src/input/commands/experience-runtime.ts +278 -0
- package/src/input/commands/git-runtime.ts +81 -0
- package/src/input/commands/guidance-runtime.ts +101 -0
- package/src/input/commands/health-runtime.ts +434 -0
- package/src/input/commands/hooks-runtime.ts +148 -0
- package/src/input/commands/incident-runtime.ts +95 -0
- package/src/input/commands/integration-runtime.ts +394 -0
- package/src/input/commands/intelligence-runtime.ts +223 -0
- package/src/input/commands/knowledge.ts +368 -0
- package/src/input/commands/local-auth-runtime.ts +105 -0
- package/src/input/commands/local-provider-runtime.ts +170 -0
- package/src/input/commands/local-runtime.ts +458 -0
- package/src/input/commands/local-setup-review.ts +192 -0
- package/src/input/commands/local-setup-transfer.ts +134 -0
- package/src/input/commands/local-setup.ts +292 -0
- package/src/input/commands/managed-runtime.ts +208 -0
- package/src/input/commands/marketplace-runtime.ts +290 -0
- package/src/input/commands/mcp-runtime.ts +202 -0
- package/src/input/commands/memory-product-runtime.ts +111 -0
- package/src/input/commands/memory.ts +151 -0
- package/src/input/commands/notify-runtime.ts +83 -0
- package/src/input/commands/operator-panel-runtime.ts +141 -0
- package/src/input/commands/operator-runtime.ts +392 -0
- package/src/input/commands/permissions-runtime.ts +104 -0
- package/src/input/commands/planning-runtime.ts +97 -0
- package/src/input/commands/platform-access-runtime.ts +422 -0
- package/src/input/commands/platform-runtime.ts +6 -0
- package/src/input/commands/platform-sandbox-qemu.ts +137 -0
- package/src/input/commands/platform-sandbox-runtime.ts +406 -0
- package/src/input/commands/platform-sandbox-session.ts +128 -0
- package/src/input/commands/platform-services-runtime.ts +246 -0
- package/src/input/commands/policy-dispatch.ts +339 -0
- package/src/input/commands/policy.ts +17 -0
- package/src/input/commands/product-runtime.ts +351 -0
- package/src/input/commands/profile-sync-runtime.ts +99 -0
- package/src/input/commands/provider-accounts-runtime.ts +113 -0
- package/src/input/commands/provider.ts +363 -0
- package/src/input/commands/quit-shared.ts +162 -0
- package/src/input/commands/recall-bundle.ts +132 -0
- package/src/input/commands/recall-capture.ts +152 -0
- package/src/input/commands/recall-query.ts +229 -0
- package/src/input/commands/recall-review.ts +98 -0
- package/src/input/commands/recall-shared.ts +22 -0
- package/src/input/commands/remote-runtime-pool.ts +106 -0
- package/src/input/commands/remote-runtime-setup.ts +199 -0
- package/src/input/commands/remote-runtime.ts +531 -0
- package/src/input/commands/replay-runtime.ts +18 -0
- package/src/input/commands/runtime-services.ts +279 -0
- package/src/input/commands/schedule-runtime.ts +332 -0
- package/src/input/commands/services-runtime.ts +207 -0
- package/src/input/commands/session-content.ts +408 -0
- package/src/input/commands/session-workflow.ts +464 -0
- package/src/input/commands/session.ts +376 -0
- package/src/input/commands/settings-sync-runtime.ts +173 -0
- package/src/input/commands/share-runtime.ts +114 -0
- package/src/input/commands/shell-core.ts +320 -0
- package/src/input/commands/skills-runtime.ts +221 -0
- package/src/input/commands/subscription-runtime.ts +434 -0
- package/src/input/commands/tasks-runtime.ts +230 -0
- package/src/input/commands/teamwork-runtime.ts +374 -0
- package/src/input/commands/teleport-runtime.ts +57 -0
- package/src/input/commands/worktree-runtime.ts +137 -0
- package/src/input/commands.ts +127 -0
- package/src/input/file-picker.ts +192 -0
- package/src/input/handler-command-route.ts +106 -0
- package/src/input/handler-content-actions.ts +465 -0
- package/src/input/handler-feed-routes.ts +541 -0
- package/src/input/handler-feed.ts +361 -0
- package/src/input/handler-modal-routes.ts +335 -0
- package/src/input/handler-modal-stack.ts +237 -0
- package/src/input/handler-modal-token-routes.ts +272 -0
- package/src/input/handler-picker-routes.ts +416 -0
- package/src/input/handler-prompt-buffer.ts +320 -0
- package/src/input/handler-shortcuts.ts +195 -0
- package/src/input/handler-ui-state.ts +294 -0
- package/src/input/handler.ts +798 -0
- package/src/input/input-history.ts +267 -0
- package/src/input/keybindings.ts +256 -0
- package/src/input/model-picker.ts +730 -0
- package/src/input/panel-integration-actions.ts +77 -0
- package/src/input/profile-picker-modal.ts +222 -0
- package/src/input/search.ts +100 -0
- package/src/input/selection-modal.ts +163 -0
- package/src/input/selection.ts +135 -0
- package/src/input/session-picker-modal.ts +136 -0
- package/src/input/settings-modal.ts +718 -0
- package/src/input/submission-intent.ts +18 -0
- package/src/input/submission-router.ts +64 -0
- package/src/integrations/index.ts +42 -0
- package/src/integrations/notifier.ts +206 -0
- package/src/integrations/webhooks.ts +177 -0
- package/src/knowledge/consolidation.ts +346 -0
- package/src/knowledge/graphql.ts +324 -0
- package/src/knowledge/index.ts +60 -0
- package/src/knowledge/ingest-compile.ts +386 -0
- package/src/knowledge/ingest-context.ts +18 -0
- package/src/knowledge/ingest-inputs.ts +387 -0
- package/src/knowledge/ingest.ts +20 -0
- package/src/knowledge/internal.ts +257 -0
- package/src/knowledge/knowledge-api.ts +432 -0
- package/src/knowledge/lint.ts +121 -0
- package/src/knowledge/memory-sync.ts +62 -0
- package/src/knowledge/packet.ts +370 -0
- package/src/knowledge/scheduling.ts +283 -0
- package/src/knowledge/service.ts +715 -0
- package/src/main.ts +798 -0
- package/src/mcp/client.ts +383 -0
- package/src/mcp/index.ts +12 -0
- package/src/mcp/mcp-api.ts +90 -0
- package/src/mcp/registry.ts +508 -0
- package/src/media/builtin-image-understanding.ts +303 -0
- package/src/media/builtin-providers.ts +26 -0
- package/src/media/index.ts +18 -0
- package/src/multimodal/index.ts +13 -0
- package/src/multimodal/service.ts +492 -0
- package/src/panels/agent-inspector-panel.ts +515 -0
- package/src/panels/agent-inspector-shared.ts +94 -0
- package/src/panels/agent-logs-panel.ts +539 -0
- package/src/panels/agent-logs-shared.ts +129 -0
- package/src/panels/approval-panel.ts +169 -0
- package/src/panels/automation-control-panel.ts +253 -0
- package/src/panels/base-panel.ts +72 -0
- package/src/panels/builtin/agent.ts +88 -0
- package/src/panels/builtin/development.ts +111 -0
- package/src/panels/builtin/knowledge.ts +26 -0
- package/src/panels/builtin/operations.ts +385 -0
- package/src/panels/builtin/session.ts +61 -0
- package/src/panels/builtin/shared.ts +240 -0
- package/src/panels/builtin-panels.ts +23 -0
- package/src/panels/cockpit-panel.ts +183 -0
- package/src/panels/communication-panel.ts +191 -0
- package/src/panels/context-visualizer-panel.ts +199 -0
- package/src/panels/control-plane-panel.ts +266 -0
- package/src/panels/cost-tracker-panel.ts +444 -0
- package/src/panels/debug-panel.ts +432 -0
- package/src/panels/diff-panel.ts +518 -0
- package/src/panels/docs-panel.ts +283 -0
- package/src/panels/eval-panel.ts +399 -0
- package/src/panels/file-explorer-panel.ts +556 -0
- package/src/panels/file-preview-panel.ts +412 -0
- package/src/panels/forensics-panel.ts +364 -0
- package/src/panels/git-panel.ts +630 -0
- package/src/panels/hooks-panel.ts +274 -0
- package/src/panels/incident-review-panel.ts +247 -0
- package/src/panels/index.ts +48 -0
- package/src/panels/intelligence-panel.ts +176 -0
- package/src/panels/knowledge-panel.ts +328 -0
- package/src/panels/local-auth-panel.ts +146 -0
- package/src/panels/marketplace-panel.ts +223 -0
- package/src/panels/mcp-panel.ts +260 -0
- package/src/panels/memory-panel.ts +293 -0
- package/src/panels/ops-control-panel.ts +184 -0
- package/src/panels/ops-strategy-panel.ts +235 -0
- package/src/panels/orchestration-panel.ts +254 -0
- package/src/panels/panel-list-panel.ts +508 -0
- package/src/panels/panel-manager.ts +538 -0
- package/src/panels/panel-picker.ts +106 -0
- package/src/panels/plan-dashboard-panel.ts +272 -0
- package/src/panels/plugins-panel.ts +201 -0
- package/src/panels/policy-panel.ts +308 -0
- package/src/panels/polish.ts +668 -0
- package/src/panels/provider-account-snapshot.ts +259 -0
- package/src/panels/provider-accounts-panel.ts +221 -0
- package/src/panels/provider-health-domains.ts +211 -0
- package/src/panels/provider-health-panel.ts +725 -0
- package/src/panels/provider-health-tracker.ts +115 -0
- package/src/panels/provider-stats-panel.ts +366 -0
- package/src/panels/remote-panel.ts +449 -0
- package/src/panels/routes-panel.ts +228 -0
- package/src/panels/sandbox-panel.ts +289 -0
- package/src/panels/schedule-panel.ts +344 -0
- package/src/panels/search-focus.ts +32 -0
- package/src/panels/security-panel.ts +329 -0
- package/src/panels/services-panel.ts +271 -0
- package/src/panels/session-browser-panel.ts +399 -0
- package/src/panels/session-maintenance.ts +125 -0
- package/src/panels/settings-sync-panel.ts +164 -0
- package/src/panels/skills-panel.ts +475 -0
- package/src/panels/subscription-panel.ts +273 -0
- package/src/panels/symbol-outline-panel.ts +486 -0
- package/src/panels/system-messages-panel.ts +224 -0
- package/src/panels/tasks-panel.ts +448 -0
- package/src/panels/thinking-panel.ts +304 -0
- package/src/panels/token-budget-panel.ts +469 -0
- package/src/panels/tool-inspector-panel.ts +434 -0
- package/src/panels/types.ts +44 -0
- package/src/panels/watchers-panel.ts +241 -0
- package/src/panels/welcome-panel.ts +64 -0
- package/src/panels/worktree-panel.ts +180 -0
- package/src/panels/wrfc-panel.ts +480 -0
- package/src/permissions/briefs/build.ts +88 -0
- package/src/permissions/manager.ts +356 -0
- package/src/permissions/prompt.ts +184 -0
- package/src/plugins/api.ts +383 -0
- package/src/plugins/loader.ts +304 -0
- package/src/plugins/manager.ts +481 -0
- package/src/profiles/shape.ts +58 -0
- package/src/providers/amazon-bedrock-mantle.ts +50 -0
- package/src/providers/amazon-bedrock.ts +61 -0
- package/src/providers/anthropic-compat.ts +373 -0
- package/src/providers/anthropic-sdk-provider.ts +230 -0
- package/src/providers/anthropic-vertex.ts +59 -0
- package/src/providers/anthropic.ts +469 -0
- package/src/providers/auto-register.ts +417 -0
- package/src/providers/builtin-catalog.ts +326 -0
- package/src/providers/builtin-registry.ts +575 -0
- package/src/providers/cache-planner.ts +258 -0
- package/src/providers/capabilities.ts +601 -0
- package/src/providers/custom-loader.ts +425 -0
- package/src/providers/discovered-compat.ts +18 -0
- package/src/providers/discovered-factory.ts +61 -0
- package/src/providers/discovered-traits.ts +138 -0
- package/src/providers/gemini.ts +462 -0
- package/src/providers/github-copilot.ts +254 -0
- package/src/providers/index.ts +47 -0
- package/src/providers/interface.ts +185 -0
- package/src/providers/llama-cpp.ts +402 -0
- package/src/providers/lm-studio-helpers.ts +367 -0
- package/src/providers/lm-studio.ts +484 -0
- package/src/providers/model-catalog-cache.ts +221 -0
- package/src/providers/model-catalog-notifications.ts +97 -0
- package/src/providers/model-catalog-synthetic.ts +202 -0
- package/src/providers/model-catalog.ts +211 -0
- package/src/providers/model-limits.ts +280 -0
- package/src/providers/ollama.ts +469 -0
- package/src/providers/openai-codex.ts +472 -0
- package/src/providers/openai-compat.ts +615 -0
- package/src/providers/openai.ts +231 -0
- package/src/providers/optimizer.ts +381 -0
- package/src/providers/provider-api.ts +553 -0
- package/src/providers/registry-helpers.ts +34 -0
- package/src/providers/registry-models.ts +77 -0
- package/src/providers/registry-types.ts +67 -0
- package/src/providers/registry.ts +729 -0
- package/src/providers/runtime-metadata.ts +149 -0
- package/src/providers/runtime-snapshot.ts +130 -0
- package/src/providers/synthetic.ts +561 -0
- package/src/providers/tier-prompts.ts +84 -0
- package/src/providers/tool-formats.ts +414 -0
- package/src/renderer/agent-detail-modal.ts +285 -0
- package/src/renderer/autocomplete-overlay.ts +154 -0
- package/src/renderer/block-actions.ts +76 -0
- package/src/renderer/bookmark-modal.ts +101 -0
- package/src/renderer/bottom-bar.ts +58 -0
- package/src/renderer/buffer.ts +34 -0
- package/src/renderer/code-block.ts +373 -0
- package/src/renderer/compositor.ts +261 -0
- package/src/renderer/context-inspector.ts +219 -0
- package/src/renderer/conversation-layout.ts +67 -0
- package/src/renderer/conversation-overlays.ts +123 -0
- package/src/renderer/conversation-surface.ts +260 -0
- package/src/renderer/diff-view.ts +132 -0
- package/src/renderer/diff.ts +122 -0
- package/src/renderer/file-picker-overlay.ts +101 -0
- package/src/renderer/file-tree.ts +153 -0
- package/src/renderer/git-status.ts +89 -0
- package/src/renderer/help-overlay.ts +247 -0
- package/src/renderer/history-search-overlay.ts +73 -0
- package/src/renderer/layout-engine.ts +97 -0
- package/src/renderer/layout.ts +32 -0
- package/src/renderer/live-tail-modal.ts +156 -0
- package/src/renderer/markdown.ts +777 -0
- package/src/renderer/modal-factory.ts +467 -0
- package/src/renderer/modal-utils.ts +24 -0
- package/src/renderer/model-picker-overlay.ts +396 -0
- package/src/renderer/overlay-box.ts +165 -0
- package/src/renderer/overlay-viewport.ts +104 -0
- package/src/renderer/panel-composite.ts +80 -0
- package/src/renderer/panel-picker-overlay.ts +202 -0
- package/src/renderer/panel-tab-bar.ts +69 -0
- package/src/renderer/panel-workspace-bar.ts +38 -0
- package/src/renderer/process-indicator.ts +96 -0
- package/src/renderer/process-modal.ts +295 -0
- package/src/renderer/profile-picker-modal.ts +129 -0
- package/src/renderer/progress.ts +98 -0
- package/src/renderer/search-overlay.ts +54 -0
- package/src/renderer/selection-modal-overlay.ts +214 -0
- package/src/renderer/semantic-diff.ts +369 -0
- package/src/renderer/session-picker-modal.ts +127 -0
- package/src/renderer/settings-modal.ts +701 -0
- package/src/renderer/shell-surface.ts +88 -0
- package/src/renderer/surface-layout.ts +101 -0
- package/src/renderer/syntax-highlighter.ts +542 -0
- package/src/renderer/system-message.ts +83 -0
- package/src/renderer/tab-strip.ts +108 -0
- package/src/renderer/text-layout.ts +31 -0
- package/src/renderer/thinking.ts +17 -0
- package/src/renderer/tool-call.ts +233 -0
- package/src/renderer/ui-factory.ts +524 -0
- package/src/renderer/ui-primitives.ts +96 -0
- package/src/runtime/auth/inspection.ts +125 -0
- package/src/runtime/bootstrap-background.ts +147 -0
- package/src/runtime/bootstrap-command-context.ts +265 -0
- package/src/runtime/bootstrap-command-parts.ts +357 -0
- package/src/runtime/bootstrap-core.ts +375 -0
- package/src/runtime/bootstrap-helpers.ts +88 -0
- package/src/runtime/bootstrap-hook-bridge.ts +271 -0
- package/src/runtime/bootstrap-runtime-events.ts +254 -0
- package/src/runtime/bootstrap-services.ts +197 -0
- package/src/runtime/bootstrap-shell.ts +262 -0
- package/src/runtime/bootstrap.ts +488 -0
- package/src/runtime/compaction/index.ts +90 -0
- package/src/runtime/compaction/lifecycle.ts +167 -0
- package/src/runtime/compaction/manager.ts +474 -0
- package/src/runtime/compaction/quality-score.ts +279 -0
- package/src/runtime/compaction/resume-repair.ts +183 -0
- package/src/runtime/compaction/strategies/autocompact.ts +65 -0
- package/src/runtime/compaction/strategies/boundary-commit.ts +106 -0
- package/src/runtime/compaction/strategies/collapse.ts +90 -0
- package/src/runtime/compaction/strategies/index.ts +23 -0
- package/src/runtime/compaction/strategies/microcompact.ts +74 -0
- package/src/runtime/compaction/strategies/reactive.ts +89 -0
- package/src/runtime/compaction/types.ts +221 -0
- package/src/runtime/context.ts +158 -0
- package/src/runtime/diagnostics/actions.ts +776 -0
- package/src/runtime/diagnostics/index.ts +99 -0
- package/src/runtime/diagnostics/panels/agents.ts +252 -0
- package/src/runtime/diagnostics/panels/events.ts +188 -0
- package/src/runtime/diagnostics/panels/health.ts +242 -0
- package/src/runtime/diagnostics/panels/index.ts +24 -0
- package/src/runtime/diagnostics/panels/ops.ts +156 -0
- package/src/runtime/diagnostics/panels/policy.ts +176 -0
- package/src/runtime/diagnostics/panels/tasks.ts +251 -0
- package/src/runtime/diagnostics/panels/tool-calls.ts +267 -0
- package/src/runtime/diagnostics/provider.ts +262 -0
- package/src/runtime/ecosystem/catalog.ts +606 -0
- package/src/runtime/ecosystem/recommendations.ts +117 -0
- package/src/runtime/emitters/agents.ts +96 -0
- package/src/runtime/emitters/automation.ts +112 -0
- package/src/runtime/emitters/communication.ts +53 -0
- package/src/runtime/emitters/compaction.ts +161 -0
- package/src/runtime/emitters/control-plane.ts +65 -0
- package/src/runtime/emitters/deliveries.ts +65 -0
- package/src/runtime/emitters/forensics.ts +17 -0
- package/src/runtime/emitters/index.ts +59 -0
- package/src/runtime/emitters/knowledge.ts +129 -0
- package/src/runtime/emitters/mcp.ts +95 -0
- package/src/runtime/emitters/ops.ts +163 -0
- package/src/runtime/emitters/orchestration.ts +87 -0
- package/src/runtime/emitters/permissions.ts +98 -0
- package/src/runtime/emitters/planner.ts +23 -0
- package/src/runtime/emitters/plugins.ts +78 -0
- package/src/runtime/emitters/providers.ts +30 -0
- package/src/runtime/emitters/routes.ts +57 -0
- package/src/runtime/emitters/security.ts +53 -0
- package/src/runtime/emitters/session.ts +93 -0
- package/src/runtime/emitters/surfaces.ts +57 -0
- package/src/runtime/emitters/tasks.ts +69 -0
- package/src/runtime/emitters/tools.ts +140 -0
- package/src/runtime/emitters/transport.ts +78 -0
- package/src/runtime/emitters/turn.ts +155 -0
- package/src/runtime/emitters/ui.ts +57 -0
- package/src/runtime/emitters/watchers.ts +57 -0
- package/src/runtime/emitters/workflows.ts +79 -0
- package/src/runtime/eval/index.ts +48 -0
- package/src/runtime/eval/runner.ts +163 -0
- package/src/runtime/eval/suites.ts +264 -0
- package/src/runtime/events/domain-map.ts +148 -0
- package/src/runtime/events/index.ts +194 -0
- package/src/runtime/events/turn.ts +60 -0
- package/src/runtime/events/workflows.ts +17 -0
- package/src/runtime/forensics/collector.ts +693 -0
- package/src/runtime/forensics/index.ts +23 -0
- package/src/runtime/foundation-clients.ts +78 -0
- package/src/runtime/foundation-services.ts +96 -0
- package/src/runtime/guidance.ts +183 -0
- package/src/runtime/health/effect-handlers.ts +189 -0
- package/src/runtime/health/index.ts +70 -0
- package/src/runtime/health/wiring.ts +115 -0
- package/src/runtime/index.ts +174 -0
- package/src/runtime/integration/helpers.ts +640 -0
- package/src/runtime/lifecycle.ts +107 -0
- package/src/runtime/mcp/index.ts +68 -0
- package/src/runtime/mcp/manager.ts +513 -0
- package/src/runtime/network/inbound.ts +131 -0
- package/src/runtime/network/index.ts +30 -0
- package/src/runtime/network/outbound.ts +292 -0
- package/src/runtime/network/shared.ts +82 -0
- package/src/runtime/operator-client.ts +235 -0
- package/src/runtime/ops/control-plane.ts +363 -0
- package/src/runtime/ops/index.ts +122 -0
- package/src/runtime/ops/playbooks/index.ts +10 -0
- package/src/runtime/ops/playbooks/session-unrecoverable.ts +196 -0
- package/src/runtime/ops/playbooks/stuck-turn.ts +197 -0
- package/src/runtime/ops/runtime-context.ts +100 -0
- package/src/runtime/ops-api.ts +27 -0
- package/src/runtime/orchestration/spawn-policy.ts +83 -0
- package/src/runtime/peer-client.ts +404 -0
- package/src/runtime/perf/index.ts +57 -0
- package/src/runtime/perf/slo-collector.ts +375 -0
- package/src/runtime/permissions/index.ts +190 -0
- package/src/runtime/permissions/policy-runtime.ts +175 -0
- package/src/runtime/permissions/preflight.ts +101 -0
- package/src/runtime/permissions/rule-suggestions.ts +36 -0
- package/src/runtime/plugins/hot-reload.ts +221 -0
- package/src/runtime/plugins/index.ts +84 -0
- package/src/runtime/plugins/lifecycle.ts +95 -0
- package/src/runtime/plugins/manager.ts +474 -0
- package/src/runtime/plugins/manifest.ts +167 -0
- package/src/runtime/plugins/quarantine.ts +202 -0
- package/src/runtime/plugins/trust.ts +291 -0
- package/src/runtime/plugins/types.ts +205 -0
- package/src/runtime/provider-accounts/registry.ts +326 -0
- package/src/runtime/remote/distributed-runtime-contract-schemas.ts +386 -0
- package/src/runtime/remote/index.ts +488 -0
- package/src/runtime/remote/runner-registry.ts +438 -0
- package/src/runtime/remote/supervisor.ts +70 -0
- package/src/runtime/runtime-hook-api.ts +5 -0
- package/src/runtime/runtime-knowledge-api.ts +14 -0
- package/src/runtime/runtime-mcp-api.ts +5 -0
- package/src/runtime/runtime-ops-api.ts +86 -0
- package/src/runtime/runtime-provider-api.ts +18 -0
- package/src/runtime/sandbox/backend.ts +291 -0
- package/src/runtime/sandbox/manager.ts +364 -0
- package/src/runtime/sandbox/provisioning.ts +422 -0
- package/src/runtime/sandbox/session-registry.ts +289 -0
- package/src/runtime/services.ts +541 -0
- package/src/runtime/session-maintenance.ts +188 -0
- package/src/runtime/session-persistence.ts +288 -0
- package/src/runtime/session-return-context.ts +195 -0
- package/src/runtime/settings/control-plane-store.ts +258 -0
- package/src/runtime/settings/control-plane.ts +599 -0
- package/src/runtime/shell-command-extensions.ts +54 -0
- package/src/runtime/shell-command-ops.ts +207 -0
- package/src/runtime/shell-command-platform.ts +47 -0
- package/src/runtime/shell-command-services.ts +143 -0
- package/src/runtime/shell-command-workspace.ts +31 -0
- package/src/runtime/store/domains/conversation.ts +181 -0
- package/src/runtime/store/domains/domain-read-matrix.ts +17 -0
- package/src/runtime/store/domains/index.ts +222 -0
- package/src/runtime/store/domains/panels.ts +117 -0
- package/src/runtime/store/domains/permissions.ts +143 -0
- package/src/runtime/store/domains/ui-perf.ts +103 -0
- package/src/runtime/store/helpers/reducers/conversation.ts +228 -0
- package/src/runtime/store/helpers/reducers/lifecycle.ts +440 -0
- package/src/runtime/store/helpers/reducers/shared.ts +60 -0
- package/src/runtime/store/helpers/reducers/sync.ts +555 -0
- package/src/runtime/store/helpers/reducers.ts +30 -0
- package/src/runtime/store/index.ts +304 -0
- package/src/runtime/store/selectors/index.ts +354 -0
- package/src/runtime/store/state.ts +137 -0
- package/src/runtime/tasks/adapters/acp-adapter.ts +211 -0
- package/src/runtime/tasks/adapters/agent-adapter.ts +208 -0
- package/src/runtime/tasks/adapters/index.ts +16 -0
- package/src/runtime/tasks/adapters/process-adapter.ts +214 -0
- package/src/runtime/tasks/adapters/scheduler-adapter.ts +193 -0
- package/src/runtime/tasks/index.ts +68 -0
- package/src/runtime/tasks/manager.ts +415 -0
- package/src/runtime/telemetry/api-helpers.ts +517 -0
- package/src/runtime/telemetry/api.ts +768 -0
- package/src/runtime/telemetry/index.ts +178 -0
- package/src/runtime/telemetry/instrumentation/domain-bridge-agent-session.ts +440 -0
- package/src/runtime/telemetry/instrumentation/domain-bridge-plugin-mcp.ts +200 -0
- package/src/runtime/telemetry/instrumentation/domain-bridge-shared.ts +18 -0
- package/src/runtime/telemetry/instrumentation/domain-bridge-transport-task.ts +204 -0
- package/src/runtime/telemetry/instrumentation/domain-bridge.ts +125 -0
- package/src/runtime/telemetry/instrumentation/index.ts +67 -0
- package/src/runtime/tools/context.ts +114 -0
- package/src/runtime/tools/index.ts +46 -0
- package/src/runtime/tools/phased-executor.ts +448 -0
- package/src/runtime/tools/phases/budget.ts +130 -0
- package/src/runtime/tools/phases/execute.ts +69 -0
- package/src/runtime/tools/phases/index.ts +13 -0
- package/src/runtime/tools/phases/map-output.ts +98 -0
- package/src/runtime/tools/phases/permission.ts +133 -0
- package/src/runtime/tools/phases/posthook.ts +57 -0
- package/src/runtime/tools/phases/prehook.ts +68 -0
- package/src/runtime/tools/phases/validate.ts +53 -0
- package/src/runtime/transports/direct.ts +73 -0
- package/src/runtime/transports/http-helpers.ts +218 -0
- package/src/runtime/transports/http-types.ts +364 -0
- package/src/runtime/transports/http.ts +629 -0
- package/src/runtime/transports/realtime.ts +50 -0
- package/src/runtime/transports/remote-events.ts +16 -0
- package/src/runtime/transports/shared.ts +39 -0
- package/src/runtime/transports/ui-runtime-events.ts +35 -0
- package/src/runtime/ui/index.ts +39 -0
- package/src/runtime/ui/model-picker/data-provider.ts +182 -0
- package/src/runtime/ui/model-picker/health-enrichment.ts +228 -0
- package/src/runtime/ui/model-picker/index.ts +59 -0
- package/src/runtime/ui/model-picker/types.ts +149 -0
- package/src/runtime/ui/provider-health/data-provider.ts +244 -0
- package/src/runtime/ui/provider-health/fallback-visualizer.ts +69 -0
- package/src/runtime/ui/provider-health/index.ts +46 -0
- package/src/runtime/ui/provider-health/types.ts +146 -0
- package/src/runtime/ui-events.ts +46 -0
- package/src/runtime/ui-read-model-helpers.ts +32 -0
- package/src/runtime/ui-read-models-core.ts +95 -0
- package/src/runtime/ui-read-models-observability-maintenance.ts +81 -0
- package/src/runtime/ui-read-models-observability-options.ts +5 -0
- package/src/runtime/ui-read-models-observability-remote.ts +73 -0
- package/src/runtime/ui-read-models-observability-security.ts +172 -0
- package/src/runtime/ui-read-models-observability-system.ts +217 -0
- package/src/runtime/ui-read-models-observability.ts +59 -0
- package/src/runtime/ui-read-models-operations.ts +203 -0
- package/src/runtime/ui-read-models.ts +61 -0
- package/src/runtime/ui-service-queries.ts +114 -0
- package/src/runtime/ui-services.ts +163 -0
- package/src/runtime/worktree/registry.ts +252 -0
- package/src/scripts/process-messages.ts +42 -0
- package/src/sessions/manager.ts +388 -0
- package/src/shell/blocking-input.ts +89 -0
- package/src/shell/ui-openers.ts +163 -0
- package/src/state/file-watcher.ts +294 -0
- package/src/state/index.ts +56 -0
- package/src/state/knowledge-injection.ts +214 -0
- package/src/state/memory-embedding-http.ts +642 -0
- package/src/state/memory-embeddings.ts +312 -0
- package/src/state/memory-ingest.ts +132 -0
- package/src/state/memory-registry.ts +111 -0
- package/src/state/memory-store-helpers.ts +160 -0
- package/src/state/memory-store.ts +728 -0
- package/src/state/memory-vector-store.ts +418 -0
- package/src/templates/manager.ts +187 -0
- package/src/tools/agent/index.ts +610 -0
- package/src/tools/agent/manager.ts +476 -0
- package/src/tools/analyze/git-modes.ts +380 -0
- package/src/tools/analyze/index.ts +128 -0
- package/src/tools/channel/agent-tools.ts +16 -0
- package/src/tools/channel/index.ts +268 -0
- package/src/tools/control/index.ts +90 -0
- package/src/tools/edit/core.ts +619 -0
- package/src/tools/edit/index.ts +4 -0
- package/src/tools/edit/phased.ts +33 -0
- package/src/tools/fetch/index.ts +3 -0
- package/src/tools/fetch/phased.ts +34 -0
- package/src/tools/fetch/runtime.ts +499 -0
- package/src/tools/index.ts +186 -0
- package/src/tools/mcp/index.ts +190 -0
- package/src/tools/remote-trigger/index.ts +130 -0
- package/src/tools/repl/index.ts +318 -0
- package/src/tools/shared/auto-heal.ts +282 -0
- package/src/tools/state/index.ts +688 -0
- package/src/tools/web-search/index.ts +38 -0
- package/src/tools/write/index.ts +604 -0
- package/src/tools/write/phased.ts +41 -0
- package/src/types/generated/foundation-client-types.ts +22 -0
- package/src/types/sql-js.d.ts +15 -0
- package/src/utils/splash-lines.ts +46 -0
- package/src/version.ts +17 -0
- package/src/watchers/index.ts +11 -0
- package/src/watchers/registry.ts +517 -0
- package/src/web-search/index.ts +26 -0
- package/src/web-search/provider-registry.ts +64 -0
- package/src/web-search/providers/brave.ts +100 -0
- package/src/web-search/providers/duckduckgo.ts +270 -0
- package/src/web-search/providers/exa.ts +77 -0
- package/src/web-search/providers/firecrawl.ts +90 -0
- package/src/web-search/providers/perplexity.ts +86 -0
- package/src/web-search/providers/searxng.ts +88 -0
- package/src/web-search/providers/shared.ts +249 -0
- package/src/web-search/providers/tavily.ts +90 -0
- package/src/web-search/service.ts +142 -0
- package/src/widget/index.ts +2 -0
- package/src/widget/types.ts +9 -0
- package/src/widget/widget.ts +8 -0
|
@@ -0,0 +1,561 @@
|
|
|
1
|
+
import type { LLMProvider, ChatRequest, ChatResponse } from './interface.ts';
|
|
2
|
+
import { ProviderError, isRateLimitOrQuotaError } from '@pellux/goodvibes-sdk/platform/types/errors';
|
|
3
|
+
import { logger } from '@pellux/goodvibes-sdk/platform/utils/logger';
|
|
4
|
+
import type { BenchmarkEntry } from '@pellux/goodvibes-sdk/platform/providers/model-benchmarks';
|
|
5
|
+
import { compositeScore } from '@pellux/goodvibes-sdk/platform/providers/model-benchmarks';
|
|
6
|
+
import type { RuntimeEventBus } from '../runtime/events/index.ts';
|
|
7
|
+
import { emitModelFallback } from '../runtime/emitters/index.ts';
|
|
8
|
+
import { summarizeError } from '@pellux/goodvibes-sdk/platform/utils/error-display';
|
|
9
|
+
|
|
10
|
+
// --- Types ---
|
|
11
|
+
|
|
12
|
+
export interface SyntheticBackend {
|
|
13
|
+
/** Provider name as registered in the provider registry. */
|
|
14
|
+
providerName: string;
|
|
15
|
+
/** Model ID as understood by the provider. */
|
|
16
|
+
modelId: string;
|
|
17
|
+
/**
|
|
18
|
+
* Compound registry key for this backend: `${providerName}:${modelId}`.
|
|
19
|
+
* Used for unambiguous routing and provider lookup.
|
|
20
|
+
*/
|
|
21
|
+
registryKey?: string;
|
|
22
|
+
/** Context window in tokens (used for backend sort order). */
|
|
23
|
+
contextWindow?: number;
|
|
24
|
+
/** Maximum output tokens (used as tiebreaker in sort order). */
|
|
25
|
+
maxOutputTokens?: number;
|
|
26
|
+
/**
|
|
27
|
+
* Environment variable names that gate this backend.
|
|
28
|
+
* Empty array or undefined means no key required (always available).
|
|
29
|
+
*/
|
|
30
|
+
envVars?: string[];
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
/**
|
|
34
|
+
* Returns true if the backend has at least one configured API key.
|
|
35
|
+
* Backends with empty envVars (or undefined) are always available (no key needed).
|
|
36
|
+
*
|
|
37
|
+
* Inlined here to avoid the circular import chain:
|
|
38
|
+
* synthetic.ts -> model-catalog.ts -> registry.ts -> synthetic.ts
|
|
39
|
+
*/
|
|
40
|
+
function hasKey(backend: SyntheticBackend): boolean {
|
|
41
|
+
const vars = backend.envVars;
|
|
42
|
+
if (!vars || vars.length === 0) return true;
|
|
43
|
+
return vars.some(v => {
|
|
44
|
+
const val = process.env[v];
|
|
45
|
+
return typeof val === 'string' && val.length > 0;
|
|
46
|
+
});
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
export type SyntheticTier = 'free' | 'paid' | 'subscription';
|
|
50
|
+
|
|
51
|
+
/**
|
|
52
|
+
* Maps normalised synthetic model IDs to their ordered backend list.
|
|
53
|
+
* Type annotation used by registry.ts for backend resolution.
|
|
54
|
+
*/
|
|
55
|
+
export type SyntheticModelMap = Record<string, SyntheticBackend[]>;
|
|
56
|
+
|
|
57
|
+
/**
|
|
58
|
+
* A canonical model offered by the SyntheticProvider.
|
|
59
|
+
* Groups backends by tier so failover never crosses tier boundaries.
|
|
60
|
+
*/
|
|
61
|
+
export interface CanonicalModel {
|
|
62
|
+
/** Canonical model ID exposed to callers (e.g. 'kimi-k2.5'). */
|
|
63
|
+
id: string;
|
|
64
|
+
/** Pricing tier — determines which backend pool is used for failover. */
|
|
65
|
+
tier: SyntheticTier;
|
|
66
|
+
/** Ordered list of backends to try within this tier. */
|
|
67
|
+
backends: SyntheticBackend[];
|
|
68
|
+
/** Total number of provider backends offering this model. */
|
|
69
|
+
backendCount: number;
|
|
70
|
+
/** Number of backends for which the user has configured API keys. */
|
|
71
|
+
keyedBackendCount: number;
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
type SyntheticCatalogAccessor = () => readonly CanonicalModel[];
|
|
75
|
+
type BenchmarkLookup = (modelId: string) => BenchmarkEntry | undefined;
|
|
76
|
+
|
|
77
|
+
/**
|
|
78
|
+
* Returns backend count metadata for a synthetic model ID.
|
|
79
|
+
* Used by the model picker to display provider availability.
|
|
80
|
+
*
|
|
81
|
+
* @returns Object with backendCount, keyedBackendCount, and tier, or null if not found.
|
|
82
|
+
*/
|
|
83
|
+
export function getSyntheticModelInfo(
|
|
84
|
+
modelId: string,
|
|
85
|
+
getCatalogModels: SyntheticCatalogAccessor,
|
|
86
|
+
): { backendCount: number; keyedBackendCount: number; tier: SyntheticTier } | null {
|
|
87
|
+
const catalog = [...getCatalogModels()];
|
|
88
|
+
const model = catalog.find(m => m.id === modelId);
|
|
89
|
+
if (!model) return null;
|
|
90
|
+
return {
|
|
91
|
+
backendCount: model.backendCount,
|
|
92
|
+
keyedBackendCount: model.keyedBackendCount,
|
|
93
|
+
tier: model.tier,
|
|
94
|
+
};
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
// --- Backend selection ---
|
|
98
|
+
|
|
99
|
+
/**
|
|
100
|
+
* Build a filtered and sorted backend list for a given synthetic model ID.
|
|
101
|
+
*
|
|
102
|
+
* Filtering rules:
|
|
103
|
+
* - Only backends matching the canonical model's tier (tier-isolated failover)
|
|
104
|
+
* - Only backends where the user has a configured API key (key-aware)
|
|
105
|
+
*
|
|
106
|
+
* Sort order: contextWindow descending → maxOutputTokens descending.
|
|
107
|
+
*
|
|
108
|
+
* Returns null if the model is not found in the catalog.
|
|
109
|
+
* Returns an empty array if the model exists but no backends have keys.
|
|
110
|
+
*/
|
|
111
|
+
function buildBackendList(
|
|
112
|
+
syntheticId: string,
|
|
113
|
+
getCatalogModels: SyntheticCatalogAccessor,
|
|
114
|
+
): { backends: SyntheticBackend[]; canonical: CanonicalModel } | null {
|
|
115
|
+
const catalog = [...getCatalogModels()];
|
|
116
|
+
const canonical = catalog.find(m => m.id === syntheticId);
|
|
117
|
+
if (!canonical) return null;
|
|
118
|
+
|
|
119
|
+
// Key-aware filtering: skip backends without configured keys
|
|
120
|
+
const keyed = canonical.backends.filter(hasKey);
|
|
121
|
+
|
|
122
|
+
// Sort: context desc → maxOutput desc
|
|
123
|
+
const sorted = keyed.slice().sort((a, b) => {
|
|
124
|
+
const ctxDiff = (b.contextWindow ?? 0) - (a.contextWindow ?? 0);
|
|
125
|
+
if (ctxDiff !== 0) return ctxDiff;
|
|
126
|
+
return (b.maxOutputTokens ?? 0) - (a.maxOutputTokens ?? 0);
|
|
127
|
+
});
|
|
128
|
+
|
|
129
|
+
return { backends: sorted, canonical };
|
|
130
|
+
}
|
|
131
|
+
|
|
132
|
+
function bestCompositeScoreForModelWithLookup(
|
|
133
|
+
model: CanonicalModel,
|
|
134
|
+
getBenchmarks: BenchmarkLookup,
|
|
135
|
+
): number {
|
|
136
|
+
let best = -1;
|
|
137
|
+
for (const b of model.backends) {
|
|
138
|
+
const entry = getBenchmarks(b.modelId);
|
|
139
|
+
if (entry) {
|
|
140
|
+
const score = compositeScore(entry.benchmarks);
|
|
141
|
+
if (score !== null && score > best) best = score;
|
|
142
|
+
}
|
|
143
|
+
}
|
|
144
|
+
return best;
|
|
145
|
+
}
|
|
146
|
+
|
|
147
|
+
/**
|
|
148
|
+
* Resolve 'best-free' to the canonical ID of the highest composite-scored
|
|
149
|
+
* free model for which the user has at least one backend key configured.
|
|
150
|
+
*
|
|
151
|
+
* Returns null if no free models have keys or benchmark data.
|
|
152
|
+
*/
|
|
153
|
+
function resolveBestFree(
|
|
154
|
+
getCatalogModels: SyntheticCatalogAccessor,
|
|
155
|
+
getBenchmarks: BenchmarkLookup,
|
|
156
|
+
): string | null {
|
|
157
|
+
const catalog = [...getCatalogModels()];
|
|
158
|
+
const freeModels = catalog.filter(m => m.tier === 'free');
|
|
159
|
+
|
|
160
|
+
let bestId: string | null = null;
|
|
161
|
+
let bestScore = -Infinity;
|
|
162
|
+
|
|
163
|
+
for (const model of freeModels) {
|
|
164
|
+
// Check if any backend for this model has a key
|
|
165
|
+
const hasAnyKey = model.backends.some(hasKey);
|
|
166
|
+
if (!hasAnyKey) continue;
|
|
167
|
+
|
|
168
|
+
const effectiveScore = bestCompositeScoreForModelWithLookup(model, getBenchmarks);
|
|
169
|
+
|
|
170
|
+
if (effectiveScore > bestScore) {
|
|
171
|
+
bestScore = effectiveScore;
|
|
172
|
+
bestId = model.id;
|
|
173
|
+
}
|
|
174
|
+
}
|
|
175
|
+
|
|
176
|
+
return bestId;
|
|
177
|
+
}
|
|
178
|
+
|
|
179
|
+
/**
|
|
180
|
+
* Resolve the next-best free model by benchmark score, excluding models in `excludeIds`.
|
|
181
|
+
* Returns the canonical model ID or null if no alternatives exist.
|
|
182
|
+
*/
|
|
183
|
+
function resolveNextBestFree(
|
|
184
|
+
excludeIds: Set<string>,
|
|
185
|
+
getCatalogModels: SyntheticCatalogAccessor,
|
|
186
|
+
getBenchmarks: BenchmarkLookup,
|
|
187
|
+
): string | null {
|
|
188
|
+
const catalog = [...getCatalogModels()];
|
|
189
|
+
const freeModels = catalog.filter(m => m.tier === 'free' && !excludeIds.has(m.id));
|
|
190
|
+
|
|
191
|
+
let bestId: string | null = null;
|
|
192
|
+
let bestScore = -Infinity;
|
|
193
|
+
|
|
194
|
+
for (const model of freeModels) {
|
|
195
|
+
const hasAnyKey = model.backends.some(hasKey);
|
|
196
|
+
if (!hasAnyKey) continue;
|
|
197
|
+
|
|
198
|
+
const effectiveScore = bestCompositeScoreForModelWithLookup(model, getBenchmarks);
|
|
199
|
+
|
|
200
|
+
if (effectiveScore > bestScore) {
|
|
201
|
+
bestScore = effectiveScore;
|
|
202
|
+
bestId = model.id;
|
|
203
|
+
}
|
|
204
|
+
}
|
|
205
|
+
|
|
206
|
+
return bestId;
|
|
207
|
+
}
|
|
208
|
+
|
|
209
|
+
// --- Default cooldown ---
|
|
210
|
+
const DEFAULT_COOLDOWN_MS = 60_000;
|
|
211
|
+
|
|
212
|
+
/** Short cooldown applied to a backend that returns a transient/server error (5xx, network, timeout). */
|
|
213
|
+
const TRANSIENT_COOLDOWN_MS = 5_000;
|
|
214
|
+
|
|
215
|
+
/**
|
|
216
|
+
* Maximum duration to transparently wait for a cooling-down backend before
|
|
217
|
+
* surfacing a 429 error to the caller. Waits of up to 2 minutes are hidden;
|
|
218
|
+
* longer cooldowns are escalated immediately.
|
|
219
|
+
*/
|
|
220
|
+
const MAX_AUTO_WAIT_MS = 120_000;
|
|
221
|
+
|
|
222
|
+
/** Buffer added to the computed wait time to absorb clock skew and scheduling jitter. */
|
|
223
|
+
const COOLDOWN_BUFFER_MS = 100;
|
|
224
|
+
|
|
225
|
+
// --- SyntheticProvider ---
|
|
226
|
+
|
|
227
|
+
export class SyntheticProvider implements LLMProvider {
|
|
228
|
+
readonly name = 'synthetic';
|
|
229
|
+
private readonly getCatalogModels: SyntheticCatalogAccessor;
|
|
230
|
+
private readonly getBenchmarks: BenchmarkLookup;
|
|
231
|
+
private readonly runtimeBus: RuntimeEventBus | null;
|
|
232
|
+
|
|
233
|
+
/** Returns a live snapshot of canonical model IDs each time it is accessed. */
|
|
234
|
+
get models(): string[] {
|
|
235
|
+
return [
|
|
236
|
+
...this.getCatalogModels().map(m => m.id),
|
|
237
|
+
'best-free',
|
|
238
|
+
];
|
|
239
|
+
}
|
|
240
|
+
|
|
241
|
+
// Track cooldowns: syntheticModelId -> array of expiresAt timestamps indexed by resolved backend position
|
|
242
|
+
private cooldowns = new Map<string, number[]>();
|
|
243
|
+
// Track active backend index per resolved model ID
|
|
244
|
+
private activeBackend = new Map<string, number>();
|
|
245
|
+
private readonly resolveProvider: (providerName: string) => LLMProvider;
|
|
246
|
+
|
|
247
|
+
constructor(options: {
|
|
248
|
+
resolveProvider: (providerName: string) => LLMProvider;
|
|
249
|
+
getCatalogModels: SyntheticCatalogAccessor;
|
|
250
|
+
getBenchmarks: BenchmarkLookup;
|
|
251
|
+
runtimeBus?: RuntimeEventBus | null;
|
|
252
|
+
}) {
|
|
253
|
+
this.resolveProvider = options.resolveProvider;
|
|
254
|
+
this.getCatalogModels = options.getCatalogModels;
|
|
255
|
+
this.getBenchmarks = options.getBenchmarks;
|
|
256
|
+
this.runtimeBus = options.runtimeBus ?? null;
|
|
257
|
+
}
|
|
258
|
+
|
|
259
|
+
async chat(params: ChatRequest): Promise<ChatResponse> {
|
|
260
|
+
let syntheticId = params.model;
|
|
261
|
+
|
|
262
|
+
// Resolve 'best-free' alias
|
|
263
|
+
if (syntheticId === 'best-free') {
|
|
264
|
+
const resolved = resolveBestFree(this.getCatalogModels, this.getBenchmarks);
|
|
265
|
+
if (!resolved) {
|
|
266
|
+
throw new ProviderError(
|
|
267
|
+
'No API keys configured for any provider offering free models',
|
|
268
|
+
{
|
|
269
|
+
statusCode: 400,
|
|
270
|
+
provider: this.name,
|
|
271
|
+
operation: 'chat',
|
|
272
|
+
phase: 'routing',
|
|
273
|
+
},
|
|
274
|
+
);
|
|
275
|
+
}
|
|
276
|
+
logger.debug(`[Synthetic] best-free resolved to: ${resolved}`);
|
|
277
|
+
syntheticId = resolved;
|
|
278
|
+
}
|
|
279
|
+
|
|
280
|
+
const result = buildBackendList(syntheticId, this.getCatalogModels);
|
|
281
|
+
|
|
282
|
+
if (!result) {
|
|
283
|
+
throw new ProviderError(`Unknown synthetic model: ${syntheticId}`, {
|
|
284
|
+
statusCode: 400,
|
|
285
|
+
provider: this.name,
|
|
286
|
+
operation: 'chat',
|
|
287
|
+
phase: 'routing',
|
|
288
|
+
});
|
|
289
|
+
}
|
|
290
|
+
|
|
291
|
+
const { backends, canonical } = result;
|
|
292
|
+
|
|
293
|
+
if (backends.length === 0) {
|
|
294
|
+
throw new ProviderError(
|
|
295
|
+
`No API keys configured for any provider offering ${canonical.id}`,
|
|
296
|
+
{
|
|
297
|
+
statusCode: 400,
|
|
298
|
+
provider: this.name,
|
|
299
|
+
operation: 'chat',
|
|
300
|
+
phase: 'routing',
|
|
301
|
+
},
|
|
302
|
+
);
|
|
303
|
+
}
|
|
304
|
+
|
|
305
|
+
const now = Date.now();
|
|
306
|
+
if (!this.cooldowns.has(syntheticId)) {
|
|
307
|
+
this.cooldowns.set(syntheticId, new Array(backends.length).fill(0));
|
|
308
|
+
}
|
|
309
|
+
|
|
310
|
+
// Resize cooldown array if backend count changed (catalog updated)
|
|
311
|
+
let cooldownArr = this.cooldowns.get(syntheticId)!;
|
|
312
|
+
if (cooldownArr.length !== backends.length) {
|
|
313
|
+
cooldownArr = new Array(backends.length).fill(0);
|
|
314
|
+
this.cooldowns.set(syntheticId, cooldownArr);
|
|
315
|
+
}
|
|
316
|
+
|
|
317
|
+
// Reset to preferred backend if its cooldown expired
|
|
318
|
+
if (cooldownArr[0] <= now) {
|
|
319
|
+
this.activeBackend.set(syntheticId, 0);
|
|
320
|
+
}
|
|
321
|
+
|
|
322
|
+
const startIndex = this.activeBackend.get(syntheticId) ?? 0;
|
|
323
|
+
const errors: Array<{ backend: SyntheticBackend; error: Error }> = [];
|
|
324
|
+
let shortestCooldown = Infinity;
|
|
325
|
+
|
|
326
|
+
// Try each backend in order, starting from active
|
|
327
|
+
for (let i = 0; i < backends.length; i++) {
|
|
328
|
+
const idx = (startIndex + i) % backends.length;
|
|
329
|
+
const backend = backends[idx];
|
|
330
|
+
|
|
331
|
+
// Skip if still in cooldown
|
|
332
|
+
if (cooldownArr[idx] > now) {
|
|
333
|
+
const remaining = cooldownArr[idx] - now;
|
|
334
|
+
if (remaining < shortestCooldown) shortestCooldown = remaining;
|
|
335
|
+
continue;
|
|
336
|
+
}
|
|
337
|
+
|
|
338
|
+
// Resolve provider
|
|
339
|
+
let provider: LLMProvider;
|
|
340
|
+
try {
|
|
341
|
+
provider = this.resolveProvider(backend.providerName);
|
|
342
|
+
} catch (err) {
|
|
343
|
+
logger.debug(`[Synthetic] Backend ${backend.providerName} not available: ${err}`);
|
|
344
|
+
continue;
|
|
345
|
+
}
|
|
346
|
+
|
|
347
|
+
// Attempt the call
|
|
348
|
+
// Note: if onDelta is set and a rate limit occurs mid-stream, partial content
|
|
349
|
+
// from this backend will already have been delivered to the caller. The next
|
|
350
|
+
// backend starts fresh, which may produce garbled output. In practice, rate
|
|
351
|
+
// limits reject before streaming begins (at the HTTP level), so this is
|
|
352
|
+
// unlikely to trigger.
|
|
353
|
+
try {
|
|
354
|
+
const response = await provider.chat({
|
|
355
|
+
...params,
|
|
356
|
+
model: backend.modelId,
|
|
357
|
+
});
|
|
358
|
+
|
|
359
|
+
// Success — update active backend
|
|
360
|
+
this.activeBackend.set(syntheticId, idx);
|
|
361
|
+
logger.info(`[Synthetic] ${syntheticId} served by ${backend.providerName} (${backend.modelId})`);
|
|
362
|
+
return response;
|
|
363
|
+
} catch (err) {
|
|
364
|
+
if (isRateLimitOrQuotaError(err)) {
|
|
365
|
+
// Record cooldown
|
|
366
|
+
const cooldownMs = (err instanceof ProviderError && err.retryAfterMs)
|
|
367
|
+
? err.retryAfterMs
|
|
368
|
+
: DEFAULT_COOLDOWN_MS;
|
|
369
|
+
cooldownArr[idx] = now + cooldownMs;
|
|
370
|
+
this.cooldowns.set(syntheticId, cooldownArr);
|
|
371
|
+
if (cooldownMs < shortestCooldown) shortestCooldown = cooldownMs;
|
|
372
|
+
|
|
373
|
+
logger.info(`[Synthetic] ${backend.providerName} rate-limited for ${syntheticId}, cooldown ${Math.round(cooldownMs / 1000)}s`);
|
|
374
|
+
errors.push({ backend, error: err as Error });
|
|
375
|
+
continue;
|
|
376
|
+
}
|
|
377
|
+
// 400 Bad Request — the request itself is malformed, no point trying other backends
|
|
378
|
+
const isBadRequest = err instanceof ProviderError
|
|
379
|
+
&& err.statusCode === 400;
|
|
380
|
+
|
|
381
|
+
if (isBadRequest) {
|
|
382
|
+
throw err;
|
|
383
|
+
}
|
|
384
|
+
|
|
385
|
+
// Other client errors (401 auth, 403 billing/forbidden, 404 model not found, etc.)
|
|
386
|
+
// are provider-specific — failover to next backend with long cooldown
|
|
387
|
+
const isProviderClientError = err instanceof ProviderError
|
|
388
|
+
&& err.statusCode !== undefined
|
|
389
|
+
&& err.statusCode > 400
|
|
390
|
+
&& err.statusCode < 500;
|
|
391
|
+
|
|
392
|
+
if (isProviderClientError) {
|
|
393
|
+
cooldownArr[idx] = now + DEFAULT_COOLDOWN_MS;
|
|
394
|
+
this.cooldowns.set(syntheticId, cooldownArr);
|
|
395
|
+
if (DEFAULT_COOLDOWN_MS < shortestCooldown) shortestCooldown = DEFAULT_COOLDOWN_MS;
|
|
396
|
+
logger.info(`[Synthetic] ${backend.providerName} returned ${(err as ProviderError).statusCode} for ${syntheticId}, trying next backend`);
|
|
397
|
+
errors.push({ backend, error: err as Error });
|
|
398
|
+
continue;
|
|
399
|
+
}
|
|
400
|
+
|
|
401
|
+
// Transient/server error — short cooldown, failover to next backend
|
|
402
|
+
cooldownArr[idx] = now + TRANSIENT_COOLDOWN_MS;
|
|
403
|
+
this.cooldowns.set(syntheticId, cooldownArr);
|
|
404
|
+
if (TRANSIENT_COOLDOWN_MS < shortestCooldown) shortestCooldown = TRANSIENT_COOLDOWN_MS;
|
|
405
|
+
logger.debug(`[Synthetic] ${backend.providerName} failed for ${syntheticId}: ${summarizeError(err) ?? err}, trying next backend`);
|
|
406
|
+
errors.push({ backend, error: err as Error });
|
|
407
|
+
continue;
|
|
408
|
+
}
|
|
409
|
+
}
|
|
410
|
+
|
|
411
|
+
// All backends exhausted — auto-wait if the shortest cooldown is within threshold
|
|
412
|
+
if (shortestCooldown !== Infinity && shortestCooldown <= MAX_AUTO_WAIT_MS) {
|
|
413
|
+
// Find the backend index with the shortest remaining cooldown
|
|
414
|
+
const nowForWait = Date.now();
|
|
415
|
+
let waitIdx = 0;
|
|
416
|
+
let minExpiry = Infinity;
|
|
417
|
+
for (let i = 0; i < cooldownArr.length; i++) {
|
|
418
|
+
if (cooldownArr[i] > nowForWait && cooldownArr[i] < minExpiry) {
|
|
419
|
+
minExpiry = cooldownArr[i];
|
|
420
|
+
waitIdx = i;
|
|
421
|
+
}
|
|
422
|
+
}
|
|
423
|
+
const waitBackend = backends[waitIdx];
|
|
424
|
+
const waitMs = minExpiry - nowForWait;
|
|
425
|
+
|
|
426
|
+
logger.debug(
|
|
427
|
+
`[Synthetic] All backends cooling down for ${syntheticId}, auto-waiting ${
|
|
428
|
+
Math.round(waitMs / 1000)
|
|
429
|
+
}s for ${waitBackend.providerName}…`,
|
|
430
|
+
);
|
|
431
|
+
|
|
432
|
+
// Wait with AbortSignal support
|
|
433
|
+
await new Promise<void>((resolve, reject) => {
|
|
434
|
+
const onAbort = () => {
|
|
435
|
+
clearTimeout(timer);
|
|
436
|
+
reject(new ProviderError('Request aborted during cooldown wait', {
|
|
437
|
+
statusCode: 499,
|
|
438
|
+
provider: this.name,
|
|
439
|
+
operation: 'chat',
|
|
440
|
+
phase: 'cooldown',
|
|
441
|
+
}));
|
|
442
|
+
};
|
|
443
|
+
const timer = setTimeout(() => {
|
|
444
|
+
if (params.signal) params.signal.removeEventListener('abort', onAbort);
|
|
445
|
+
resolve();
|
|
446
|
+
}, waitMs + COOLDOWN_BUFFER_MS);
|
|
447
|
+
if (params.signal) {
|
|
448
|
+
if (params.signal.aborted) {
|
|
449
|
+
clearTimeout(timer);
|
|
450
|
+
reject(new ProviderError('Request aborted during cooldown wait', {
|
|
451
|
+
statusCode: 499,
|
|
452
|
+
provider: this.name,
|
|
453
|
+
operation: 'chat',
|
|
454
|
+
phase: 'cooldown',
|
|
455
|
+
}));
|
|
456
|
+
return;
|
|
457
|
+
}
|
|
458
|
+
params.signal.addEventListener('abort', onAbort, { once: true });
|
|
459
|
+
}
|
|
460
|
+
});
|
|
461
|
+
|
|
462
|
+
// Single retry attempt on the backend that just came off cooldown
|
|
463
|
+
try {
|
|
464
|
+
const waitProvider = this.resolveProvider(waitBackend.providerName);
|
|
465
|
+
const response = await waitProvider.chat({
|
|
466
|
+
...params,
|
|
467
|
+
model: waitBackend.modelId,
|
|
468
|
+
});
|
|
469
|
+
this.activeBackend.set(syntheticId, waitIdx);
|
|
470
|
+
logger.info(
|
|
471
|
+
`[Synthetic] ${syntheticId} served by ${waitBackend.providerName} (${waitBackend.modelId}) after auto-wait`,
|
|
472
|
+
);
|
|
473
|
+
return response;
|
|
474
|
+
} catch (retryErr) {
|
|
475
|
+
// Retry failed — fall through to throw below
|
|
476
|
+
logger.debug(
|
|
477
|
+
`[Synthetic] Auto-wait retry failed for ${syntheticId} via ${
|
|
478
|
+
waitBackend.providerName
|
|
479
|
+
}: ${retryErr}`,
|
|
480
|
+
);
|
|
481
|
+
}
|
|
482
|
+
}
|
|
483
|
+
|
|
484
|
+
// All backends exhausted (or cooldown exceeded threshold, or retry failed)
|
|
485
|
+
const cooldownSec = shortestCooldown === Infinity ? '?' : Math.round(shortestCooldown / 1000);
|
|
486
|
+
|
|
487
|
+
// Cross-model failover for free tier only
|
|
488
|
+
if (canonical.tier === 'free') {
|
|
489
|
+
const tried = new Set<string>([syntheticId]);
|
|
490
|
+
let fallbackId = resolveNextBestFree(tried, this.getCatalogModels, this.getBenchmarks);
|
|
491
|
+
|
|
492
|
+
while (fallbackId) {
|
|
493
|
+
tried.add(fallbackId);
|
|
494
|
+
const fallbackResult = buildBackendList(fallbackId, this.getCatalogModels);
|
|
495
|
+
if (!fallbackResult || fallbackResult.backends.length === 0) {
|
|
496
|
+
fallbackId = resolveNextBestFree(tried, this.getCatalogModels, this.getBenchmarks);
|
|
497
|
+
continue;
|
|
498
|
+
}
|
|
499
|
+
|
|
500
|
+
for (const backend of fallbackResult.backends) {
|
|
501
|
+
try {
|
|
502
|
+
const provider = this.resolveProvider(backend.providerName);
|
|
503
|
+
const response = await provider.chat({
|
|
504
|
+
...params,
|
|
505
|
+
model: backend.modelId,
|
|
506
|
+
});
|
|
507
|
+
|
|
508
|
+
this.activeBackend.set(fallbackId, 0);
|
|
509
|
+
logger.info(`[Synthetic] ${syntheticId} exhausted, fell back to ${fallbackId} via ${backend.providerName}`);
|
|
510
|
+
|
|
511
|
+
if (this.runtimeBus) {
|
|
512
|
+
try {
|
|
513
|
+
emitModelFallback(this.runtimeBus, {
|
|
514
|
+
sessionId: 'system',
|
|
515
|
+
traceId: `synthetic:fallback:${syntheticId}:${fallbackId}`,
|
|
516
|
+
source: 'synthetic-provider',
|
|
517
|
+
}, {
|
|
518
|
+
from: syntheticId,
|
|
519
|
+
to: fallbackId,
|
|
520
|
+
provider: backend.providerName,
|
|
521
|
+
});
|
|
522
|
+
} catch (e) {
|
|
523
|
+
logger.debug('[Synthetic] runtime bus emit failed', { error: summarizeError(e) });
|
|
524
|
+
}
|
|
525
|
+
}
|
|
526
|
+
|
|
527
|
+
return response;
|
|
528
|
+
} catch (err) {
|
|
529
|
+
logger.debug(`[Synthetic] Fallback ${fallbackId} via ${backend.providerName} failed: ${summarizeError(err)}`);
|
|
530
|
+
continue;
|
|
531
|
+
}
|
|
532
|
+
}
|
|
533
|
+
|
|
534
|
+
// All backends for this fallback exhausted, try next model
|
|
535
|
+
fallbackId = resolveNextBestFree(tried, this.getCatalogModels, this.getBenchmarks);
|
|
536
|
+
}
|
|
537
|
+
|
|
538
|
+
// All free models exhausted
|
|
539
|
+
throw new ProviderError(
|
|
540
|
+
`All free models exhausted. No alternatives available. Last tried: ${[...tried].join(', ')}`,
|
|
541
|
+
{
|
|
542
|
+
statusCode: 429,
|
|
543
|
+
provider: this.name,
|
|
544
|
+
operation: 'chat',
|
|
545
|
+
phase: 'routing',
|
|
546
|
+
},
|
|
547
|
+
);
|
|
548
|
+
}
|
|
549
|
+
|
|
550
|
+
throw new ProviderError(
|
|
551
|
+
`All backends for ${syntheticId} exhausted. Shortest cooldown expires in ${cooldownSec}s. ` +
|
|
552
|
+
`Tried: ${errors.map(e => `${e.backend.providerName} (${e.error.message})`).join(', ')}`,
|
|
553
|
+
{
|
|
554
|
+
statusCode: 429,
|
|
555
|
+
provider: this.name,
|
|
556
|
+
operation: 'chat',
|
|
557
|
+
phase: 'routing',
|
|
558
|
+
},
|
|
559
|
+
);
|
|
560
|
+
}
|
|
561
|
+
}
|
|
@@ -0,0 +1,84 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Tier-based system prompt supplements.
|
|
3
|
+
*
|
|
4
|
+
* Model capability tiers drive how much extra guidance is injected into
|
|
5
|
+
* the system prompt. All features remain available regardless of tier;
|
|
6
|
+
* only the verbosity of the guidance changes.
|
|
7
|
+
*/
|
|
8
|
+
|
|
9
|
+
import type { ModelTier } from './registry.ts';
|
|
10
|
+
export type { ModelTier };
|
|
11
|
+
|
|
12
|
+
/**
|
|
13
|
+
* Derive the model tier from a model's context window size.
|
|
14
|
+
*
|
|
15
|
+
* - small (<32K) → 'free' (needs most guidance)
|
|
16
|
+
* - medium (32K–128K) → 'standard' (brief reminders)
|
|
17
|
+
* - large (>128K) → 'premium' (no extra guidance needed)
|
|
18
|
+
*
|
|
19
|
+
* This is used instead of the static ModelDefinition.tier field so that
|
|
20
|
+
* tier-prompt selection is driven by actual model capabilities.
|
|
21
|
+
*/
|
|
22
|
+
export function getTierForContextWindow(contextWindow: number): ModelTier {
|
|
23
|
+
if (contextWindow > 128_000) return 'premium';
|
|
24
|
+
if (contextWindow >= 32_000) return 'standard';
|
|
25
|
+
return 'free';
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
/**
|
|
29
|
+
* Returns supplemental system prompt content based on the model's capability
|
|
30
|
+
* tier. The returned string is appended to the base system prompt before
|
|
31
|
+
* each LLM call.
|
|
32
|
+
*
|
|
33
|
+
* - free — explicit tool-call examples, multi-agent reminders, structured
|
|
34
|
+
* output enforcement (~300 tokens)
|
|
35
|
+
* - standard — brief reminders about tool usage and plan adherence (~80 tokens)
|
|
36
|
+
* - premium — empty; capable models need no extra hand-holding
|
|
37
|
+
*/
|
|
38
|
+
export function getTierPromptSupplement(tier: ModelTier): string {
|
|
39
|
+
switch (tier) {
|
|
40
|
+
case 'free':
|
|
41
|
+
return FREE_SUPPLEMENT;
|
|
42
|
+
case 'standard':
|
|
43
|
+
return STANDARD_SUPPLEMENT;
|
|
44
|
+
case 'premium':
|
|
45
|
+
return '';
|
|
46
|
+
case 'subscription':
|
|
47
|
+
return '';
|
|
48
|
+
}
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
// ---------------------------------------------------------------------------
|
|
52
|
+
// Supplement text
|
|
53
|
+
// ---------------------------------------------------------------------------
|
|
54
|
+
|
|
55
|
+
const FREE_SUPPLEMENT = `## Agent Guidance
|
|
56
|
+
|
|
57
|
+
You are operating in a multi-agent system. Follow these rules carefully:
|
|
58
|
+
|
|
59
|
+
**Tool calls — required format:**
|
|
60
|
+
Every tool call must include ALL required parameters. Missing parameters cause
|
|
61
|
+
silent failures. When in doubt, check the tool's schema before calling it.
|
|
62
|
+
|
|
63
|
+
Example — correct agent spawn:
|
|
64
|
+
\`\`\`json
|
|
65
|
+
{ "name": "agent", "input": { "task": "<task>", "mode": "engineer" } }
|
|
66
|
+
\`\`\`
|
|
67
|
+
|
|
68
|
+
**Multi-agent workflows:**
|
|
69
|
+
When a plan requires multiple parallel agents, spawn ALL of them before
|
|
70
|
+
waiting for results — do not spawn one, wait, then spawn the next. Parallel
|
|
71
|
+
spawns run concurrently and complete faster.
|
|
72
|
+
|
|
73
|
+
**Structured output:**
|
|
74
|
+
Your final message MUST end with the required JSON completion block. Omitting
|
|
75
|
+
it causes the orchestrator to treat your run as failed.
|
|
76
|
+
|
|
77
|
+
**Plan adherence:**
|
|
78
|
+
Complete the full plan. Do not stop after the first step and ask for
|
|
79
|
+
confirmation — there is no human watching. Make the best choice and continue.`;
|
|
80
|
+
|
|
81
|
+
const STANDARD_SUPPLEMENT = `## Reminders
|
|
82
|
+
- Include all required parameters in every tool call.
|
|
83
|
+
- When spawning agents, use the correct \`mode\` parameter.
|
|
84
|
+
- Complete your full plan before reporting results.`;
|