@build-astron-co/nimbus 0.3.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 (441) hide show
  1. package/bin/nimbus.cmd +41 -0
  2. package/bin/nimbus.mjs +70 -0
  3. package/completions/nimbus.bash +38 -0
  4. package/completions/nimbus.fish +48 -0
  5. package/completions/nimbus.zsh +81 -0
  6. package/dist/src/agent/compaction-agent.js +215 -0
  7. package/dist/src/agent/context-manager.js +385 -0
  8. package/dist/src/agent/context.js +322 -0
  9. package/dist/src/agent/deploy-preview.js +395 -0
  10. package/dist/src/agent/expand-files.js +95 -0
  11. package/dist/src/agent/index.js +18 -0
  12. package/dist/src/agent/loop.js +1535 -0
  13. package/dist/src/agent/modes.js +347 -0
  14. package/dist/src/agent/permissions.js +396 -0
  15. package/dist/src/agent/subagents/base.js +67 -0
  16. package/dist/src/agent/subagents/cost.js +45 -0
  17. package/dist/src/agent/subagents/explore.js +36 -0
  18. package/dist/src/agent/subagents/general.js +41 -0
  19. package/dist/src/agent/subagents/index.js +88 -0
  20. package/dist/src/agent/subagents/infra.js +52 -0
  21. package/dist/src/agent/subagents/security.js +60 -0
  22. package/dist/src/agent/system-prompt.js +860 -0
  23. package/dist/src/app.js +152 -0
  24. package/dist/src/audit/activity-log.js +209 -0
  25. package/dist/src/audit/compliance-checker.js +419 -0
  26. package/dist/src/audit/cost-tracker.js +231 -0
  27. package/dist/src/audit/index.js +10 -0
  28. package/dist/src/audit/security-scanner.js +490 -0
  29. package/dist/src/auth/guard.js +64 -0
  30. package/dist/src/auth/index.js +19 -0
  31. package/dist/src/auth/keychain.js +79 -0
  32. package/dist/src/auth/oauth.js +389 -0
  33. package/dist/src/auth/providers.js +415 -0
  34. package/dist/src/auth/sso.js +87 -0
  35. package/dist/src/auth/store.js +424 -0
  36. package/dist/src/auth/types.js +5 -0
  37. package/dist/src/cli/index.js +8 -0
  38. package/dist/src/cli/init.js +1048 -0
  39. package/dist/src/cli/openapi-spec.js +346 -0
  40. package/dist/src/cli/run.js +505 -0
  41. package/dist/src/cli/serve-auth.js +56 -0
  42. package/dist/src/cli/serve.js +432 -0
  43. package/dist/src/cli/web.js +50 -0
  44. package/dist/src/cli.js +1574 -0
  45. package/dist/src/clients/core-engine-client.js +156 -0
  46. package/dist/src/clients/enterprise-client.js +246 -0
  47. package/dist/src/clients/generator-client.js +219 -0
  48. package/dist/src/clients/git-client.js +367 -0
  49. package/dist/src/clients/github-client.js +229 -0
  50. package/dist/src/clients/helm-client.js +299 -0
  51. package/dist/src/clients/index.js +18 -0
  52. package/dist/src/clients/k8s-client.js +270 -0
  53. package/dist/src/clients/llm-client.js +119 -0
  54. package/dist/src/clients/rest-client.js +104 -0
  55. package/dist/src/clients/service-discovery.js +35 -0
  56. package/dist/src/clients/terraform-client.js +302 -0
  57. package/dist/src/clients/tools-client.js +1227 -0
  58. package/dist/src/clients/ws-client.js +93 -0
  59. package/dist/src/commands/alias.js +91 -0
  60. package/dist/src/commands/analyze/index.js +313 -0
  61. package/dist/src/commands/apply/helm.js +375 -0
  62. package/dist/src/commands/apply/index.js +176 -0
  63. package/dist/src/commands/apply/k8s.js +350 -0
  64. package/dist/src/commands/apply/terraform.js +465 -0
  65. package/dist/src/commands/ask.js +137 -0
  66. package/dist/src/commands/audit/index.js +322 -0
  67. package/dist/src/commands/auth-cloud.js +345 -0
  68. package/dist/src/commands/auth-list.js +112 -0
  69. package/dist/src/commands/auth-profile.js +104 -0
  70. package/dist/src/commands/auth-refresh.js +161 -0
  71. package/dist/src/commands/auth-status.js +122 -0
  72. package/dist/src/commands/aws/ec2.js +402 -0
  73. package/dist/src/commands/aws/iam.js +304 -0
  74. package/dist/src/commands/aws/index.js +108 -0
  75. package/dist/src/commands/aws/lambda.js +317 -0
  76. package/dist/src/commands/aws/rds.js +345 -0
  77. package/dist/src/commands/aws/s3.js +346 -0
  78. package/dist/src/commands/aws/vpc.js +302 -0
  79. package/dist/src/commands/aws-discover.js +413 -0
  80. package/dist/src/commands/aws-terraform.js +618 -0
  81. package/dist/src/commands/azure/aks.js +305 -0
  82. package/dist/src/commands/azure/functions.js +200 -0
  83. package/dist/src/commands/azure/index.js +93 -0
  84. package/dist/src/commands/azure/storage.js +378 -0
  85. package/dist/src/commands/azure/vm.js +291 -0
  86. package/dist/src/commands/billing/index.js +224 -0
  87. package/dist/src/commands/chat.js +259 -0
  88. package/dist/src/commands/completions.js +255 -0
  89. package/dist/src/commands/config.js +291 -0
  90. package/dist/src/commands/cost/cloud-cost-estimator.js +211 -0
  91. package/dist/src/commands/cost/estimator.js +73 -0
  92. package/dist/src/commands/cost/index.js +625 -0
  93. package/dist/src/commands/cost/parsers/terraform.js +234 -0
  94. package/dist/src/commands/cost/parsers/types.js +4 -0
  95. package/dist/src/commands/cost/pricing/aws.js +501 -0
  96. package/dist/src/commands/cost/pricing/azure.js +462 -0
  97. package/dist/src/commands/cost/pricing/gcp.js +359 -0
  98. package/dist/src/commands/cost/pricing/index.js +24 -0
  99. package/dist/src/commands/demo.js +196 -0
  100. package/dist/src/commands/deploy.js +215 -0
  101. package/dist/src/commands/doctor.js +1291 -0
  102. package/dist/src/commands/drift/index.js +674 -0
  103. package/dist/src/commands/explain.js +235 -0
  104. package/dist/src/commands/export.js +120 -0
  105. package/dist/src/commands/feedback.js +319 -0
  106. package/dist/src/commands/fix.js +263 -0
  107. package/dist/src/commands/fs/index.js +338 -0
  108. package/dist/src/commands/gcp/compute.js +266 -0
  109. package/dist/src/commands/gcp/functions.js +221 -0
  110. package/dist/src/commands/gcp/gke.js +357 -0
  111. package/dist/src/commands/gcp/iam.js +295 -0
  112. package/dist/src/commands/gcp/index.js +105 -0
  113. package/dist/src/commands/gcp/storage.js +232 -0
  114. package/dist/src/commands/generate-helm.js +1026 -0
  115. package/dist/src/commands/generate-k8s.js +1263 -0
  116. package/dist/src/commands/generate-terraform.js +1058 -0
  117. package/dist/src/commands/gh/index.js +663 -0
  118. package/dist/src/commands/git/index.js +1208 -0
  119. package/dist/src/commands/helm/index.js +985 -0
  120. package/dist/src/commands/help.js +639 -0
  121. package/dist/src/commands/history.js +120 -0
  122. package/dist/src/commands/import.js +782 -0
  123. package/dist/src/commands/incident.js +144 -0
  124. package/dist/src/commands/index.js +109 -0
  125. package/dist/src/commands/init.js +955 -0
  126. package/dist/src/commands/k8s/index.js +979 -0
  127. package/dist/src/commands/login.js +588 -0
  128. package/dist/src/commands/logout.js +61 -0
  129. package/dist/src/commands/logs.js +160 -0
  130. package/dist/src/commands/onboarding.js +382 -0
  131. package/dist/src/commands/pipeline.js +153 -0
  132. package/dist/src/commands/plan/display.js +216 -0
  133. package/dist/src/commands/plan/index.js +525 -0
  134. package/dist/src/commands/plugin.js +325 -0
  135. package/dist/src/commands/preview.js +356 -0
  136. package/dist/src/commands/profile.js +297 -0
  137. package/dist/src/commands/questionnaire.js +1021 -0
  138. package/dist/src/commands/resume.js +35 -0
  139. package/dist/src/commands/rollback.js +259 -0
  140. package/dist/src/commands/rollout.js +74 -0
  141. package/dist/src/commands/runbook.js +307 -0
  142. package/dist/src/commands/schedule.js +202 -0
  143. package/dist/src/commands/status.js +213 -0
  144. package/dist/src/commands/team/index.js +309 -0
  145. package/dist/src/commands/team-context.js +200 -0
  146. package/dist/src/commands/template.js +204 -0
  147. package/dist/src/commands/tf/index.js +989 -0
  148. package/dist/src/commands/upgrade.js +515 -0
  149. package/dist/src/commands/usage/index.js +118 -0
  150. package/dist/src/commands/version.js +145 -0
  151. package/dist/src/commands/watch.js +127 -0
  152. package/dist/src/compat/index.js +2 -0
  153. package/dist/src/compat/runtime.js +10 -0
  154. package/dist/src/compat/sqlite.js +144 -0
  155. package/dist/src/config/index.js +6 -0
  156. package/dist/src/config/manager.js +469 -0
  157. package/dist/src/config/mode-store.js +57 -0
  158. package/dist/src/config/profiles.js +66 -0
  159. package/dist/src/config/safety-policy.js +251 -0
  160. package/dist/src/config/schema.js +107 -0
  161. package/dist/src/config/types.js +311 -0
  162. package/dist/src/config/workspace-state.js +38 -0
  163. package/dist/src/context/context-db.js +138 -0
  164. package/dist/src/demo/index.js +295 -0
  165. package/dist/src/demo/scenarios/full-journey.js +226 -0
  166. package/dist/src/demo/scenarios/getting-started.js +124 -0
  167. package/dist/src/demo/scenarios/helm-release.js +334 -0
  168. package/dist/src/demo/scenarios/k8s-deployment.js +190 -0
  169. package/dist/src/demo/scenarios/terraform-vpc.js +167 -0
  170. package/dist/src/demo/types.js +6 -0
  171. package/dist/src/engine/cost-estimator.js +334 -0
  172. package/dist/src/engine/diagram-generator.js +192 -0
  173. package/dist/src/engine/drift-detector.js +688 -0
  174. package/dist/src/engine/executor.js +832 -0
  175. package/dist/src/engine/index.js +39 -0
  176. package/dist/src/engine/orchestrator.js +436 -0
  177. package/dist/src/engine/planner.js +616 -0
  178. package/dist/src/engine/safety.js +609 -0
  179. package/dist/src/engine/verifier.js +664 -0
  180. package/dist/src/enterprise/audit.js +241 -0
  181. package/dist/src/enterprise/auth.js +189 -0
  182. package/dist/src/enterprise/billing.js +512 -0
  183. package/dist/src/enterprise/index.js +16 -0
  184. package/dist/src/enterprise/teams.js +315 -0
  185. package/dist/src/generator/best-practices.js +1375 -0
  186. package/dist/src/generator/helm.js +495 -0
  187. package/dist/src/generator/index.js +11 -0
  188. package/dist/src/generator/intent-parser.js +420 -0
  189. package/dist/src/generator/kubernetes.js +773 -0
  190. package/dist/src/generator/terraform.js +1472 -0
  191. package/dist/src/history/index.js +6 -0
  192. package/dist/src/history/manager.js +199 -0
  193. package/dist/src/history/types.js +6 -0
  194. package/dist/src/hooks/config.js +318 -0
  195. package/dist/src/hooks/engine.js +317 -0
  196. package/dist/src/hooks/index.js +2 -0
  197. package/dist/src/llm/auth-bridge.js +157 -0
  198. package/dist/src/llm/circuit-breaker.js +116 -0
  199. package/dist/src/llm/config-loader.js +172 -0
  200. package/dist/src/llm/cost-calculator.js +137 -0
  201. package/dist/src/llm/index.js +7 -0
  202. package/dist/src/llm/model-aliases.js +99 -0
  203. package/dist/src/llm/provider-registry.js +57 -0
  204. package/dist/src/llm/providers/anthropic.js +430 -0
  205. package/dist/src/llm/providers/bedrock.js +409 -0
  206. package/dist/src/llm/providers/google.js +344 -0
  207. package/dist/src/llm/providers/ollama.js +661 -0
  208. package/dist/src/llm/providers/openai-compatible.js +289 -0
  209. package/dist/src/llm/providers/openai.js +284 -0
  210. package/dist/src/llm/providers/openrouter.js +293 -0
  211. package/dist/src/llm/router.js +844 -0
  212. package/dist/src/llm/types.js +69 -0
  213. package/dist/src/lsp/client.js +239 -0
  214. package/dist/src/lsp/languages.js +95 -0
  215. package/dist/src/lsp/manager.js +243 -0
  216. package/dist/src/mcp/client.js +289 -0
  217. package/dist/src/mcp/index.js +5 -0
  218. package/dist/src/mcp/manager.js +113 -0
  219. package/dist/src/nimbus.js +212 -0
  220. package/dist/src/plugins/index.js +13 -0
  221. package/dist/src/plugins/loader.js +280 -0
  222. package/dist/src/plugins/manager.js +282 -0
  223. package/dist/src/plugins/types.js +23 -0
  224. package/dist/src/scanners/cicd-scanner.js +230 -0
  225. package/dist/src/scanners/cloud-scanner.js +415 -0
  226. package/dist/src/scanners/framework-scanner.js +430 -0
  227. package/dist/src/scanners/iac-scanner.js +350 -0
  228. package/dist/src/scanners/index.js +454 -0
  229. package/dist/src/scanners/language-scanner.js +258 -0
  230. package/dist/src/scanners/package-manager-scanner.js +252 -0
  231. package/dist/src/scanners/types.js +6 -0
  232. package/dist/src/sessions/manager.js +395 -0
  233. package/dist/src/sessions/types.js +4 -0
  234. package/dist/src/sharing/sync.js +238 -0
  235. package/dist/src/sharing/viewer.js +131 -0
  236. package/dist/src/snapshots/index.js +1 -0
  237. package/dist/src/snapshots/manager.js +432 -0
  238. package/dist/src/state/artifacts.js +94 -0
  239. package/dist/src/state/audit.js +73 -0
  240. package/dist/src/state/billing.js +126 -0
  241. package/dist/src/state/checkpoints.js +81 -0
  242. package/dist/src/state/config.js +58 -0
  243. package/dist/src/state/conversations.js +7 -0
  244. package/dist/src/state/credentials.js +96 -0
  245. package/dist/src/state/db.js +53 -0
  246. package/dist/src/state/index.js +23 -0
  247. package/dist/src/state/messages.js +76 -0
  248. package/dist/src/state/projects.js +92 -0
  249. package/dist/src/state/schema.js +233 -0
  250. package/dist/src/state/sessions.js +79 -0
  251. package/dist/src/state/teams.js +131 -0
  252. package/dist/src/telemetry.js +91 -0
  253. package/dist/src/tools/aws-ops.js +747 -0
  254. package/dist/src/tools/azure-ops.js +491 -0
  255. package/dist/src/tools/file-ops.js +451 -0
  256. package/dist/src/tools/gcp-ops.js +559 -0
  257. package/dist/src/tools/git-ops.js +557 -0
  258. package/dist/src/tools/github-ops.js +460 -0
  259. package/dist/src/tools/helm-ops.js +634 -0
  260. package/dist/src/tools/index.js +16 -0
  261. package/dist/src/tools/k8s-ops.js +579 -0
  262. package/dist/src/tools/schemas/converter.js +129 -0
  263. package/dist/src/tools/schemas/devops.js +3319 -0
  264. package/dist/src/tools/schemas/index.js +19 -0
  265. package/dist/src/tools/schemas/standard.js +966 -0
  266. package/dist/src/tools/schemas/types.js +409 -0
  267. package/dist/src/tools/spawn-exec.js +109 -0
  268. package/dist/src/tools/terraform-ops.js +627 -0
  269. package/dist/src/types/config.js +1 -0
  270. package/dist/src/types/drift.js +4 -0
  271. package/dist/src/types/enterprise.js +5 -0
  272. package/dist/src/types/index.js +14 -0
  273. package/dist/src/types/plan.js +1 -0
  274. package/dist/src/types/request.js +1 -0
  275. package/dist/src/types/response.js +1 -0
  276. package/dist/src/types/service.js +1 -0
  277. package/dist/src/ui/App.js +1672 -0
  278. package/dist/src/ui/DeployPreview.js +60 -0
  279. package/dist/src/ui/FileDiffModal.js +108 -0
  280. package/dist/src/ui/Header.js +46 -0
  281. package/dist/src/ui/HelpModal.js +9 -0
  282. package/dist/src/ui/InputBox.js +408 -0
  283. package/dist/src/ui/MessageList.js +795 -0
  284. package/dist/src/ui/PermissionPrompt.js +72 -0
  285. package/dist/src/ui/StatusBar.js +109 -0
  286. package/dist/src/ui/TerminalPane.js +31 -0
  287. package/dist/src/ui/ToolCallDisplay.js +303 -0
  288. package/dist/src/ui/TreePane.js +83 -0
  289. package/dist/src/ui/chat-ui.js +721 -0
  290. package/dist/src/ui/index.js +11 -0
  291. package/dist/src/ui/ink/index.js +1325 -0
  292. package/dist/src/ui/streaming.js +137 -0
  293. package/dist/src/ui/theme.js +78 -0
  294. package/dist/src/ui/types.js +7 -0
  295. package/dist/src/utils/analytics.js +61 -0
  296. package/dist/src/utils/cost-warning.js +25 -0
  297. package/dist/src/utils/env.js +42 -0
  298. package/dist/src/utils/errors.js +54 -0
  299. package/dist/src/utils/event-bus.js +22 -0
  300. package/dist/src/utils/index.js +16 -0
  301. package/dist/src/utils/logger.js +150 -0
  302. package/dist/src/utils/rate-limiter.js +90 -0
  303. package/dist/src/utils/service-auth.js +36 -0
  304. package/dist/src/utils/validation.js +39 -0
  305. package/dist/src/version.js +3 -0
  306. package/dist/src/watcher/index.js +192 -0
  307. package/dist/src/wizard/approval.js +275 -0
  308. package/dist/src/wizard/index.js +13 -0
  309. package/dist/src/wizard/prompts.js +273 -0
  310. package/dist/src/wizard/types.js +4 -0
  311. package/dist/src/wizard/ui.js +453 -0
  312. package/dist/src/wizard/wizard.js +227 -0
  313. package/package.json +22 -15
  314. package/src/__tests__/alias.test.ts +133 -0
  315. package/src/__tests__/cli-run.test.ts +237 -1
  316. package/src/__tests__/compat-sqlite.test.ts +68 -0
  317. package/src/__tests__/context-manager.test.ts +130 -0
  318. package/src/__tests__/devops-terminal-gaps.test.ts +718 -0
  319. package/src/__tests__/doctor.test.ts +48 -0
  320. package/src/__tests__/export.test.ts +236 -0
  321. package/src/__tests__/gap-11-18-20.test.ts +958 -0
  322. package/src/__tests__/helm-streaming.test.ts +127 -0
  323. package/src/__tests__/incident.test.ts +179 -0
  324. package/src/__tests__/init.test.ts +54 -3
  325. package/src/__tests__/logs.test.ts +107 -0
  326. package/src/__tests__/loop-errors.test.ts +244 -0
  327. package/src/__tests__/perf-optimizations.test.ts +847 -0
  328. package/src/__tests__/pipeline.test.ts +50 -0
  329. package/src/__tests__/polish-phase3.test.ts +340 -0
  330. package/src/__tests__/profile.test.ts +237 -0
  331. package/src/__tests__/rollback.test.ts +83 -0
  332. package/src/__tests__/runbook.test.ts +219 -0
  333. package/src/__tests__/schedule.test.ts +206 -0
  334. package/src/__tests__/sessions.test.ts +95 -0
  335. package/src/__tests__/standalone-migration.test.ts +199 -0
  336. package/src/__tests__/status.test.ts +158 -0
  337. package/src/__tests__/stream-with-tools.test.ts +49 -1
  338. package/src/__tests__/system-prompt.test.ts +81 -2
  339. package/src/__tests__/terminal-gap-v2.test.ts +395 -0
  340. package/src/__tests__/terminal-parity.test.ts +393 -0
  341. package/src/__tests__/tf-apply.test.ts +187 -0
  342. package/src/__tests__/tool-schemas.test.ts +209 -4
  343. package/src/__tests__/version-json.test.ts +184 -0
  344. package/src/__tests__/watch.test.ts +129 -0
  345. package/src/agent/compaction-agent.ts +40 -1
  346. package/src/agent/context-manager.ts +67 -3
  347. package/src/agent/deploy-preview.ts +62 -1
  348. package/src/agent/expand-files.ts +108 -0
  349. package/src/agent/loop.ts +1312 -31
  350. package/src/agent/permissions.ts +51 -4
  351. package/src/agent/system-prompt.ts +573 -19
  352. package/src/app.ts +58 -0
  353. package/src/audit/security-scanner.ts +45 -0
  354. package/src/auth/keychain.ts +82 -0
  355. package/src/cli/init.ts +378 -5
  356. package/src/cli/run.ts +407 -16
  357. package/src/cli/serve.ts +78 -1
  358. package/src/cli/web.ts +10 -6
  359. package/src/cli.ts +312 -1
  360. package/src/clients/service-discovery.ts +30 -25
  361. package/src/commands/alias.ts +100 -0
  362. package/src/commands/audit/index.ts +121 -2
  363. package/src/commands/auth-cloud.ts +113 -0
  364. package/src/commands/auth-refresh.ts +187 -0
  365. package/src/commands/aws-discover.ts +144 -251
  366. package/src/commands/aws-terraform.ts +68 -118
  367. package/src/commands/chat.ts +9 -3
  368. package/src/commands/completions.ts +268 -0
  369. package/src/commands/config.ts +26 -0
  370. package/src/commands/cost/index.ts +218 -2
  371. package/src/commands/deploy.ts +260 -0
  372. package/src/commands/doctor.ts +744 -152
  373. package/src/commands/drift/index.ts +371 -23
  374. package/src/commands/export.ts +146 -0
  375. package/src/commands/generate-k8s.ts +9 -61
  376. package/src/commands/generate-terraform.ts +191 -449
  377. package/src/commands/help.ts +212 -36
  378. package/src/commands/history.ts +8 -1
  379. package/src/commands/incident.ts +166 -0
  380. package/src/commands/init.ts +5 -0
  381. package/src/commands/login.ts +86 -1
  382. package/src/commands/logs.ts +167 -0
  383. package/src/commands/onboarding.ts +211 -34
  384. package/src/commands/pipeline.ts +186 -0
  385. package/src/commands/plugin.ts +398 -0
  386. package/src/commands/profile.ts +342 -0
  387. package/src/commands/questionnaire.ts +0 -98
  388. package/src/commands/resume.ts +26 -34
  389. package/src/commands/rollback.ts +315 -0
  390. package/src/commands/rollout.ts +88 -0
  391. package/src/commands/runbook.ts +346 -0
  392. package/src/commands/schedule.ts +236 -0
  393. package/src/commands/status.ts +252 -0
  394. package/src/commands/team-context.ts +220 -0
  395. package/src/commands/template.ts +58 -57
  396. package/src/commands/tf/index.ts +70 -11
  397. package/src/commands/upgrade.ts +57 -0
  398. package/src/commands/version.ts +54 -50
  399. package/src/commands/watch.ts +153 -0
  400. package/src/compat/sqlite.ts +75 -5
  401. package/src/config/mode-store.ts +62 -0
  402. package/src/config/profiles.ts +84 -0
  403. package/src/config/types.ts +83 -1
  404. package/src/config/workspace-state.ts +53 -0
  405. package/src/engine/cost-estimator.ts +52 -10
  406. package/src/engine/executor.ts +33 -2
  407. package/src/engine/planner.ts +68 -1
  408. package/src/generator/terraform.ts +8 -0
  409. package/src/history/manager.ts +2 -74
  410. package/src/llm/cost-calculator.ts +2 -2
  411. package/src/llm/providers/anthropic.ts +50 -21
  412. package/src/llm/router.ts +76 -7
  413. package/src/lsp/languages.ts +3 -0
  414. package/src/lsp/manager.ts +20 -4
  415. package/src/nimbus.ts +34 -1
  416. package/src/sessions/manager.ts +108 -1
  417. package/src/sharing/viewer.ts +66 -0
  418. package/src/tools/file-ops.ts +22 -0
  419. package/src/tools/schemas/devops.ts +3007 -117
  420. package/src/tools/schemas/standard.ts +5 -1
  421. package/src/tools/schemas/types.ts +31 -1
  422. package/src/tools/spawn-exec.ts +148 -0
  423. package/src/ui/App.tsx +1183 -66
  424. package/src/ui/DeployPreview.tsx +62 -57
  425. package/src/ui/FileDiffModal.tsx +162 -0
  426. package/src/ui/Header.tsx +87 -24
  427. package/src/ui/HelpModal.tsx +57 -0
  428. package/src/ui/InputBox.tsx +163 -10
  429. package/src/ui/MessageList.tsx +487 -40
  430. package/src/ui/PermissionPrompt.tsx +17 -5
  431. package/src/ui/StatusBar.tsx +122 -3
  432. package/src/ui/TerminalPane.tsx +84 -0
  433. package/src/ui/ToolCallDisplay.tsx +252 -18
  434. package/src/ui/TreePane.tsx +132 -0
  435. package/src/ui/chat-ui.ts +41 -44
  436. package/src/ui/ink/index.ts +771 -38
  437. package/src/ui/theme.ts +104 -0
  438. package/src/ui/types.ts +18 -0
  439. package/src/version.ts +1 -1
  440. package/src/watcher/index.ts +66 -15
  441. package/src/wizard/types.ts +1 -0
