@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.
- package/LICENSE +21 -0
- package/README.md +628 -0
- package/bin/nimbus +38 -0
- package/package.json +80 -0
- package/src/__tests__/app.test.ts +76 -0
- package/src/__tests__/audit.test.ts +877 -0
- package/src/__tests__/circuit-breaker.test.ts +116 -0
- package/src/__tests__/cli-run.test.ts +115 -0
- package/src/__tests__/context-manager.test.ts +502 -0
- package/src/__tests__/context.test.ts +242 -0
- package/src/__tests__/enterprise.test.ts +401 -0
- package/src/__tests__/generator.test.ts +433 -0
- package/src/__tests__/hooks.test.ts +582 -0
- package/src/__tests__/init.test.ts +436 -0
- package/src/__tests__/intent-parser.test.ts +229 -0
- package/src/__tests__/llm-router.test.ts +209 -0
- package/src/__tests__/lsp.test.ts +293 -0
- package/src/__tests__/modes.test.ts +336 -0
- package/src/__tests__/permissions.test.ts +338 -0
- package/src/__tests__/serve.test.ts +275 -0
- package/src/__tests__/sessions.test.ts +227 -0
- package/src/__tests__/sharing.test.ts +288 -0
- package/src/__tests__/snapshots.test.ts +581 -0
- package/src/__tests__/state-db.test.ts +334 -0
- package/src/__tests__/stream-with-tools.test.ts +732 -0
- package/src/__tests__/subagents.test.ts +176 -0
- package/src/__tests__/system-prompt.test.ts +169 -0
- package/src/__tests__/tool-converter.test.ts +256 -0
- package/src/__tests__/tool-schemas.test.ts +397 -0
- package/src/__tests__/tools.test.ts +143 -0
- package/src/__tests__/version.test.ts +49 -0
- package/src/agent/compaction-agent.ts +227 -0
- package/src/agent/context-manager.ts +435 -0
- package/src/agent/context.ts +427 -0
- package/src/agent/deploy-preview.ts +426 -0
- package/src/agent/index.ts +68 -0
- package/src/agent/loop.ts +717 -0
- package/src/agent/modes.ts +429 -0
- package/src/agent/permissions.ts +466 -0
- package/src/agent/subagents/base.ts +116 -0
- package/src/agent/subagents/cost.ts +51 -0
- package/src/agent/subagents/explore.ts +42 -0
- package/src/agent/subagents/general.ts +54 -0
- package/src/agent/subagents/index.ts +102 -0
- package/src/agent/subagents/infra.ts +59 -0
- package/src/agent/subagents/security.ts +69 -0
- package/src/agent/system-prompt.ts +436 -0
- package/src/app.ts +122 -0
- package/src/audit/activity-log.ts +290 -0
- package/src/audit/compliance-checker.ts +540 -0
- package/src/audit/cost-tracker.ts +318 -0
- package/src/audit/index.ts +23 -0
- package/src/audit/security-scanner.ts +596 -0
- package/src/auth/guard.ts +75 -0
- package/src/auth/index.ts +56 -0
- package/src/auth/oauth.ts +455 -0
- package/src/auth/providers.ts +470 -0
- package/src/auth/sso.ts +113 -0
- package/src/auth/store.ts +505 -0
- package/src/auth/types.ts +187 -0
- package/src/build.ts +141 -0
- package/src/cli/index.ts +16 -0
- package/src/cli/init.ts +854 -0
- package/src/cli/openapi-spec.ts +356 -0
- package/src/cli/run.ts +237 -0
- package/src/cli/serve-auth.ts +80 -0
- package/src/cli/serve.ts +462 -0
- package/src/cli/web.ts +67 -0
- package/src/cli.ts +1417 -0
- package/src/clients/core-engine-client.ts +227 -0
- package/src/clients/enterprise-client.ts +334 -0
- package/src/clients/generator-client.ts +351 -0
- package/src/clients/git-client.ts +627 -0
- package/src/clients/github-client.ts +410 -0
- package/src/clients/helm-client.ts +504 -0
- package/src/clients/index.ts +80 -0
- package/src/clients/k8s-client.ts +497 -0
- package/src/clients/llm-client.ts +161 -0
- package/src/clients/rest-client.ts +130 -0
- package/src/clients/service-discovery.ts +33 -0
- package/src/clients/terraform-client.ts +482 -0
- package/src/clients/tools-client.ts +1843 -0
- package/src/clients/ws-client.ts +115 -0
- package/src/commands/analyze/index.ts +352 -0
- package/src/commands/apply/helm.ts +473 -0
- package/src/commands/apply/index.ts +213 -0
- package/src/commands/apply/k8s.ts +454 -0
- package/src/commands/apply/terraform.ts +582 -0
- package/src/commands/ask.ts +167 -0
- package/src/commands/audit/index.ts +238 -0
- package/src/commands/auth-cloud.ts +294 -0
- package/src/commands/auth-list.ts +134 -0
- package/src/commands/auth-profile.ts +121 -0
- package/src/commands/auth-status.ts +141 -0
- package/src/commands/aws/ec2.ts +501 -0
- package/src/commands/aws/iam.ts +397 -0
- package/src/commands/aws/index.ts +133 -0
- package/src/commands/aws/lambda.ts +396 -0
- package/src/commands/aws/rds.ts +439 -0
- package/src/commands/aws/s3.ts +439 -0
- package/src/commands/aws/vpc.ts +393 -0
- package/src/commands/aws-discover.ts +649 -0
- package/src/commands/aws-terraform.ts +805 -0
- package/src/commands/azure/aks.ts +376 -0
- package/src/commands/azure/functions.ts +253 -0
- package/src/commands/azure/index.ts +116 -0
- package/src/commands/azure/storage.ts +478 -0
- package/src/commands/azure/vm.ts +355 -0
- package/src/commands/billing/index.ts +256 -0
- package/src/commands/chat.ts +314 -0
- package/src/commands/config.ts +346 -0
- package/src/commands/cost/cloud-cost-estimator.ts +266 -0
- package/src/commands/cost/estimator.ts +79 -0
- package/src/commands/cost/index.ts +594 -0
- package/src/commands/cost/parsers/terraform.ts +273 -0
- package/src/commands/cost/parsers/types.ts +25 -0
- package/src/commands/cost/pricing/aws.ts +544 -0
- package/src/commands/cost/pricing/azure.ts +499 -0
- package/src/commands/cost/pricing/gcp.ts +396 -0
- package/src/commands/cost/pricing/index.ts +40 -0
- package/src/commands/demo.ts +250 -0
- package/src/commands/doctor.ts +794 -0
- package/src/commands/drift/index.ts +439 -0
- package/src/commands/explain.ts +277 -0
- package/src/commands/feedback.ts +389 -0
- package/src/commands/fix.ts +324 -0
- package/src/commands/fs/index.ts +402 -0
- package/src/commands/gcp/compute.ts +325 -0
- package/src/commands/gcp/functions.ts +271 -0
- package/src/commands/gcp/gke.ts +438 -0
- package/src/commands/gcp/iam.ts +344 -0
- package/src/commands/gcp/index.ts +129 -0
- package/src/commands/gcp/storage.ts +284 -0
- package/src/commands/generate-helm.ts +1249 -0
- package/src/commands/generate-k8s.ts +1560 -0
- package/src/commands/generate-terraform.ts +1460 -0
- package/src/commands/gh/index.ts +863 -0
- package/src/commands/git/index.ts +1343 -0
- package/src/commands/helm/index.ts +1126 -0
- package/src/commands/help.ts +539 -0
- package/src/commands/history.ts +142 -0
- package/src/commands/import.ts +868 -0
- package/src/commands/index.ts +367 -0
- package/src/commands/init.ts +1046 -0
- package/src/commands/k8s/index.ts +1137 -0
- package/src/commands/login.ts +631 -0
- package/src/commands/logout.ts +83 -0
- package/src/commands/onboarding.ts +228 -0
- package/src/commands/plan/display.ts +279 -0
- package/src/commands/plan/index.ts +599 -0
- package/src/commands/preview.ts +452 -0
- package/src/commands/questionnaire.ts +1270 -0
- package/src/commands/resume.ts +55 -0
- package/src/commands/team/index.ts +346 -0
- package/src/commands/template.ts +232 -0
- package/src/commands/tf/index.ts +1034 -0
- package/src/commands/upgrade.ts +550 -0
- package/src/commands/usage/index.ts +134 -0
- package/src/commands/version.ts +170 -0
- package/src/compat/index.ts +2 -0
- package/src/compat/runtime.ts +12 -0
- package/src/compat/sqlite.ts +107 -0
- package/src/config/index.ts +17 -0
- package/src/config/manager.ts +530 -0
- package/src/config/safety-policy.ts +358 -0
- package/src/config/schema.ts +125 -0
- package/src/config/types.ts +527 -0
- package/src/context/context-db.ts +199 -0
- package/src/demo/index.ts +349 -0
- package/src/demo/scenarios/full-journey.ts +229 -0
- package/src/demo/scenarios/getting-started.ts +127 -0
- package/src/demo/scenarios/helm-release.ts +341 -0
- package/src/demo/scenarios/k8s-deployment.ts +194 -0
- package/src/demo/scenarios/terraform-vpc.ts +170 -0
- package/src/demo/types.ts +92 -0
- package/src/engine/cost-estimator.ts +438 -0
- package/src/engine/diagram-generator.ts +256 -0
- package/src/engine/drift-detector.ts +902 -0
- package/src/engine/executor.ts +1035 -0
- package/src/engine/index.ts +76 -0
- package/src/engine/orchestrator.ts +636 -0
- package/src/engine/planner.ts +720 -0
- package/src/engine/safety.ts +743 -0
- package/src/engine/verifier.ts +770 -0
- package/src/enterprise/audit.ts +348 -0
- package/src/enterprise/auth.ts +270 -0
- package/src/enterprise/billing.ts +822 -0
- package/src/enterprise/index.ts +17 -0
- package/src/enterprise/teams.ts +443 -0
- package/src/generator/best-practices.ts +1608 -0
- package/src/generator/helm.ts +630 -0
- package/src/generator/index.ts +37 -0
- package/src/generator/intent-parser.ts +514 -0
- package/src/generator/kubernetes.ts +976 -0
- package/src/generator/terraform.ts +1867 -0
- package/src/history/index.ts +8 -0
- package/src/history/manager.ts +322 -0
- package/src/history/types.ts +34 -0
- package/src/hooks/config.ts +432 -0
- package/src/hooks/engine.ts +391 -0
- package/src/hooks/index.ts +4 -0
- package/src/llm/auth-bridge.ts +198 -0
- package/src/llm/circuit-breaker.ts +140 -0
- package/src/llm/config-loader.ts +201 -0
- package/src/llm/cost-calculator.ts +171 -0
- package/src/llm/index.ts +8 -0
- package/src/llm/model-aliases.ts +115 -0
- package/src/llm/provider-registry.ts +63 -0
- package/src/llm/providers/anthropic.ts +433 -0
- package/src/llm/providers/bedrock.ts +477 -0
- package/src/llm/providers/google.ts +405 -0
- package/src/llm/providers/ollama.ts +767 -0
- package/src/llm/providers/openai-compatible.ts +340 -0
- package/src/llm/providers/openai.ts +328 -0
- package/src/llm/providers/openrouter.ts +338 -0
- package/src/llm/router.ts +1035 -0
- package/src/llm/types.ts +232 -0
- package/src/lsp/client.ts +298 -0
- package/src/lsp/languages.ts +116 -0
- package/src/lsp/manager.ts +278 -0
- package/src/mcp/client.ts +402 -0
- package/src/mcp/index.ts +5 -0
- package/src/mcp/manager.ts +133 -0
- package/src/nimbus.ts +214 -0
- package/src/plugins/index.ts +27 -0
- package/src/plugins/loader.ts +334 -0
- package/src/plugins/manager.ts +376 -0
- package/src/plugins/types.ts +284 -0
- package/src/scanners/cicd-scanner.ts +258 -0
- package/src/scanners/cloud-scanner.ts +466 -0
- package/src/scanners/framework-scanner.ts +469 -0
- package/src/scanners/iac-scanner.ts +388 -0
- package/src/scanners/index.ts +539 -0
- package/src/scanners/language-scanner.ts +276 -0
- package/src/scanners/package-manager-scanner.ts +277 -0
- package/src/scanners/types.ts +172 -0
- package/src/sessions/manager.ts +365 -0
- package/src/sessions/types.ts +44 -0
- package/src/sharing/sync.ts +296 -0
- package/src/sharing/viewer.ts +97 -0
- package/src/snapshots/index.ts +2 -0
- package/src/snapshots/manager.ts +530 -0
- package/src/state/artifacts.ts +147 -0
- package/src/state/audit.ts +137 -0
- package/src/state/billing.ts +240 -0
- package/src/state/checkpoints.ts +117 -0
- package/src/state/config.ts +67 -0
- package/src/state/conversations.ts +14 -0
- package/src/state/credentials.ts +154 -0
- package/src/state/db.ts +58 -0
- package/src/state/index.ts +26 -0
- package/src/state/messages.ts +115 -0
- package/src/state/projects.ts +123 -0
- package/src/state/schema.ts +236 -0
- package/src/state/sessions.ts +147 -0
- package/src/state/teams.ts +200 -0
- package/src/telemetry.ts +108 -0
- package/src/tools/aws-ops.ts +952 -0
- package/src/tools/azure-ops.ts +579 -0
- package/src/tools/file-ops.ts +593 -0
- package/src/tools/gcp-ops.ts +625 -0
- package/src/tools/git-ops.ts +773 -0
- package/src/tools/github-ops.ts +799 -0
- package/src/tools/helm-ops.ts +943 -0
- package/src/tools/index.ts +17 -0
- package/src/tools/k8s-ops.ts +819 -0
- package/src/tools/schemas/converter.ts +184 -0
- package/src/tools/schemas/devops.ts +612 -0
- package/src/tools/schemas/index.ts +73 -0
- package/src/tools/schemas/standard.ts +1144 -0
- package/src/tools/schemas/types.ts +705 -0
- package/src/tools/terraform-ops.ts +862 -0
- package/src/types/ambient.d.ts +193 -0
- package/src/types/config.ts +83 -0
- package/src/types/drift.ts +116 -0
- package/src/types/enterprise.ts +335 -0
- package/src/types/index.ts +20 -0
- package/src/types/plan.ts +44 -0
- package/src/types/request.ts +65 -0
- package/src/types/response.ts +54 -0
- package/src/types/service.ts +51 -0
- package/src/ui/App.tsx +997 -0
- package/src/ui/DeployPreview.tsx +169 -0
- package/src/ui/Header.tsx +68 -0
- package/src/ui/InputBox.tsx +350 -0
- package/src/ui/MessageList.tsx +585 -0
- package/src/ui/PermissionPrompt.tsx +151 -0
- package/src/ui/StatusBar.tsx +158 -0
- package/src/ui/ToolCallDisplay.tsx +409 -0
- package/src/ui/chat-ui.ts +853 -0
- package/src/ui/index.ts +33 -0
- package/src/ui/ink/index.ts +711 -0
- package/src/ui/streaming.ts +176 -0
- package/src/ui/types.ts +57 -0
- package/src/utils/analytics.ts +72 -0
- package/src/utils/cost-warning.ts +27 -0
- package/src/utils/env.ts +46 -0
- package/src/utils/errors.ts +69 -0
- package/src/utils/event-bus.ts +38 -0
- package/src/utils/index.ts +24 -0
- package/src/utils/logger.ts +171 -0
- package/src/utils/rate-limiter.ts +121 -0
- package/src/utils/service-auth.ts +49 -0
- package/src/utils/validation.ts +53 -0
- package/src/version.ts +4 -0
- package/src/watcher/index.ts +163 -0
- package/src/wizard/approval.ts +383 -0
- package/src/wizard/index.ts +25 -0
- package/src/wizard/prompts.ts +338 -0
- package/src/wizard/types.ts +171 -0
- package/src/wizard/ui.ts +556 -0
- package/src/wizard/wizard.ts +304 -0
- 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;
|