@build-astron-co/nimbus 0.2.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 (313) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +628 -0
  3. package/bin/nimbus +38 -0
  4. package/package.json +80 -0
  5. package/src/__tests__/app.test.ts +76 -0
  6. package/src/__tests__/audit.test.ts +877 -0
  7. package/src/__tests__/circuit-breaker.test.ts +116 -0
  8. package/src/__tests__/cli-run.test.ts +115 -0
  9. package/src/__tests__/context-manager.test.ts +502 -0
  10. package/src/__tests__/context.test.ts +242 -0
  11. package/src/__tests__/enterprise.test.ts +401 -0
  12. package/src/__tests__/generator.test.ts +433 -0
  13. package/src/__tests__/hooks.test.ts +582 -0
  14. package/src/__tests__/init.test.ts +436 -0
  15. package/src/__tests__/intent-parser.test.ts +229 -0
  16. package/src/__tests__/llm-router.test.ts +209 -0
  17. package/src/__tests__/lsp.test.ts +293 -0
  18. package/src/__tests__/modes.test.ts +336 -0
  19. package/src/__tests__/permissions.test.ts +338 -0
  20. package/src/__tests__/serve.test.ts +275 -0
  21. package/src/__tests__/sessions.test.ts +227 -0
  22. package/src/__tests__/sharing.test.ts +288 -0
  23. package/src/__tests__/snapshots.test.ts +581 -0
  24. package/src/__tests__/state-db.test.ts +334 -0
  25. package/src/__tests__/stream-with-tools.test.ts +732 -0
  26. package/src/__tests__/subagents.test.ts +176 -0
  27. package/src/__tests__/system-prompt.test.ts +169 -0
  28. package/src/__tests__/tool-converter.test.ts +256 -0
  29. package/src/__tests__/tool-schemas.test.ts +397 -0
  30. package/src/__tests__/tools.test.ts +143 -0
  31. package/src/__tests__/version.test.ts +49 -0
  32. package/src/agent/compaction-agent.ts +227 -0
  33. package/src/agent/context-manager.ts +435 -0
  34. package/src/agent/context.ts +427 -0
  35. package/src/agent/deploy-preview.ts +426 -0
  36. package/src/agent/index.ts +68 -0
  37. package/src/agent/loop.ts +717 -0
  38. package/src/agent/modes.ts +429 -0
  39. package/src/agent/permissions.ts +466 -0
  40. package/src/agent/subagents/base.ts +116 -0
  41. package/src/agent/subagents/cost.ts +51 -0
  42. package/src/agent/subagents/explore.ts +42 -0
  43. package/src/agent/subagents/general.ts +54 -0
  44. package/src/agent/subagents/index.ts +102 -0
  45. package/src/agent/subagents/infra.ts +59 -0
  46. package/src/agent/subagents/security.ts +69 -0
  47. package/src/agent/system-prompt.ts +436 -0
  48. package/src/app.ts +122 -0
  49. package/src/audit/activity-log.ts +290 -0
  50. package/src/audit/compliance-checker.ts +540 -0
  51. package/src/audit/cost-tracker.ts +318 -0
  52. package/src/audit/index.ts +23 -0
  53. package/src/audit/security-scanner.ts +596 -0
  54. package/src/auth/guard.ts +75 -0
  55. package/src/auth/index.ts +56 -0
  56. package/src/auth/oauth.ts +455 -0
  57. package/src/auth/providers.ts +470 -0
  58. package/src/auth/sso.ts +113 -0
  59. package/src/auth/store.ts +505 -0
  60. package/src/auth/types.ts +187 -0
  61. package/src/build.ts +141 -0
  62. package/src/cli/index.ts +16 -0
  63. package/src/cli/init.ts +854 -0
  64. package/src/cli/openapi-spec.ts +356 -0
  65. package/src/cli/run.ts +237 -0
  66. package/src/cli/serve-auth.ts +80 -0
  67. package/src/cli/serve.ts +462 -0
  68. package/src/cli/web.ts +67 -0
  69. package/src/cli.ts +1417 -0
  70. package/src/clients/core-engine-client.ts +227 -0
  71. package/src/clients/enterprise-client.ts +334 -0
  72. package/src/clients/generator-client.ts +351 -0
  73. package/src/clients/git-client.ts +627 -0
  74. package/src/clients/github-client.ts +410 -0
  75. package/src/clients/helm-client.ts +504 -0
  76. package/src/clients/index.ts +80 -0
  77. package/src/clients/k8s-client.ts +497 -0
  78. package/src/clients/llm-client.ts +161 -0
  79. package/src/clients/rest-client.ts +130 -0
  80. package/src/clients/service-discovery.ts +33 -0
  81. package/src/clients/terraform-client.ts +482 -0
  82. package/src/clients/tools-client.ts +1843 -0
  83. package/src/clients/ws-client.ts +115 -0
  84. package/src/commands/analyze/index.ts +352 -0
  85. package/src/commands/apply/helm.ts +473 -0
  86. package/src/commands/apply/index.ts +213 -0
  87. package/src/commands/apply/k8s.ts +454 -0
  88. package/src/commands/apply/terraform.ts +582 -0
  89. package/src/commands/ask.ts +167 -0
  90. package/src/commands/audit/index.ts +238 -0
  91. package/src/commands/auth-cloud.ts +294 -0
  92. package/src/commands/auth-list.ts +134 -0
  93. package/src/commands/auth-profile.ts +121 -0
  94. package/src/commands/auth-status.ts +141 -0
  95. package/src/commands/aws/ec2.ts +501 -0
  96. package/src/commands/aws/iam.ts +397 -0
  97. package/src/commands/aws/index.ts +133 -0
  98. package/src/commands/aws/lambda.ts +396 -0
  99. package/src/commands/aws/rds.ts +439 -0
  100. package/src/commands/aws/s3.ts +439 -0
  101. package/src/commands/aws/vpc.ts +393 -0
  102. package/src/commands/aws-discover.ts +649 -0
  103. package/src/commands/aws-terraform.ts +805 -0
  104. package/src/commands/azure/aks.ts +376 -0
  105. package/src/commands/azure/functions.ts +253 -0
  106. package/src/commands/azure/index.ts +116 -0
  107. package/src/commands/azure/storage.ts +478 -0
  108. package/src/commands/azure/vm.ts +355 -0
  109. package/src/commands/billing/index.ts +256 -0
  110. package/src/commands/chat.ts +314 -0
  111. package/src/commands/config.ts +346 -0
  112. package/src/commands/cost/cloud-cost-estimator.ts +266 -0
  113. package/src/commands/cost/estimator.ts +79 -0
  114. package/src/commands/cost/index.ts +594 -0
  115. package/src/commands/cost/parsers/terraform.ts +273 -0
  116. package/src/commands/cost/parsers/types.ts +25 -0
  117. package/src/commands/cost/pricing/aws.ts +544 -0
  118. package/src/commands/cost/pricing/azure.ts +499 -0
  119. package/src/commands/cost/pricing/gcp.ts +396 -0
  120. package/src/commands/cost/pricing/index.ts +40 -0
  121. package/src/commands/demo.ts +250 -0
  122. package/src/commands/doctor.ts +794 -0
  123. package/src/commands/drift/index.ts +439 -0
  124. package/src/commands/explain.ts +277 -0
  125. package/src/commands/feedback.ts +389 -0
  126. package/src/commands/fix.ts +324 -0
  127. package/src/commands/fs/index.ts +402 -0
  128. package/src/commands/gcp/compute.ts +325 -0
  129. package/src/commands/gcp/functions.ts +271 -0
  130. package/src/commands/gcp/gke.ts +438 -0
  131. package/src/commands/gcp/iam.ts +344 -0
  132. package/src/commands/gcp/index.ts +129 -0
  133. package/src/commands/gcp/storage.ts +284 -0
  134. package/src/commands/generate-helm.ts +1249 -0
  135. package/src/commands/generate-k8s.ts +1560 -0
  136. package/src/commands/generate-terraform.ts +1460 -0
  137. package/src/commands/gh/index.ts +863 -0
  138. package/src/commands/git/index.ts +1343 -0
  139. package/src/commands/helm/index.ts +1126 -0
  140. package/src/commands/help.ts +539 -0
  141. package/src/commands/history.ts +142 -0
  142. package/src/commands/import.ts +868 -0
  143. package/src/commands/index.ts +367 -0
  144. package/src/commands/init.ts +1046 -0
  145. package/src/commands/k8s/index.ts +1137 -0
  146. package/src/commands/login.ts +631 -0
  147. package/src/commands/logout.ts +83 -0
  148. package/src/commands/onboarding.ts +228 -0
  149. package/src/commands/plan/display.ts +279 -0
  150. package/src/commands/plan/index.ts +599 -0
  151. package/src/commands/preview.ts +452 -0
  152. package/src/commands/questionnaire.ts +1270 -0
  153. package/src/commands/resume.ts +55 -0
  154. package/src/commands/team/index.ts +346 -0
  155. package/src/commands/template.ts +232 -0
  156. package/src/commands/tf/index.ts +1034 -0
  157. package/src/commands/upgrade.ts +550 -0
  158. package/src/commands/usage/index.ts +134 -0
  159. package/src/commands/version.ts +170 -0
  160. package/src/compat/index.ts +2 -0
  161. package/src/compat/runtime.ts +12 -0
  162. package/src/compat/sqlite.ts +107 -0
  163. package/src/config/index.ts +17 -0
  164. package/src/config/manager.ts +530 -0
  165. package/src/config/safety-policy.ts +358 -0
  166. package/src/config/schema.ts +125 -0
  167. package/src/config/types.ts +527 -0
  168. package/src/context/context-db.ts +199 -0
  169. package/src/demo/index.ts +349 -0
  170. package/src/demo/scenarios/full-journey.ts +229 -0
  171. package/src/demo/scenarios/getting-started.ts +127 -0
  172. package/src/demo/scenarios/helm-release.ts +341 -0
  173. package/src/demo/scenarios/k8s-deployment.ts +194 -0
  174. package/src/demo/scenarios/terraform-vpc.ts +170 -0
  175. package/src/demo/types.ts +92 -0
  176. package/src/engine/cost-estimator.ts +438 -0
  177. package/src/engine/diagram-generator.ts +256 -0
  178. package/src/engine/drift-detector.ts +902 -0
  179. package/src/engine/executor.ts +1035 -0
  180. package/src/engine/index.ts +76 -0
  181. package/src/engine/orchestrator.ts +636 -0
  182. package/src/engine/planner.ts +720 -0
  183. package/src/engine/safety.ts +743 -0
  184. package/src/engine/verifier.ts +770 -0
  185. package/src/enterprise/audit.ts +348 -0
  186. package/src/enterprise/auth.ts +270 -0
  187. package/src/enterprise/billing.ts +822 -0
  188. package/src/enterprise/index.ts +17 -0
  189. package/src/enterprise/teams.ts +443 -0
  190. package/src/generator/best-practices.ts +1608 -0
  191. package/src/generator/helm.ts +630 -0
  192. package/src/generator/index.ts +37 -0
  193. package/src/generator/intent-parser.ts +514 -0
  194. package/src/generator/kubernetes.ts +976 -0
  195. package/src/generator/terraform.ts +1867 -0
  196. package/src/history/index.ts +8 -0
  197. package/src/history/manager.ts +322 -0
  198. package/src/history/types.ts +34 -0
  199. package/src/hooks/config.ts +432 -0
  200. package/src/hooks/engine.ts +391 -0
  201. package/src/hooks/index.ts +4 -0
  202. package/src/llm/auth-bridge.ts +198 -0
  203. package/src/llm/circuit-breaker.ts +140 -0
  204. package/src/llm/config-loader.ts +201 -0
  205. package/src/llm/cost-calculator.ts +171 -0
  206. package/src/llm/index.ts +8 -0
  207. package/src/llm/model-aliases.ts +115 -0
  208. package/src/llm/provider-registry.ts +63 -0
  209. package/src/llm/providers/anthropic.ts +433 -0
  210. package/src/llm/providers/bedrock.ts +477 -0
  211. package/src/llm/providers/google.ts +405 -0
  212. package/src/llm/providers/ollama.ts +767 -0
  213. package/src/llm/providers/openai-compatible.ts +340 -0
  214. package/src/llm/providers/openai.ts +328 -0
  215. package/src/llm/providers/openrouter.ts +338 -0
  216. package/src/llm/router.ts +1035 -0
  217. package/src/llm/types.ts +232 -0
  218. package/src/lsp/client.ts +298 -0
  219. package/src/lsp/languages.ts +116 -0
  220. package/src/lsp/manager.ts +278 -0
  221. package/src/mcp/client.ts +402 -0
  222. package/src/mcp/index.ts +5 -0
  223. package/src/mcp/manager.ts +133 -0
  224. package/src/nimbus.ts +214 -0
  225. package/src/plugins/index.ts +27 -0
  226. package/src/plugins/loader.ts +334 -0
  227. package/src/plugins/manager.ts +376 -0
  228. package/src/plugins/types.ts +284 -0
  229. package/src/scanners/cicd-scanner.ts +258 -0
  230. package/src/scanners/cloud-scanner.ts +466 -0
  231. package/src/scanners/framework-scanner.ts +469 -0
  232. package/src/scanners/iac-scanner.ts +388 -0
  233. package/src/scanners/index.ts +539 -0
  234. package/src/scanners/language-scanner.ts +276 -0
  235. package/src/scanners/package-manager-scanner.ts +277 -0
  236. package/src/scanners/types.ts +172 -0
  237. package/src/sessions/manager.ts +365 -0
  238. package/src/sessions/types.ts +44 -0
  239. package/src/sharing/sync.ts +296 -0
  240. package/src/sharing/viewer.ts +97 -0
  241. package/src/snapshots/index.ts +2 -0
  242. package/src/snapshots/manager.ts +530 -0
  243. package/src/state/artifacts.ts +147 -0
  244. package/src/state/audit.ts +137 -0
  245. package/src/state/billing.ts +240 -0
  246. package/src/state/checkpoints.ts +117 -0
  247. package/src/state/config.ts +67 -0
  248. package/src/state/conversations.ts +14 -0
  249. package/src/state/credentials.ts +154 -0
  250. package/src/state/db.ts +58 -0
  251. package/src/state/index.ts +26 -0
  252. package/src/state/messages.ts +115 -0
  253. package/src/state/projects.ts +123 -0
  254. package/src/state/schema.ts +236 -0
  255. package/src/state/sessions.ts +147 -0
  256. package/src/state/teams.ts +200 -0
  257. package/src/telemetry.ts +108 -0
  258. package/src/tools/aws-ops.ts +952 -0
  259. package/src/tools/azure-ops.ts +579 -0
  260. package/src/tools/file-ops.ts +593 -0
  261. package/src/tools/gcp-ops.ts +625 -0
  262. package/src/tools/git-ops.ts +773 -0
  263. package/src/tools/github-ops.ts +799 -0
  264. package/src/tools/helm-ops.ts +943 -0
  265. package/src/tools/index.ts +17 -0
  266. package/src/tools/k8s-ops.ts +819 -0
  267. package/src/tools/schemas/converter.ts +184 -0
  268. package/src/tools/schemas/devops.ts +612 -0
  269. package/src/tools/schemas/index.ts +73 -0
  270. package/src/tools/schemas/standard.ts +1144 -0
  271. package/src/tools/schemas/types.ts +705 -0
  272. package/src/tools/terraform-ops.ts +862 -0
  273. package/src/types/ambient.d.ts +193 -0
  274. package/src/types/config.ts +83 -0
  275. package/src/types/drift.ts +116 -0
  276. package/src/types/enterprise.ts +335 -0
  277. package/src/types/index.ts +20 -0
  278. package/src/types/plan.ts +44 -0
  279. package/src/types/request.ts +65 -0
  280. package/src/types/response.ts +54 -0
  281. package/src/types/service.ts +51 -0
  282. package/src/ui/App.tsx +997 -0
  283. package/src/ui/DeployPreview.tsx +169 -0
  284. package/src/ui/Header.tsx +68 -0
  285. package/src/ui/InputBox.tsx +350 -0
  286. package/src/ui/MessageList.tsx +585 -0
  287. package/src/ui/PermissionPrompt.tsx +151 -0
  288. package/src/ui/StatusBar.tsx +158 -0
  289. package/src/ui/ToolCallDisplay.tsx +409 -0
  290. package/src/ui/chat-ui.ts +853 -0
  291. package/src/ui/index.ts +33 -0
  292. package/src/ui/ink/index.ts +711 -0
  293. package/src/ui/streaming.ts +176 -0
  294. package/src/ui/types.ts +57 -0
  295. package/src/utils/analytics.ts +72 -0
  296. package/src/utils/cost-warning.ts +27 -0
  297. package/src/utils/env.ts +46 -0
  298. package/src/utils/errors.ts +69 -0
  299. package/src/utils/event-bus.ts +38 -0
  300. package/src/utils/index.ts +24 -0
  301. package/src/utils/logger.ts +171 -0
  302. package/src/utils/rate-limiter.ts +121 -0
  303. package/src/utils/service-auth.ts +49 -0
  304. package/src/utils/validation.ts +53 -0
  305. package/src/version.ts +4 -0
  306. package/src/watcher/index.ts +163 -0
  307. package/src/wizard/approval.ts +383 -0
  308. package/src/wizard/index.ts +25 -0
  309. package/src/wizard/prompts.ts +338 -0
  310. package/src/wizard/types.ts +171 -0
  311. package/src/wizard/ui.ts +556 -0
  312. package/src/wizard/wizard.ts +304 -0
  313. package/tsconfig.json +24 -0