@@ -0,0 +1,199 @@
1
+ /**
2
+ * Standalone Migration Tests
3
+ *
4
+ * Source-level checks verifying that microservice-dependent commands
5
+ * (RestClient, CoreEngineClient, localhost:300X) have been replaced
6
+ * with standalone CLI/SQLite/generator implementations.
7
+ *
8
+ * These are intentionally static — no runtime startup needed.
9
+ */
10
+
11
+ import { readFileSync } from 'node:fs';
12
+ import { join } from 'node:path';
13
+ import { describe, it, expect } from 'vitest';
14
+
15
+ const ROOT = join(__dirname, '..', '..');
16
+
17
+ function src(relativePath: string): string {
18
+ return readFileSync(join(ROOT, relativePath), 'utf8');
19
+ }
20
+
21
+ // ---------------------------------------------------------------------------
22
+ // C1 — generate-terraform: standalone
23
+ // ---------------------------------------------------------------------------
24
+ describe('C1 — generate-terraform standalone', () => {
25
+ it('has no new RestClient call', () => {
26
+ expect(src('src/commands/generate-terraform.ts')).not.toContain('new RestClient');
27
+ });
28
+ it('imports generateTerraformProject from generator', () => {
29
+ expect(src('src/commands/generate-terraform.ts')).toContain('generateTerraformProject');
30
+ });
31
+ it('has no localhost:300 reference', () => {
32
+ expect(src('src/commands/generate-terraform.ts')).not.toMatch(/localhost:300\d/);
33
+ });
34
+ });
35
+
36
+ // ---------------------------------------------------------------------------
37
+ // C2 — generate-k8s: standalone
38
+ // ---------------------------------------------------------------------------
39
+ describe('C2 — generate-k8s standalone', () => {
40
+ it('has no new RestClient call', () => {
41
+ expect(src('src/commands/generate-k8s.ts')).not.toContain('new RestClient');
42
+ });
43
+ it('uses generateManifestsLocally or kubernetes generator', () => {
44
+ expect(src('src/commands/generate-k8s.ts')).toMatch(/generateManifestsLocally|generateK8sManifests|K8sGeneratorConfig/);
45
+ });
46
+ it('has no localhost:300 reference', () => {
47
+ expect(src('src/commands/generate-k8s.ts')).not.toMatch(/localhost:300\d/);
48
+ });
49
+ });
50
+
51
+ // ---------------------------------------------------------------------------
52
+ // C3 — aws-discover: standalone
53
+ // ---------------------------------------------------------------------------
54
+ describe('C3 — aws-discover standalone', () => {
55
+ it('has no new RestClient call', () => {
56
+ expect(src('src/commands/aws-discover.ts')).not.toContain('new RestClient');
57
+ });
58
+ it('uses aws configure or execFile for profiles', () => {
59
+ expect(src('src/commands/aws-discover.ts')).toMatch(/configure|execFile|execFileSync|cliGetAwsProfiles/);
60
+ });
61
+ it('has no localhost:300 reference', () => {
62
+ expect(src('src/commands/aws-discover.ts')).not.toMatch(/localhost:300\d/);
63
+ });
64
+ });
65
+
66
+ // ---------------------------------------------------------------------------
67
+ // C4 — aws-terraform: standalone
68
+ // ---------------------------------------------------------------------------
69
+ describe('C4 — aws-terraform standalone', () => {
70
+ it('has no new RestClient call', () => {
71
+ expect(src('src/commands/aws-terraform.ts')).not.toContain('new RestClient');
72
+ });
73
+ it('imports generateTerraformProject from generator', () => {
74
+ expect(src('src/commands/aws-terraform.ts')).toContain('generateTerraformProject');
75
+ });
76
+ it('has no localhost:300 reference', () => {
77
+ expect(src('src/commands/aws-terraform.ts')).not.toMatch(/localhost:300\d/);
78
+ });
79
+ });
80
+
81
+ // ---------------------------------------------------------------------------
82
+ // C5 — resume: standalone
83
+ // ---------------------------------------------------------------------------
84
+ describe('C5 — resume command standalone', () => {
85
+ it('has no CoreEngineClient import', () => {
86
+ expect(src('src/commands/resume.ts')).not.toContain('CoreEngineClient');
87
+ });
88
+ it('references chatCommand or getDb for session lookup', () => {
89
+ expect(src('src/commands/resume.ts')).toMatch(/chatCommand|getDb|SessionManager/);
90
+ });
91
+ it('has no localhost:300 reference', () => {
92
+ expect(src('src/commands/resume.ts')).not.toMatch(/localhost:300\d/);
93
+ });
94
+ });
95
+
96
+ // ---------------------------------------------------------------------------
97
+ // C6 — version: DevOps CLI tools (not localhost services)
98
+ // ---------------------------------------------------------------------------
99
+ describe('C6 — version command DevOps tools', () => {
100
+ it('has no localhost:300 references in fetchComponentVersions', () => {
101
+ expect(src('src/commands/version.ts')).not.toMatch(/localhost:300\d/);
102
+ });
103
+ it('includes terraform version check', () => {
104
+ expect(src('src/commands/version.ts')).toContain('terraform');
105
+ });
106
+ it('includes kubectl version check', () => {
107
+ expect(src('src/commands/version.ts')).toContain('kubectl');
108
+ });
109
+ it('includes helm version check', () => {
110
+ expect(src('src/commands/version.ts')).toContain('helm');
111
+ });
112
+ it('shows [+] / [-] icon format', () => {
113
+ expect(src('src/commands/version.ts')).toMatch(/\[\+\]|\[-\]/);
114
+ });
115
+ });
116
+
117
+ // ---------------------------------------------------------------------------
118
+ // C7 — drift: no CoreEngineClient
119
+ // ---------------------------------------------------------------------------
120
+ describe('C7 — drift no CoreEngineClient', () => {
121
+ it('drift/index.ts has no CoreEngineClient import', () => {
122
+ const content = src('src/commands/drift/index.ts');
123
+ // Must not have an import statement (comments are fine)
124
+ expect(content).not.toMatch(/^import.*CoreEngineClient/m);
125
+ });
126
+ it('drift/index.ts uses execFileSync or execFile for terraform', () => {
127
+ expect(src('src/commands/drift/index.ts')).toMatch(/execFileSync|execFile|spawnExec/);
128
+ });
129
+ });
130
+
131
+ // ---------------------------------------------------------------------------
132
+ // C8 — dead client code deleted/stubbed
133
+ // ---------------------------------------------------------------------------
134
+ describe('C8 — dead client code', () => {
135
+ it('service-discovery.ts is either deleted or a stub without ghost URLs', () => {
136
+ try {
137
+ const content = src('src/clients/service-discovery.ts');
138
+ // If it exists, it must not contain the old ghost localhost service URLs
139
+ expect(content).not.toContain('localhost:3001');
140
+ expect(content).not.toContain('localhost:3003');
141
+ expect(content).not.toContain('localhost:3009');
142
+ } catch {
143
+ // File deleted — that's fine
144
+ }
145
+ });
146
+ });
147
+
148
+ // ---------------------------------------------------------------------------
149
+ // H1 — Emoji removed, replaced with ASCII icons
150
+ // ---------------------------------------------------------------------------
151
+ describe('H1 — emoji removed from UI', () => {
152
+ it('TreePane.tsx has no folder emoji', () => {
153
+ expect(src('src/ui/TreePane.tsx')).not.toContain('\u{1F4C1}'); // 📁
154
+ });
155
+ it('TreePane.tsx uses [/] for directories', () => {
156
+ expect(src('src/ui/TreePane.tsx')).toContain('[/]');
157
+ });
158
+ it('TerminalPane.tsx has no right-pointing triangle', () => {
159
+ expect(src('src/ui/TerminalPane.tsx')).not.toContain('\u25B6'); // ▶
160
+ });
161
+ it('TerminalPane.tsx uses [>] icon', () => {
162
+ expect(src('src/ui/TerminalPane.tsx')).toContain('[>]');
163
+ });
164
+ it('StatusBar.tsx has no Greek capital delta (Δ)', () => {
165
+ expect(src('src/ui/StatusBar.tsx')).not.toContain('\u0394'); // Δ
166
+ });
167
+ it('StatusBar.tsx uses delta: text', () => {
168
+ expect(src('src/ui/StatusBar.tsx')).toContain('delta:');
169
+ });
170
+ });
171
+
172
+ // ---------------------------------------------------------------------------
173
+ // M1 — Secret redaction in spawn-exec
174
+ // ---------------------------------------------------------------------------
175
+ describe('M1 — secret redaction in spawn-exec', () => {
176
+ it('spawn-exec.ts has SECRET_PATTERNS constant', () => {
177
+ expect(src('src/tools/spawn-exec.ts')).toContain('SECRET_PATTERNS');
178
+ });
179
+ it('spawn-exec.ts has redactSecrets function', () => {
180
+ expect(src('src/tools/spawn-exec.ts')).toContain('redactSecrets');
181
+ });
182
+ it('spawn-exec.ts redacts AKIA keys', () => {
183
+ expect(src('src/tools/spawn-exec.ts')).toContain('AKIA');
184
+ });
185
+ it('spawn-exec.ts redacts Bearer tokens', () => {
186
+ expect(src('src/tools/spawn-exec.ts')).toContain('Bearer');
187
+ });
188
+ });
189
+
190
+ // ---------------------------------------------------------------------------
191
+ // M2 — Doctor shows inline versions
192
+ // ---------------------------------------------------------------------------
193
+ describe('M2 — doctor inline versions', () => {
194
+ it('doctor.ts references version in available tools output', () => {
195
+ // The available list should include t.version or similar version string
196
+ const content = src('src/commands/doctor.ts');
197
+ expect(content).toMatch(/t\.version|tool\.version|\.version\b/);
198
+ });
199
+ });
@@ -0,0 +1,158 @@
1
+ /**
2
+ * Tests for the nimbus status command (G18).
3
+ *
4
+ * The statusCommand runs concurrent CLI checks. We test the module structure
5
+ * and basic argument parsing rather than the actual CLI calls to avoid
6
+ * platform dependencies.
7
+ */
8
+
9
+ import { describe, test, expect } from 'vitest';
10
+ import { statusCommand, type StatusOptions } from '../commands/status';
11
+
12
+ describe('statusCommand (G18)', () => {
13
+ test('exports statusCommand function', () => {
14
+ expect(typeof statusCommand).toBe('function');
15
+ });
16
+
17
+ test('StatusOptions accepts json flag', () => {
18
+ const opts: StatusOptions = { json: true };
19
+ expect(opts.json).toBe(true);
20
+ });
21
+
22
+ test('StatusOptions json defaults to undefined', () => {
23
+ const opts: StatusOptions = {};
24
+ expect(opts.json).toBeUndefined();
25
+ });
26
+
27
+ test('statusCommand resolves without throwing when CLIs are unavailable', async () => {
28
+ // Mock console.log to suppress output in test
29
+ const original = console.log;
30
+ const logs: string[] = [];
31
+ console.log = (msg: string) => logs.push(msg);
32
+
33
+ try {
34
+ // Should not throw even if kubectl/terraform/aws/gcloud are not installed
35
+ // (the command catches errors gracefully)
36
+ await expect(statusCommand({ json: true })).resolves.toBeUndefined();
37
+ } finally {
38
+ console.log = original;
39
+ }
40
+ });
41
+
42
+ test('statusCommand with json option outputs valid JSON', async () => {
43
+ const original = console.log;
44
+ let jsonOutput = '';
45
+ console.log = (msg: string) => { jsonOutput = msg; };
46
+
47
+ try {
48
+ await statusCommand({ json: true });
49
+ // Should produce valid JSON
50
+ const parsed = JSON.parse(jsonOutput);
51
+ expect(parsed).toBeTypeOf('object');
52
+ // Should have an errors array
53
+ expect(Array.isArray(parsed.errors)).toBe(true);
54
+ } finally {
55
+ console.log = original;
56
+ }
57
+ });
58
+ });
59
+
60
+ describe('statusCommand C2 enhancements', () => {
61
+ test('JSON output includes model field with default fallback', async () => {
62
+ const original = console.log;
63
+ let jsonOutput = '';
64
+ console.log = (msg: string) => { jsonOutput = msg; };
65
+
66
+ try {
67
+ await statusCommand({ json: true });
68
+ const parsed = JSON.parse(jsonOutput) as Record<string, unknown>;
69
+ // model should be set (either from config.json or default 'claude-sonnet-4-6')
70
+ expect(typeof parsed.model).toBe('string');
71
+ expect((parsed.model as string).length).toBeGreaterThan(0);
72
+ } finally {
73
+ console.log = original;
74
+ }
75
+ });
76
+
77
+ test('JSON output includes provider field with default fallback', async () => {
78
+ const original = console.log;
79
+ let jsonOutput = '';
80
+ console.log = (msg: string) => { jsonOutput = msg; };
81
+
82
+ try {
83
+ await statusCommand({ json: true });
84
+ const parsed = JSON.parse(jsonOutput) as Record<string, unknown>;
85
+ // provider should be set (either from config.json or default 'anthropic')
86
+ expect(typeof parsed.provider).toBe('string');
87
+ expect((parsed.provider as string).length).toBeGreaterThan(0);
88
+ } finally {
89
+ console.log = original;
90
+ }
91
+ });
92
+
93
+ test('JSON output includes nimbusMdFound field', async () => {
94
+ const original = console.log;
95
+ let jsonOutput = '';
96
+ console.log = (msg: string) => { jsonOutput = msg; };
97
+
98
+ try {
99
+ await statusCommand({ json: true });
100
+ const parsed = JSON.parse(jsonOutput) as Record<string, unknown>;
101
+ // nimbusMdFound should be a boolean (true or undefined/false)
102
+ expect(
103
+ parsed.nimbusMdFound === undefined || typeof parsed.nimbusMdFound === 'boolean'
104
+ ).toBe(true);
105
+ } finally {
106
+ console.log = original;
107
+ }
108
+ });
109
+
110
+ test('JSON output includes sessionCount field', async () => {
111
+ const original = console.log;
112
+ let jsonOutput = '';
113
+ console.log = (msg: string) => { jsonOutput = msg; };
114
+
115
+ try {
116
+ await statusCommand({ json: true });
117
+ const parsed = JSON.parse(jsonOutput) as Record<string, unknown>;
118
+ // sessionCount should be a number or 'N/A'
119
+ expect(
120
+ typeof parsed.sessionCount === 'number' || parsed.sessionCount === 'N/A'
121
+ ).toBe(true);
122
+ } finally {
123
+ console.log = original;
124
+ }
125
+ });
126
+
127
+ test('default model is claude-sonnet-4-6 when config.json is absent', async () => {
128
+ const original = console.log;
129
+ let jsonOutput = '';
130
+ console.log = (msg: string) => { jsonOutput = msg; };
131
+
132
+ // We can test this by verifying the model is never empty
133
+ try {
134
+ await statusCommand({ json: true });
135
+ const parsed = JSON.parse(jsonOutput) as Record<string, unknown>;
136
+ // Model should be a non-empty string
137
+ expect(typeof parsed.model).toBe('string');
138
+ expect((parsed.model as string).length).toBeGreaterThan(0);
139
+ } finally {
140
+ console.log = original;
141
+ }
142
+ });
143
+
144
+ test('default provider is anthropic when config.json is absent', async () => {
145
+ const original = console.log;
146
+ let jsonOutput = '';
147
+ console.log = (msg: string) => { jsonOutput = msg; };
148
+
149
+ try {
150
+ await statusCommand({ json: true });
151
+ const parsed = JSON.parse(jsonOutput) as Record<string, unknown>;
152
+ expect(typeof parsed.provider).toBe('string');
153
+ expect((parsed.provider as string).length).toBeGreaterThan(0);
154
+ } finally {
155
+ console.log = original;
156
+ }
157
+ });
158
+ });
@@ -14,7 +14,7 @@
14
14
  * All tests use mocks -- no real API calls are made.
