@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.
Files changed (469) hide show
  1. package/bin/nimbus +26 -10
  2. package/bin/nimbus.cmd +41 -0
  3. package/bin/nimbus.mjs +70 -0
  4. package/completions/nimbus.bash +38 -0
  5. package/completions/nimbus.fish +48 -0
  6. package/completions/nimbus.zsh +81 -0
  7. package/dist/src/agent/compaction-agent.js +215 -0
  8. package/dist/src/agent/context-manager.js +385 -0
  9. package/dist/src/agent/context.js +322 -0
  10. package/dist/src/agent/deploy-preview.js +395 -0
  11. package/dist/src/agent/expand-files.js +95 -0
  12. package/dist/src/agent/index.js +18 -0
  13. package/dist/src/agent/loop.js +1535 -0
  14. package/dist/src/agent/modes.js +347 -0
  15. package/dist/src/agent/permissions.js +396 -0
  16. package/dist/src/agent/subagents/base.js +67 -0
  17. package/dist/src/agent/subagents/cost.js +45 -0
  18. package/dist/src/agent/subagents/explore.js +36 -0
  19. package/dist/src/agent/subagents/general.js +41 -0
  20. package/dist/src/agent/subagents/index.js +88 -0
  21. package/dist/src/agent/subagents/infra.js +52 -0
  22. package/dist/src/agent/subagents/security.js +60 -0
  23. package/dist/src/agent/system-prompt.js +860 -0
  24. package/dist/src/app.js +152 -0
  25. package/dist/src/audit/activity-log.js +209 -0
  26. package/dist/src/audit/compliance-checker.js +419 -0
  27. package/dist/src/audit/cost-tracker.js +231 -0
  28. package/dist/src/audit/index.js +10 -0
  29. package/dist/src/audit/security-scanner.js +490 -0
  30. package/dist/src/auth/guard.js +64 -0
  31. package/dist/src/auth/index.js +19 -0
  32. package/dist/src/auth/keychain.js +79 -0
  33. package/dist/src/auth/oauth.js +389 -0
  34. package/dist/src/auth/providers.js +415 -0
  35. package/dist/src/auth/sso.js +87 -0
  36. package/dist/src/auth/store.js +424 -0
  37. package/dist/src/auth/types.js +5 -0
  38. package/dist/src/cli/index.js +8 -0
  39. package/dist/src/cli/init.js +1048 -0
  40. package/dist/src/cli/openapi-spec.js +346 -0
  41. package/dist/src/cli/run.js +505 -0
  42. package/dist/src/cli/serve-auth.js +56 -0
  43. package/dist/src/cli/serve.js +432 -0
  44. package/dist/src/cli/web.js +50 -0
  45. package/dist/src/cli.js +1574 -0
  46. package/dist/src/clients/core-engine-client.js +156 -0
  47. package/dist/src/clients/enterprise-client.js +246 -0
  48. package/dist/src/clients/generator-client.js +219 -0
  49. package/dist/src/clients/git-client.js +367 -0
  50. package/dist/src/clients/github-client.js +229 -0
  51. package/dist/src/clients/helm-client.js +299 -0
  52. package/dist/src/clients/index.js +18 -0
  53. package/dist/src/clients/k8s-client.js +270 -0
  54. package/dist/src/clients/llm-client.js +119 -0
  55. package/dist/src/clients/rest-client.js +104 -0
  56. package/dist/src/clients/service-discovery.js +35 -0
  57. package/dist/src/clients/terraform-client.js +302 -0
  58. package/dist/src/clients/tools-client.js +1227 -0
  59. package/dist/src/clients/ws-client.js +93 -0
  60. package/dist/src/commands/alias.js +91 -0
  61. package/dist/src/commands/analyze/index.js +313 -0
  62. package/dist/src/commands/apply/helm.js +375 -0
  63. package/dist/src/commands/apply/index.js +176 -0
  64. package/dist/src/commands/apply/k8s.js +350 -0
  65. package/dist/src/commands/apply/terraform.js +465 -0
  66. package/dist/src/commands/ask.js +137 -0
  67. package/dist/src/commands/audit/index.js +322 -0
  68. package/dist/src/commands/auth-cloud.js +345 -0
  69. package/dist/src/commands/auth-list.js +112 -0
  70. package/dist/src/commands/auth-profile.js +104 -0
  71. package/dist/src/commands/auth-refresh.js +161 -0
  72. package/dist/src/commands/auth-status.js +122 -0
  73. package/dist/src/commands/aws/ec2.js +402 -0
  74. package/dist/src/commands/aws/iam.js +304 -0
  75. package/dist/src/commands/aws/index.js +108 -0
  76. package/dist/src/commands/aws/lambda.js +317 -0
  77. package/dist/src/commands/aws/rds.js +345 -0
  78. package/dist/src/commands/aws/s3.js +346 -0
  79. package/dist/src/commands/aws/vpc.js +302 -0
  80. package/dist/src/commands/aws-discover.js +413 -0
  81. package/dist/src/commands/aws-terraform.js +618 -0
  82. package/dist/src/commands/azure/aks.js +305 -0
  83. package/dist/src/commands/azure/functions.js +200 -0
  84. package/dist/src/commands/azure/index.js +93 -0
  85. package/dist/src/commands/azure/storage.js +378 -0
  86. package/dist/src/commands/azure/vm.js +291 -0
  87. package/dist/src/commands/billing/index.js +224 -0
  88. package/dist/src/commands/chat.js +259 -0
  89. package/dist/src/commands/completions.js +255 -0
  90. package/dist/src/commands/config.js +291 -0
  91. package/dist/src/commands/cost/cloud-cost-estimator.js +211 -0
  92. package/dist/src/commands/cost/estimator.js +73 -0
  93. package/dist/src/commands/cost/index.js +625 -0
  94. package/dist/src/commands/cost/parsers/terraform.js +234 -0
  95. package/dist/src/commands/cost/parsers/types.js +4 -0
  96. package/dist/src/commands/cost/pricing/aws.js +501 -0
  97. package/dist/src/commands/cost/pricing/azure.js +462 -0
  98. package/dist/src/commands/cost/pricing/gcp.js +359 -0
  99. package/dist/src/commands/cost/pricing/index.js +24 -0
  100. package/dist/src/commands/demo.js +196 -0
  101. package/dist/src/commands/deploy.js +215 -0
  102. package/dist/src/commands/doctor.js +1291 -0
  103. package/dist/src/commands/drift/index.js +674 -0
  104. package/dist/src/commands/explain.js +235 -0
  105. package/dist/src/commands/export.js +120 -0
  106. package/dist/src/commands/feedback.js +319 -0
  107. package/dist/src/commands/fix.js +263 -0
  108. package/dist/src/commands/fs/index.js +338 -0
  109. package/dist/src/commands/gcp/compute.js +266 -0
  110. package/dist/src/commands/gcp/functions.js +221 -0
  111. package/dist/src/commands/gcp/gke.js +357 -0
  112. package/dist/src/commands/gcp/iam.js +295 -0
  113. package/dist/src/commands/gcp/index.js +105 -0
  114. package/dist/src/commands/gcp/storage.js +232 -0
  115. package/dist/src/commands/generate-helm.js +1026 -0
  116. package/dist/src/commands/generate-k8s.js +1263 -0
  117. package/dist/src/commands/generate-terraform.js +1058 -0
  118. package/dist/src/commands/gh/index.js +663 -0
  119. package/dist/src/commands/git/index.js +1208 -0
  120. package/dist/src/commands/helm/index.js +985 -0
  121. package/dist/src/commands/help.js +639 -0
  122. package/dist/src/commands/history.js +120 -0
  123. package/dist/src/commands/import.js +782 -0
  124. package/dist/src/commands/incident.js +144 -0
  125. package/dist/src/commands/index.js +109 -0
  126. package/dist/src/commands/init.js +955 -0
  127. package/dist/src/commands/k8s/index.js +979 -0
  128. package/dist/src/commands/login.js +588 -0
  129. package/dist/src/commands/logout.js +61 -0
  130. package/dist/src/commands/logs.js +160 -0
  131. package/dist/src/commands/onboarding.js +382 -0
  132. package/dist/src/commands/pipeline.js +153 -0
  133. package/dist/src/commands/plan/display.js +216 -0
  134. package/dist/src/commands/plan/index.js +525 -0
  135. package/dist/src/commands/plugin.js +325 -0
  136. package/dist/src/commands/preview.js +356 -0
  137. package/dist/src/commands/profile.js +297 -0
  138. package/dist/src/commands/questionnaire.js +1021 -0
  139. package/dist/src/commands/resume.js +35 -0
  140. package/dist/src/commands/rollback.js +259 -0
  141. package/dist/src/commands/rollout.js +74 -0
  142. package/dist/src/commands/runbook.js +307 -0
  143. package/dist/src/commands/schedule.js +202 -0
  144. package/dist/src/commands/status.js +213 -0
  145. package/dist/src/commands/team/index.js +309 -0
  146. package/dist/src/commands/team-context.js +200 -0
  147. package/dist/src/commands/template.js +204 -0
  148. package/dist/src/commands/tf/index.js +989 -0
  149. package/dist/src/commands/upgrade.js +515 -0
  150. package/dist/src/commands/usage/index.js +118 -0
  151. package/dist/src/commands/version.js +145 -0
  152. package/dist/src/commands/watch.js +127 -0
  153. package/dist/src/compat/index.js +2 -0
  154. package/dist/src/compat/runtime.js +10 -0
  155. package/dist/src/compat/sqlite.js +144 -0
  156. package/dist/src/config/index.js +6 -0
  157. package/dist/src/config/manager.js +469 -0
  158. package/dist/src/config/mode-store.js +57 -0
  159. package/dist/src/config/profiles.js +66 -0
  160. package/dist/src/config/safety-policy.js +251 -0
  161. package/dist/src/config/schema.js +107 -0
  162. package/dist/src/config/types.js +311 -0
  163. package/dist/src/config/workspace-state.js +38 -0
  164. package/dist/src/context/context-db.js +138 -0
  165. package/dist/src/demo/index.js +295 -0
  166. package/dist/src/demo/scenarios/full-journey.js +226 -0
  167. package/dist/src/demo/scenarios/getting-started.js +124 -0
  168. package/dist/src/demo/scenarios/helm-release.js +334 -0
  169. package/dist/src/demo/scenarios/k8s-deployment.js +190 -0
  170. package/dist/src/demo/scenarios/terraform-vpc.js +167 -0
  171. package/dist/src/demo/types.js +6 -0
  172. package/dist/src/engine/cost-estimator.js +334 -0
  173. package/dist/src/engine/diagram-generator.js +192 -0
  174. package/dist/src/engine/drift-detector.js +688 -0
  175. package/dist/src/engine/executor.js +832 -0
  176. package/dist/src/engine/index.js +39 -0
  177. package/dist/src/engine/orchestrator.js +436 -0
  178. package/dist/src/engine/planner.js +616 -0
  179. package/dist/src/engine/safety.js +609 -0
  180. package/dist/src/engine/verifier.js +664 -0
  181. package/dist/src/enterprise/audit.js +241 -0
  182. package/dist/src/enterprise/auth.js +189 -0
  183. package/dist/src/enterprise/billing.js +512 -0
  184. package/dist/src/enterprise/index.js +16 -0
  185. package/dist/src/enterprise/teams.js +315 -0
  186. package/dist/src/generator/best-practices.js +1375 -0
  187. package/dist/src/generator/helm.js +495 -0
  188. package/dist/src/generator/index.js +11 -0
  189. package/dist/src/generator/intent-parser.js +420 -0
  190. package/dist/src/generator/kubernetes.js +773 -0
  191. package/dist/src/generator/terraform.js +1472 -0
  192. package/dist/src/history/index.js +6 -0
  193. package/dist/src/history/manager.js +199 -0
  194. package/dist/src/history/types.js +6 -0
  195. package/dist/src/hooks/config.js +318 -0
  196. package/dist/src/hooks/engine.js +317 -0
  197. package/dist/src/hooks/index.js +2 -0
  198. package/dist/src/llm/auth-bridge.js +157 -0
  199. package/dist/src/llm/circuit-breaker.js +116 -0
  200. package/dist/src/llm/config-loader.js +172 -0
  201. package/dist/src/llm/cost-calculator.js +137 -0
  202. package/dist/src/llm/index.js +7 -0
  203. package/dist/src/llm/model-aliases.js +99 -0
  204. package/dist/src/llm/provider-registry.js +57 -0
  205. package/dist/src/llm/providers/anthropic.js +430 -0
  206. package/dist/src/llm/providers/bedrock.js +409 -0
  207. package/dist/src/llm/providers/google.js +344 -0
  208. package/dist/src/llm/providers/ollama.js +661 -0
  209. package/dist/src/llm/providers/openai-compatible.js +289 -0
  210. package/dist/src/llm/providers/openai.js +284 -0
  211. package/dist/src/llm/providers/openrouter.js +293 -0
  212. package/dist/src/llm/router.js +844 -0
  213. package/dist/src/llm/types.js +69 -0
  214. package/dist/src/lsp/client.js +239 -0
  215. package/dist/src/lsp/languages.js +95 -0
  216. package/dist/src/lsp/manager.js +243 -0
  217. package/dist/src/mcp/client.js +289 -0
  218. package/dist/src/mcp/index.js +5 -0
  219. package/dist/src/mcp/manager.js +113 -0
  220. package/dist/src/nimbus.js +212 -0
  221. package/dist/src/plugins/index.js +13 -0
  222. package/dist/src/plugins/loader.js +280 -0
  223. package/dist/src/plugins/manager.js +282 -0
  224. package/dist/src/plugins/types.js +23 -0
  225. package/dist/src/scanners/cicd-scanner.js +230 -0
  226. package/dist/src/scanners/cloud-scanner.js +415 -0
  227. package/dist/src/scanners/framework-scanner.js +430 -0
  228. package/dist/src/scanners/iac-scanner.js +350 -0
  229. package/dist/src/scanners/index.js +454 -0
  230. package/dist/src/scanners/language-scanner.js +258 -0
  231. package/dist/src/scanners/package-manager-scanner.js +252 -0
  232. package/dist/src/scanners/types.js +6 -0
  233. package/dist/src/sessions/manager.js +395 -0
  234. package/dist/src/sessions/types.js +4 -0
  235. package/dist/src/sharing/sync.js +238 -0
  236. package/dist/src/sharing/viewer.js +131 -0
  237. package/dist/src/snapshots/index.js +1 -0
  238. package/dist/src/snapshots/manager.js +432 -0
  239. package/dist/src/state/artifacts.js +94 -0
  240. package/dist/src/state/audit.js +73 -0
  241. package/dist/src/state/billing.js +126 -0
  242. package/dist/src/state/checkpoints.js +81 -0
  243. package/dist/src/state/config.js +58 -0
  244. package/dist/src/state/conversations.js +7 -0
  245. package/dist/src/state/credentials.js +96 -0
  246. package/dist/src/state/db.js +53 -0
  247. package/dist/src/state/index.js +23 -0
  248. package/dist/src/state/messages.js +76 -0
  249. package/dist/src/state/projects.js +92 -0
  250. package/dist/src/state/schema.js +233 -0
  251. package/dist/src/state/sessions.js +79 -0
  252. package/dist/src/state/teams.js +131 -0
  253. package/dist/src/telemetry.js +91 -0
  254. package/dist/src/tools/aws-ops.js +747 -0
  255. package/dist/src/tools/azure-ops.js +491 -0
  256. package/dist/src/tools/file-ops.js +451 -0
  257. package/dist/src/tools/gcp-ops.js +559 -0
  258. package/dist/src/tools/git-ops.js +557 -0
  259. package/dist/src/tools/github-ops.js +460 -0
  260. package/dist/src/tools/helm-ops.js +634 -0
  261. package/dist/src/tools/index.js +16 -0
  262. package/dist/src/tools/k8s-ops.js +579 -0
  263. package/dist/src/tools/schemas/converter.js +129 -0
  264. package/dist/src/tools/schemas/devops.js +3319 -0
  265. package/dist/src/tools/schemas/index.js +19 -0
  266. package/dist/src/tools/schemas/standard.js +966 -0
  267. package/dist/src/tools/schemas/types.js +409 -0
  268. package/dist/src/tools/spawn-exec.js +109 -0
  269. package/dist/src/tools/terraform-ops.js +627 -0
  270. package/dist/src/types/config.js +1 -0
  271. package/dist/src/types/drift.js +4 -0
  272. package/dist/src/types/enterprise.js +5 -0
  273. package/dist/src/types/index.js +14 -0
  274. package/dist/src/types/plan.js +1 -0
  275. package/dist/src/types/request.js +1 -0
  276. package/dist/src/types/response.js +1 -0
  277. package/dist/src/types/service.js +1 -0
  278. package/dist/src/ui/App.js +1672 -0
  279. package/dist/src/ui/DeployPreview.js +60 -0
  280. package/dist/src/ui/FileDiffModal.js +108 -0
  281. package/dist/src/ui/Header.js +46 -0
  282. package/dist/src/ui/HelpModal.js +9 -0
  283. package/dist/src/ui/InputBox.js +408 -0
  284. package/dist/src/ui/MessageList.js +795 -0
  285. package/dist/src/ui/PermissionPrompt.js +72 -0
  286. package/dist/src/ui/StatusBar.js +109 -0
  287. package/dist/src/ui/TerminalPane.js +31 -0
  288. package/dist/src/ui/ToolCallDisplay.js +303 -0
  289. package/dist/src/ui/TreePane.js +83 -0
  290. package/dist/src/ui/chat-ui.js +721 -0
  291. package/dist/src/ui/index.js +11 -0
  292. package/dist/src/ui/ink/index.js +1325 -0
  293. package/dist/src/ui/streaming.js +137 -0
  294. package/dist/src/ui/theme.js +78 -0
  295. package/dist/src/ui/types.js +7 -0
  296. package/dist/src/utils/analytics.js +61 -0
  297. package/dist/src/utils/cost-warning.js +25 -0
  298. package/dist/src/utils/env.js +42 -0
  299. package/dist/src/utils/errors.js +54 -0
  300. package/dist/src/utils/event-bus.js +22 -0
  301. package/dist/src/utils/index.js +16 -0
  302. package/dist/src/utils/logger.js +150 -0
  303. package/dist/src/utils/rate-limiter.js +90 -0
  304. package/dist/src/utils/service-auth.js +36 -0
  305. package/dist/src/utils/validation.js +39 -0
  306. package/dist/src/version.js +3 -0
  307. package/dist/src/watcher/index.js +192 -0
  308. package/dist/src/wizard/approval.js +275 -0
  309. package/dist/src/wizard/index.js +13 -0
  310. package/dist/src/wizard/prompts.js +273 -0
  311. package/dist/src/wizard/types.js +4 -0
  312. package/dist/src/wizard/ui.js +453 -0
  313. package/dist/src/wizard/wizard.js +227 -0
  314. package/package.json +31 -23
  315. package/src/__tests__/alias.test.ts +133 -0
  316. package/src/__tests__/app.test.ts +1 -1
  317. package/src/__tests__/audit.test.ts +1 -1
  318. package/src/__tests__/circuit-breaker.test.ts +1 -1
  319. package/src/__tests__/cli-run.test.ts +237 -1
  320. package/src/__tests__/compat-sqlite.test.ts +68 -0
  321. package/src/__tests__/context-manager.test.ts +131 -1
  322. package/src/__tests__/context.test.ts +1 -1
  323. package/src/__tests__/devops-terminal-gaps.test.ts +718 -0
  324. package/src/__tests__/doctor.test.ts +48 -0
  325. package/src/__tests__/enterprise.test.ts +1 -1
  326. package/src/__tests__/export.test.ts +236 -0
  327. package/src/__tests__/gap-11-18-20.test.ts +958 -0
  328. package/src/__tests__/generator.test.ts +1 -1
  329. package/src/__tests__/helm-streaming.test.ts +127 -0
  330. package/src/__tests__/hooks.test.ts +1 -1
  331. package/src/__tests__/incident.test.ts +179 -0
  332. package/src/__tests__/init.test.ts +55 -4
  333. package/src/__tests__/intent-parser.test.ts +1 -1
  334. package/src/__tests__/llm-router.test.ts +1 -1
  335. package/src/__tests__/logs.test.ts +107 -0
  336. package/src/__tests__/loop-errors.test.ts +244 -0
  337. package/src/__tests__/lsp.test.ts +1 -1
  338. package/src/__tests__/modes.test.ts +1 -1
  339. package/src/__tests__/perf-optimizations.test.ts +847 -0
  340. package/src/__tests__/permissions.test.ts +1 -1
  341. package/src/__tests__/pipeline.test.ts +50 -0
  342. package/src/__tests__/polish-phase3.test.ts +340 -0
  343. package/src/__tests__/profile.test.ts +237 -0
  344. package/src/__tests__/rollback.test.ts +83 -0
  345. package/src/__tests__/runbook.test.ts +219 -0
  346. package/src/__tests__/schedule.test.ts +206 -0
  347. package/src/__tests__/serve.test.ts +1 -1
  348. package/src/__tests__/sessions.test.ts +96 -1
  349. package/src/__tests__/sharing.test.ts +53 -1
  350. package/src/__tests__/snapshots.test.ts +1 -1
  351. package/src/__tests__/standalone-migration.test.ts +199 -0
  352. package/src/__tests__/state-db.test.ts +1 -1
  353. package/src/__tests__/status.test.ts +158 -0
  354. package/src/__tests__/stream-with-tools.test.ts +71 -25
  355. package/src/__tests__/subagents.test.ts +1 -1
  356. package/src/__tests__/system-prompt.test.ts +82 -3
  357. package/src/__tests__/terminal-gap-v2.test.ts +395 -0
  358. package/src/__tests__/terminal-parity.test.ts +393 -0
  359. package/src/__tests__/tf-apply.test.ts +187 -0
  360. package/src/__tests__/tool-converter.test.ts +1 -1
  361. package/src/__tests__/tool-schemas.test.ts +209 -4
  362. package/src/__tests__/tools.test.ts +4 -3
  363. package/src/__tests__/version-json.test.ts +184 -0
  364. package/src/__tests__/version.test.ts +1 -1
  365. package/src/__tests__/watch.test.ts +129 -0
  366. package/src/agent/compaction-agent.ts +40 -1
  367. package/src/agent/context-manager.ts +67 -3
  368. package/src/agent/deploy-preview.ts +62 -1
  369. package/src/agent/expand-files.ts +108 -0
  370. package/src/agent/loop.ts +1312 -31
  371. package/src/agent/permissions.ts +51 -4
  372. package/src/agent/system-prompt.ts +573 -19
  373. package/src/app.ts +58 -0
  374. package/src/audit/security-scanner.ts +45 -0
  375. package/src/auth/keychain.ts +82 -0
  376. package/src/auth/oauth.ts +15 -5
  377. package/src/cli/init.ts +378 -5
  378. package/src/cli/run.ts +407 -16
  379. package/src/cli/serve.ts +78 -1
  380. package/src/cli/web.ts +10 -6
  381. package/src/cli.ts +312 -1
  382. package/src/clients/service-discovery.ts +30 -25
  383. package/src/commands/alias.ts +100 -0
  384. package/src/commands/audit/index.ts +121 -2
  385. package/src/commands/auth-cloud.ts +113 -0
  386. package/src/commands/auth-refresh.ts +187 -0
  387. package/src/commands/aws-discover.ts +144 -251
  388. package/src/commands/aws-terraform.ts +68 -118
  389. package/src/commands/chat.ts +9 -3
  390. package/src/commands/completions.ts +268 -0
  391. package/src/commands/config.ts +26 -0
  392. package/src/commands/cost/index.ts +218 -2
  393. package/src/commands/deploy.ts +260 -0
  394. package/src/commands/doctor.ts +744 -152
  395. package/src/commands/drift/index.ts +371 -23
  396. package/src/commands/export.ts +146 -0
  397. package/src/commands/generate-k8s.ts +9 -61
  398. package/src/commands/generate-terraform.ts +191 -449
  399. package/src/commands/help.ts +212 -36
  400. package/src/commands/history.ts +8 -1
  401. package/src/commands/incident.ts +166 -0
  402. package/src/commands/init.ts +5 -0
  403. package/src/commands/login.ts +86 -1
  404. package/src/commands/logs.ts +167 -0
  405. package/src/commands/onboarding.ts +211 -34
  406. package/src/commands/pipeline.ts +186 -0
  407. package/src/commands/plugin.ts +398 -0
  408. package/src/commands/profile.ts +342 -0
  409. package/src/commands/questionnaire.ts +0 -98
  410. package/src/commands/resume.ts +26 -34
  411. package/src/commands/rollback.ts +315 -0
  412. package/src/commands/rollout.ts +88 -0
  413. package/src/commands/runbook.ts +346 -0
  414. package/src/commands/schedule.ts +236 -0
  415. package/src/commands/status.ts +252 -0
  416. package/src/commands/team-context.ts +220 -0
  417. package/src/commands/template.ts +58 -57
  418. package/src/commands/tf/index.ts +70 -11
  419. package/src/commands/upgrade.ts +57 -0
  420. package/src/commands/version.ts +54 -50
  421. package/src/commands/watch.ts +153 -0
  422. package/src/compat/runtime.ts +1 -1
  423. package/src/compat/sqlite.ts +75 -5
  424. package/src/config/mode-store.ts +62 -0
  425. package/src/config/profiles.ts +84 -0
  426. package/src/config/types.ts +83 -1
  427. package/src/config/workspace-state.ts +53 -0
  428. package/src/engine/cost-estimator.ts +52 -10
  429. package/src/engine/executor.ts +33 -2
  430. package/src/engine/planner.ts +68 -1
  431. package/src/generator/terraform.ts +8 -0
  432. package/src/history/manager.ts +2 -74
  433. package/src/hooks/engine.ts +5 -4
  434. package/src/llm/cost-calculator.ts +2 -2
  435. package/src/llm/providers/anthropic.ts +50 -21
  436. package/src/llm/router.ts +76 -7
  437. package/src/lsp/languages.ts +3 -0
  438. package/src/lsp/manager.ts +21 -5
  439. package/src/nimbus.ts +37 -18
  440. package/src/sessions/manager.ts +108 -1
  441. package/src/sharing/sync.ts +4 -0
  442. package/src/sharing/viewer.ts +66 -0
  443. package/src/tools/file-ops.ts +22 -0
  444. package/src/tools/schemas/devops.ts +3007 -117
  445. package/src/tools/schemas/standard.ts +5 -1
  446. package/src/tools/schemas/types.ts +31 -1
  447. package/src/tools/spawn-exec.ts +148 -0
  448. package/src/ui/App.tsx +1183 -66
  449. package/src/ui/DeployPreview.tsx +62 -57
  450. package/src/ui/FileDiffModal.tsx +162 -0
  451. package/src/ui/Header.tsx +87 -24
  452. package/src/ui/HelpModal.tsx +57 -0
  453. package/src/ui/InputBox.tsx +163 -10
  454. package/src/ui/MessageList.tsx +487 -40
  455. package/src/ui/PermissionPrompt.tsx +17 -5
  456. package/src/ui/StatusBar.tsx +122 -3
  457. package/src/ui/TerminalPane.tsx +84 -0
  458. package/src/ui/ToolCallDisplay.tsx +252 -18
  459. package/src/ui/TreePane.tsx +132 -0
  460. package/src/ui/chat-ui.ts +41 -44
  461. package/src/ui/ink/index.ts +771 -38
  462. package/src/ui/streaming.ts +1 -1
  463. package/src/ui/theme.ts +104 -0
  464. package/src/ui/types.ts +18 -0
  465. package/src/version.ts +1 -1
  466. package/src/watcher/index.ts +66 -15
  467. package/src/wizard/types.ts +1 -0
  468. package/src/wizard/ui.ts +1 -1
  469. 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
+ }