@build-astron-co/nimbus 0.2.0 → 0.4.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/bin/nimbus +26 -10
- package/bin/nimbus.cmd +41 -0
- package/bin/nimbus.mjs +70 -0
- package/completions/nimbus.bash +38 -0
- package/completions/nimbus.fish +48 -0
- package/completions/nimbus.zsh +81 -0
- package/dist/src/agent/compaction-agent.js +215 -0
- package/dist/src/agent/context-manager.js +385 -0
- package/dist/src/agent/context.js +322 -0
- package/dist/src/agent/deploy-preview.js +395 -0
- package/dist/src/agent/expand-files.js +95 -0
- package/dist/src/agent/index.js +18 -0
- package/dist/src/agent/loop.js +1535 -0
- package/dist/src/agent/modes.js +347 -0
- package/dist/src/agent/permissions.js +396 -0
- package/dist/src/agent/subagents/base.js +67 -0
- package/dist/src/agent/subagents/cost.js +45 -0
- package/dist/src/agent/subagents/explore.js +36 -0
- package/dist/src/agent/subagents/general.js +41 -0
- package/dist/src/agent/subagents/index.js +88 -0
- package/dist/src/agent/subagents/infra.js +52 -0
- package/dist/src/agent/subagents/security.js +60 -0
- package/dist/src/agent/system-prompt.js +860 -0
- package/dist/src/app.js +152 -0
- package/dist/src/audit/activity-log.js +209 -0
- package/dist/src/audit/compliance-checker.js +419 -0
- package/dist/src/audit/cost-tracker.js +231 -0
- package/dist/src/audit/index.js +10 -0
- package/dist/src/audit/security-scanner.js +490 -0
- package/dist/src/auth/guard.js +64 -0
- package/dist/src/auth/index.js +19 -0
- package/dist/src/auth/keychain.js +79 -0
- package/dist/src/auth/oauth.js +389 -0
- package/dist/src/auth/providers.js +415 -0
- package/dist/src/auth/sso.js +87 -0
- package/dist/src/auth/store.js +424 -0
- package/dist/src/auth/types.js +5 -0
- package/dist/src/cli/index.js +8 -0
- package/dist/src/cli/init.js +1048 -0
- package/dist/src/cli/openapi-spec.js +346 -0
- package/dist/src/cli/run.js +505 -0
- package/dist/src/cli/serve-auth.js +56 -0
- package/dist/src/cli/serve.js +432 -0
- package/dist/src/cli/web.js +50 -0
- package/dist/src/cli.js +1574 -0
- package/dist/src/clients/core-engine-client.js +156 -0
- package/dist/src/clients/enterprise-client.js +246 -0
- package/dist/src/clients/generator-client.js +219 -0
- package/dist/src/clients/git-client.js +367 -0
- package/dist/src/clients/github-client.js +229 -0
- package/dist/src/clients/helm-client.js +299 -0
- package/dist/src/clients/index.js +18 -0
- package/dist/src/clients/k8s-client.js +270 -0
- package/dist/src/clients/llm-client.js +119 -0
- package/dist/src/clients/rest-client.js +104 -0
- package/dist/src/clients/service-discovery.js +35 -0
- package/dist/src/clients/terraform-client.js +302 -0
- package/dist/src/clients/tools-client.js +1227 -0
- package/dist/src/clients/ws-client.js +93 -0
- package/dist/src/commands/alias.js +91 -0
- package/dist/src/commands/analyze/index.js +313 -0
- package/dist/src/commands/apply/helm.js +375 -0
- package/dist/src/commands/apply/index.js +176 -0
- package/dist/src/commands/apply/k8s.js +350 -0
- package/dist/src/commands/apply/terraform.js +465 -0
- package/dist/src/commands/ask.js +137 -0
- package/dist/src/commands/audit/index.js +322 -0
- package/dist/src/commands/auth-cloud.js +345 -0
- package/dist/src/commands/auth-list.js +112 -0
- package/dist/src/commands/auth-profile.js +104 -0
- package/dist/src/commands/auth-refresh.js +161 -0
- package/dist/src/commands/auth-status.js +122 -0
- package/dist/src/commands/aws/ec2.js +402 -0
- package/dist/src/commands/aws/iam.js +304 -0
- package/dist/src/commands/aws/index.js +108 -0
- package/dist/src/commands/aws/lambda.js +317 -0
- package/dist/src/commands/aws/rds.js +345 -0
- package/dist/src/commands/aws/s3.js +346 -0
- package/dist/src/commands/aws/vpc.js +302 -0
- package/dist/src/commands/aws-discover.js +413 -0
- package/dist/src/commands/aws-terraform.js +618 -0
- package/dist/src/commands/azure/aks.js +305 -0
- package/dist/src/commands/azure/functions.js +200 -0
- package/dist/src/commands/azure/index.js +93 -0
- package/dist/src/commands/azure/storage.js +378 -0
- package/dist/src/commands/azure/vm.js +291 -0
- package/dist/src/commands/billing/index.js +224 -0
- package/dist/src/commands/chat.js +259 -0
- package/dist/src/commands/completions.js +255 -0
- package/dist/src/commands/config.js +291 -0
- package/dist/src/commands/cost/cloud-cost-estimator.js +211 -0
- package/dist/src/commands/cost/estimator.js +73 -0
- package/dist/src/commands/cost/index.js +625 -0
- package/dist/src/commands/cost/parsers/terraform.js +234 -0
- package/dist/src/commands/cost/parsers/types.js +4 -0
- package/dist/src/commands/cost/pricing/aws.js +501 -0
- package/dist/src/commands/cost/pricing/azure.js +462 -0
- package/dist/src/commands/cost/pricing/gcp.js +359 -0
- package/dist/src/commands/cost/pricing/index.js +24 -0
- package/dist/src/commands/demo.js +196 -0
- package/dist/src/commands/deploy.js +215 -0
- package/dist/src/commands/doctor.js +1291 -0
- package/dist/src/commands/drift/index.js +674 -0
- package/dist/src/commands/explain.js +235 -0
- package/dist/src/commands/export.js +120 -0
- package/dist/src/commands/feedback.js +319 -0
- package/dist/src/commands/fix.js +263 -0
- package/dist/src/commands/fs/index.js +338 -0
- package/dist/src/commands/gcp/compute.js +266 -0
- package/dist/src/commands/gcp/functions.js +221 -0
- package/dist/src/commands/gcp/gke.js +357 -0
- package/dist/src/commands/gcp/iam.js +295 -0
- package/dist/src/commands/gcp/index.js +105 -0
- package/dist/src/commands/gcp/storage.js +232 -0
- package/dist/src/commands/generate-helm.js +1026 -0
- package/dist/src/commands/generate-k8s.js +1263 -0
- package/dist/src/commands/generate-terraform.js +1058 -0
- package/dist/src/commands/gh/index.js +663 -0
- package/dist/src/commands/git/index.js +1208 -0
- package/dist/src/commands/helm/index.js +985 -0
- package/dist/src/commands/help.js +639 -0
- package/dist/src/commands/history.js +120 -0
- package/dist/src/commands/import.js +782 -0
- package/dist/src/commands/incident.js +144 -0
- package/dist/src/commands/index.js +109 -0
- package/dist/src/commands/init.js +955 -0
- package/dist/src/commands/k8s/index.js +979 -0
- package/dist/src/commands/login.js +588 -0
- package/dist/src/commands/logout.js +61 -0
- package/dist/src/commands/logs.js +160 -0
- package/dist/src/commands/onboarding.js +382 -0
- package/dist/src/commands/pipeline.js +153 -0
- package/dist/src/commands/plan/display.js +216 -0
- package/dist/src/commands/plan/index.js +525 -0
- package/dist/src/commands/plugin.js +325 -0
- package/dist/src/commands/preview.js +356 -0
- package/dist/src/commands/profile.js +297 -0
- package/dist/src/commands/questionnaire.js +1021 -0
- package/dist/src/commands/resume.js +35 -0
- package/dist/src/commands/rollback.js +259 -0
- package/dist/src/commands/rollout.js +74 -0
- package/dist/src/commands/runbook.js +307 -0
- package/dist/src/commands/schedule.js +202 -0
- package/dist/src/commands/status.js +213 -0
- package/dist/src/commands/team/index.js +309 -0
- package/dist/src/commands/team-context.js +200 -0
- package/dist/src/commands/template.js +204 -0
- package/dist/src/commands/tf/index.js +989 -0
- package/dist/src/commands/upgrade.js +515 -0
- package/dist/src/commands/usage/index.js +118 -0
- package/dist/src/commands/version.js +145 -0
- package/dist/src/commands/watch.js +127 -0
- package/dist/src/compat/index.js +2 -0
- package/dist/src/compat/runtime.js +10 -0
- package/dist/src/compat/sqlite.js +144 -0
- package/dist/src/config/index.js +6 -0
- package/dist/src/config/manager.js +469 -0
- package/dist/src/config/mode-store.js +57 -0
- package/dist/src/config/profiles.js +66 -0
- package/dist/src/config/safety-policy.js +251 -0
- package/dist/src/config/schema.js +107 -0
- package/dist/src/config/types.js +311 -0
- package/dist/src/config/workspace-state.js +38 -0
- package/dist/src/context/context-db.js +138 -0
- package/dist/src/demo/index.js +295 -0
- package/dist/src/demo/scenarios/full-journey.js +226 -0
- package/dist/src/demo/scenarios/getting-started.js +124 -0
- package/dist/src/demo/scenarios/helm-release.js +334 -0
- package/dist/src/demo/scenarios/k8s-deployment.js +190 -0
- package/dist/src/demo/scenarios/terraform-vpc.js +167 -0
- package/dist/src/demo/types.js +6 -0
- package/dist/src/engine/cost-estimator.js +334 -0
- package/dist/src/engine/diagram-generator.js +192 -0
- package/dist/src/engine/drift-detector.js +688 -0
- package/dist/src/engine/executor.js +832 -0
- package/dist/src/engine/index.js +39 -0
- package/dist/src/engine/orchestrator.js +436 -0
- package/dist/src/engine/planner.js +616 -0
- package/dist/src/engine/safety.js +609 -0
- package/dist/src/engine/verifier.js +664 -0
- package/dist/src/enterprise/audit.js +241 -0
- package/dist/src/enterprise/auth.js +189 -0
- package/dist/src/enterprise/billing.js +512 -0
- package/dist/src/enterprise/index.js +16 -0
- package/dist/src/enterprise/teams.js +315 -0
- package/dist/src/generator/best-practices.js +1375 -0
- package/dist/src/generator/helm.js +495 -0
- package/dist/src/generator/index.js +11 -0
- package/dist/src/generator/intent-parser.js +420 -0
- package/dist/src/generator/kubernetes.js +773 -0
- package/dist/src/generator/terraform.js +1472 -0
- package/dist/src/history/index.js +6 -0
- package/dist/src/history/manager.js +199 -0
- package/dist/src/history/types.js +6 -0
- package/dist/src/hooks/config.js +318 -0
- package/dist/src/hooks/engine.js +317 -0
- package/dist/src/hooks/index.js +2 -0
- package/dist/src/llm/auth-bridge.js +157 -0
- package/dist/src/llm/circuit-breaker.js +116 -0
- package/dist/src/llm/config-loader.js +172 -0
- package/dist/src/llm/cost-calculator.js +137 -0
- package/dist/src/llm/index.js +7 -0
- package/dist/src/llm/model-aliases.js +99 -0
- package/dist/src/llm/provider-registry.js +57 -0
- package/dist/src/llm/providers/anthropic.js +430 -0
- package/dist/src/llm/providers/bedrock.js +409 -0
- package/dist/src/llm/providers/google.js +344 -0
- package/dist/src/llm/providers/ollama.js +661 -0
- package/dist/src/llm/providers/openai-compatible.js +289 -0
- package/dist/src/llm/providers/openai.js +284 -0
- package/dist/src/llm/providers/openrouter.js +293 -0
- package/dist/src/llm/router.js +844 -0
- package/dist/src/llm/types.js +69 -0
- package/dist/src/lsp/client.js +239 -0
- package/dist/src/lsp/languages.js +95 -0
- package/dist/src/lsp/manager.js +243 -0
- package/dist/src/mcp/client.js +289 -0
- package/dist/src/mcp/index.js +5 -0
- package/dist/src/mcp/manager.js +113 -0
- package/dist/src/nimbus.js +212 -0
- package/dist/src/plugins/index.js +13 -0
- package/dist/src/plugins/loader.js +280 -0
- package/dist/src/plugins/manager.js +282 -0
- package/dist/src/plugins/types.js +23 -0
- package/dist/src/scanners/cicd-scanner.js +230 -0
- package/dist/src/scanners/cloud-scanner.js +415 -0
- package/dist/src/scanners/framework-scanner.js +430 -0
- package/dist/src/scanners/iac-scanner.js +350 -0
- package/dist/src/scanners/index.js +454 -0
- package/dist/src/scanners/language-scanner.js +258 -0
- package/dist/src/scanners/package-manager-scanner.js +252 -0
- package/dist/src/scanners/types.js +6 -0
- package/dist/src/sessions/manager.js +395 -0
- package/dist/src/sessions/types.js +4 -0
- package/dist/src/sharing/sync.js +238 -0
- package/dist/src/sharing/viewer.js +131 -0
- package/dist/src/snapshots/index.js +1 -0
- package/dist/src/snapshots/manager.js +432 -0
- package/dist/src/state/artifacts.js +94 -0
- package/dist/src/state/audit.js +73 -0
- package/dist/src/state/billing.js +126 -0
- package/dist/src/state/checkpoints.js +81 -0
- package/dist/src/state/config.js +58 -0
- package/dist/src/state/conversations.js +7 -0
- package/dist/src/state/credentials.js +96 -0
- package/dist/src/state/db.js +53 -0
- package/dist/src/state/index.js +23 -0
- package/dist/src/state/messages.js +76 -0
- package/dist/src/state/projects.js +92 -0
- package/dist/src/state/schema.js +233 -0
- package/dist/src/state/sessions.js +79 -0
- package/dist/src/state/teams.js +131 -0
- package/dist/src/telemetry.js +91 -0
- package/dist/src/tools/aws-ops.js +747 -0
- package/dist/src/tools/azure-ops.js +491 -0
- package/dist/src/tools/file-ops.js +451 -0
- package/dist/src/tools/gcp-ops.js +559 -0
- package/dist/src/tools/git-ops.js +557 -0
- package/dist/src/tools/github-ops.js +460 -0
- package/dist/src/tools/helm-ops.js +634 -0
- package/dist/src/tools/index.js +16 -0
- package/dist/src/tools/k8s-ops.js +579 -0
- package/dist/src/tools/schemas/converter.js +129 -0
- package/dist/src/tools/schemas/devops.js +3319 -0
- package/dist/src/tools/schemas/index.js +19 -0
- package/dist/src/tools/schemas/standard.js +966 -0
- package/dist/src/tools/schemas/types.js +409 -0
- package/dist/src/tools/spawn-exec.js +109 -0
- package/dist/src/tools/terraform-ops.js +627 -0
- package/dist/src/types/config.js +1 -0
- package/dist/src/types/drift.js +4 -0
- package/dist/src/types/enterprise.js +5 -0
- package/dist/src/types/index.js +14 -0
- package/dist/src/types/plan.js +1 -0
- package/dist/src/types/request.js +1 -0
- package/dist/src/types/response.js +1 -0
- package/dist/src/types/service.js +1 -0
- package/dist/src/ui/App.js +1672 -0
- package/dist/src/ui/DeployPreview.js +60 -0
- package/dist/src/ui/FileDiffModal.js +108 -0
- package/dist/src/ui/Header.js +46 -0
- package/dist/src/ui/HelpModal.js +9 -0
- package/dist/src/ui/InputBox.js +408 -0
- package/dist/src/ui/MessageList.js +795 -0
- package/dist/src/ui/PermissionPrompt.js +72 -0
- package/dist/src/ui/StatusBar.js +109 -0
- package/dist/src/ui/TerminalPane.js +31 -0
- package/dist/src/ui/ToolCallDisplay.js +303 -0
- package/dist/src/ui/TreePane.js +83 -0
- package/dist/src/ui/chat-ui.js +721 -0
- package/dist/src/ui/index.js +11 -0
- package/dist/src/ui/ink/index.js +1325 -0
- package/dist/src/ui/streaming.js +137 -0
- package/dist/src/ui/theme.js +78 -0
- package/dist/src/ui/types.js +7 -0
- package/dist/src/utils/analytics.js +61 -0
- package/dist/src/utils/cost-warning.js +25 -0
- package/dist/src/utils/env.js +42 -0
- package/dist/src/utils/errors.js +54 -0
- package/dist/src/utils/event-bus.js +22 -0
- package/dist/src/utils/index.js +16 -0
- package/dist/src/utils/logger.js +150 -0
- package/dist/src/utils/rate-limiter.js +90 -0
- package/dist/src/utils/service-auth.js +36 -0
- package/dist/src/utils/validation.js +39 -0
- package/dist/src/version.js +3 -0
- package/dist/src/watcher/index.js +192 -0
- package/dist/src/wizard/approval.js +275 -0
- package/dist/src/wizard/index.js +13 -0
- package/dist/src/wizard/prompts.js +273 -0
- package/dist/src/wizard/types.js +4 -0
- package/dist/src/wizard/ui.js +453 -0
- package/dist/src/wizard/wizard.js +227 -0
- package/package.json +31 -23
- package/src/__tests__/alias.test.ts +133 -0
- package/src/__tests__/app.test.ts +1 -1
- package/src/__tests__/audit.test.ts +1 -1
- package/src/__tests__/circuit-breaker.test.ts +1 -1
- package/src/__tests__/cli-run.test.ts +237 -1
- package/src/__tests__/compat-sqlite.test.ts +68 -0
- package/src/__tests__/context-manager.test.ts +131 -1
- package/src/__tests__/context.test.ts +1 -1
- package/src/__tests__/devops-terminal-gaps.test.ts +718 -0
- package/src/__tests__/doctor.test.ts +48 -0
- package/src/__tests__/enterprise.test.ts +1 -1
- package/src/__tests__/export.test.ts +236 -0
- package/src/__tests__/gap-11-18-20.test.ts +958 -0
- package/src/__tests__/generator.test.ts +1 -1
- package/src/__tests__/helm-streaming.test.ts +127 -0
- package/src/__tests__/hooks.test.ts +1 -1
- package/src/__tests__/incident.test.ts +179 -0
- package/src/__tests__/init.test.ts +55 -4
- package/src/__tests__/intent-parser.test.ts +1 -1
- package/src/__tests__/llm-router.test.ts +1 -1
- package/src/__tests__/logs.test.ts +107 -0
- package/src/__tests__/loop-errors.test.ts +244 -0
- package/src/__tests__/lsp.test.ts +1 -1
- package/src/__tests__/modes.test.ts +1 -1
- package/src/__tests__/perf-optimizations.test.ts +847 -0
- package/src/__tests__/permissions.test.ts +1 -1
- package/src/__tests__/pipeline.test.ts +50 -0
- package/src/__tests__/polish-phase3.test.ts +340 -0
- package/src/__tests__/profile.test.ts +237 -0
- package/src/__tests__/rollback.test.ts +83 -0
- package/src/__tests__/runbook.test.ts +219 -0
- package/src/__tests__/schedule.test.ts +206 -0
- package/src/__tests__/serve.test.ts +1 -1
- package/src/__tests__/sessions.test.ts +96 -1
- package/src/__tests__/sharing.test.ts +53 -1
- package/src/__tests__/snapshots.test.ts +1 -1
- package/src/__tests__/standalone-migration.test.ts +199 -0
- package/src/__tests__/state-db.test.ts +1 -1
- package/src/__tests__/status.test.ts +158 -0
- package/src/__tests__/stream-with-tools.test.ts +71 -25
- package/src/__tests__/subagents.test.ts +1 -1
- package/src/__tests__/system-prompt.test.ts +82 -3
- package/src/__tests__/terminal-gap-v2.test.ts +395 -0
- package/src/__tests__/terminal-parity.test.ts +393 -0
- package/src/__tests__/tf-apply.test.ts +187 -0
- package/src/__tests__/tool-converter.test.ts +1 -1
- package/src/__tests__/tool-schemas.test.ts +209 -4
- package/src/__tests__/tools.test.ts +4 -3
- package/src/__tests__/version-json.test.ts +184 -0
- package/src/__tests__/version.test.ts +1 -1
- package/src/__tests__/watch.test.ts +129 -0
- package/src/agent/compaction-agent.ts +40 -1
- package/src/agent/context-manager.ts +67 -3
- package/src/agent/deploy-preview.ts +62 -1
- package/src/agent/expand-files.ts +108 -0
- package/src/agent/loop.ts +1312 -31
- package/src/agent/permissions.ts +51 -4
- package/src/agent/system-prompt.ts +573 -19
- package/src/app.ts +58 -0
- package/src/audit/security-scanner.ts +45 -0
- package/src/auth/keychain.ts +82 -0
- package/src/auth/oauth.ts +15 -5
- package/src/cli/init.ts +378 -5
- package/src/cli/run.ts +407 -16
- package/src/cli/serve.ts +78 -1
- package/src/cli/web.ts +10 -6
- package/src/cli.ts +312 -1
- package/src/clients/service-discovery.ts +30 -25
- package/src/commands/alias.ts +100 -0
- package/src/commands/audit/index.ts +121 -2
- package/src/commands/auth-cloud.ts +113 -0
- package/src/commands/auth-refresh.ts +187 -0
- package/src/commands/aws-discover.ts +144 -251
- package/src/commands/aws-terraform.ts +68 -118
- package/src/commands/chat.ts +9 -3
- package/src/commands/completions.ts +268 -0
- package/src/commands/config.ts +26 -0
- package/src/commands/cost/index.ts +218 -2
- package/src/commands/deploy.ts +260 -0
- package/src/commands/doctor.ts +744 -152
- package/src/commands/drift/index.ts +371 -23
- package/src/commands/export.ts +146 -0
- package/src/commands/generate-k8s.ts +9 -61
- package/src/commands/generate-terraform.ts +191 -449
- package/src/commands/help.ts +212 -36
- package/src/commands/history.ts +8 -1
- package/src/commands/incident.ts +166 -0
- package/src/commands/init.ts +5 -0
- package/src/commands/login.ts +86 -1
- package/src/commands/logs.ts +167 -0
- package/src/commands/onboarding.ts +211 -34
- package/src/commands/pipeline.ts +186 -0
- package/src/commands/plugin.ts +398 -0
- package/src/commands/profile.ts +342 -0
- package/src/commands/questionnaire.ts +0 -98
- package/src/commands/resume.ts +26 -34
- package/src/commands/rollback.ts +315 -0
- package/src/commands/rollout.ts +88 -0
- package/src/commands/runbook.ts +346 -0
- package/src/commands/schedule.ts +236 -0
- package/src/commands/status.ts +252 -0
- package/src/commands/team-context.ts +220 -0
- package/src/commands/template.ts +58 -57
- package/src/commands/tf/index.ts +70 -11
- package/src/commands/upgrade.ts +57 -0
- package/src/commands/version.ts +54 -50
- package/src/commands/watch.ts +153 -0
- package/src/compat/runtime.ts +1 -1
- package/src/compat/sqlite.ts +75 -5
- package/src/config/mode-store.ts +62 -0
- package/src/config/profiles.ts +84 -0
- package/src/config/types.ts +83 -1
- package/src/config/workspace-state.ts +53 -0
- package/src/engine/cost-estimator.ts +52 -10
- package/src/engine/executor.ts +33 -2
- package/src/engine/planner.ts +68 -1
- package/src/generator/terraform.ts +8 -0
- package/src/history/manager.ts +2 -74
- package/src/hooks/engine.ts +5 -4
- package/src/llm/cost-calculator.ts +2 -2
- package/src/llm/providers/anthropic.ts +50 -21
- package/src/llm/router.ts +76 -7
- package/src/lsp/languages.ts +3 -0
- package/src/lsp/manager.ts +21 -5
- package/src/nimbus.ts +37 -18
- package/src/sessions/manager.ts +108 -1
- package/src/sharing/sync.ts +4 -0
- package/src/sharing/viewer.ts +66 -0
- package/src/tools/file-ops.ts +22 -0
- package/src/tools/schemas/devops.ts +3007 -117
- package/src/tools/schemas/standard.ts +5 -1
- package/src/tools/schemas/types.ts +31 -1
- package/src/tools/spawn-exec.ts +148 -0
- package/src/ui/App.tsx +1183 -66
- package/src/ui/DeployPreview.tsx +62 -57
- package/src/ui/FileDiffModal.tsx +162 -0
- package/src/ui/Header.tsx +87 -24
- package/src/ui/HelpModal.tsx +57 -0
- package/src/ui/InputBox.tsx +163 -10
- package/src/ui/MessageList.tsx +487 -40
- package/src/ui/PermissionPrompt.tsx +17 -5
- package/src/ui/StatusBar.tsx +122 -3
- package/src/ui/TerminalPane.tsx +84 -0
- package/src/ui/ToolCallDisplay.tsx +252 -18
- package/src/ui/TreePane.tsx +132 -0
- package/src/ui/chat-ui.ts +41 -44
- package/src/ui/ink/index.ts +771 -38
- package/src/ui/streaming.ts +1 -1
- package/src/ui/theme.ts +104 -0
- package/src/ui/types.ts +18 -0
- package/src/version.ts +1 -1
- package/src/watcher/index.ts +66 -15
- package/src/wizard/types.ts +1 -0
- package/src/wizard/ui.ts +1 -1
- package/tsconfig.json +2 -2
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
/**
|
|
2
2
|
* Tests for Session Sharing
|
|
3
3
|
*/
|
|
4
|
-
import { describe, test, expect, beforeEach, afterEach } from '
|
|
4
|
+
import { describe, test, expect, beforeEach, afterEach } from 'vitest';
|
|
5
5
|
import {
|
|
6
6
|
shareSession,
|
|
7
7
|
getSharedSession,
|
|
@@ -39,7 +39,58 @@ const mockMessages = [
|
|
|
39
39
|
// Inject test dependencies (no module-level mocks — avoids cross-file leaks)
|
|
40
40
|
// ---------------------------------------------------------------------------
|
|
41
41
|
|
|
42
|
+
/** Minimal in-memory DB that mimics the SQLite API used by sharing/sync.ts */
|
|
43
|
+
function makeInMemoryDb() {
|
|
44
|
+
const rows = new Map<string, any>();
|
|
45
|
+
|
|
46
|
+
return {
|
|
47
|
+
run(sql: string, params: any[]) {
|
|
48
|
+
if (/INSERT INTO shares/.test(sql)) {
|
|
49
|
+
const [id, session_id, name, messages, model, mode, cost_usd, token_count, is_live, write_token, created_at, expires_at] = params;
|
|
50
|
+
rows.set(id, { id, session_id, name, messages, model, mode, cost_usd, token_count, is_live, write_token, created_at, expires_at });
|
|
51
|
+
return { changes: 1 };
|
|
52
|
+
} else if (/UPDATE shares SET messages/.test(sql)) {
|
|
53
|
+
const [messages, id] = params;
|
|
54
|
+
const row = rows.get(id);
|
|
55
|
+
if (row) { row.messages = messages; return { changes: 1 }; }
|
|
56
|
+
return { changes: 0 };
|
|
57
|
+
} else if (/DELETE FROM shares WHERE expires_at/.test(sql)) {
|
|
58
|
+
const [now] = params;
|
|
59
|
+
let count = 0;
|
|
60
|
+
for (const [id, row] of rows) {
|
|
61
|
+
if (row.expires_at <= now) { rows.delete(id); count++; }
|
|
62
|
+
}
|
|
63
|
+
return { changes: count };
|
|
64
|
+
} else if (/DELETE FROM shares WHERE id/.test(sql)) {
|
|
65
|
+
const [id] = params;
|
|
66
|
+
const existed = rows.has(id);
|
|
67
|
+
rows.delete(id);
|
|
68
|
+
return { changes: existed ? 1 : 0 };
|
|
69
|
+
}
|
|
70
|
+
return { changes: 0 };
|
|
71
|
+
},
|
|
72
|
+
query(sql: string) {
|
|
73
|
+
return {
|
|
74
|
+
get(...params: any[]) {
|
|
75
|
+
if (/WHERE id = \? AND expires_at/.test(sql)) {
|
|
76
|
+
const [id, now] = params;
|
|
77
|
+
const row = rows.get(id);
|
|
78
|
+
return row && row.expires_at > now ? row : undefined;
|
|
79
|
+
}
|
|
80
|
+
return undefined;
|
|
81
|
+
},
|
|
82
|
+
all(..._params: any[]) {
|
|
83
|
+
// listShares uses DELETE first then SELECT with ORDER BY (no WHERE params)
|
|
84
|
+
return [...rows.values()];
|
|
85
|
+
},
|
|
86
|
+
};
|
|
87
|
+
},
|
|
88
|
+
};
|
|
89
|
+
}
|
|
90
|
+
|
|
42
91
|
beforeEach(() => {
|
|
92
|
+
const db = makeInMemoryDb();
|
|
93
|
+
_deps.getDb = () => db;
|
|
43
94
|
_deps.getConversation = (id: string) =>
|
|
44
95
|
id === 'test-session-id' ? { messages: mockMessages } : null;
|
|
45
96
|
_deps.getSessionManager = () => ({
|
|
@@ -48,6 +99,7 @@ beforeEach(() => {
|
|
|
48
99
|
});
|
|
49
100
|
|
|
50
101
|
afterEach(() => {
|
|
102
|
+
_deps.getDb = undefined;
|
|
51
103
|
_deps.getConversation = undefined;
|
|
52
104
|
_deps.getSessionManager = undefined;
|
|
53
105
|
});
|
|
@@ -9,7 +9,7 @@
|
|
|
9
9
|
* git write-tree / read-tree / checkout-index workflow.
|
|
10
10
|
*/
|
|
11
11
|
|
|
12
|
-
import { describe, test, expect, beforeEach, afterEach } from '
|
|
12
|
+
import { describe, test, expect, beforeEach, afterEach } from 'vitest';
|
|
13
13
|
import * as fs from 'node:fs';
|
|
14
14
|
import * as path from 'node:path';
|
|
15
15
|
import * as os from 'node:os';
|
|
@@ -0,0 +1,199 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Standalone Migration Tests
|
|
3
|
+
*
|
|
4
|
+
* Source-level checks verifying that microservice-dependent commands
|
|
5
|
+
* (RestClient, CoreEngineClient, localhost:300X) have been replaced
|
|
6
|
+
* with standalone CLI/SQLite/generator implementations.
|
|
7
|
+
*
|
|
8
|
+
* These are intentionally static — no runtime startup needed.
|
|
9
|
+
*/
|
|
10
|
+
|
|
11
|
+
import { readFileSync } from 'node:fs';
|
|
12
|
+
import { join } from 'node:path';
|
|
13
|
+
import { describe, it, expect } from 'vitest';
|
|
14
|
+
|
|
15
|
+
const ROOT = join(__dirname, '..', '..');
|
|
16
|
+
|
|
17
|
+
function src(relativePath: string): string {
|
|
18
|
+
return readFileSync(join(ROOT, relativePath), 'utf8');
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
// ---------------------------------------------------------------------------
|
|
22
|
+
// C1 — generate-terraform: standalone
|
|
23
|
+
// ---------------------------------------------------------------------------
|
|
24
|
+
describe('C1 — generate-terraform standalone', () => {
|
|
25
|
+
it('has no new RestClient call', () => {
|
|
26
|
+
expect(src('src/commands/generate-terraform.ts')).not.toContain('new RestClient');
|
|
27
|
+
});
|
|
28
|
+
it('imports generateTerraformProject from generator', () => {
|
|
29
|
+
expect(src('src/commands/generate-terraform.ts')).toContain('generateTerraformProject');
|
|
30
|
+
});
|
|
31
|
+
it('has no localhost:300 reference', () => {
|
|
32
|
+
expect(src('src/commands/generate-terraform.ts')).not.toMatch(/localhost:300\d/);
|
|
33
|
+
});
|
|
34
|
+
});
|
|
35
|
+
|
|
36
|
+
// ---------------------------------------------------------------------------
|
|
37
|
+
// C2 — generate-k8s: standalone
|
|
38
|
+
// ---------------------------------------------------------------------------
|
|
39
|
+
describe('C2 — generate-k8s standalone', () => {
|
|
40
|
+
it('has no new RestClient call', () => {
|
|
41
|
+
expect(src('src/commands/generate-k8s.ts')).not.toContain('new RestClient');
|
|
42
|
+
});
|
|
43
|
+
it('uses generateManifestsLocally or kubernetes generator', () => {
|
|
44
|
+
expect(src('src/commands/generate-k8s.ts')).toMatch(/generateManifestsLocally|generateK8sManifests|K8sGeneratorConfig/);
|
|
45
|
+
});
|
|
46
|
+
it('has no localhost:300 reference', () => {
|
|
47
|
+
expect(src('src/commands/generate-k8s.ts')).not.toMatch(/localhost:300\d/);
|
|
48
|
+
});
|
|
49
|
+
});
|
|
50
|
+
|
|
51
|
+
// ---------------------------------------------------------------------------
|
|
52
|
+
// C3 — aws-discover: standalone
|
|
53
|
+
// ---------------------------------------------------------------------------
|
|
54
|
+
describe('C3 — aws-discover standalone', () => {
|
|
55
|
+
it('has no new RestClient call', () => {
|
|
56
|
+
expect(src('src/commands/aws-discover.ts')).not.toContain('new RestClient');
|
|
57
|
+
});
|
|
58
|
+
it('uses aws configure or execFile for profiles', () => {
|
|
59
|
+
expect(src('src/commands/aws-discover.ts')).toMatch(/configure|execFile|execFileSync|cliGetAwsProfiles/);
|
|
60
|
+
});
|
|
61
|
+
it('has no localhost:300 reference', () => {
|
|
62
|
+
expect(src('src/commands/aws-discover.ts')).not.toMatch(/localhost:300\d/);
|
|
63
|
+
});
|
|
64
|
+
});
|
|
65
|
+
|
|
66
|
+
// ---------------------------------------------------------------------------
|
|
67
|
+
// C4 — aws-terraform: standalone
|
|
68
|
+
// ---------------------------------------------------------------------------
|
|
69
|
+
describe('C4 — aws-terraform standalone', () => {
|
|
70
|
+
it('has no new RestClient call', () => {
|
|
71
|
+
expect(src('src/commands/aws-terraform.ts')).not.toContain('new RestClient');
|
|
72
|
+
});
|
|
73
|
+
it('imports generateTerraformProject from generator', () => {
|
|
74
|
+
expect(src('src/commands/aws-terraform.ts')).toContain('generateTerraformProject');
|
|
75
|
+
});
|
|
76
|
+
it('has no localhost:300 reference', () => {
|
|
77
|
+
expect(src('src/commands/aws-terraform.ts')).not.toMatch(/localhost:300\d/);
|
|
78
|
+
});
|
|
79
|
+
});
|
|
80
|
+
|
|
81
|
+
// ---------------------------------------------------------------------------
|
|
82
|
+
// C5 — resume: standalone
|
|
83
|
+
// ---------------------------------------------------------------------------
|
|
84
|
+
describe('C5 — resume command standalone', () => {
|
|
85
|
+
it('has no CoreEngineClient import', () => {
|
|
86
|
+
expect(src('src/commands/resume.ts')).not.toContain('CoreEngineClient');
|
|
87
|
+
});
|
|
88
|
+
it('references chatCommand or getDb for session lookup', () => {
|
|
89
|
+
expect(src('src/commands/resume.ts')).toMatch(/chatCommand|getDb|SessionManager/);
|
|
90
|
+
});
|
|
91
|
+
it('has no localhost:300 reference', () => {
|
|
92
|
+
expect(src('src/commands/resume.ts')).not.toMatch(/localhost:300\d/);
|
|
93
|
+
});
|
|
94
|
+
});
|
|
95
|
+
|
|
96
|
+
// ---------------------------------------------------------------------------
|
|
97
|
+
// C6 — version: DevOps CLI tools (not localhost services)
|
|
98
|
+
// ---------------------------------------------------------------------------
|
|
99
|
+
describe('C6 — version command DevOps tools', () => {
|
|
100
|
+
it('has no localhost:300 references in fetchComponentVersions', () => {
|
|
101
|
+
expect(src('src/commands/version.ts')).not.toMatch(/localhost:300\d/);
|
|
102
|
+
});
|
|
103
|
+
it('includes terraform version check', () => {
|
|
104
|
+
expect(src('src/commands/version.ts')).toContain('terraform');
|
|
105
|
+
});
|
|
106
|
+
it('includes kubectl version check', () => {
|
|
107
|
+
expect(src('src/commands/version.ts')).toContain('kubectl');
|
|
108
|
+
});
|
|
109
|
+
it('includes helm version check', () => {
|
|
110
|
+
expect(src('src/commands/version.ts')).toContain('helm');
|
|
111
|
+
});
|
|
112
|
+
it('shows [+] / [-] icon format', () => {
|
|
113
|
+
expect(src('src/commands/version.ts')).toMatch(/\[\+\]|\[-\]/);
|
|
114
|
+
});
|
|
115
|
+
});
|
|
116
|
+
|
|
117
|
+
// ---------------------------------------------------------------------------
|
|
118
|
+
// C7 — drift: no CoreEngineClient
|
|
119
|
+
// ---------------------------------------------------------------------------
|
|
120
|
+
describe('C7 — drift no CoreEngineClient', () => {
|
|
121
|
+
it('drift/index.ts has no CoreEngineClient import', () => {
|
|
122
|
+
const content = src('src/commands/drift/index.ts');
|
|
123
|
+
// Must not have an import statement (comments are fine)
|
|
124
|
+
expect(content).not.toMatch(/^import.*CoreEngineClient/m);
|
|
125
|
+
});
|
|
126
|
+
it('drift/index.ts uses execFileSync or execFile for terraform', () => {
|
|
127
|
+
expect(src('src/commands/drift/index.ts')).toMatch(/execFileSync|execFile|spawnExec/);
|
|
128
|
+
});
|
|
129
|
+
});
|
|
130
|
+
|
|
131
|
+
// ---------------------------------------------------------------------------
|
|
132
|
+
// C8 — dead client code deleted/stubbed
|
|
133
|
+
// ---------------------------------------------------------------------------
|
|
134
|
+
describe('C8 — dead client code', () => {
|
|
135
|
+
it('service-discovery.ts is either deleted or a stub without ghost URLs', () => {
|
|
136
|
+
try {
|
|
137
|
+
const content = src('src/clients/service-discovery.ts');
|
|
138
|
+
// If it exists, it must not contain the old ghost localhost service URLs
|
|
139
|
+
expect(content).not.toContain('localhost:3001');
|
|
140
|
+
expect(content).not.toContain('localhost:3003');
|
|
141
|
+
expect(content).not.toContain('localhost:3009');
|
|
142
|
+
} catch {
|
|
143
|
+
// File deleted — that's fine
|
|
144
|
+
}
|
|
145
|
+
});
|
|
146
|
+
});
|
|
147
|
+
|
|
148
|
+
// ---------------------------------------------------------------------------
|
|
149
|
+
// H1 — Emoji removed, replaced with ASCII icons
|
|
150
|
+
// ---------------------------------------------------------------------------
|
|
151
|
+
describe('H1 — emoji removed from UI', () => {
|
|
152
|
+
it('TreePane.tsx has no folder emoji', () => {
|
|
153
|
+
expect(src('src/ui/TreePane.tsx')).not.toContain('\u{1F4C1}'); // 📁
|
|
154
|
+
});
|
|
155
|
+
it('TreePane.tsx uses [/] for directories', () => {
|
|
156
|
+
expect(src('src/ui/TreePane.tsx')).toContain('[/]');
|
|
157
|
+
});
|
|
158
|
+
it('TerminalPane.tsx has no right-pointing triangle', () => {
|
|
159
|
+
expect(src('src/ui/TerminalPane.tsx')).not.toContain('\u25B6'); // ▶
|
|
160
|
+
});
|
|
161
|
+
it('TerminalPane.tsx uses [>] icon', () => {
|
|
162
|
+
expect(src('src/ui/TerminalPane.tsx')).toContain('[>]');
|
|
163
|
+
});
|
|
164
|
+
it('StatusBar.tsx has no Greek capital delta (Δ)', () => {
|
|
165
|
+
expect(src('src/ui/StatusBar.tsx')).not.toContain('\u0394'); // Δ
|
|
166
|
+
});
|
|
167
|
+
it('StatusBar.tsx uses delta: text', () => {
|
|
168
|
+
expect(src('src/ui/StatusBar.tsx')).toContain('delta:');
|
|
169
|
+
});
|
|
170
|
+
});
|
|
171
|
+
|
|
172
|
+
// ---------------------------------------------------------------------------
|
|
173
|
+
// M1 — Secret redaction in spawn-exec
|
|
174
|
+
// ---------------------------------------------------------------------------
|
|
175
|
+
describe('M1 — secret redaction in spawn-exec', () => {
|
|
176
|
+
it('spawn-exec.ts has SECRET_PATTERNS constant', () => {
|
|
177
|
+
expect(src('src/tools/spawn-exec.ts')).toContain('SECRET_PATTERNS');
|
|
178
|
+
});
|
|
179
|
+
it('spawn-exec.ts has redactSecrets function', () => {
|
|
180
|
+
expect(src('src/tools/spawn-exec.ts')).toContain('redactSecrets');
|
|
181
|
+
});
|
|
182
|
+
it('spawn-exec.ts redacts AKIA keys', () => {
|
|
183
|
+
expect(src('src/tools/spawn-exec.ts')).toContain('AKIA');
|
|
184
|
+
});
|
|
185
|
+
it('spawn-exec.ts redacts Bearer tokens', () => {
|
|
186
|
+
expect(src('src/tools/spawn-exec.ts')).toContain('Bearer');
|
|
187
|
+
});
|
|
188
|
+
});
|
|
189
|
+
|
|
190
|
+
// ---------------------------------------------------------------------------
|
|
191
|
+
// M2 — Doctor shows inline versions
|
|
192
|
+
// ---------------------------------------------------------------------------
|
|
193
|
+
describe('M2 — doctor inline versions', () => {
|
|
194
|
+
it('doctor.ts references version in available tools output', () => {
|
|
195
|
+
// The available list should include t.version or similar version string
|
|
196
|
+
const content = src('src/commands/doctor.ts');
|
|
197
|
+
expect(content).toMatch(/t\.version|tool\.version|\.version\b/);
|
|
198
|
+
});
|
|
199
|
+
});
|
|
@@ -10,7 +10,7 @@
|
|
|
10
10
|
* is never touched, and tests are fully isolated and fast.
|
|
11
11
|
*/
|
|
12
12
|
|
|
13
|
-
import { describe, it, expect, beforeEach } from '
|
|
13
|
+
import { describe, it, expect, beforeEach } from 'vitest';
|
|
14
14
|
import type { Database } from '../compat/sqlite';
|
|
15
15
|
import { getTestDb } from '../state/db';
|
|
16
16
|
import {
|
|
@@ -0,0 +1,158 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Tests for the nimbus status command (G18).
|
|
3
|
+
*
|
|
4
|
+
* The statusCommand runs concurrent CLI checks. We test the module structure
|
|
5
|
+
* and basic argument parsing rather than the actual CLI calls to avoid
|
|
6
|
+
* platform dependencies.
|
|
7
|
+
*/
|
|
8
|
+
|
|
9
|
+
import { describe, test, expect } from 'vitest';
|
|
10
|
+
import { statusCommand, type StatusOptions } from '../commands/status';
|
|
11
|
+
|
|
12
|
+
describe('statusCommand (G18)', () => {
|
|
13
|
+
test('exports statusCommand function', () => {
|
|
14
|
+
expect(typeof statusCommand).toBe('function');
|
|
15
|
+
});
|
|
16
|
+
|
|
17
|
+
test('StatusOptions accepts json flag', () => {
|
|
18
|
+
const opts: StatusOptions = { json: true };
|
|
19
|
+
expect(opts.json).toBe(true);
|
|
20
|
+
});
|
|
21
|
+
|
|
22
|
+
test('StatusOptions json defaults to undefined', () => {
|
|
23
|
+
const opts: StatusOptions = {};
|
|
24
|
+
expect(opts.json).toBeUndefined();
|
|
25
|
+
});
|
|
26
|
+
|
|
27
|
+
test('statusCommand resolves without throwing when CLIs are unavailable', async () => {
|
|
28
|
+
// Mock console.log to suppress output in test
|
|
29
|
+
const original = console.log;
|
|
30
|
+
const logs: string[] = [];
|
|
31
|
+
console.log = (msg: string) => logs.push(msg);
|
|
32
|
+
|
|
33
|
+
try {
|
|
34
|
+
// Should not throw even if kubectl/terraform/aws/gcloud are not installed
|
|
35
|
+
// (the command catches errors gracefully)
|
|
36
|
+
await expect(statusCommand({ json: true })).resolves.toBeUndefined();
|
|
37
|
+
} finally {
|
|
38
|
+
console.log = original;
|
|
39
|
+
}
|
|
40
|
+
});
|
|
41
|
+
|
|
42
|
+
test('statusCommand with json option outputs valid JSON', async () => {
|
|
43
|
+
const original = console.log;
|
|
44
|
+
let jsonOutput = '';
|
|
45
|
+
console.log = (msg: string) => { jsonOutput = msg; };
|
|
46
|
+
|
|
47
|
+
try {
|
|
48
|
+
await statusCommand({ json: true });
|
|
49
|
+
// Should produce valid JSON
|
|
50
|
+
const parsed = JSON.parse(jsonOutput);
|
|
51
|
+
expect(parsed).toBeTypeOf('object');
|
|
52
|
+
// Should have an errors array
|
|
53
|
+
expect(Array.isArray(parsed.errors)).toBe(true);
|
|
54
|
+
} finally {
|
|
55
|
+
console.log = original;
|
|
56
|
+
}
|
|
57
|
+
});
|
|
58
|
+
});
|
|
59
|
+
|
|
60
|
+
describe('statusCommand C2 enhancements', () => {
|
|
61
|
+
test('JSON output includes model field with default fallback', async () => {
|
|
62
|
+
const original = console.log;
|
|
63
|
+
let jsonOutput = '';
|
|
64
|
+
console.log = (msg: string) => { jsonOutput = msg; };
|
|
65
|
+
|
|
66
|
+
try {
|
|
67
|
+
await statusCommand({ json: true });
|
|
68
|
+
const parsed = JSON.parse(jsonOutput) as Record<string, unknown>;
|
|
69
|
+
// model should be set (either from config.json or default 'claude-sonnet-4-6')
|
|
70
|
+
expect(typeof parsed.model).toBe('string');
|
|
71
|
+
expect((parsed.model as string).length).toBeGreaterThan(0);
|
|
72
|
+
} finally {
|
|
73
|
+
console.log = original;
|
|
74
|
+
}
|
|
75
|
+
});
|
|
76
|
+
|
|
77
|
+
test('JSON output includes provider field with default fallback', async () => {
|
|
78
|
+
const original = console.log;
|
|
79
|
+
let jsonOutput = '';
|
|
80
|
+
console.log = (msg: string) => { jsonOutput = msg; };
|
|
81
|
+
|
|
82
|
+
try {
|
|
83
|
+
await statusCommand({ json: true });
|
|
84
|
+
const parsed = JSON.parse(jsonOutput) as Record<string, unknown>;
|
|
85
|
+
// provider should be set (either from config.json or default 'anthropic')
|
|
86
|
+
expect(typeof parsed.provider).toBe('string');
|
|
87
|
+
expect((parsed.provider as string).length).toBeGreaterThan(0);
|
|
88
|
+
} finally {
|
|
89
|
+
console.log = original;
|
|
90
|
+
}
|
|
91
|
+
});
|
|
92
|
+
|
|
93
|
+
test('JSON output includes nimbusMdFound field', async () => {
|
|
94
|
+
const original = console.log;
|
|
95
|
+
let jsonOutput = '';
|
|
96
|
+
console.log = (msg: string) => { jsonOutput = msg; };
|
|
97
|
+
|
|
98
|
+
try {
|
|
99
|
+
await statusCommand({ json: true });
|
|
100
|
+
const parsed = JSON.parse(jsonOutput) as Record<string, unknown>;
|
|
101
|
+
// nimbusMdFound should be a boolean (true or undefined/false)
|
|
102
|
+
expect(
|
|
103
|
+
parsed.nimbusMdFound === undefined || typeof parsed.nimbusMdFound === 'boolean'
|
|
104
|
+
).toBe(true);
|
|
105
|
+
} finally {
|
|
106
|
+
console.log = original;
|
|
107
|
+
}
|
|
108
|
+
});
|
|
109
|
+
|
|
110
|
+
test('JSON output includes sessionCount field', async () => {
|
|
111
|
+
const original = console.log;
|
|
112
|
+
let jsonOutput = '';
|
|
113
|
+
console.log = (msg: string) => { jsonOutput = msg; };
|
|
114
|
+
|
|
115
|
+
try {
|
|
116
|
+
await statusCommand({ json: true });
|
|
117
|
+
const parsed = JSON.parse(jsonOutput) as Record<string, unknown>;
|
|
118
|
+
// sessionCount should be a number or 'N/A'
|
|
119
|
+
expect(
|
|
120
|
+
typeof parsed.sessionCount === 'number' || parsed.sessionCount === 'N/A'
|
|
121
|
+
).toBe(true);
|
|
122
|
+
} finally {
|
|
123
|
+
console.log = original;
|
|
124
|
+
}
|
|
125
|
+
});
|
|
126
|
+
|
|
127
|
+
test('default model is claude-sonnet-4-6 when config.json is absent', async () => {
|
|
128
|
+
const original = console.log;
|
|
129
|
+
let jsonOutput = '';
|
|
130
|
+
console.log = (msg: string) => { jsonOutput = msg; };
|
|
131
|
+
|
|
132
|
+
// We can test this by verifying the model is never empty
|
|
133
|
+
try {
|
|
134
|
+
await statusCommand({ json: true });
|
|
135
|
+
const parsed = JSON.parse(jsonOutput) as Record<string, unknown>;
|
|
136
|
+
// Model should be a non-empty string
|
|
137
|
+
expect(typeof parsed.model).toBe('string');
|
|
138
|
+
expect((parsed.model as string).length).toBeGreaterThan(0);
|
|
139
|
+
} finally {
|
|
140
|
+
console.log = original;
|
|
141
|
+
}
|
|
142
|
+
});
|
|
143
|
+
|
|
144
|
+
test('default provider is anthropic when config.json is absent', async () => {
|
|
145
|
+
const original = console.log;
|
|
146
|
+
let jsonOutput = '';
|
|
147
|
+
console.log = (msg: string) => { jsonOutput = msg; };
|
|
148
|
+
|
|
149
|
+
try {
|
|
150
|
+
await statusCommand({ json: true });
|
|
151
|
+
const parsed = JSON.parse(jsonOutput) as Record<string, unknown>;
|
|
152
|
+
expect(typeof parsed.provider).toBe('string');
|
|
153
|
+
expect((parsed.provider as string).length).toBeGreaterThan(0);
|
|
154
|
+
} finally {
|
|
155
|
+
console.log = original;
|
|
156
|
+
}
|
|
157
|
+
});
|
|
158
|
+
});
|
|
@@ -14,7 +14,7 @@
|
|
|
14
14
|
* All tests use mocks -- no real API calls are made.
|
|
15
15
|
*/
|
|
16
16
|
|
|
17
|
-
import { describe, test, expect,
|
|
17
|
+
import { describe, test, it, expect, vi, beforeEach } from 'vitest';
|
|
18
18
|
import type { ToolCompletionRequest, StreamChunk } from '../llm/types';
|
|
19
19
|
|
|
20
20
|
// ---------------------------------------------------------------------------
|
|
@@ -120,7 +120,7 @@ describe('OllamaProvider.streamWithTools', () => {
|
|
|
120
120
|
'data: [DONE]\n\n',
|
|
121
121
|
];
|
|
122
122
|
|
|
123
|
-
globalThis.fetch =
|
|
123
|
+
globalThis.fetch = vi.fn(() =>
|
|
124
124
|
Promise.resolve(
|
|
125
125
|
new Response(buildReadableStream(sseLines), {
|
|
126
126
|
status: 200,
|
|
@@ -170,7 +170,7 @@ describe('OllamaProvider.streamWithTools', () => {
|
|
|
170
170
|
'data: [DONE]\n\n',
|
|
171
171
|
];
|
|
172
172
|
|
|
173
|
-
globalThis.fetch =
|
|
173
|
+
globalThis.fetch = vi.fn(() =>
|
|
174
174
|
Promise.resolve(
|
|
175
175
|
new Response(buildReadableStream(sseLines), {
|
|
176
176
|
status: 200,
|
|
@@ -202,7 +202,7 @@ describe('OllamaProvider.streamWithTools', () => {
|
|
|
202
202
|
test('fallback: when native streaming fails, falls back to completeWithTools', async () => {
|
|
203
203
|
let _callCount = 0;
|
|
204
204
|
|
|
205
|
-
globalThis.fetch =
|
|
205
|
+
globalThis.fetch = vi.fn((url: string | URL | Request) => {
|
|
206
206
|
_callCount++;
|
|
207
207
|
const urlStr = typeof url === 'string' ? url : url instanceof URL ? url.toString() : url.url;
|
|
208
208
|
|
|
@@ -268,7 +268,7 @@ describe('OllamaProvider.streamWithTools', () => {
|
|
|
268
268
|
'data: [DONE]\n\n',
|
|
269
269
|
];
|
|
270
270
|
|
|
271
|
-
globalThis.fetch =
|
|
271
|
+
globalThis.fetch = vi.fn(() =>
|
|
272
272
|
Promise.resolve(
|
|
273
273
|
new Response(buildReadableStream(sseLines), {
|
|
274
274
|
status: 200,
|
|
@@ -311,7 +311,7 @@ describe('OpenRouterProvider.streamWithTools', () => {
|
|
|
311
311
|
},
|
|
312
312
|
]);
|
|
313
313
|
|
|
314
|
-
const mockCreate =
|
|
314
|
+
const mockCreate = vi.fn(() => Promise.resolve(streamChunks));
|
|
315
315
|
|
|
316
316
|
const { OpenRouterProvider } = await import('../llm/providers/openrouter');
|
|
317
317
|
const provider = new OpenRouterProvider('test-api-key');
|
|
@@ -392,7 +392,7 @@ describe('OpenRouterProvider.streamWithTools', () => {
|
|
|
392
392
|
},
|
|
393
393
|
]);
|
|
394
394
|
|
|
395
|
-
const mockCreate =
|
|
395
|
+
const mockCreate = vi.fn(() => Promise.resolve(streamChunks));
|
|
396
396
|
|
|
397
397
|
const { OpenRouterProvider } = await import('../llm/providers/openrouter');
|
|
398
398
|
const provider = new OpenRouterProvider('test-api-key');
|
|
@@ -415,7 +415,7 @@ describe('OpenRouterProvider.streamWithTools', () => {
|
|
|
415
415
|
});
|
|
416
416
|
|
|
417
417
|
test('fallback: when SDK stream creation throws, the generator yields nothing', async () => {
|
|
418
|
-
const mockCreate =
|
|
418
|
+
const mockCreate = vi.fn(() => Promise.reject(new Error('API unavailable')));
|
|
419
419
|
|
|
420
420
|
const { OpenRouterProvider } = await import('../llm/providers/openrouter');
|
|
421
421
|
const provider = new OpenRouterProvider('test-api-key');
|
|
@@ -462,7 +462,7 @@ describe('OpenRouterProvider.streamWithTools', () => {
|
|
|
462
462
|
},
|
|
463
463
|
]);
|
|
464
464
|
|
|
465
|
-
const mockCreate =
|
|
465
|
+
const mockCreate = vi.fn(() => Promise.resolve(streamChunks));
|
|
466
466
|
|
|
467
467
|
const { OpenRouterProvider } = await import('../llm/providers/openrouter');
|
|
468
468
|
const provider = new OpenRouterProvider('test-api-key');
|
|
@@ -495,10 +495,8 @@ describe('OpenRouterProvider.streamWithTools', () => {
|
|
|
495
495
|
// ===========================================================================
|
|
496
496
|
|
|
497
497
|
describe('OpenAICompatibleProvider.streamWithTools', () => {
|
|
498
|
-
function createProvider() {
|
|
499
|
-
|
|
500
|
-
// eslint-disable-next-line @typescript-eslint/no-var-requires
|
|
501
|
-
const { OpenAICompatibleProvider } = require('../llm/providers/openai-compatible');
|
|
498
|
+
async function createProvider() {
|
|
499
|
+
const { OpenAICompatibleProvider } = await import('../llm/providers/openai-compatible');
|
|
502
500
|
return new OpenAICompatibleProvider({
|
|
503
501
|
name: 'test-compat',
|
|
504
502
|
apiKey: 'test-key',
|
|
@@ -517,9 +515,9 @@ describe('OpenAICompatibleProvider.streamWithTools', () => {
|
|
|
517
515
|
},
|
|
518
516
|
]);
|
|
519
517
|
|
|
520
|
-
const mockCreate =
|
|
518
|
+
const mockCreate = vi.fn(() => Promise.resolve(streamChunks));
|
|
521
519
|
|
|
522
|
-
const provider = createProvider();
|
|
520
|
+
const provider = await createProvider();
|
|
523
521
|
(provider as any).client = {
|
|
524
522
|
chat: { completions: { create: mockCreate } },
|
|
525
523
|
};
|
|
@@ -578,9 +576,9 @@ describe('OpenAICompatibleProvider.streamWithTools', () => {
|
|
|
578
576
|
},
|
|
579
577
|
]);
|
|
580
578
|
|
|
581
|
-
const mockCreate =
|
|
579
|
+
const mockCreate = vi.fn(() => Promise.resolve(streamChunks));
|
|
582
580
|
|
|
583
|
-
const provider = createProvider();
|
|
581
|
+
const provider = await createProvider();
|
|
584
582
|
(provider as any).client = {
|
|
585
583
|
chat: { completions: { create: mockCreate } },
|
|
586
584
|
};
|
|
@@ -600,9 +598,9 @@ describe('OpenAICompatibleProvider.streamWithTools', () => {
|
|
|
600
598
|
});
|
|
601
599
|
|
|
602
600
|
test('fallback: when SDK stream creation throws, the error propagates', async () => {
|
|
603
|
-
const mockCreate =
|
|
601
|
+
const mockCreate = vi.fn(() => Promise.reject(new Error('Provider down')));
|
|
604
602
|
|
|
605
|
-
const provider = createProvider();
|
|
603
|
+
const provider = await createProvider();
|
|
606
604
|
(provider as any).client = {
|
|
607
605
|
chat: { completions: { create: mockCreate } },
|
|
608
606
|
};
|
|
@@ -655,9 +653,9 @@ describe('OpenAICompatibleProvider.streamWithTools', () => {
|
|
|
655
653
|
},
|
|
656
654
|
]);
|
|
657
655
|
|
|
658
|
-
const mockCreate =
|
|
656
|
+
const mockCreate = vi.fn(() => Promise.resolve(streamChunks));
|
|
659
657
|
|
|
660
|
-
const provider = createProvider();
|
|
658
|
+
const provider = await createProvider();
|
|
661
659
|
(provider as any).client = {
|
|
662
660
|
chat: { completions: { create: mockCreate } },
|
|
663
661
|
};
|
|
@@ -686,9 +684,9 @@ describe('OpenAICompatibleProvider.streamWithTools', () => {
|
|
|
686
684
|
},
|
|
687
685
|
]);
|
|
688
686
|
|
|
689
|
-
const mockCreate =
|
|
687
|
+
const mockCreate = vi.fn(() => Promise.resolve(streamChunks));
|
|
690
688
|
|
|
691
|
-
const provider = createProvider();
|
|
689
|
+
const provider = await createProvider();
|
|
692
690
|
(provider as any).client = {
|
|
693
691
|
chat: { completions: { create: mockCreate } },
|
|
694
692
|
};
|
|
@@ -709,9 +707,9 @@ describe('OpenAICompatibleProvider.streamWithTools', () => {
|
|
|
709
707
|
{ choices: [{ delta: { content: 'done' }, finish_reason: 'stop' }] },
|
|
710
708
|
]);
|
|
711
709
|
|
|
712
|
-
const mockCreate =
|
|
710
|
+
const mockCreate = vi.fn(() => Promise.resolve(streamChunks));
|
|
713
711
|
|
|
714
|
-
const provider = createProvider();
|
|
712
|
+
const provider = await createProvider();
|
|
715
713
|
(provider as any).client = {
|
|
716
714
|
chat: { completions: { create: mockCreate } },
|
|
717
715
|
};
|
|
@@ -730,3 +728,51 @@ describe('OpenAICompatibleProvider.streamWithTools', () => {
|
|
|
730
728
|
expect(createArg.max_tokens).toBe(1024);
|
|
731
729
|
});
|
|
732
730
|
});
|
|
731
|
+
|
|
732
|
+
// ---------------------------------------------------------------------------
|
|
733
|
+
// PERF-2c: Unbuffered streaming in LLMRouter fallback branch
|
|
734
|
+
// ---------------------------------------------------------------------------
|
|
735
|
+
|
|
736
|
+
describe('PERF-2c: LLMRouter unbuffered streaming', () => {
|
|
737
|
+
it('streamWithTools fallback loop no longer buffers chunks before yielding', async () => {
|
|
738
|
+
const { readFileSync } = await import('node:fs');
|
|
739
|
+
const { join } = await import('node:path');
|
|
740
|
+
const src = readFileSync(join(process.cwd(), 'src/llm/router.ts'), 'utf-8');
|
|
741
|
+
// Extract only the streamWithTools section (between the streamWithTools guards)
|
|
742
|
+
const swStart = src.indexOf('Use native streaming-with-tools if providers support it');
|
|
743
|
+
const swEnd = src.indexOf('If all providers with streamWithTools failed', swStart);
|
|
744
|
+
const swSection = swStart > 0 && swEnd > swStart ? src.slice(swStart, swEnd) : '';
|
|
745
|
+
// The streamWithTools fallback section should NOT contain buffering
|
|
746
|
+
expect(swSection).not.toContain('bufferedChunks');
|
|
747
|
+
// It should contain yield chunk directly
|
|
748
|
+
expect(swSection).toContain('yield chunk;');
|
|
749
|
+
});
|
|
750
|
+
|
|
751
|
+
it('router.ts yields each chunk immediately inside the for-await loop', async () => {
|
|
752
|
+
const { readFileSync } = await import('node:fs');
|
|
753
|
+
const { join } = await import('node:path');
|
|
754
|
+
const src = readFileSync(join(process.cwd(), 'src/llm/router.ts'), 'utf-8');
|
|
755
|
+
// The yield statement should appear inside the for-await (before circuitBreaker.recordSuccess)
|
|
756
|
+
expect(src).toContain('yield chunk;');
|
|
757
|
+
// circuitBreaker.recordSuccess comes after the loop ends (done chunk received)
|
|
758
|
+
expect(src).toContain('circuitBreaker.recordSuccess(p.name);');
|
|
759
|
+
});
|
|
760
|
+
|
|
761
|
+
it('circuitBreaker.recordSuccess is called after stream ends (not before yield)', async () => {
|
|
762
|
+
const { readFileSync } = await import('node:fs');
|
|
763
|
+
const { join } = await import('node:path');
|
|
764
|
+
const src = readFileSync(join(process.cwd(), 'src/llm/router.ts'), 'utf-8');
|
|
765
|
+
// Find the fallback for-loop block that has yield chunk
|
|
766
|
+
const yieldIdx = src.indexOf('yield chunk;');
|
|
767
|
+
const recordSuccessIdx = src.indexOf('circuitBreaker.recordSuccess(p.name);');
|
|
768
|
+
// recordSuccess should appear AFTER the yield in the source
|
|
769
|
+
expect(recordSuccessIdx).toBeGreaterThan(yieldIdx);
|
|
770
|
+
});
|
|
771
|
+
|
|
772
|
+
it('stream error path still calls circuitBreaker.recordFailure', async () => {
|
|
773
|
+
const { readFileSync } = await import('node:fs');
|
|
774
|
+
const { join } = await import('node:path');
|
|
775
|
+
const src = readFileSync(join(process.cwd(), 'src/llm/router.ts'), 'utf-8');
|
|
776
|
+
expect(src).toContain('circuitBreaker.recordFailure(p.name);');
|
|
777
|
+
});
|
|
778
|
+
});
|