@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,230 @@
1
+ /**
2
+ * CI/CD Scanner
3
+ *
4
+ * Detects CI/CD platforms and workflows in a project
5
+ */
6
+ import * as fs from 'fs';
7
+ import * as path from 'path';
8
+ const CICD_PATTERNS = [
9
+ {
10
+ name: 'github-actions',
11
+ displayName: 'GitHub Actions',
12
+ configPaths: ['.github/workflows'],
13
+ workflowPattern: /\.ya?ml$/,
14
+ },
15
+ {
16
+ name: 'gitlab-ci',
17
+ displayName: 'GitLab CI',
18
+ configPaths: ['.gitlab-ci.yml', '.gitlab-ci.yaml'],
19
+ },
20
+ {
21
+ name: 'jenkins',
22
+ displayName: 'Jenkins',
23
+ configPaths: ['Jenkinsfile', 'jenkins/Jenkinsfile', '.jenkins'],
24
+ },
25
+ {
26
+ name: 'circleci',
27
+ displayName: 'CircleCI',
28
+ configPaths: ['.circleci/config.yml', '.circleci/config.yaml'],
29
+ },
30
+ {
31
+ name: 'travis-ci',
32
+ displayName: 'Travis CI',
33
+ configPaths: ['.travis.yml', '.travis.yaml'],
34
+ },
35
+ {
36
+ name: 'azure-pipelines',
37
+ displayName: 'Azure Pipelines',
38
+ configPaths: ['azure-pipelines.yml', 'azure-pipelines.yaml', '.azure-pipelines'],
39
+ },
40
+ {
41
+ name: 'bitbucket-pipelines',
42
+ displayName: 'Bitbucket Pipelines',
43
+ configPaths: ['bitbucket-pipelines.yml', 'bitbucket-pipelines.yaml'],
44
+ },
45
+ {
46
+ name: 'drone',
47
+ displayName: 'Drone CI',
48
+ configPaths: ['.drone.yml', '.drone.yaml'],
49
+ },
50
+ {
51
+ name: 'buildkite',
52
+ displayName: 'Buildkite',
53
+ configPaths: ['.buildkite/pipeline.yml', '.buildkite/pipeline.yaml', 'buildkite.yml'],
54
+ },
55
+ {
56
+ name: 'teamcity',
57
+ displayName: 'TeamCity',
58
+ configPaths: ['.teamcity'],
59
+ },
60
+ {
61
+ name: 'concourse',
62
+ displayName: 'Concourse CI',
63
+ configPaths: ['ci/pipeline.yml', 'ci/pipeline.yaml', 'concourse.yml'],
64
+ },
65
+ {
66
+ name: 'codebuild',
67
+ displayName: 'AWS CodeBuild',
68
+ configPaths: ['buildspec.yml', 'buildspec.yaml'],
69
+ },
70
+ {
71
+ name: 'cloudbuild',
72
+ displayName: 'Google Cloud Build',
73
+ configPaths: ['cloudbuild.yaml', 'cloudbuild.yml'],
74
+ },
75
+ {
76
+ name: 'tekton',
77
+ displayName: 'Tekton',
78
+ configPaths: ['.tekton', 'tekton'],
79
+ },
80
+ {
81
+ name: 'argo-workflows',
82
+ displayName: 'Argo Workflows',
83
+ configPaths: ['.argo'],
84
+ },
85
+ {
86
+ name: 'woodpecker',
87
+ displayName: 'Woodpecker CI',
88
+ configPaths: ['.woodpecker.yml', '.woodpecker.yaml', '.woodpecker'],
89
+ },
90
+ {
91
+ name: 'earthly',
92
+ displayName: 'Earthly',
93
+ configPaths: ['Earthfile'],
94
+ },
95
+ {
96
+ name: 'dagger',
97
+ displayName: 'Dagger',
98
+ configPaths: ['dagger.json', 'dagger'],
99
+ },
100
+ {
101
+ name: 'taskfile',
102
+ displayName: 'Task (Taskfile)',
103
+ configPaths: ['Taskfile.yml', 'Taskfile.yaml', 'Taskfile.dist.yml'],
104
+ },
105
+ {
106
+ name: 'makefile',
107
+ displayName: 'Make',
108
+ configPaths: ['Makefile', 'makefile', 'GNUmakefile'],
109
+ },
110
+ {
111
+ name: 'justfile',
112
+ displayName: 'Just',
113
+ configPaths: ['justfile', 'Justfile', '.justfile'],
114
+ },
115
+ ];
116
+ export class CICDScanner {
117
+ name = 'cicd';
118
+ async scan(cwd, _options) {
119
+ const cicd = await this.detectCICD(cwd);
120
+ return {
121
+ detected: cicd.length > 0,
122
+ confidence: cicd.length > 0 ? cicd[0].confidence : 'low',
123
+ details: {
124
+ cicd,
125
+ },
126
+ };
127
+ }
128
+ async detectCICD(cwd) {
129
+ const detected = [];
130
+ for (const pattern of CICD_PATTERNS) {
131
+ const result = await this.detectCICDPlatform(cwd, pattern);
132
+ if (result) {
133
+ detected.push(result);
134
+ }
135
+ }
136
+ // Sort by confidence, then prioritize CI/CD over build tools
137
+ return detected.sort((a, b) => {
138
+ const order = { high: 3, medium: 2, low: 1 };
139
+ const confDiff = order[b.confidence] - order[a.confidence];
140
+ if (confDiff !== 0) {
141
+ return confDiff;
142
+ }
143
+ // Prioritize CI/CD platforms over local build tools
144
+ const buildTools = ['makefile', 'taskfile', 'justfile'];
145
+ const aIsBuildTool = buildTools.includes(a.platform);
146
+ const bIsBuildTool = buildTools.includes(b.platform);
147
+ if (aIsBuildTool && !bIsBuildTool) {
148
+ return 1;
149
+ }
150
+ if (!aIsBuildTool && bIsBuildTool) {
151
+ return -1;
152
+ }
153
+ return 0;
154
+ });
155
+ }
156
+ async detectCICDPlatform(cwd, pattern) {
157
+ let confidence = 'low';
158
+ const workflows = [];
159
+ for (const configPath of pattern.configPaths) {
160
+ const fullPath = path.join(cwd, configPath);
161
+ if (fs.existsSync(fullPath)) {
162
+ const stat = fs.statSync(fullPath);
163
+ if (stat.isDirectory()) {
164
+ // It's a directory, look for workflow files
165
+ try {
166
+ const files = fs.readdirSync(fullPath);
167
+ for (const file of files) {
168
+ if (pattern.workflowPattern) {
169
+ if (pattern.workflowPattern.test(file)) {
170
+ workflows.push(path.join(configPath, file));
171
+ confidence = 'high';
172
+ }
173
+ }
174
+ else {
175
+ workflows.push(path.join(configPath, file));
176
+ confidence = 'high';
177
+ }
178
+ }
179
+ }
180
+ catch {
181
+ // Ignore read errors
182
+ }
183
+ }
184
+ else {
185
+ // It's a file
186
+ workflows.push(configPath);
187
+ confidence = 'high';
188
+ }
189
+ }
190
+ }
191
+ if (workflows.length === 0) {
192
+ return null;
193
+ }
194
+ return {
195
+ platform: pattern.name,
196
+ workflows,
197
+ confidence,
198
+ };
199
+ }
200
+ /**
201
+ * Get all CI/CD workflow files
202
+ */
203
+ async getCICDFiles(cwd) {
204
+ const files = [];
205
+ const cicdInfo = await this.detectCICD(cwd);
206
+ for (const info of cicdInfo) {
207
+ files.push(...info.workflows);
208
+ }
209
+ return [...new Set(files)];
210
+ }
211
+ /**
212
+ * Get the primary CI/CD platform
213
+ */
214
+ async getPrimaryCICDPlatform(cwd) {
215
+ const cicdInfo = await this.detectCICD(cwd);
216
+ // Filter out local build tools
217
+ const buildTools = ['makefile', 'taskfile', 'justfile'];
218
+ const platforms = cicdInfo.filter(info => !buildTools.includes(info.platform));
219
+ if (platforms.length === 0) {
220
+ return null;
221
+ }
222
+ return platforms[0].platform;
223
+ }
224
+ }
225
+ /**
226
+ * Create CI/CD scanner instance
227
+ */
228
+ export function createCICDScanner() {
229
+ return new CICDScanner();
230
+ }
@@ -0,0 +1,415 @@
1
+ /**
2
+ * Cloud Scanner
3
+ *
4
+ * Detects cloud providers and configurations in a project
5
+ */
6
+ import * as fs from 'fs';
7
+ import * as path from 'path';
8
+ const CLOUD_PATTERNS = [
9
+ {
10
+ name: 'aws',
11
+ displayName: 'Amazon Web Services',
12
+ configFiles: [
13
+ '.aws/credentials',
14
+ '.aws/config',
15
+ 'samconfig.toml',
16
+ 'cdk.json',
17
+ 'serverless.yml',
18
+ 'template.yaml',
19
+ 'buildspec.yml',
20
+ ],
21
+ envVarPrefixes: ['AWS_'],
22
+ sdkPackages: [
23
+ '@aws-sdk',
24
+ 'aws-sdk',
25
+ 'boto3',
26
+ 'botocore',
27
+ 'aws-cdk-lib',
28
+ 'github.com/aws/aws-sdk-go',
29
+ ],
30
+ regions: [
31
+ 'us-east-1',
32
+ 'us-east-2',
33
+ 'us-west-1',
34
+ 'us-west-2',
35
+ 'eu-west-1',
36
+ 'eu-west-2',
37
+ 'eu-west-3',
38
+ 'eu-central-1',
39
+ 'eu-north-1',
40
+ 'ap-southeast-1',
41
+ 'ap-southeast-2',
42
+ 'ap-northeast-1',
43
+ 'ap-northeast-2',
44
+ 'ap-south-1',
45
+ 'sa-east-1',
46
+ 'ca-central-1',
47
+ ],
48
+ },
49
+ {
50
+ name: 'gcp',
51
+ displayName: 'Google Cloud Platform',
52
+ configFiles: ['.gcloud/credentials.json', 'app.yaml', 'cloudbuild.yaml', 'cloudrun.yaml'],
53
+ envVarPrefixes: ['GOOGLE_', 'GCLOUD_', 'GCP_'],
54
+ sdkPackages: [
55
+ '@google-cloud',
56
+ 'google-cloud',
57
+ 'google-api-python-client',
58
+ 'cloud.google.com/go',
59
+ ],
60
+ regions: [
61
+ 'us-central1',
62
+ 'us-east1',
63
+ 'us-east4',
64
+ 'us-west1',
65
+ 'us-west2',
66
+ 'europe-west1',
67
+ 'europe-west2',
68
+ 'europe-west3',
69
+ 'europe-west4',
70
+ 'asia-east1',
71
+ 'asia-east2',
72
+ 'asia-northeast1',
73
+ 'asia-southeast1',
74
+ ],
75
+ },
76
+ {
77
+ name: 'azure',
78
+ displayName: 'Microsoft Azure',
79
+ configFiles: ['.azure/credentials', 'azure-pipelines.yml', 'host.json', 'local.settings.json'],
80
+ envVarPrefixes: ['AZURE_', 'ARM_'],
81
+ sdkPackages: [
82
+ '@azure',
83
+ 'azure-sdk',
84
+ 'azure-identity',
85
+ 'azure-mgmt',
86
+ 'github.com/Azure/azure-sdk-for-go',
87
+ ],
88
+ regions: [
89
+ 'eastus',
90
+ 'eastus2',
91
+ 'westus',
92
+ 'westus2',
93
+ 'centralus',
94
+ 'westeurope',
95
+ 'northeurope',
96
+ 'uksouth',
97
+ 'ukwest',
98
+ 'southeastasia',
99
+ 'eastasia',
100
+ 'japaneast',
101
+ 'australiaeast',
102
+ ],
103
+ },
104
+ {
105
+ name: 'digitalocean',
106
+ displayName: 'DigitalOcean',
107
+ configFiles: ['.do/app.yaml', 'do.yaml'],
108
+ envVarPrefixes: ['DIGITALOCEAN_', 'DO_'],
109
+ sdkPackages: ['digitalocean', 'do-spaces'],
110
+ regions: [
111
+ 'nyc1',
112
+ 'nyc2',
113
+ 'nyc3',
114
+ 'sfo1',
115
+ 'sfo2',
116
+ 'sfo3',
117
+ 'ams2',
118
+ 'ams3',
119
+ 'sgp1',
120
+ 'lon1',
121
+ 'fra1',
122
+ 'tor1',
123
+ 'blr1',
124
+ ],
125
+ },
126
+ {
127
+ name: 'heroku',
128
+ displayName: 'Heroku',
129
+ configFiles: ['Procfile', 'app.json', 'heroku.yml'],
130
+ envVarPrefixes: ['HEROKU_'],
131
+ sdkPackages: [],
132
+ regions: ['us', 'eu'],
133
+ },
134
+ {
135
+ name: 'vercel',
136
+ displayName: 'Vercel',
137
+ configFiles: ['vercel.json', '.vercel/project.json'],
138
+ envVarPrefixes: ['VERCEL_'],
139
+ sdkPackages: ['vercel', '@vercel/node'],
140
+ regions: ['iad1', 'sfo1', 'hnd1', 'cdg1'],
141
+ },
142
+ {
143
+ name: 'netlify',
144
+ displayName: 'Netlify',
145
+ configFiles: ['netlify.toml', '.netlify/state.json'],
146
+ envVarPrefixes: ['NETLIFY_'],
147
+ sdkPackages: ['netlify', '@netlify/functions'],
148
+ regions: [],
149
+ },
150
+ {
151
+ name: 'cloudflare',
152
+ displayName: 'Cloudflare',
153
+ configFiles: ['wrangler.toml', 'wrangler.json'],
154
+ envVarPrefixes: ['CLOUDFLARE_', 'CF_'],
155
+ sdkPackages: ['@cloudflare/workers-types', 'wrangler'],
156
+ regions: [],
157
+ },
158
+ {
159
+ name: 'fly',
160
+ displayName: 'Fly.io',
161
+ configFiles: ['fly.toml'],
162
+ envVarPrefixes: ['FLY_'],
163
+ sdkPackages: ['flyctl'],
164
+ regions: ['iad', 'lax', 'sjc', 'ord', 'dfw', 'sea', 'lhr', 'ams', 'fra', 'sin', 'syd', 'nrt'],
165
+ },
166
+ {
167
+ name: 'render',
168
+ displayName: 'Render',
169
+ configFiles: ['render.yaml'],
170
+ envVarPrefixes: ['RENDER_'],
171
+ sdkPackages: [],
172
+ regions: ['oregon', 'ohio', 'virginia', 'frankfurt', 'singapore'],
173
+ },
174
+ {
175
+ name: 'railway',
176
+ displayName: 'Railway',
177
+ configFiles: ['railway.json', 'railway.toml'],
178
+ envVarPrefixes: ['RAILWAY_'],
179
+ sdkPackages: [],
180
+ regions: [],
181
+ },
182
+ ];
183
+ export class CloudScanner {
184
+ name = 'cloud';
185
+ async scan(cwd, _options) {
186
+ const cloud = await this.detectCloud(cwd);
187
+ return {
188
+ detected: cloud.length > 0,
189
+ confidence: cloud.length > 0 ? cloud[0].confidence : 'low',
190
+ details: {
191
+ cloud,
192
+ },
193
+ };
194
+ }
195
+ async detectCloud(cwd) {
196
+ const detected = [];
197
+ // Load package.json for SDK detection
198
+ const packageJson = this.loadPackageJson(cwd);
199
+ const requirements = this.loadRequirements(cwd);
200
+ const goMod = this.loadGoMod(cwd);
201
+ const terraformFiles = this.scanTerraformProviders(cwd);
202
+ for (const pattern of CLOUD_PATTERNS) {
203
+ const result = await this.detectCloudProvider(cwd, pattern, {
204
+ packageJson,
205
+ requirements,
206
+ goMod,
207
+ terraformFiles,
208
+ });
209
+ if (result) {
210
+ detected.push(result);
211
+ }
212
+ }
213
+ // Sort by confidence
214
+ return detected.sort((a, b) => {
215
+ const order = { high: 3, medium: 2, low: 1 };
216
+ return order[b.confidence] - order[a.confidence];
217
+ });
218
+ }
219
+ loadPackageJson(cwd) {
220
+ const packageJsonPath = path.join(cwd, 'package.json');
221
+ if (fs.existsSync(packageJsonPath)) {
222
+ try {
223
+ return JSON.parse(fs.readFileSync(packageJsonPath, 'utf-8'));
224
+ }
225
+ catch {
226
+ return null;
227
+ }
228
+ }
229
+ return null;
230
+ }
231
+ loadRequirements(cwd) {
232
+ const requirementsPath = path.join(cwd, 'requirements.txt');
233
+ if (fs.existsSync(requirementsPath)) {
234
+ try {
235
+ return fs.readFileSync(requirementsPath, 'utf-8');
236
+ }
237
+ catch {
238
+ return null;
239
+ }
240
+ }
241
+ return null;
242
+ }
243
+ loadGoMod(cwd) {
244
+ const goModPath = path.join(cwd, 'go.mod');
245
+ if (fs.existsSync(goModPath)) {
246
+ try {
247
+ return fs.readFileSync(goModPath, 'utf-8');
248
+ }
249
+ catch {
250
+ return null;
251
+ }
252
+ }
253
+ return null;
254
+ }
255
+ scanTerraformProviders(cwd) {
256
+ const result = { providers: [], regions: [] };
257
+ const tfDirs = ['.', 'terraform', 'infra', 'infrastructure'];
258
+ for (const dir of tfDirs) {
259
+ const dirPath = path.join(cwd, dir);
260
+ if (!fs.existsSync(dirPath)) {
261
+ continue;
262
+ }
263
+ try {
264
+ const files = fs.readdirSync(dirPath);
265
+ for (const file of files) {
266
+ if (file.endsWith('.tf')) {
267
+ const content = fs.readFileSync(path.join(dirPath, file), 'utf-8');
268
+ // Check for providers
269
+ if (content.includes('provider "aws"') ||
270
+ content.includes('source = "hashicorp/aws"')) {
271
+ if (!result.providers.includes('aws')) {
272
+ result.providers.push('aws');
273
+ }
274
+ }
275
+ if (content.includes('provider "google"') ||
276
+ content.includes('source = "hashicorp/google"')) {
277
+ if (!result.providers.includes('gcp')) {
278
+ result.providers.push('gcp');
279
+ }
280
+ }
281
+ if (content.includes('provider "azurerm"') ||
282
+ content.includes('source = "hashicorp/azurerm"')) {
283
+ if (!result.providers.includes('azure')) {
284
+ result.providers.push('azure');
285
+ }
286
+ }
287
+ // Extract regions
288
+ const regionMatches = content.match(/region\s*=\s*["']([^"']+)["']/g);
289
+ if (regionMatches) {
290
+ for (const match of regionMatches) {
291
+ const region = match.match(/["']([^"']+)["']/)?.[1];
292
+ if (region && !result.regions.includes(region)) {
293
+ result.regions.push(region);
294
+ }
295
+ }
296
+ }
297
+ }
298
+ }
299
+ }
300
+ catch {
301
+ // Ignore errors
302
+ }
303
+ }
304
+ return result;
305
+ }
306
+ async detectCloudProvider(cwd, pattern, deps) {
307
+ let confidence = 'low';
308
+ const detectedRegions = [];
309
+ const detectedServices = [];
310
+ // Check Terraform providers first (highest confidence)
311
+ if (deps.terraformFiles.providers.includes(pattern.name)) {
312
+ confidence = 'high';
313
+ detectedRegions.push(...deps.terraformFiles.regions.filter(r => pattern.regions.includes(r)));
314
+ }
315
+ // Check config files
316
+ for (const configFile of pattern.configFiles) {
317
+ const configPath = path.join(cwd, configFile);
318
+ if (fs.existsSync(configPath)) {
319
+ confidence = 'high';
320
+ break;
321
+ }
322
+ }
323
+ // Check SDK packages in package.json
324
+ if (deps.packageJson) {
325
+ const allDeps = {
326
+ ...(deps.packageJson.dependencies || {}),
327
+ ...(deps.packageJson.devDependencies || {}),
328
+ };
329
+ for (const pkg of pattern.sdkPackages) {
330
+ for (const depName of Object.keys(allDeps)) {
331
+ if (depName.startsWith(pkg) || depName === pkg) {
332
+ if (confidence === 'low') {
333
+ confidence = 'medium';
334
+ }
335
+ detectedServices.push(depName);
336
+ }
337
+ }
338
+ }
339
+ }
340
+ // Check Python requirements
341
+ if (deps.requirements) {
342
+ for (const pkg of pattern.sdkPackages) {
343
+ if (deps.requirements.toLowerCase().includes(pkg.toLowerCase())) {
344
+ if (confidence === 'low') {
345
+ confidence = 'medium';
346
+ }
347
+ }
348
+ }
349
+ }
350
+ // Check Go modules
351
+ if (deps.goMod) {
352
+ for (const pkg of pattern.sdkPackages) {
353
+ if (deps.goMod.includes(pkg)) {
354
+ if (confidence === 'low') {
355
+ confidence = 'medium';
356
+ }
357
+ }
358
+ }
359
+ }
360
+ // Check for .env files with cloud env vars
361
+ const envFiles = ['.env', '.env.local', '.env.development', '.env.production'];
362
+ for (const envFile of envFiles) {
363
+ const envPath = path.join(cwd, envFile);
364
+ if (fs.existsSync(envPath)) {
365
+ try {
366
+ const content = fs.readFileSync(envPath, 'utf-8');
367
+ for (const prefix of pattern.envVarPrefixes) {
368
+ if (content.includes(prefix)) {
369
+ if (confidence === 'low') {
370
+ confidence = 'medium';
371
+ }
372
+ break;
373
+ }
374
+ }
375
+ }
376
+ catch {
377
+ // Ignore errors
378
+ }
379
+ }
380
+ }
381
+ if (confidence === 'low') {
382
+ return null;
383
+ }
384
+ return {
385
+ provider: pattern.name,
386
+ regions: [...new Set(detectedRegions)],
387
+ services: [...new Set(detectedServices)],
388
+ confidence,
389
+ };
390
+ }
391
+ /**
392
+ * Get all detected cloud providers
393
+ */
394
+ async getProviders(cwd) {
395
+ const cloudInfo = await this.detectCloud(cwd);
396
+ return cloudInfo.map(info => info.provider);
397
+ }
398
+ /**
399
+ * Get all detected cloud regions
400
+ */
401
+ async getRegions(cwd) {
402
+ const cloudInfo = await this.detectCloud(cwd);
403
+ const regions = [];
404
+ for (const info of cloudInfo) {
405
+ regions.push(...info.regions);
406
+ }
407
+ return [...new Set(regions)];
408
+ }
409
+ }
410
+ /**
411
+ * Create cloud scanner instance
412
+ */
413
+ export function createCloudScanner() {
414
+ return new CloudScanner();
415
+ }