15
15
  */
16
16
 
17
- import { describe, test, expect, vi, beforeEach } from 'vitest';
17
+ import { describe, test, it, expect, vi, beforeEach } from 'vitest';
18
18
  import type { ToolCompletionRequest, StreamChunk } from '../llm/types';
19
19
 
20
20
  // ---------------------------------------------------------------------------
@@ -728,3 +728,51 @@ describe('OpenAICompatibleProvider.streamWithTools', () => {
728
728
  expect(createArg.max_tokens).toBe(1024);
729
729
  });
730
730
  });
731
+
732
+ // ---------------------------------------------------------------------------
733
+ // PERF-2c: Unbuffered streaming in LLMRouter fallback branch
734
+ // ---------------------------------------------------------------------------
735
+
736
+ describe('PERF-2c: LLMRouter unbuffered streaming', () => {
737
+ it('streamWithTools fallback loop no longer buffers chunks before yielding', async () => {
738
+ const { readFileSync } = await import('node:fs');
739
+ const { join } = await import('node:path');
740
+ const src = readFileSync(join(process.cwd(), 'src/llm/router.ts'), 'utf-8');
741
+ // Extract only the streamWithTools section (between the streamWithTools guards)
742
+ const swStart = src.indexOf('Use native streaming-with-tools if providers support it');
743
+ const swEnd = src.indexOf('If all providers with streamWithTools failed', swStart);
744
+ const swSection = swStart > 0 && swEnd > swStart ? src.slice(swStart, swEnd) : '';
745
+ // The streamWithTools fallback section should NOT contain buffering
746
+ expect(swSection).not.toContain('bufferedChunks');
747
+ // It should contain yield chunk directly
748
+ expect(swSection).toContain('yield chunk;');
749
+ });
750
+
751
+ it('router.ts yields each chunk immediately inside the for-await loop', async () => {
752
+ const { readFileSync } = await import('node:fs');
753
+ const { join } = await import('node:path');
754
+ const src = readFileSync(join(process.cwd(), 'src/llm/router.ts'), 'utf-8');
755
+ // The yield statement should appear inside the for-await (before circuitBreaker.recordSuccess)
756
+ expect(src).toContain('yield chunk;');
757
+ // circuitBreaker.recordSuccess comes after the loop ends (done chunk received)
758
+ expect(src).toContain('circuitBreaker.recordSuccess(p.name);');
759
+ });
760
+
761
+ it('circuitBreaker.recordSuccess is called after stream ends (not before yield)', async () => {
762
+ const { readFileSync } = await import('node:fs');
763
+ const { join } = await import('node:path');
764
+ const src = readFileSync(join(process.cwd(), 'src/llm/router.ts'), 'utf-8');
765
+ // Find the fallback for-loop block that has yield chunk
766
+ const yieldIdx = src.indexOf('yield chunk;');
767
+ const recordSuccessIdx = src.indexOf('circuitBreaker.recordSuccess(p.name);');
768
+ // recordSuccess should appear AFTER the yield in the source
769
+ expect(recordSuccessIdx).toBeGreaterThan(yieldIdx);
770
+ });
771
+
772
+ it('stream error path still calls circuitBreaker.recordFailure', async () => {
773
+ const { readFileSync } = await import('node:fs');
774
+ const { join } = await import('node:path');
775
+ const src = readFileSync(join(process.cwd(), 'src/llm/router.ts'), 'utf-8');
776
+ expect(src).toContain('circuitBreaker.recordFailure(p.name);');
777
+ });
778
+ });
@@ -9,7 +9,7 @@ import { describe, test, expect, beforeEach, afterEach } from 'vitest';
9
9
  import * as fs from 'node:fs';
