@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
|
@@ -0,0 +1,718 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* DevOps Terminal Gap Fix Tests
|
|
3
|
+
*
|
|
4
|
+
* Validates source-level fixes for the gap fix plan:
|
|
5
|
+
* C1–C3 (critical), H1–H5 (high), M1–M3, L1 (medium/low)
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
import { describe, it, expect } from 'vitest';
|
|
9
|
+
import { readFileSync } from 'node:fs';
|
|
10
|
+
import { join } from 'node:path';
|
|
11
|
+
|
|
12
|
+
const ROOT = join(__dirname, '../..');
|
|
13
|
+
|
|
14
|
+
function src(rel: string): string {
|
|
15
|
+
return readFileSync(join(ROOT, rel), 'utf-8');
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
// ---------------------------------------------------------------------------
|
|
19
|
+
// C1 — template.ts uses SQLite, not REST
|
|
20
|
+
// ---------------------------------------------------------------------------
|
|
21
|
+
describe('C1 — template.ts SQLite migration', () => {
|
|
22
|
+
it('no longer calls localhost:3004', () => {
|
|
23
|
+
expect(src('src/commands/template.ts')).not.toContain('localhost:3004');
|
|
24
|
+
});
|
|
25
|
+
|
|
26
|
+
it('uses getDb from state/db', () => {
|
|
27
|
+
expect(src('src/commands/template.ts')).toContain('getDb');
|
|
28
|
+
});
|
|
29
|
+
|
|
30
|
+
it('uses randomUUID for new template IDs', () => {
|
|
31
|
+
expect(src('src/commands/template.ts')).toContain('randomUUID');
|
|
32
|
+
});
|
|
33
|
+
|
|
34
|
+
it('does not import RestClient', () => {
|
|
35
|
+
expect(src('src/commands/template.ts')).not.toContain('RestClient');
|
|
36
|
+
});
|
|
37
|
+
});
|
|
38
|
+
|
|
39
|
+
// ---------------------------------------------------------------------------
|
|
40
|
+
// C2 — questionnaire.ts no dead service check
|
|
41
|
+
// ---------------------------------------------------------------------------
|
|
42
|
+
describe('C2 — questionnaire.ts dead service check removed', () => {
|
|
43
|
+
it('does not call generatorClient.isAvailable', () => {
|
|
44
|
+
expect(src('src/commands/questionnaire.ts')).not.toContain('generatorClient.isAvailable');
|
|
45
|
+
});
|
|
46
|
+
|
|
47
|
+
it('does not define runWithGeneratorService', () => {
|
|
48
|
+
expect(src('src/commands/questionnaire.ts')).not.toContain('runWithGeneratorService');
|
|
49
|
+
});
|
|
50
|
+
});
|
|
51
|
+
|
|
52
|
+
// ---------------------------------------------------------------------------
|
|
53
|
+
// C3 — history/manager.ts no noisy warning
|
|
54
|
+
// ---------------------------------------------------------------------------
|
|
55
|
+
describe('C3 — history manager no noisy warning', () => {
|
|
56
|
+
it('does not log [nimbus] Failed to sync warning', () => {
|
|
57
|
+
expect(src('src/history/manager.ts')).not.toContain('[nimbus] Failed to sync');
|
|
58
|
+
});
|
|
59
|
+
|
|
60
|
+
it('does not have loadFromStateService method', () => {
|
|
61
|
+
expect(src('src/history/manager.ts')).not.toContain('loadFromStateService');
|
|
62
|
+
});
|
|
63
|
+
|
|
64
|
+
it('does not call syncToStateService', () => {
|
|
65
|
+
expect(src('src/history/manager.ts')).not.toContain('syncToStateService');
|
|
66
|
+
});
|
|
67
|
+
});
|
|
68
|
+
|
|
69
|
+
// ---------------------------------------------------------------------------
|
|
70
|
+
// H1 — run.ts discovers infra context
|
|
71
|
+
// ---------------------------------------------------------------------------
|
|
72
|
+
describe('H1 — run.ts infra context discovery', () => {
|
|
73
|
+
it('calls discoverInfraContext', () => {
|
|
74
|
+
expect(src('src/cli/run.ts')).toContain('discoverInfraContext');
|
|
75
|
+
});
|
|
76
|
+
|
|
77
|
+
it('passes infraContext to runAgentLoop', () => {
|
|
78
|
+
expect(src('src/cli/run.ts')).toContain('infraContext');
|
|
79
|
+
});
|
|
80
|
+
});
|
|
81
|
+
|
|
82
|
+
// ---------------------------------------------------------------------------
|
|
83
|
+
// H2 — app.ts warns about missing NIMBUS.md
|
|
84
|
+
// ---------------------------------------------------------------------------
|
|
85
|
+
describe('H2 — app.ts NIMBUS.md warning', () => {
|
|
86
|
+
it('checks for NIMBUS.md', () => {
|
|
87
|
+
expect(src('src/app.ts')).toContain('NIMBUS.md');
|
|
88
|
+
});
|
|
89
|
+
|
|
90
|
+
it('adds startup warning when NIMBUS.md is missing', () => {
|
|
91
|
+
expect(src('src/app.ts')).toContain('No NIMBUS.md found');
|
|
92
|
+
});
|
|
93
|
+
});
|
|
94
|
+
|
|
95
|
+
// ---------------------------------------------------------------------------
|
|
96
|
+
// H3 — loop.ts auto-discovers infra context
|
|
97
|
+
// ---------------------------------------------------------------------------
|
|
98
|
+
describe('H3 — loop.ts infra context auto-discovery', () => {
|
|
99
|
+
it('calls discoverInfraContext in agent loop', () => {
|
|
100
|
+
expect(src('src/agent/loop.ts')).toContain('discoverInfraContext');
|
|
101
|
+
});
|
|
102
|
+
});
|
|
103
|
+
|
|
104
|
+
// ---------------------------------------------------------------------------
|
|
105
|
+
// H4 — ToolCallDisplay streaming output visible
|
|
106
|
+
// ---------------------------------------------------------------------------
|
|
107
|
+
describe('H4 — ToolCallDisplay streaming output', () => {
|
|
108
|
+
it('has [LIVE] label prefix for streaming output', () => {
|
|
109
|
+
expect(src('src/ui/ToolCallDisplay.tsx')).toContain('[LIVE]');
|
|
110
|
+
});
|
|
111
|
+
|
|
112
|
+
it('does not use dimColor on streaming section outer box', () => {
|
|
113
|
+
// The new streaming section should not have dimColor wrapping the whole box
|
|
114
|
+
const content = src('src/ui/ToolCallDisplay.tsx');
|
|
115
|
+
// We check that the streaming indicator uses green bold instead of dimColor
|
|
116
|
+
expect(content).toContain('color="green" bold');
|
|
117
|
+
});
|
|
118
|
+
});
|
|
119
|
+
|
|
120
|
+
// ---------------------------------------------------------------------------
|
|
121
|
+
// H5 — App.tsx has ? key handler
|
|
122
|
+
// ---------------------------------------------------------------------------
|
|
123
|
+
describe('H5 — App.tsx ? key handler', () => {
|
|
124
|
+
it('handles ? keypress to open help', () => {
|
|
125
|
+
expect(src('src/ui/App.tsx')).toContain("input === '?'");
|
|
126
|
+
});
|
|
127
|
+
});
|
|
128
|
+
|
|
129
|
+
// ---------------------------------------------------------------------------
|
|
130
|
+
// M1 — status.ts helm + pod health
|
|
131
|
+
// ---------------------------------------------------------------------------
|
|
132
|
+
describe('M1 — status.ts helm and pod health', () => {
|
|
133
|
+
it('checks helm release health', () => {
|
|
134
|
+
// Uses run('helm', ['list', '-A', ...])
|
|
135
|
+
expect(src('src/commands/status.ts')).toContain("'helm'");
|
|
136
|
+
expect(src('src/commands/status.ts')).toContain("'list'");
|
|
137
|
+
});
|
|
138
|
+
|
|
139
|
+
it('checks unhealthy pod count', () => {
|
|
140
|
+
expect(src('src/commands/status.ts')).toContain('unhealthyPodCount');
|
|
141
|
+
});
|
|
142
|
+
|
|
143
|
+
it('reports helm failed releases', () => {
|
|
144
|
+
expect(src('src/commands/status.ts')).toContain('helmFailedCount');
|
|
145
|
+
});
|
|
146
|
+
});
|
|
147
|
+
|
|
148
|
+
// ---------------------------------------------------------------------------
|
|
149
|
+
// M3 — ProjectConfig in types
|
|
150
|
+
// ---------------------------------------------------------------------------
|
|
151
|
+
describe('M3 — ProjectConfig in config/types.ts', () => {
|
|
152
|
+
it('exports ProjectConfig interface', () => {
|
|
153
|
+
expect(src('src/config/types.ts')).toContain('ProjectConfig');
|
|
154
|
+
});
|
|
155
|
+
|
|
156
|
+
it('exports loadProjectConfig function', () => {
|
|
157
|
+
expect(src('src/config/types.ts')).toContain('loadProjectConfig');
|
|
158
|
+
});
|
|
159
|
+
|
|
160
|
+
it('supports protectedEnvironments field', () => {
|
|
161
|
+
expect(src('src/config/types.ts')).toContain('protectedEnvironments');
|
|
162
|
+
});
|
|
163
|
+
});
|
|
164
|
+
|
|
165
|
+
// ---------------------------------------------------------------------------
|
|
166
|
+
// L1 — StatusBar has ? hint
|
|
167
|
+
// ---------------------------------------------------------------------------
|
|
168
|
+
describe('L1 — StatusBar ? help hint', () => {
|
|
169
|
+
it('shows ? help in StatusBar', () => {
|
|
170
|
+
expect(src('src/ui/StatusBar.tsx')).toContain('? help');
|
|
171
|
+
});
|
|
172
|
+
});
|
|
173
|
+
|
|
174
|
+
// ---------------------------------------------------------------------------
|
|
175
|
+
// L2 — completions.ts dynamic completions
|
|
176
|
+
// ---------------------------------------------------------------------------
|
|
177
|
+
describe('L2 — completions.ts dynamic completions', () => {
|
|
178
|
+
it('exports dynamicComplete function', () => {
|
|
179
|
+
expect(src('src/commands/completions.ts')).toContain('dynamicComplete');
|
|
180
|
+
});
|
|
181
|
+
|
|
182
|
+
it('gets terraform workspaces for completion', () => {
|
|
183
|
+
expect(src('src/commands/completions.ts')).toContain('workspace');
|
|
184
|
+
expect(src('src/commands/completions.ts')).toContain("'list'");
|
|
185
|
+
});
|
|
186
|
+
|
|
187
|
+
it('gets kubectl contexts for completion', () => {
|
|
188
|
+
expect(src('src/commands/completions.ts')).toContain('get-contexts');
|
|
189
|
+
});
|
|
190
|
+
|
|
191
|
+
it('gets helm releases for completion', () => {
|
|
192
|
+
expect(src('src/commands/completions.ts')).toContain('--short');
|
|
193
|
+
});
|
|
194
|
+
});
|
|
195
|
+
|
|
196
|
+
// ---------------------------------------------------------------------------
|
|
197
|
+
// M4 — init.ts generates CI/CD section in NIMBUS.md
|
|
198
|
+
// ---------------------------------------------------------------------------
|
|
199
|
+
describe('M4 — init.ts CI/CD section in NIMBUS.md', () => {
|
|
200
|
+
it('generates ## CI/CD section when pipeline detected', () => {
|
|
201
|
+
expect(src('src/cli/init.ts')).toContain("## CI/CD");
|
|
202
|
+
});
|
|
203
|
+
|
|
204
|
+
it('includes pipeline conventions', () => {
|
|
205
|
+
expect(src('src/cli/init.ts')).toContain('terraform plan');
|
|
206
|
+
});
|
|
207
|
+
});
|
|
208
|
+
|
|
209
|
+
// ---------------------------------------------------------------------------
|
|
210
|
+
// M5 — TUI watcher notifies on DevOps file changes
|
|
211
|
+
// ---------------------------------------------------------------------------
|
|
212
|
+
describe('M5 — TUI FileWatcher DevOps notifications', () => {
|
|
213
|
+
it('debounces DevOps file changes', () => {
|
|
214
|
+
expect(src('src/ui/ink/index.ts')).toContain('devopsChangeDebounce');
|
|
215
|
+
});
|
|
216
|
+
|
|
217
|
+
it('matches .tf, .yaml and Dockerfile extensions', () => {
|
|
218
|
+
expect(src('src/ui/ink/index.ts')).toContain('isDevOps');
|
|
219
|
+
});
|
|
220
|
+
|
|
221
|
+
it('prompts user to /plan on file change', () => {
|
|
222
|
+
expect(src('src/ui/ink/index.ts')).toContain('type ${hint} to review drift impact');
|
|
223
|
+
});
|
|
224
|
+
});
|
|
225
|
+
|
|
226
|
+
// ---------------------------------------------------------------------------
|
|
227
|
+
// M2 — doctor.ts LLM connectivity check
|
|
228
|
+
// ---------------------------------------------------------------------------
|
|
229
|
+
describe('M2 — doctor.ts LLM connectivity', () => {
|
|
230
|
+
it('has checkLLMConnectivity function', () => {
|
|
231
|
+
expect(src('src/commands/doctor.ts')).toContain('checkLLMConnectivity');
|
|
232
|
+
});
|
|
233
|
+
|
|
234
|
+
it('includes LLM Connectivity in diagnostic checks', () => {
|
|
235
|
+
expect(src('src/commands/doctor.ts')).toContain("'LLM Connectivity'");
|
|
236
|
+
});
|
|
237
|
+
});
|
|
238
|
+
|
|
239
|
+
// ---------------------------------------------------------------------------
|
|
240
|
+
// DevOps Identity — help, welcome splash, header, onboarding
|
|
241
|
+
// ---------------------------------------------------------------------------
|
|
242
|
+
describe('DevOps Identity — help command reorganized', () => {
|
|
243
|
+
it('puts DevOps Operations section first', () => {
|
|
244
|
+
const content = src('src/commands/help.ts');
|
|
245
|
+
const devopsIdx = content.indexOf('DevOps Operations:');
|
|
246
|
+
const chatIdx = content.indexOf('AI Agent:');
|
|
247
|
+
expect(devopsIdx).toBeGreaterThan(-1);
|
|
248
|
+
expect(chatIdx).toBeGreaterThan(-1);
|
|
249
|
+
// DevOps section must appear before Chat/AI section
|
|
250
|
+
expect(devopsIdx).toBeLessThan(chatIdx);
|
|
251
|
+
});
|
|
252
|
+
|
|
253
|
+
it('lists terraform, k8s, helm, drift, cost in primary section', () => {
|
|
254
|
+
const content = src('src/commands/help.ts');
|
|
255
|
+
const devopsSection = content.slice(content.indexOf('DevOps Operations:'), content.indexOf('Incident & Automation:'));
|
|
256
|
+
expect(devopsSection).toContain('tf <cmd>');
|
|
257
|
+
expect(devopsSection).toContain('k8s <cmd>');
|
|
258
|
+
expect(devopsSection).toContain('helm <cmd>');
|
|
259
|
+
expect(devopsSection).toContain('drift');
|
|
260
|
+
expect(devopsSection).toContain('cost');
|
|
261
|
+
});
|
|
262
|
+
|
|
263
|
+
it('has Incident & Automation section', () => {
|
|
264
|
+
expect(src('src/commands/help.ts')).toContain('Incident & Automation:');
|
|
265
|
+
});
|
|
266
|
+
|
|
267
|
+
it('has DevOps-first header title', () => {
|
|
268
|
+
expect(src('src/commands/help.ts')).toContain('AI-Powered DevOps Terminal');
|
|
269
|
+
});
|
|
270
|
+
});
|
|
271
|
+
|
|
272
|
+
describe('DevOps Identity — welcome splash in MessageList', () => {
|
|
273
|
+
it('shows DevOps quick-start examples on empty screen', () => {
|
|
274
|
+
expect(src('src/ui/MessageList.tsx')).toContain('What would you like to do?');
|
|
275
|
+
});
|
|
276
|
+
|
|
277
|
+
it('includes terraform plan example', () => {
|
|
278
|
+
expect(src('src/ui/MessageList.tsx')).toContain('terraform plan');
|
|
279
|
+
});
|
|
280
|
+
|
|
281
|
+
it('includes kubernetes example', () => {
|
|
282
|
+
expect(src('src/ui/MessageList.tsx')).toContain('pod restart');
|
|
283
|
+
});
|
|
284
|
+
|
|
285
|
+
it('includes help hint', () => {
|
|
286
|
+
expect(src('src/ui/MessageList.tsx')).toContain('/init to set up context');
|
|
287
|
+
});
|
|
288
|
+
});
|
|
289
|
+
|
|
290
|
+
describe('DevOps Identity — Header production warning', () => {
|
|
291
|
+
it('shows PROD badge for production environments', () => {
|
|
292
|
+
expect(src('src/ui/Header.tsx')).toContain('[PROD]');
|
|
293
|
+
});
|
|
294
|
+
|
|
295
|
+
it('shows deploy mode production warning banner', () => {
|
|
296
|
+
expect(src('src/ui/Header.tsx')).toContain('DEPLOY MODE — targeting production');
|
|
297
|
+
});
|
|
298
|
+
|
|
299
|
+
it('showProdWarning requires both deploy mode and prod env', () => {
|
|
300
|
+
const content = src('src/ui/Header.tsx');
|
|
301
|
+
expect(content).toContain("session.mode === 'deploy'");
|
|
302
|
+
expect(content).toContain('showProdWarning');
|
|
303
|
+
});
|
|
304
|
+
});
|
|
305
|
+
|
|
306
|
+
describe('DevOps Identity — onboarding infra detection', () => {
|
|
307
|
+
it('detects terraform files during onboarding', () => {
|
|
308
|
+
expect(src('src/commands/onboarding.ts')).toContain('.tf');
|
|
309
|
+
});
|
|
310
|
+
|
|
311
|
+
it('shows detected infrastructure stack to user', () => {
|
|
312
|
+
expect(src('src/commands/onboarding.ts')).toContain('Detected infrastructure:');
|
|
313
|
+
});
|
|
314
|
+
|
|
315
|
+
it('has DevOps-first welcome banner', () => {
|
|
316
|
+
expect(src('src/commands/onboarding.ts')).toContain('AI-Powered DevOps Terminal');
|
|
317
|
+
});
|
|
318
|
+
});
|
|
319
|
+
|
|
320
|
+
// ---------------------------------------------------------------------------
|
|
321
|
+
// Polish Plan — H1-H5, M1-M7, L1-L5
|
|
322
|
+
// ---------------------------------------------------------------------------
|
|
323
|
+
|
|
324
|
+
describe('H1 — InputBox DevOps placeholder', () => {
|
|
325
|
+
it('shows DevOps-specific placeholder text', () => {
|
|
326
|
+
expect(src('src/ui/InputBox.tsx')).toContain('Type a DevOps command');
|
|
327
|
+
});
|
|
328
|
+
});
|
|
329
|
+
|
|
330
|
+
describe('H2 — /context shows infra + token breakdown', () => {
|
|
331
|
+
it('shows LLM Model in context snapshot', () => {
|
|
332
|
+
expect(src('src/ui/App.tsx')).toContain('LLM Model:');
|
|
333
|
+
});
|
|
334
|
+
|
|
335
|
+
it('shows TF Workspace and K8s Context', () => {
|
|
336
|
+
const content = src('src/ui/App.tsx');
|
|
337
|
+
expect(content).toContain('TF Workspace:');
|
|
338
|
+
expect(content).toContain('K8s Context:');
|
|
339
|
+
});
|
|
340
|
+
});
|
|
341
|
+
|
|
342
|
+
describe('H3 — Deploy mode confirmation state', () => {
|
|
343
|
+
it('has pendingDeployConfirm state', () => {
|
|
344
|
+
expect(src('src/ui/App.tsx')).toContain('pendingDeployConfirm');
|
|
345
|
+
});
|
|
346
|
+
|
|
347
|
+
it('shows confirmation box with destructive operations warning', () => {
|
|
348
|
+
const content = src('src/ui/App.tsx');
|
|
349
|
+
expect(content).toContain('!! Switch to DEPLOY mode?');
|
|
350
|
+
expect(content).toContain('terraform apply/destroy, kubectl delete, helm uninstall');
|
|
351
|
+
});
|
|
352
|
+
});
|
|
353
|
+
|
|
354
|
+
describe('H4 — doctor DevOps CLI version check', () => {
|
|
355
|
+
it('has checkDevOpsCLIs function', () => {
|
|
356
|
+
expect(src('src/commands/doctor.ts')).toContain('checkDevOpsCLIs');
|
|
357
|
+
});
|
|
358
|
+
|
|
359
|
+
it('includes DevOps CLIs in diagnostic checks', () => {
|
|
360
|
+
expect(src('src/commands/doctor.ts')).toContain("'DevOps CLIs'");
|
|
361
|
+
});
|
|
362
|
+
});
|
|
363
|
+
|
|
364
|
+
describe('H5 — cost delta hint after terraform apply / helm upgrade', () => {
|
|
365
|
+
it('has extractCostHintFromToolOutput function', () => {
|
|
366
|
+
expect(src('src/agent/loop.ts')).toContain('extractCostHintFromToolOutput');
|
|
367
|
+
});
|
|
368
|
+
|
|
369
|
+
it('parses terraform apply resource counts', () => {
|
|
370
|
+
expect(src('src/agent/loop.ts')).toContain('resources created');
|
|
371
|
+
});
|
|
372
|
+
|
|
373
|
+
it('emits cost hint for helm releases', () => {
|
|
374
|
+
expect(src('src/agent/loop.ts')).toContain('nimbus cost');
|
|
375
|
+
});
|
|
376
|
+
});
|
|
377
|
+
|
|
378
|
+
describe('M1 — run.ts JSON output infraContext + toolsUsed', () => {
|
|
379
|
+
it('has infraContext field in RunJsonOutput', () => {
|
|
380
|
+
expect(src('src/cli/run.ts')).toContain('infraContext');
|
|
381
|
+
});
|
|
382
|
+
|
|
383
|
+
it('has toolsUsed field in RunJsonOutput', () => {
|
|
384
|
+
expect(src('src/cli/run.ts')).toContain('toolsUsed');
|
|
385
|
+
});
|
|
386
|
+
});
|
|
387
|
+
|
|
388
|
+
describe('M2 — streaming output coloring', () => {
|
|
389
|
+
it('colors lines containing "will be created"', () => {
|
|
390
|
+
expect(src('src/ui/ToolCallDisplay.tsx')).toContain('will be created');
|
|
391
|
+
});
|
|
392
|
+
|
|
393
|
+
it('applies green color for created lines', () => {
|
|
394
|
+
expect(src('src/ui/ToolCallDisplay.tsx')).toContain("'green'");
|
|
395
|
+
});
|
|
396
|
+
});
|
|
397
|
+
|
|
398
|
+
describe('M3 — /profile slash command in TUI', () => {
|
|
399
|
+
it('handles /profile command', () => {
|
|
400
|
+
expect(src('src/ui/App.tsx')).toContain("'/profile '");
|
|
401
|
+
});
|
|
402
|
+
|
|
403
|
+
it('calls profileCommand', () => {
|
|
404
|
+
expect(src('src/ui/App.tsx')).toContain('profileCommand');
|
|
405
|
+
});
|
|
406
|
+
});
|
|
407
|
+
|
|
408
|
+
describe('M4 — recurring error persistence', () => {
|
|
409
|
+
it('has trackAndPersistError function', () => {
|
|
410
|
+
expect(src('src/agent/loop.ts')).toContain('trackAndPersistError');
|
|
411
|
+
});
|
|
412
|
+
|
|
413
|
+
it('has sessionErrorCounts map', () => {
|
|
414
|
+
expect(src('src/agent/loop.ts')).toContain('sessionErrorCounts');
|
|
415
|
+
});
|
|
416
|
+
|
|
417
|
+
it('appends to Observed Issues section', () => {
|
|
418
|
+
expect(src('src/agent/loop.ts')).toContain('Observed Issues');
|
|
419
|
+
});
|
|
420
|
+
});
|
|
421
|
+
|
|
422
|
+
describe('M5 — nimbus init next steps', () => {
|
|
423
|
+
it('prints Next steps after init', () => {
|
|
424
|
+
expect(src('src/cli/init.ts')).toContain('Next steps:');
|
|
425
|
+
});
|
|
426
|
+
|
|
427
|
+
it('mentions nimbus plan in next steps', () => {
|
|
428
|
+
expect(src('src/cli/init.ts')).toContain('nimbus plan');
|
|
429
|
+
});
|
|
430
|
+
|
|
431
|
+
it('mentions nimbus doctor in next steps', () => {
|
|
432
|
+
expect(src('src/cli/init.ts')).toContain('nimbus doctor');
|
|
433
|
+
});
|
|
434
|
+
});
|
|
435
|
+
|
|
436
|
+
describe('M6 — destructive action guard', () => {
|
|
437
|
+
it('has isDestructiveAction function', () => {
|
|
438
|
+
expect(src('src/agent/loop.ts')).toContain('isDestructiveAction');
|
|
439
|
+
});
|
|
440
|
+
|
|
441
|
+
it('warns on terraform destroy', () => {
|
|
442
|
+
expect(src('src/agent/loop.ts')).toContain('PERMANENTLY DELETE all managed infrastructure');
|
|
443
|
+
});
|
|
444
|
+
|
|
445
|
+
it('warns on kubectl delete', () => {
|
|
446
|
+
expect(src('src/agent/loop.ts')).toContain('IRREVERSIBLE');
|
|
447
|
+
});
|
|
448
|
+
});
|
|
449
|
+
|
|
450
|
+
describe('M7 — completions install in postinstall', () => {
|
|
451
|
+
it('mentions completions install in postinstall', () => {
|
|
452
|
+
expect(src('package.json')).toContain('completions install');
|
|
453
|
+
});
|
|
454
|
+
|
|
455
|
+
it('has 4 steps in postinstall', () => {
|
|
456
|
+
const content = src('package.json');
|
|
457
|
+
expect(content).toContain('1.');
|
|
458
|
+
expect(content).toContain('2.');
|
|
459
|
+
expect(content).toContain('3.');
|
|
460
|
+
expect(content).toContain('4.');
|
|
461
|
+
});
|
|
462
|
+
});
|
|
463
|
+
|
|
464
|
+
describe('L1 — @agent completions in InputBox', () => {
|
|
465
|
+
it('has AGENT_NAMES constant', () => {
|
|
466
|
+
expect(src('src/ui/InputBox.tsx')).toContain('AGENT_NAMES');
|
|
467
|
+
});
|
|
468
|
+
|
|
469
|
+
it('includes known agent names', () => {
|
|
470
|
+
const content = src('src/ui/InputBox.tsx');
|
|
471
|
+
expect(content).toContain('explore');
|
|
472
|
+
expect(content).toContain('infra');
|
|
473
|
+
expect(content).toContain('security');
|
|
474
|
+
});
|
|
475
|
+
});
|
|
476
|
+
|
|
477
|
+
describe('L2 — status shows LLM model', () => {
|
|
478
|
+
it('uses info.model in status output', () => {
|
|
479
|
+
expect(src('src/commands/status.ts')).toContain('info.model');
|
|
480
|
+
});
|
|
481
|
+
|
|
482
|
+
it('shows model in status dashboard', () => {
|
|
483
|
+
expect(src('src/commands/status.ts')).toContain('Model:');
|
|
484
|
+
});
|
|
485
|
+
});
|
|
486
|
+
|
|
487
|
+
describe('L3 — nimbus context command', () => {
|
|
488
|
+
it('handles context command in cli.ts', () => {
|
|
489
|
+
expect(src('src/cli.ts')).toContain("command === 'context'");
|
|
490
|
+
});
|
|
491
|
+
|
|
492
|
+
it('StatusOptions has verbose field', () => {
|
|
493
|
+
expect(src('src/commands/status.ts')).toContain('verbose');
|
|
494
|
+
});
|
|
495
|
+
});
|
|
496
|
+
|
|
497
|
+
describe('L4 — session resume hint in welcome message', () => {
|
|
498
|
+
it('shows Previous session available hint', () => {
|
|
499
|
+
expect(src('src/ui/ink/index.ts')).toContain('Previous session available');
|
|
500
|
+
});
|
|
501
|
+
|
|
502
|
+
it('mentions /sessions command in resume hint', () => {
|
|
503
|
+
expect(src('src/ui/ink/index.ts')).toContain('type /sessions to resume');
|
|
504
|
+
});
|
|
505
|
+
});
|
|
506
|
+
|
|
507
|
+
// ---------------------------------------------------------------------------
|
|
508
|
+
// Polish Plan — Additional completeness checks
|
|
509
|
+
// ---------------------------------------------------------------------------
|
|
510
|
+
|
|
511
|
+
describe('H3 — deploy confirmation key handler', () => {
|
|
512
|
+
it('handles y key to confirm deploy', () => {
|
|
513
|
+
const content = src('src/ui/App.tsx');
|
|
514
|
+
expect(content).toContain("input === 'y'");
|
|
515
|
+
expect(content).toContain("input === 'Y'");
|
|
516
|
+
});
|
|
517
|
+
|
|
518
|
+
it('handles n key to cancel deploy', () => {
|
|
519
|
+
const content = src('src/ui/App.tsx');
|
|
520
|
+
expect(content).toContain("input === 'n'");
|
|
521
|
+
});
|
|
522
|
+
|
|
523
|
+
it('uses isActive with pendingDeployConfirm', () => {
|
|
524
|
+
expect(src('src/ui/App.tsx')).toContain('isActive: pendingDeployConfirm');
|
|
525
|
+
});
|
|
526
|
+
});
|
|
527
|
+
|
|
528
|
+
describe('M6 — destructive guard injected before tool', () => {
|
|
529
|
+
it('injects SAFETY message into tool context', () => {
|
|
530
|
+
expect(src('src/agent/loop.ts')).toContain('[SAFETY]');
|
|
531
|
+
});
|
|
532
|
+
|
|
533
|
+
it('warns on helm uninstall', () => {
|
|
534
|
+
expect(src('src/agent/loop.ts')).toContain('helm uninstall will remove the release');
|
|
535
|
+
});
|
|
536
|
+
});
|
|
537
|
+
|
|
538
|
+
describe('L1 — @agent Tab completion suggestions', () => {
|
|
539
|
+
it('maps agent names to @name format for suggestions', () => {
|
|
540
|
+
expect(src('src/ui/InputBox.tsx')).toContain('@${a} ');
|
|
541
|
+
});
|
|
542
|
+
});
|
|
543
|
+
|
|
544
|
+
describe('H5 — cost hint integration in loop', () => {
|
|
545
|
+
it('calls extractCostHintFromToolOutput after tool completes', () => {
|
|
546
|
+
const content = src('src/agent/loop.ts');
|
|
547
|
+
expect(content).toContain('const costHint = extractCostHintFromToolOutput');
|
|
548
|
+
});
|
|
549
|
+
});
|
|
550
|
+
|
|
551
|
+
describe('M6/M7/L2/L6 gaps', () => {
|
|
552
|
+
it('M6: subagent tag parsing in MessageList', () => {
|
|
553
|
+
const src_content = readFileSync(join(ROOT, 'src/ui/MessageList.tsx'), 'utf-8');
|
|
554
|
+
expect(src_content).toContain('[subagent:');
|
|
555
|
+
expect(src_content).toContain('parseSubagentTag');
|
|
556
|
+
});
|
|
557
|
+
it('M7: /explain in SLASH_COMMANDS', () => {
|
|
558
|
+
const src_content = readFileSync(join(ROOT, 'src/ui/InputBox.tsx'), 'utf-8');
|
|
559
|
+
expect(src_content).toContain("'/explain'");
|
|
560
|
+
});
|
|
561
|
+
it('M7: /explain handler in App.tsx', () => {
|
|
562
|
+
const src_content = readFileSync(join(ROOT, 'src/ui/App.tsx'), 'utf-8');
|
|
563
|
+
expect(src_content).toContain('/explain');
|
|
564
|
+
});
|
|
565
|
+
it('L2: Ctrl+Z maps to undo', () => {
|
|
566
|
+
const src_content = readFileSync(join(ROOT, 'src/ui/App.tsx'), 'utf-8');
|
|
567
|
+
expect(src_content).toContain("'z' && key.ctrl");
|
|
568
|
+
});
|
|
569
|
+
it('L6: runbook auto-generation after terraform apply', () => {
|
|
570
|
+
const src_content = readFileSync(join(ROOT, 'src/agent/loop.ts'), 'utf-8');
|
|
571
|
+
expect(src_content).toContain('runbook');
|
|
572
|
+
});
|
|
573
|
+
});
|
|
574
|
+
|
|
575
|
+
// ---------------------------------------------------------------------------
|
|
576
|
+
// New gaps: C1 history, C4 cmd, C5 SQLite, H7 Node version, L5 progress bar, M5 watch, M3 branch, L8 monorepo
|
|
577
|
+
// ---------------------------------------------------------------------------
|
|
578
|
+
|
|
579
|
+
describe('C1 — persistent input history', () => {
|
|
580
|
+
it('InputBox.tsx loads history from disk', () => {
|
|
581
|
+
const content = src('src/ui/InputBox.tsx');
|
|
582
|
+
expect(content).toContain('loadHistory');
|
|
583
|
+
expect(content).toContain('saveHistory');
|
|
584
|
+
expect(content).toContain('input-history.json');
|
|
585
|
+
});
|
|
586
|
+
it('history is saved on submit', () => {
|
|
587
|
+
const content = src('src/ui/InputBox.tsx');
|
|
588
|
+
expect(content).toContain('saveHistory(h)');
|
|
589
|
+
});
|
|
590
|
+
it('history is saved on unmount via useEffect', () => {
|
|
591
|
+
const content = src('src/ui/InputBox.tsx');
|
|
592
|
+
expect(content).toContain('return () => {');
|
|
593
|
+
expect(content).toContain('saveHistory(history.current)');
|
|
594
|
+
});
|
|
595
|
+
});
|
|
596
|
+
|
|
597
|
+
describe('C4 — Windows cmd Node preference', () => {
|
|
598
|
+
it('bin/nimbus.cmd uses Node as primary runtime', () => {
|
|
599
|
+
const content = src('bin/nimbus.cmd');
|
|
600
|
+
expect(content).not.toContain('Prefer Bun');
|
|
601
|
+
expect(content).toContain('Node.js >= 18');
|
|
602
|
+
});
|
|
603
|
+
it('bin/nimbus.cmd removes Bun-first preference', () => {
|
|
604
|
+
const content = src('bin/nimbus.cmd');
|
|
605
|
+
// Bun check should not appear before node check
|
|
606
|
+
const nodeIdx = content.indexOf('where node');
|
|
607
|
+
const bunIdx = content.indexOf('where bun');
|
|
608
|
+
// bun should not appear before node (or not appear at all)
|
|
609
|
+
expect(nodeIdx).toBeGreaterThan(-1);
|
|
610
|
+
expect(bunIdx === -1 || bunIdx > nodeIdx).toBe(true);
|
|
611
|
+
});
|
|
612
|
+
});
|
|
613
|
+
|
|
614
|
+
describe('C5 — SQLite failure visibility', () => {
|
|
615
|
+
it('session error is pushed to _startupWarnings for TUI display', () => {
|
|
616
|
+
const content = src('src/ui/ink/index.ts');
|
|
617
|
+
expect(content).toContain('_startupWarnings.push');
|
|
618
|
+
expect(content).toContain('Session persistence unavailable');
|
|
619
|
+
});
|
|
620
|
+
});
|
|
621
|
+
|
|
622
|
+
describe('H7 — doctor Node version check', () => {
|
|
623
|
+
it('checkNodeRuntime function exists in doctor.ts', () => {
|
|
624
|
+
const content = src('src/commands/doctor.ts');
|
|
625
|
+
expect(content).toContain('checkNodeRuntime');
|
|
626
|
+
});
|
|
627
|
+
it('checks node version >= 18', () => {
|
|
628
|
+
const content = src('src/commands/doctor.ts');
|
|
629
|
+
expect(content).toContain('major < 18');
|
|
630
|
+
});
|
|
631
|
+
it('Node.js Runtime check in DIAGNOSTIC_CHECKS', () => {
|
|
632
|
+
const content = src('src/commands/doctor.ts');
|
|
633
|
+
expect(content).toContain("'Node.js Runtime'");
|
|
634
|
+
});
|
|
635
|
+
it('checks tsx availability', () => {
|
|
636
|
+
const content = src('src/commands/doctor.ts');
|
|
637
|
+
expect(content).toContain('tsx');
|
|
638
|
+
});
|
|
639
|
+
});
|
|
640
|
+
|
|
641
|
+
describe('H1 — infra context persisted on exit', () => {
|
|
642
|
+
it('exit handler writes infraStatePath on exit', () => {
|
|
643
|
+
const content = src('src/ui/ink/index.ts');
|
|
644
|
+
expect(content).toContain('currentInfraContext');
|
|
645
|
+
expect(content).toContain('infraStatePath');
|
|
646
|
+
// Should save in exit handler
|
|
647
|
+
const exitIdx = content.indexOf("process.on('exit'");
|
|
648
|
+
const lastExitIdx = content.lastIndexOf("process.on('exit'");
|
|
649
|
+
expect(lastExitIdx).toBeGreaterThan(exitIdx); // Multiple exit handlers
|
|
650
|
+
});
|
|
651
|
+
});
|
|
652
|
+
|
|
653
|
+
describe('L5 — visual context budget progress bar', () => {
|
|
654
|
+
it('StatusBar renders a progress bar with filled/empty chars', () => {
|
|
655
|
+
const content = src('src/ui/StatusBar.tsx');
|
|
656
|
+
expect(content).toContain('progressBar');
|
|
657
|
+
expect(content).toContain('█');
|
|
658
|
+
expect(content).toContain('░');
|
|
659
|
+
expect(content).toContain('BAR_WIDTH');
|
|
660
|
+
});
|
|
661
|
+
});
|
|
662
|
+
|
|
663
|
+
describe('M5 — /watch TUI integration', () => {
|
|
664
|
+
it('/watch handler exists in App.tsx', () => {
|
|
665
|
+
const content = src('src/ui/App.tsx');
|
|
666
|
+
expect(content).toContain("'/watch'");
|
|
667
|
+
expect(content).toContain('watchPattern');
|
|
668
|
+
expect(content).toContain('watchAbortRef');
|
|
669
|
+
});
|
|
670
|
+
it('/watch in InputBox SLASH_COMMANDS', () => {
|
|
671
|
+
const content = src('src/ui/InputBox.tsx');
|
|
672
|
+
expect(content).toContain("'/watch'");
|
|
673
|
+
});
|
|
674
|
+
});
|
|
675
|
+
|
|
676
|
+
describe('M3 — conversation branching /branch', () => {
|
|
677
|
+
it('/branch handler saves conversation checkpoint', () => {
|
|
678
|
+
const content = src('src/ui/App.tsx');
|
|
679
|
+
expect(content).toContain("'/branch'");
|
|
680
|
+
expect(content).toContain('branches');
|
|
681
|
+
expect(content).toContain('branchName');
|
|
682
|
+
});
|
|
683
|
+
it('/branch in InputBox SLASH_COMMANDS', () => {
|
|
684
|
+
const content = src('src/ui/InputBox.tsx');
|
|
685
|
+
expect(content).toContain("'/branch'");
|
|
686
|
+
});
|
|
687
|
+
});
|
|
688
|
+
|
|
689
|
+
describe('M2 — keyboard context switcher hint in Header', () => {
|
|
690
|
+
it('Header shows /k8s-ctx | /tf-ws hint when context is active', () => {
|
|
691
|
+
const content = src('src/ui/Header.tsx');
|
|
692
|
+
expect(content).toContain('/k8s-ctx | /tf-ws to switch');
|
|
693
|
+
});
|
|
694
|
+
});
|
|
695
|
+
|
|
696
|
+
describe('L8 — monorepo-aware nimbus init', () => {
|
|
697
|
+
it('init.ts scans subdirs for terraform roots', () => {
|
|
698
|
+
const content = src('src/cli/init.ts');
|
|
699
|
+
expect(content).toContain('tfRoots');
|
|
700
|
+
expect(content).toContain('Terraform Modules (Monorepo)');
|
|
701
|
+
});
|
|
702
|
+
});
|
|
703
|
+
|
|
704
|
+
describe('H5 — Homebrew formula version', () => {
|
|
705
|
+
it('Homebrew formula is updated to 0.3.0 (in sibling repo)', () => {
|
|
706
|
+
const { readFileSync: rfs, existsSync: exists } = require('node:fs');
|
|
707
|
+
const { join: j } = require('node:path');
|
|
708
|
+
// The Homebrew tap lives in a sibling repository
|
|
709
|
+
const formulaPath = j(__dirname, '../../../homebrew-tap/Formula/nimbus.rb');
|
|
710
|
+
if (!exists(formulaPath)) {
|
|
711
|
+
// Skip gracefully if the tap repo isn't checked out alongside
|
|
712
|
+
return;
|
|
713
|
+
}
|
|
714
|
+
const formula = rfs(formulaPath, 'utf-8') as string;
|
|
715
|
+
expect(formula).toContain('0.4.0');
|
|
716
|
+
expect(formula).not.toContain("version \"0.2.0\"");
|
|
717
|
+
});
|
|
718
|
+
});
|