@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,263 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Fix Command
|
|
3
|
+
*
|
|
4
|
+
* AI-assisted error fixing
|
|
5
|
+
*
|
|
6
|
+
* Usage: nimbus fix <error-or-file> [options]
|
|
7
|
+
*/
|
|
8
|
+
import { logger } from '../utils';
|
|
9
|
+
import { ui, confirm } from '../wizard';
|
|
10
|
+
import { llmClient } from '../clients';
|
|
11
|
+
/**
|
|
12
|
+
* Parse fix response from AI
|
|
13
|
+
*/
|
|
14
|
+
function parseFixResponse(response) {
|
|
15
|
+
// Try to extract structured sections from the response
|
|
16
|
+
const problemMatch = response.match(/(?:problem|issue|error):\s*(.+?)(?=\n(?:explanation|fix|solution)|$)/is);
|
|
17
|
+
const explanationMatch = response.match(/(?:explanation|cause|reason):\s*(.+?)(?=\n(?:fix|solution)|$)/is);
|
|
18
|
+
const fixMatch = response.match(/(?:fix|solution|resolution):\s*(.+?)(?=\n(?:original|fixed)|$)/is);
|
|
19
|
+
const originalMatch = response.match(/(?:original|before)[^:]*:\s*```[\w]*\n([\s\S]*?)```/i);
|
|
20
|
+
const fixedMatch = response.match(/(?:fixed|after|corrected)[^:]*:\s*```[\w]*\n([\s\S]*?)```/i);
|
|
21
|
+
return {
|
|
22
|
+
problem: problemMatch?.[1]?.trim() || 'Unable to parse problem description',
|
|
23
|
+
explanation: explanationMatch?.[1]?.trim() || response.split('\n').slice(0, 3).join('\n'),
|
|
24
|
+
fix: fixMatch?.[1]?.trim() || 'See suggested code below',
|
|
25
|
+
originalCode: originalMatch?.[1]?.trim(),
|
|
26
|
+
fixedCode: fixedMatch?.[1]?.trim(),
|
|
27
|
+
};
|
|
28
|
+
}
|
|
29
|
+
/**
|
|
30
|
+
* Build the fix prompt
|
|
31
|
+
*/
|
|
32
|
+
function buildFixPrompt(errorContent, fileContent, filePath) {
|
|
33
|
+
let prompt = `Please help fix this error. Analyze the problem and provide a solution.
|
|
34
|
+
|
|
35
|
+
Error:
|
|
36
|
+
\`\`\`
|
|
37
|
+
${errorContent}
|
|
38
|
+
\`\`\`
|
|
39
|
+
`;
|
|
40
|
+
if (fileContent && filePath) {
|
|
41
|
+
prompt += `
|
|
42
|
+
Source file (${filePath}):
|
|
43
|
+
\`\`\`
|
|
44
|
+
${fileContent}
|
|
45
|
+
\`\`\`
|
|
46
|
+
`;
|
|
47
|
+
}
|
|
48
|
+
prompt += `
|
|
49
|
+
Please provide:
|
|
50
|
+
1. **Problem**: A brief description of what's wrong
|
|
51
|
+
2. **Explanation**: Why this error occurs
|
|
52
|
+
3. **Fix**: How to fix it
|
|
53
|
+
|
|
54
|
+
If you can provide code changes, please show:
|
|
55
|
+
- **Original**: The problematic code
|
|
56
|
+
- **Fixed**: The corrected code
|
|
57
|
+
|
|
58
|
+
Format your response with clear section headers.`;
|
|
59
|
+
return prompt;
|
|
60
|
+
}
|
|
61
|
+
/**
|
|
62
|
+
* Display fix suggestion
|
|
63
|
+
*/
|
|
64
|
+
function displayFixSuggestion(suggestion) {
|
|
65
|
+
ui.newLine();
|
|
66
|
+
// Problem
|
|
67
|
+
ui.print(ui.color('Problem:', 'yellow'));
|
|
68
|
+
ui.print(` ${suggestion.problem}`);
|
|
69
|
+
ui.newLine();
|
|
70
|
+
// Explanation
|
|
71
|
+
ui.print(ui.color('Explanation:', 'blue'));
|
|
72
|
+
for (const line of suggestion.explanation.split('\n')) {
|
|
73
|
+
ui.print(` ${line}`);
|
|
74
|
+
}
|
|
75
|
+
ui.newLine();
|
|
76
|
+
// Fix
|
|
77
|
+
ui.print(ui.color('Suggested Fix:', 'green'));
|
|
78
|
+
for (const line of suggestion.fix.split('\n')) {
|
|
79
|
+
ui.print(` ${line}`);
|
|
80
|
+
}
|
|
81
|
+
// Show diff if we have original and fixed code
|
|
82
|
+
if (suggestion.originalCode && suggestion.fixedCode) {
|
|
83
|
+
ui.newLine();
|
|
84
|
+
ui.print(ui.color('Code Changes:', 'cyan'));
|
|
85
|
+
ui.newLine();
|
|
86
|
+
// Show original
|
|
87
|
+
ui.print(ui.color('Before:', 'red'));
|
|
88
|
+
ui.print('```');
|
|
89
|
+
for (const line of suggestion.originalCode.split('\n')) {
|
|
90
|
+
ui.print(ui.color(`- ${line}`, 'red'));
|
|
91
|
+
}
|
|
92
|
+
ui.print('```');
|
|
93
|
+
ui.newLine();
|
|
94
|
+
// Show fixed
|
|
95
|
+
ui.print(ui.color('After:', 'green'));
|
|
96
|
+
ui.print('```');
|
|
97
|
+
for (const line of suggestion.fixedCode.split('\n')) {
|
|
98
|
+
ui.print(ui.color(`+ ${line}`, 'green'));
|
|
99
|
+
}
|
|
100
|
+
ui.print('```');
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
/**
|
|
104
|
+
* Apply the fix to a file
|
|
105
|
+
*/
|
|
106
|
+
async function applyFix(suggestion, filePath) {
|
|
107
|
+
if (!suggestion.originalCode || !suggestion.fixedCode) {
|
|
108
|
+
ui.warning('Cannot auto-apply: No code diff provided');
|
|
109
|
+
return false;
|
|
110
|
+
}
|
|
111
|
+
try {
|
|
112
|
+
const fs = await import('fs/promises');
|
|
113
|
+
const content = await fs.readFile(filePath, 'utf-8');
|
|
114
|
+
// Try to find and replace the original code
|
|
115
|
+
if (content.includes(suggestion.originalCode)) {
|
|
116
|
+
const newContent = content.replace(suggestion.originalCode, suggestion.fixedCode);
|
|
117
|
+
await fs.writeFile(filePath, newContent, 'utf-8');
|
|
118
|
+
return true;
|
|
119
|
+
}
|
|
120
|
+
else {
|
|
121
|
+
ui.warning('Could not find the original code in the file');
|
|
122
|
+
ui.info('The file may have been modified since the analysis');
|
|
123
|
+
return false;
|
|
124
|
+
}
|
|
125
|
+
}
|
|
126
|
+
catch (error) {
|
|
127
|
+
ui.error(`Failed to apply fix: ${error.message}`);
|
|
128
|
+
return false;
|
|
129
|
+
}
|
|
130
|
+
}
|
|
131
|
+
/**
|
|
132
|
+
* Run the fix command
|
|
133
|
+
*/
|
|
134
|
+
export async function fixCommand(errorOrFile, options = {}) {
|
|
135
|
+
logger.info('Running fix command', { errorOrFile, options });
|
|
136
|
+
let errorContent;
|
|
137
|
+
let fileContent;
|
|
138
|
+
let filePath;
|
|
139
|
+
// Determine what we're fixing
|
|
140
|
+
if (options.file) {
|
|
141
|
+
// Error content with explicit file
|
|
142
|
+
errorContent = errorOrFile;
|
|
143
|
+
filePath = options.file;
|
|
144
|
+
try {
|
|
145
|
+
const fs = await import('fs/promises');
|
|
146
|
+
fileContent = await fs.readFile(filePath, 'utf-8');
|
|
147
|
+
}
|
|
148
|
+
catch (error) {
|
|
149
|
+
ui.warning(`Could not read file ${filePath}: ${error.message}`);
|
|
150
|
+
}
|
|
151
|
+
}
|
|
152
|
+
else if (errorOrFile) {
|
|
153
|
+
// Check if it's a file path
|
|
154
|
+
try {
|
|
155
|
+
const fs = await import('fs/promises');
|
|
156
|
+
const stat = await fs.stat(errorOrFile);
|
|
157
|
+
if (stat.isFile()) {
|
|
158
|
+
filePath = errorOrFile;
|
|
159
|
+
fileContent = await fs.readFile(errorOrFile, 'utf-8');
|
|
160
|
+
// For a file without explicit error, we'll analyze the whole file
|
|
161
|
+
errorContent = `Please analyze this file for potential issues and errors:\n${errorOrFile}`;
|
|
162
|
+
}
|
|
163
|
+
else {
|
|
164
|
+
errorContent = errorOrFile;
|
|
165
|
+
}
|
|
166
|
+
}
|
|
167
|
+
catch {
|
|
168
|
+
// Not a file, treat as error message
|
|
169
|
+
errorContent = errorOrFile;
|
|
170
|
+
}
|
|
171
|
+
}
|
|
172
|
+
else {
|
|
173
|
+
ui.error('Please provide an error message or file to fix');
|
|
174
|
+
ui.newLine();
|
|
175
|
+
ui.print('Usage: nimbus fix <error-or-file> [options]');
|
|
176
|
+
ui.newLine();
|
|
177
|
+
ui.print('Examples:');
|
|
178
|
+
ui.print(' nimbus fix "Error: undefined variable"');
|
|
179
|
+
ui.print(' nimbus fix ./broken.tf');
|
|
180
|
+
ui.print(' nimbus fix "Error: invalid syntax" --file ./app.py');
|
|
181
|
+
ui.print(' nimbus fix ./config.yaml --auto-apply');
|
|
182
|
+
process.exit(1);
|
|
183
|
+
}
|
|
184
|
+
// Display header
|
|
185
|
+
ui.header('Nimbus Fix');
|
|
186
|
+
if (filePath) {
|
|
187
|
+
ui.info(`File: ${filePath}`);
|
|
188
|
+
}
|
|
189
|
+
ui.info(`Error: ${errorContent.slice(0, 100)}${errorContent.length > 100 ? '...' : ''}`);
|
|
190
|
+
ui.newLine();
|
|
191
|
+
// Check if LLM is available
|
|
192
|
+
const llmAvailable = await llmClient.isAvailable();
|
|
193
|
+
if (!llmAvailable) {
|
|
194
|
+
ui.error('LLM service is not available');
|
|
195
|
+
ui.info('Make sure you have configured an LLM provider with "nimbus login"');
|
|
196
|
+
process.exit(1);
|
|
197
|
+
}
|
|
198
|
+
// Build prompt
|
|
199
|
+
const prompt = buildFixPrompt(errorContent, fileContent, filePath);
|
|
200
|
+
ui.startSpinner({ message: 'Analyzing error...' });
|
|
201
|
+
try {
|
|
202
|
+
let response = '';
|
|
203
|
+
for await (const chunk of llmClient.chat(prompt, [])) {
|
|
204
|
+
if (chunk.type === 'content' && chunk.content) {
|
|
205
|
+
response += chunk.content;
|
|
206
|
+
}
|
|
207
|
+
else if (chunk.type === 'error') {
|
|
208
|
+
ui.stopSpinnerFail('Error');
|
|
209
|
+
ui.error(chunk.message || chunk.error || 'Unknown error');
|
|
210
|
+
process.exit(1);
|
|
211
|
+
}
|
|
212
|
+
}
|
|
213
|
+
ui.stopSpinnerSuccess('Analysis complete');
|
|
214
|
+
// Parse and display the suggestion
|
|
215
|
+
const suggestion = parseFixResponse(response);
|
|
216
|
+
suggestion.filePath = filePath;
|
|
217
|
+
displayFixSuggestion(suggestion);
|
|
218
|
+
// JSON output mode
|
|
219
|
+
if (options.json) {
|
|
220
|
+
console.log(JSON.stringify(suggestion, null, 2));
|
|
221
|
+
return;
|
|
222
|
+
}
|
|
223
|
+
// Dry run - don't apply
|
|
224
|
+
if (options.dryRun) {
|
|
225
|
+
ui.newLine();
|
|
226
|
+
ui.info('Dry run mode - no changes applied');
|
|
227
|
+
return;
|
|
228
|
+
}
|
|
229
|
+
// Apply the fix if requested and possible
|
|
230
|
+
if (filePath && suggestion.originalCode && suggestion.fixedCode) {
|
|
231
|
+
ui.newLine();
|
|
232
|
+
const shouldApply = options.autoApply ||
|
|
233
|
+
(await confirm({
|
|
234
|
+
message: 'Apply this fix?',
|
|
235
|
+
defaultValue: false,
|
|
236
|
+
}));
|
|
237
|
+
if (shouldApply) {
|
|
238
|
+
ui.startSpinner({ message: 'Applying fix...' });
|
|
239
|
+
const applied = await applyFix(suggestion, filePath);
|
|
240
|
+
if (applied) {
|
|
241
|
+
ui.stopSpinnerSuccess('Fix applied successfully!');
|
|
242
|
+
ui.newLine();
|
|
243
|
+
ui.info(`File updated: ${filePath}`);
|
|
244
|
+
ui.info('Please review the changes and test your code');
|
|
245
|
+
}
|
|
246
|
+
else {
|
|
247
|
+
ui.stopSpinnerFail('Could not apply fix automatically');
|
|
248
|
+
ui.info('Please apply the suggested changes manually');
|
|
249
|
+
}
|
|
250
|
+
}
|
|
251
|
+
else {
|
|
252
|
+
ui.info('Fix not applied');
|
|
253
|
+
}
|
|
254
|
+
}
|
|
255
|
+
}
|
|
256
|
+
catch (error) {
|
|
257
|
+
ui.stopSpinnerFail('Failed');
|
|
258
|
+
ui.error(error.message);
|
|
259
|
+
process.exit(1);
|
|
260
|
+
}
|
|
261
|
+
}
|
|
262
|
+
// Export as default
|
|
263
|
+
export default fixCommand;
|
|
@@ -0,0 +1,338 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* File System Commands
|
|
3
|
+
*
|
|
4
|
+
* CLI commands for file system operations via FS Tools Service
|
|
5
|
+
*/
|
|
6
|
+
import { ToolsClient } from '../../clients';
|
|
7
|
+
import { ui } from '../../wizard/ui';
|
|
8
|
+
const toolsClient = new ToolsClient();
|
|
9
|
+
/**
|
|
10
|
+
* List directory contents
|
|
11
|
+
*/
|
|
12
|
+
export async function fsListCommand(dirPath, options = {}) {
|
|
13
|
+
ui.header('Files List');
|
|
14
|
+
ui.info(`Path: ${dirPath}`);
|
|
15
|
+
if (options.recursive) {
|
|
16
|
+
ui.info('Recursive: yes');
|
|
17
|
+
}
|
|
18
|
+
ui.startSpinner({ message: `Listing ${dirPath}...` });
|
|
19
|
+
try {
|
|
20
|
+
const result = await toolsClient.fs.list(dirPath, {
|
|
21
|
+
recursive: options.recursive,
|
|
22
|
+
pattern: options.pattern,
|
|
23
|
+
});
|
|
24
|
+
if (result.success) {
|
|
25
|
+
const data = result.data;
|
|
26
|
+
const entries = data?.entries || [];
|
|
27
|
+
ui.stopSpinnerSuccess(`Found ${entries.length} entries`);
|
|
28
|
+
if (entries.length > 0) {
|
|
29
|
+
for (const entry of entries) {
|
|
30
|
+
const icon = entry.type === 'directory' ? '📁' : '📄';
|
|
31
|
+
const size = entry.size ? ` (${formatSize(entry.size)})` : '';
|
|
32
|
+
ui.print(` ${icon} ${entry.name}${size}`);
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
else {
|
|
36
|
+
ui.info('Directory is empty');
|
|
37
|
+
}
|
|
38
|
+
}
|
|
39
|
+
else {
|
|
40
|
+
ui.stopSpinnerFail('Failed to list directory');
|
|
41
|
+
if (result.error) {
|
|
42
|
+
ui.error(result.error.message);
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
catch (error) {
|
|
47
|
+
ui.stopSpinnerFail('Error listing directory');
|
|
48
|
+
ui.error(error.message);
|
|
49
|
+
}
|
|
50
|
+
}
|
|
51
|
+
/**
|
|
52
|
+
* Search for files
|
|
53
|
+
*/
|
|
54
|
+
export async function fsSearchCommand(pattern, searchPath = '.', options = {}) {
|
|
55
|
+
ui.header('Files Search');
|
|
56
|
+
ui.info(`Pattern: ${pattern}`);
|
|
57
|
+
ui.info(`Path: ${searchPath}`);
|
|
58
|
+
ui.startSpinner({ message: `Searching for ${pattern}...` });
|
|
59
|
+
try {
|
|
60
|
+
const result = await toolsClient.fs.search(pattern, {
|
|
61
|
+
path: searchPath,
|
|
62
|
+
maxResults: options.maxResults,
|
|
63
|
+
});
|
|
64
|
+
if (result.success) {
|
|
65
|
+
const data = result.data;
|
|
66
|
+
const matches = data?.matches || data?.entries || [];
|
|
67
|
+
ui.stopSpinnerSuccess(`Found ${matches.length} match(es)`);
|
|
68
|
+
if (matches.length > 0) {
|
|
69
|
+
for (const match of matches) {
|
|
70
|
+
const name = typeof match === 'string' ? match : match.path || match.name;
|
|
71
|
+
ui.print(` ${ui.color('•', 'green')} ${name}`);
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
else {
|
|
76
|
+
ui.stopSpinnerFail('Search failed');
|
|
77
|
+
if (result.error) {
|
|
78
|
+
ui.error(result.error.message);
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
catch (error) {
|
|
83
|
+
ui.stopSpinnerFail('Error searching files');
|
|
84
|
+
ui.error(error.message);
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
/**
|
|
88
|
+
* Read file contents
|
|
89
|
+
*/
|
|
90
|
+
export async function fsReadCommand(filePath, _options = {}) {
|
|
91
|
+
ui.header('Files Read');
|
|
92
|
+
ui.info(`File: ${filePath}`);
|
|
93
|
+
ui.startSpinner({ message: `Reading ${filePath}...` });
|
|
94
|
+
try {
|
|
95
|
+
const result = await toolsClient.fs.read(filePath);
|
|
96
|
+
if (result.success) {
|
|
97
|
+
ui.stopSpinnerSuccess('File read successfully');
|
|
98
|
+
const data = result.data;
|
|
99
|
+
if (data?.content !== undefined) {
|
|
100
|
+
console.log(data.content);
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
else {
|
|
104
|
+
ui.stopSpinnerFail('Failed to read file');
|
|
105
|
+
if (result.error) {
|
|
106
|
+
ui.error(result.error.message);
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
catch (error) {
|
|
111
|
+
ui.stopSpinnerFail('Error reading file');
|
|
112
|
+
ui.error(error.message);
|
|
113
|
+
}
|
|
114
|
+
}
|
|
115
|
+
/**
|
|
116
|
+
* Display directory tree using the dedicated tree endpoint
|
|
117
|
+
*/
|
|
118
|
+
export async function fsTreeCommand(dirPath, options = {}) {
|
|
119
|
+
ui.header('File Tree');
|
|
120
|
+
ui.info(`Path: ${dirPath}`);
|
|
121
|
+
if (options.maxDepth) {
|
|
122
|
+
ui.info(`Max depth: ${options.maxDepth}`);
|
|
123
|
+
}
|
|
124
|
+
ui.startSpinner({ message: `Building tree for ${dirPath}...` });
|
|
125
|
+
try {
|
|
126
|
+
const result = await toolsClient.fs.tree(dirPath, {
|
|
127
|
+
maxDepth: options.maxDepth,
|
|
128
|
+
includeFiles: true,
|
|
129
|
+
});
|
|
130
|
+
if (result.success) {
|
|
131
|
+
const data = result.data;
|
|
132
|
+
const tree = data?.tree || data?.entries || [];
|
|
133
|
+
ui.stopSpinnerSuccess(`Tree for ${dirPath}`);
|
|
134
|
+
const printTree = (nodes, prefix = '') => {
|
|
135
|
+
for (let i = 0; i < nodes.length; i++) {
|
|
136
|
+
const node = nodes[i];
|
|
137
|
+
const isLast = i === nodes.length - 1;
|
|
138
|
+
const connector = isLast ? '└── ' : '├── ';
|
|
139
|
+
const icon = node.type === 'directory' ? '📁' : '📄';
|
|
140
|
+
ui.print(`${prefix}${connector}${icon} ${node.name}`);
|
|
141
|
+
if (node.children && node.children.length > 0) {
|
|
142
|
+
const childPrefix = prefix + (isLast ? ' ' : '│ ');
|
|
143
|
+
printTree(node.children, childPrefix);
|
|
144
|
+
}
|
|
145
|
+
}
|
|
146
|
+
};
|
|
147
|
+
if (tree.length > 0) {
|
|
148
|
+
printTree(tree);
|
|
149
|
+
}
|
|
150
|
+
else {
|
|
151
|
+
ui.info('Directory is empty');
|
|
152
|
+
}
|
|
153
|
+
}
|
|
154
|
+
else {
|
|
155
|
+
ui.stopSpinnerFail('Failed to build tree');
|
|
156
|
+
if (result.error) {
|
|
157
|
+
ui.error(result.error.message);
|
|
158
|
+
}
|
|
159
|
+
}
|
|
160
|
+
}
|
|
161
|
+
catch (error) {
|
|
162
|
+
ui.stopSpinnerFail('Error building tree');
|
|
163
|
+
ui.error(error.message);
|
|
164
|
+
}
|
|
165
|
+
}
|
|
166
|
+
/**
|
|
167
|
+
* Write content to a file
|
|
168
|
+
*/
|
|
169
|
+
export async function fsWriteCommand(filePath, content, options = {}) {
|
|
170
|
+
ui.header('Files Write');
|
|
171
|
+
ui.info(`File: ${filePath}`);
|
|
172
|
+
ui.startSpinner({ message: `Writing to ${filePath}...` });
|
|
173
|
+
try {
|
|
174
|
+
const result = await toolsClient.fs.write(filePath, content, {
|
|
175
|
+
createDirs: options.createDirs,
|
|
176
|
+
});
|
|
177
|
+
if (result.success) {
|
|
178
|
+
ui.stopSpinnerSuccess(`File written successfully: ${filePath}`);
|
|
179
|
+
}
|
|
180
|
+
else {
|
|
181
|
+
ui.stopSpinnerFail('Failed to write file');
|
|
182
|
+
if (result.error) {
|
|
183
|
+
ui.error(result.error.message);
|
|
184
|
+
}
|
|
185
|
+
}
|
|
186
|
+
}
|
|
187
|
+
catch (error) {
|
|
188
|
+
ui.stopSpinnerFail('Error writing file');
|
|
189
|
+
ui.error(error.message);
|
|
190
|
+
}
|
|
191
|
+
}
|
|
192
|
+
/**
|
|
193
|
+
* Show diff between two files
|
|
194
|
+
*/
|
|
195
|
+
export async function fsDiffCommand(file1, file2) {
|
|
196
|
+
ui.header('Files Diff');
|
|
197
|
+
ui.info(`File 1: ${file1}`);
|
|
198
|
+
ui.info(`File 2: ${file2}`);
|
|
199
|
+
ui.startSpinner({ message: 'Computing diff...' });
|
|
200
|
+
try {
|
|
201
|
+
// Read both files and compare via the FS tools service diff endpoint
|
|
202
|
+
const result = await toolsClient.fs.read(file1);
|
|
203
|
+
const result2 = await toolsClient.fs.read(file2);
|
|
204
|
+
if (!result.success) {
|
|
205
|
+
ui.stopSpinnerFail(`Failed to read file: ${file1}`);
|
|
206
|
+
if (result.error) {
|
|
207
|
+
ui.error(result.error.message);
|
|
208
|
+
}
|
|
209
|
+
return;
|
|
210
|
+
}
|
|
211
|
+
if (!result2.success) {
|
|
212
|
+
ui.stopSpinnerFail(`Failed to read file: ${file2}`);
|
|
213
|
+
if (result2.error) {
|
|
214
|
+
ui.error(result2.error.message);
|
|
215
|
+
}
|
|
216
|
+
return;
|
|
217
|
+
}
|
|
218
|
+
const data1 = result.data;
|
|
219
|
+
const data2 = result2.data;
|
|
220
|
+
const content1 = data1?.content || '';
|
|
221
|
+
const content2 = data2?.content || '';
|
|
222
|
+
if (content1 === content2) {
|
|
223
|
+
ui.stopSpinnerSuccess('Files are identical');
|
|
224
|
+
}
|
|
225
|
+
else {
|
|
226
|
+
ui.stopSpinnerSuccess('Diff computed');
|
|
227
|
+
// Simple line-by-line diff display
|
|
228
|
+
const lines1 = content1.split('\n');
|
|
229
|
+
const lines2 = content2.split('\n');
|
|
230
|
+
const maxLines = Math.max(lines1.length, lines2.length);
|
|
231
|
+
const diffLines = [];
|
|
232
|
+
for (let i = 0; i < maxLines; i++) {
|
|
233
|
+
const l1 = lines1[i];
|
|
234
|
+
const l2 = lines2[i];
|
|
235
|
+
if (l1 === undefined) {
|
|
236
|
+
diffLines.push(ui.color(`+ ${l2}`, 'green'));
|
|
237
|
+
}
|
|
238
|
+
else if (l2 === undefined) {
|
|
239
|
+
diffLines.push(ui.color(`- ${l1}`, 'red'));
|
|
240
|
+
}
|
|
241
|
+
else if (l1 !== l2) {
|
|
242
|
+
diffLines.push(ui.color(`- ${l1}`, 'red'));
|
|
243
|
+
diffLines.push(ui.color(`+ ${l2}`, 'green'));
|
|
244
|
+
}
|
|
245
|
+
}
|
|
246
|
+
if (diffLines.length > 0) {
|
|
247
|
+
ui.box({ title: 'Diff', content: diffLines.join('\n') });
|
|
248
|
+
}
|
|
249
|
+
}
|
|
250
|
+
}
|
|
251
|
+
catch (error) {
|
|
252
|
+
ui.stopSpinnerFail('Error computing diff');
|
|
253
|
+
ui.error(error.message);
|
|
254
|
+
}
|
|
255
|
+
}
|
|
256
|
+
/**
|
|
257
|
+
* Main fs command router
|
|
258
|
+
*/
|
|
259
|
+
export async function fsCommand(subcommand, args) {
|
|
260
|
+
const options = {};
|
|
261
|
+
const positionalArgs = [];
|
|
262
|
+
for (let i = 0; i < args.length; i++) {
|
|
263
|
+
const arg = args[i];
|
|
264
|
+
if (arg === '-r' || arg === '--recursive') {
|
|
265
|
+
options.recursive = true;
|
|
266
|
+
}
|
|
267
|
+
else if (arg === '-p' || arg === '--pattern') {
|
|
268
|
+
options.pattern = args[++i];
|
|
269
|
+
}
|
|
270
|
+
else if (arg === '-n' || arg === '--max-results') {
|
|
271
|
+
options.maxResults = parseInt(args[++i], 10);
|
|
272
|
+
}
|
|
273
|
+
else if (arg === '--depth') {
|
|
274
|
+
options.maxDepth = parseInt(args[++i], 10);
|
|
275
|
+
}
|
|
276
|
+
else if (arg === '--create-dirs') {
|
|
277
|
+
options.createDirs = true;
|
|
278
|
+
}
|
|
279
|
+
else if (!arg.startsWith('-')) {
|
|
280
|
+
positionalArgs.push(arg);
|
|
281
|
+
}
|
|
282
|
+
}
|
|
283
|
+
switch (subcommand) {
|
|
284
|
+
case 'list':
|
|
285
|
+
case 'ls':
|
|
286
|
+
await fsListCommand(positionalArgs[0] || '.', options);
|
|
287
|
+
break;
|
|
288
|
+
case 'tree':
|
|
289
|
+
await fsTreeCommand(positionalArgs[0] || '.', options);
|
|
290
|
+
break;
|
|
291
|
+
case 'search':
|
|
292
|
+
case 'find':
|
|
293
|
+
if (positionalArgs.length < 1) {
|
|
294
|
+
ui.error('Usage: nimbus fs search <pattern> [path]');
|
|
295
|
+
return;
|
|
296
|
+
}
|
|
297
|
+
await fsSearchCommand(positionalArgs[0], positionalArgs[1] || '.', options);
|
|
298
|
+
break;
|
|
299
|
+
case 'read':
|
|
300
|
+
case 'cat':
|
|
301
|
+
if (positionalArgs.length < 1) {
|
|
302
|
+
ui.error('Usage: nimbus fs read <file>');
|
|
303
|
+
return;
|
|
304
|
+
}
|
|
305
|
+
await fsReadCommand(positionalArgs[0], options);
|
|
306
|
+
break;
|
|
307
|
+
case 'write':
|
|
308
|
+
if (positionalArgs.length < 2) {
|
|
309
|
+
ui.error('Usage: nimbus fs write <path> <content>');
|
|
310
|
+
return;
|
|
311
|
+
}
|
|
312
|
+
await fsWriteCommand(positionalArgs[0], positionalArgs.slice(1).join(' '), options);
|
|
313
|
+
break;
|
|
314
|
+
case 'diff':
|
|
315
|
+
if (positionalArgs.length < 2) {
|
|
316
|
+
ui.error('Usage: nimbus fs diff <file1> <file2>');
|
|
317
|
+
return;
|
|
318
|
+
}
|
|
319
|
+
await fsDiffCommand(positionalArgs[0], positionalArgs[1]);
|
|
320
|
+
break;
|
|
321
|
+
default:
|
|
322
|
+
ui.error(`Unknown fs subcommand: ${subcommand}`);
|
|
323
|
+
ui.info('Available commands: list, tree, search, read, write, diff');
|
|
324
|
+
}
|
|
325
|
+
}
|
|
326
|
+
/**
|
|
327
|
+
* Format file size in human-readable format
|
|
328
|
+
*/
|
|
329
|
+
function formatSize(bytes) {
|
|
330
|
+
const units = ['B', 'KB', 'MB', 'GB'];
|
|
331
|
+
let size = bytes;
|
|
332
|
+
let unitIndex = 0;
|
|
333
|
+
while (size >= 1024 && unitIndex < units.length - 1) {
|
|
334
|
+
size /= 1024;
|
|
335
|
+
unitIndex++;
|
|
336
|
+
}
|
|
337
|
+
return `${size.toFixed(unitIndex > 0 ? 1 : 0)} ${units[unitIndex]}`;
|
|
338
|
+
}
|