10
10
  import * as path from 'node:path';
11
11
  import * as os from 'node:os';
12
- import { buildSystemPrompt, loadNimbusMd } from '../agent/system-prompt';
12
+ import { buildSystemPrompt, loadNimbusMd, extractForbiddenRules } from '../agent/system-prompt';
13
13
  import type { ToolDefinition } from '../tools/schemas/types';
14
14
  import { z } from 'zod';
15
15
 
@@ -51,7 +51,9 @@ describe('buildSystemPrompt', () => {
51
51
  test('includes base identity', () => {
52
52
  const prompt = buildSystemPrompt({ mode: 'build', tools: [] });
53
53
  expect(prompt).toContain('You are Nimbus');
54
- expect(prompt).toContain('AI-powered DevOps');
54
+ // C2: new DevOps-operator-first framing
55
+ expect(prompt).toContain('autonomous DevOps operator');
56
+ expect(prompt).toContain('RUN commands and query live state');
55
57
  });
56
58
 
57
59
  test('includes mode-specific instructions for "plan"', () => {
@@ -167,3 +169,80 @@ describe('loadNimbusMd', () => {
167
169
  expect(result).toBe('root-level');
168
170
  });
169
171
  });
172
+
173
+ // ===========================================================================
174
+ // G14: extractForbiddenRules
175
+ // ===========================================================================
176
+
177
+ describe('extractForbiddenRules (G14)', () => {
178
+ test('extracts bullet items from ## Forbidden section', () => {
179
+ const content = `
180
+ ## Safety Rules
181
+
182
+ - Do not break prod
183
+
184
+ ## Forbidden
185
+
186
+ - Never destroy the production database
187
+ - Never delete the S3 bucket
188
+ - Never run rm -rf /
189
+
190
+ ## Custom Instructions
191
+
192
+ Some other stuff
193
+ `;
194
+ const rules = extractForbiddenRules(content);
195
+ expect(rules).toHaveLength(3);
196
+ expect(rules[0]).toBe('Never destroy the production database');
197
+ expect(rules[1]).toBe('Never delete the S3 bucket');
198
+ expect(rules[2]).toBe('Never run rm -rf /');
199
+ });
200
+
201
+ test('returns empty array when no Forbidden section', () => {
202
+ const content = '## Safety Rules\n\n- Be careful\n';
203
+ expect(extractForbiddenRules(content)).toEqual([]);
204
+ });
205
+
206
+ test('ignores HTML comment lines', () => {
207
+ const content = `
208
+ ## Forbidden
209
+
210
+ <!-- Example: - Never destroy the database -->
211
+ - Never touch prod
212
+ `;
213
+ const rules = extractForbiddenRules(content);
214
+ expect(rules).toHaveLength(1);
215
+ expect(rules[0]).toBe('Never touch prod');
216
+ });
217
+
218
+ test('G14: prompt includes HARD CONSTRAINTS block when Forbidden section has entries', () => {
219
+ const nimbusContent = `
220
+ ## Forbidden
221
+
222
+ - Never destroy the production database
223
+ - Never run terraform destroy in prod
224
+ `;
225
+ const prompt = buildSystemPrompt({
226
+ mode: 'build',
227
+ tools: [],
228
+ nimbusInstructions: nimbusContent,
229
+ });
230
+ expect(prompt).toContain('HARD CONSTRAINTS');
231
+ expect(prompt).toContain('Never destroy the production database');
232
+ expect(prompt).toContain('STRICTLY FORBIDDEN');
233
+ });
234
+
235
+ test('G14: prompt does not include HARD CONSTRAINTS when Forbidden section is empty/comments', () => {
236
+ const nimbusContent = `
237
+ ## Forbidden
238
+
239
+ <!-- List operations Nimbus must never perform in this project -->
240
+ `;
241
+ const prompt = buildSystemPrompt({
242
+ mode: 'build',
243
+ tools: [],
244
+ nimbusInstructions: nimbusContent,
245
+ });
246
+ expect(prompt).not.toContain('HARD CONSTRAINTS');
247
+ });
248
+ });