@@ -0,0 +1,501 @@
1
+ /**
2
+ * AWS EC2 Commands
3
+ *
4
+ * EC2 instance operations with cost warnings before billable actions
5
+ *
6
+ * Usage:
7
+ * nimbus aws ec2 list
8
+ * nimbus aws ec2 describe <instance-id>
9
+ * nimbus aws ec2 start <instance-id>
10
+ * nimbus aws ec2 stop <instance-id>
11
+ * nimbus aws ec2 terminate <instance-id>
12
+ */
13
+
14
+ import { logger } from '../../utils';
15
+ import { ui } from '../../wizard/ui';
16
+ import { confirm } from '../../wizard/prompts';
17
+ import {
18
+ loadSafetyPolicy,
19
+ evaluateSafety,
20
+ type SafetyContext,
21
+ type SafetyCheckResult,
22
+ } from '../../config/safety-policy';
23
+ import { promptForApproval } from '../../wizard/approval';
24
+ import { estimateCloudCost, formatCostWarning } from '../cost/cloud-cost-estimator';
25
+ import type { AwsCommandOptions } from './index';
26
+
27
+ interface EC2Instance {
28
+ InstanceId: string;
29
+ InstanceType: string;
30
+ State: { Name: string };
31
+ PublicIpAddress?: string;
32
+ PrivateIpAddress?: string;
33
+ Tags?: Array<{ Key: string; Value: string }>;
34
+ LaunchTime?: string;
35
+ }
36
+
37
+ /**
38
+ * EC2 command router
39
+ */
40
+ export async function ec2Command(
41
+ action: string,
42
+ args: string[],
43
+ options: AwsCommandOptions
44
+ ): Promise<void> {
45
+ logger.info('Running EC2 command', { action, args, options });
46
+
47
+ switch (action) {
48
+ case 'list':
49
+ case 'ls':
50
+ await listInstances(options);
51
+ break;
52
+
53
+ case 'describe':
54
+ if (!args[0]) {
55
+ ui.error('Instance ID is required');
56
+ ui.print('Usage: nimbus aws ec2 describe <instance-id>');
57
+ return;
58
+ }
59
+ await describeInstance(args[0], options);
60
+ break;
61
+
62
+ case 'start':
63
+ if (!args[0]) {
64
+ ui.error('Instance ID is required');
65
+ ui.print('Usage: nimbus aws ec2 start <instance-id>');
66
+ return;
67
+ }
68
+ await startInstance(args[0], options);
69
+ break;
70
+
71
+ case 'stop':
72
+ if (!args[0]) {
73
+ ui.error('Instance ID is required');
74
+ ui.print('Usage: nimbus aws ec2 stop <instance-id>');
75
+ return;
76
+ }
77
+ await stopInstance(args[0], options);
78
+ break;
79
+
80
+ case 'terminate':
81
+ if (!args[0]) {
82
+ ui.error('Instance ID is required');
83
+ ui.print('Usage: nimbus aws ec2 terminate <instance-id>');
84
+ return;
85
+ }
86
+ await terminateInstance(args[0], options);
87
+ break;
88
+
89
+ default:
90
+ showEc2Help();
91
+ break;
92
+ }
93
+ }
94
+
95
+ /**
96
+ * List all EC2 instances
97
+ */
98
+ async function listInstances(options: AwsCommandOptions): Promise<void> {
99
+ ui.header('EC2 Instances');
100
+
101
+ ui.startSpinner({ message: 'Fetching EC2 instances...' });
102
+
103
+ try {
104
+ const instances = await runAwsCommand<EC2Instance[]>(
105
+ 'ec2 describe-instances --query "Reservations[].Instances[]"',
106
+ options
107
+ );
108
+
109
+ ui.stopSpinnerSuccess(`Found ${instances.length} instance(s)`);
110
+ ui.newLine();
111
+
112
+ if (instances.length === 0) {
113
+ ui.info('No EC2 instances found');
114
+ return;
115
+ }
116
+
117
+ // Display table
118
+ displayInstanceTable(instances);
119
+ } catch (error) {
120
+ ui.stopSpinnerFail('Failed to list instances');
121
+ ui.error((error as Error).message);
122
+ }
123
+ }
124
+
125
+ /**
126
+ * Describe a specific EC2 instance
127
+ */
128
+ async function describeInstance(instanceId: string, options: AwsCommandOptions): Promise<void> {
129
+ ui.header(`EC2 Instance: ${instanceId}`);
130
+
131
+ ui.startSpinner({ message: 'Fetching instance details...' });
132
+
133
+ try {
134
+ const instances = await runAwsCommand<EC2Instance[]>(
135
+ `ec2 describe-instances --instance-ids ${instanceId} --query "Reservations[].Instances[]"`,
136
+ options
137
+ );
138
+
139
+ ui.stopSpinnerSuccess('Instance details retrieved');
140
+ ui.newLine();
141
+
142
+ if (instances.length === 0) {
143
+ ui.error(`Instance ${instanceId} not found`);
144
+ return;
145
+ }
146
+
147
+ const instance = instances[0];
148
+
149
+ // Display instance details
150
+ ui.print(ui.bold('Instance Details:'));
151
+ ui.newLine();
152
+ ui.print(` Instance ID: ${instance.InstanceId}`);
153
+ ui.print(` Instance Type: ${instance.InstanceType}`);
154
+ ui.print(` State: ${formatState(instance.State.Name)}`);
155
+ ui.print(` Public IP: ${instance.PublicIpAddress || 'N/A'}`);
156
+ ui.print(` Private IP: ${instance.PrivateIpAddress || 'N/A'}`);
157
+ ui.print(` Launch Time: ${instance.LaunchTime || 'N/A'}`);
158
+
159
+ if (instance.Tags && instance.Tags.length > 0) {
160
+ ui.newLine();
161
+ ui.print(ui.bold('Tags:'));
162
+ for (const tag of instance.Tags) {
163
+ ui.print(` ${tag.Key}: ${tag.Value}`);
164
+ }
165
+ }
166
+
167
+ // Show cost estimate for running instances
168
+ displayCostWarning(instance.InstanceType);
169
+ } catch (error) {
170
+ ui.stopSpinnerFail('Failed to describe instance');
171
+ ui.error((error as Error).message);
172
+ }
173
+ }
174
+
175
+ /**
176
+ * Display a cost warning for an EC2 instance type using the cloud cost estimator.
177
+ */
178
+ function displayCostWarning(instanceType: string): void {
179
+ const estimate = estimateCloudCost('ec2:StartInstances', { instanceType });
180
+ if (estimate) {
181
+ const color = estimate.monthly > 200 ? 'red' : estimate.monthly > 50 ? 'yellow' : 'green';
182
+ ui.newLine();
183
+ ui.print(ui.bold(' Estimated Cost:'));
184
+ ui.print(` Hourly: ${ui.color(`$${estimate.hourly.toFixed(4)}/hr`, color)}`);
185
+ ui.print(
186
+ ` Monthly: ${ui.color(`$${estimate.monthly.toFixed(2)}/mo`, color)} (on-demand, approximate)`
187
+ );
188
+ ui.newLine();
189
+ }
190
+ }
191
+
192
+ /**
193
+ * Start an EC2 instance
194
+ */
195
+ async function startInstance(instanceId: string, options: AwsCommandOptions): Promise<void> {
196
+ ui.header(`Start EC2 Instance: ${instanceId}`);
197
+
198
+ // Try to get instance type for cost estimate
199
+ try {
200
+ const instances = await runAwsCommand<EC2Instance[]>(
201
+ `ec2 describe-instances --instance-ids ${instanceId} --query "Reservations[].Instances[]"`,
202
+ options
203
+ );
204
+ if (instances.length > 0) {
205
+ const instanceType = instances[0].InstanceType;
206
+ const estimate = estimateCloudCost('ec2:StartInstances', { instanceType });
207
+ if (estimate) {
208
+ ui.newLine();
209
+ ui.warning(formatCostWarning(estimate));
210
+ ui.newLine();
211
+ }
212
+ }
213
+ } catch {
214
+ // Non-critical, continue without cost estimate
215
+ }
216
+
217
+ // Confirm action
218
+ const proceed = await confirm({
219
+ message: `Start instance ${instanceId}?`,
220
+ defaultValue: true,
221
+ });
222
+
223
+ if (!proceed) {
224
+ ui.info('Operation cancelled');
225
+ return;
226
+ }
227
+
228
+ ui.startSpinner({ message: 'Starting instance...' });
229
+
230
+ try {
231
+ await runAwsCommand(`ec2 start-instances --instance-ids ${instanceId}`, options);
232
+
233
+ ui.stopSpinnerSuccess('Instance started');
234
+ ui.info(`Instance ${instanceId} is now starting`);
235
+ } catch (error) {
236
+ ui.stopSpinnerFail('Failed to start instance');
237
+ ui.error((error as Error).message);
238
+ }
239
+ }
240
+
241
+ /**
242
+ * Stop an EC2 instance
243
+ */
244
+ async function stopInstance(instanceId: string, options: AwsCommandOptions): Promise<void> {
245
+ ui.header(`Stop EC2 Instance: ${instanceId}`);
246
+
247
+ // Show cost estimate for the instance being stopped
248
+ try {
249
+ const instances = await runAwsCommand<EC2Instance[]>(
250
+ `ec2 describe-instances --instance-ids ${instanceId} --query "Reservations[].Instances[]"`,
251
+ options
252
+ );
253
+ if (instances.length > 0) {
254
+ displayCostWarning(instances[0].InstanceType);
255
+ ui.info('Stopping this instance will stop incurring compute charges.');
256
+ ui.newLine();
257
+ }
258
+ } catch {
259
+ // Non-critical, continue without cost estimate
260
+ }
261
+
262
+ // Run safety checks
263
+ const safetyResult = await runSafetyCheck('stop', instanceId, options);
264
+
265
+ if (!safetyResult.passed) {
266
+ ui.error('Safety checks failed');
267
+ return;
268
+ }
269
+
270
+ if (safetyResult.requiresApproval) {
271
+ const approval = await promptForApproval({
272
+ title: 'Stop EC2 Instance',
273
+ operation: `ec2 stop ${instanceId}`,
274
+ risks: safetyResult.risks,
275
+ });
276
+
277
+ if (!approval.approved) {
278
+ ui.info('Operation cancelled');
279
+ return;
280
+ }
281
+ } else {
282
+ const proceed = await confirm({
283
+ message: `Stop instance ${instanceId}?`,
284
+ defaultValue: false,
285
+ });
286
+
287
+ if (!proceed) {
288
+ ui.info('Operation cancelled');
289
+ return;
290
+ }
291
+ }
292
+
293
+ ui.startSpinner({ message: 'Stopping instance...' });
294
+
295
+ try {
296
+ await runAwsCommand(`ec2 stop-instances --instance-ids ${instanceId}`, options);
297
+
298
+ ui.stopSpinnerSuccess('Instance stopped');
299
+ ui.info(`Instance ${instanceId} is now stopping`);
300
+ } catch (error) {
301
+ ui.stopSpinnerFail('Failed to stop instance');
302
+ ui.error((error as Error).message);
303
+ }
304
+ }
305
+
306
+ /**
307
+ * Terminate an EC2 instance
308
+ */
309
+ async function terminateInstance(instanceId: string, options: AwsCommandOptions): Promise<void> {
310
+ ui.header(`Terminate EC2 Instance: ${instanceId}`);
311
+ ui.warning('This action cannot be undone!');
312
+ ui.newLine();
313
+
314
+ // Show cost estimate for the instance being terminated
315
+ try {
316
+ const instances = await runAwsCommand<EC2Instance[]>(
317
+ `ec2 describe-instances --instance-ids ${instanceId} --query "Reservations[].Instances[]"`,
318
+ options
319
+ );
320
+ if (instances.length > 0) {
321
+ displayCostWarning(instances[0].InstanceType);
322
+ ui.info('Terminating this instance will permanently stop all charges.');
323
+ ui.newLine();
324
+ }
325
+ } catch {
326
+ // Non-critical, continue without cost estimate
327
+ }
328
+
329
+ // Run safety checks
330
+ const safetyResult = await runSafetyCheck('terminate', instanceId, options);
331
+
332
+ if (!safetyResult.passed) {
333
+ ui.error('Safety checks failed - operation blocked');
334
+ for (const blocker of safetyResult.blockers) {
335
+ ui.print(` ${ui.color('x', 'red')} ${blocker.message}`);
336
+ }
337
+ return;
338
+ }
339
+
340
+ // Always require approval for terminate
341
+ const approval = await promptForApproval({
342
+ title: 'Terminate EC2 Instance',
343
+ operation: `ec2 terminate ${instanceId}`,
344
+ risks: safetyResult.risks,
345
+ requireConfirmation: true,
346
+ confirmationWord: 'terminate',
347
+ });
348
+
349
+ if (!approval.approved) {
350
+ ui.info('Operation cancelled');
351
+ return;
352
+ }
353
+
354
+ ui.startSpinner({ message: 'Terminating instance...' });
355
+
356
+ try {
357
+ await runAwsCommand(`ec2 terminate-instances --instance-ids ${instanceId}`, options);
358
+
359
+ ui.stopSpinnerSuccess('Instance terminated');
360
+ ui.info(`Instance ${instanceId} has been terminated`);
361
+ } catch (error) {
362
+ ui.stopSpinnerFail('Failed to terminate instance');
363
+ ui.error((error as Error).message);
364
+ }
365
+ }
366
+
367
+ /**
368
+ * Run safety check for EC2 operation
369
+ */
370
+ async function runSafetyCheck(
371
+ operation: string,
372
+ instanceId: string,
373
+ options: AwsCommandOptions
374
+ ): Promise<SafetyCheckResult> {
375
+ const policy = loadSafetyPolicy();
376
+
377
+ const context: SafetyContext = {
378
+ operation,
379
+ type: 'aws',
380
+ resources: [instanceId],
381
+ metadata: {
382
+ service: 'ec2',
383
+ region: options.region,
384
+ },
385
+ };
386
+
387
+ return evaluateSafety(context, policy);
388
+ }
389
+
390
+ /**
391
+ * Run AWS CLI command and parse JSON output
392
+ */
393
+ async function runAwsCommand<T>(command: string, options: AwsCommandOptions): Promise<T> {
394
+ const { execFile } = await import('child_process');
395
+ const { promisify } = await import('util');
396
+ const execFileAsync = promisify(execFile);
397
+
398
+ const args = command.split(' ');
399
+ const baseCommand = args[0];
400
+ const commandArgs = args.slice(1);
401
+
402
+ // Add common options
403
+ if (options.profile) {
404
+ commandArgs.push('--profile', options.profile);
405
+ }
406
+ if (options.region) {
407
+ commandArgs.push('--region', options.region);
408
+ }
409
+ commandArgs.push('--output', 'json');
410
+
411
+ const { stdout } = await execFileAsync('aws', [baseCommand, ...commandArgs]);
412
+ return JSON.parse(stdout) as T;
413
+ }
414
+
415
+ /**
416
+ * Display instances in a table format
417
+ */
418
+ function displayInstanceTable(instances: EC2Instance[]): void {
419
+ // Calculate column widths
420
+ const headers = ['Instance ID', 'Name', 'Type', 'State', 'Public IP', 'Private IP'];
421
+ const rows = instances.map(inst => {
422
+ const nameTag = inst.Tags?.find(t => t.Key === 'Name');
423
+ return [
424
+ inst.InstanceId,
425
+ nameTag?.Value || '-',
426
+ inst.InstanceType,
427
+ inst.State.Name,
428
+ inst.PublicIpAddress || '-',
429
+ inst.PrivateIpAddress || '-',
430
+ ];
431
+ });
432
+
433
+ // Print header
434
+ const headerRow = headers
435
+ .map((h, i) => {
436
+ const maxWidth = Math.max(h.length, ...rows.map(r => r[i].length));
437
+ return h.padEnd(maxWidth);
438
+ })
439
+ .join(' ');
440
+
441
+ ui.print(ui.bold(headerRow));
442
+ ui.print('-'.repeat(headerRow.length));
443
+
444
+ // Print rows
445
+ for (const row of rows) {
446
+ const formattedRow = row
447
+ .map((cell, i) => {
448
+ const maxWidth = Math.max(headers[i].length, ...rows.map(r => r[i].length));
449
+ if (i === 3) {
450
+ // State column - colorize
451
+ return formatState(cell).padEnd(maxWidth + 10); // Extra for color codes
452
+ }
453
+ return cell.padEnd(maxWidth);
454
+ })
455
+ .join(' ');
456
+
457
+ ui.print(formattedRow);
458
+ }
459
+ }
460
+
461
+ /**
462
+ * Format instance state with color
463
+ */
464
+ function formatState(state: string): string {
465
+ switch (state) {
466
+ case 'running':
467
+ return ui.color(state, 'green');
468
+ case 'stopped':
469
+ return ui.color(state, 'red');
470
+ case 'pending':
471
+ case 'stopping':
472
+ return ui.color(state, 'yellow');
473
+ case 'terminated':
474
+ return ui.color(state, 'gray');
475
+ default:
476
+ return state;
477
+ }
478
+ }
479
+
480
+ /**
481
+ * Show EC2 command help
482
+ */
483
+ function showEc2Help(): void {
484
+ ui.print('Usage: nimbus aws ec2 <action> [args]');
485
+ ui.newLine();
486
+
487
+ ui.print(ui.bold('Actions:'));
488
+ ui.print(' list List all EC2 instances');
489
+ ui.print(' describe <id> Describe a specific instance');
490
+ ui.print(' start <id> Start an instance');
491
+ ui.print(' stop <id> Stop an instance');
492
+ ui.print(' terminate <id> Terminate an instance (requires approval)');
493
+ ui.newLine();
494
+
495
+ ui.print(ui.bold('Examples:'));
496
+ ui.print(' nimbus aws ec2 list');
497
+ ui.print(' nimbus aws ec2 describe i-1234567890abcdef0');
498
+ ui.print(' nimbus aws ec2 stop i-1234567890abcdef0');
499
+ }
500
+
501
+ export default ec2Command;