@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,454 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Apply Kubernetes Command
|
|
3
|
+
*
|
|
4
|
+
* Apply Kubernetes manifests to a cluster
|
|
5
|
+
*
|
|
6
|
+
* Usage: nimbus apply k8s <manifests> [options]
|
|
7
|
+
*/
|
|
8
|
+
|
|
9
|
+
import { logger } from '../../utils';
|
|
10
|
+
import { ui, confirm } from '../../wizard';
|
|
11
|
+
import { k8sClient } from '../../clients';
|
|
12
|
+
import {
|
|
13
|
+
loadSafetyPolicy,
|
|
14
|
+
evaluateSafety,
|
|
15
|
+
type SafetyContext,
|
|
16
|
+
type SafetyCheckResult,
|
|
17
|
+
} from '../../config/safety-policy';
|
|
18
|
+
import { promptForApproval, displaySafetySummary } from '../../wizard/approval';
|
|
19
|
+
|
|
20
|
+
/**
|
|
21
|
+
* Command options
|
|
22
|
+
*/
|
|
23
|
+
export interface ApplyK8sOptions {
|
|
24
|
+
manifests?: string;
|
|
25
|
+
namespace?: string;
|
|
26
|
+
dryRun?: boolean;
|
|
27
|
+
wait?: boolean;
|
|
28
|
+
prune?: boolean;
|
|
29
|
+
force?: boolean;
|
|
30
|
+
recursive?: boolean;
|
|
31
|
+
selector?: string;
|
|
32
|
+
/** Skip safety checks */
|
|
33
|
+
skipSafety?: boolean;
|
|
34
|
+
/** Environment name (for safety policy) */
|
|
35
|
+
environment?: string;
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
/**
|
|
39
|
+
* Run kubectl apply command
|
|
40
|
+
*/
|
|
41
|
+
export async function applyK8sCommand(options: ApplyK8sOptions = {}): Promise<void> {
|
|
42
|
+
logger.info('Running kubectl apply', { options });
|
|
43
|
+
|
|
44
|
+
const manifests = options.manifests || '.';
|
|
45
|
+
|
|
46
|
+
ui.header('Kubernetes Apply');
|
|
47
|
+
ui.info(`Manifests: ${manifests}`);
|
|
48
|
+
if (options.namespace) {
|
|
49
|
+
ui.info(`Namespace: ${options.namespace}`);
|
|
50
|
+
}
|
|
51
|
+
ui.newLine();
|
|
52
|
+
|
|
53
|
+
// Check if k8s client is available
|
|
54
|
+
const clientAvailable = await k8sClient.isAvailable();
|
|
55
|
+
|
|
56
|
+
if (clientAvailable) {
|
|
57
|
+
// Use K8s tools service
|
|
58
|
+
await applyWithService(options);
|
|
59
|
+
} else {
|
|
60
|
+
// Fall back to local kubectl CLI
|
|
61
|
+
await applyWithLocalCLI(options);
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
/**
|
|
66
|
+
* Apply using K8s Tools Service
|
|
67
|
+
*/
|
|
68
|
+
async function applyWithService(options: ApplyK8sOptions): Promise<void> {
|
|
69
|
+
const manifests = options.manifests || '.';
|
|
70
|
+
const fs = await import('fs/promises');
|
|
71
|
+
const path = await import('path');
|
|
72
|
+
|
|
73
|
+
// Read manifest files
|
|
74
|
+
let manifestContent: string;
|
|
75
|
+
|
|
76
|
+
try {
|
|
77
|
+
const stat = await fs.stat(manifests);
|
|
78
|
+
|
|
79
|
+
if (stat.isDirectory()) {
|
|
80
|
+
// Read all YAML files in directory
|
|
81
|
+
const files = await fs.readdir(manifests);
|
|
82
|
+
const yamlFiles = files.filter(f => f.endsWith('.yaml') || f.endsWith('.yml'));
|
|
83
|
+
|
|
84
|
+
if (yamlFiles.length === 0) {
|
|
85
|
+
ui.error('No YAML manifests found in directory');
|
|
86
|
+
process.exit(1);
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
const contents = await Promise.all(
|
|
90
|
+
yamlFiles.map(f => fs.readFile(path.join(manifests, f), 'utf-8'))
|
|
91
|
+
);
|
|
92
|
+
manifestContent = contents.join('\n---\n');
|
|
93
|
+
|
|
94
|
+
ui.info(`Found ${yamlFiles.length} manifest file(s)`);
|
|
95
|
+
} else {
|
|
96
|
+
manifestContent = await fs.readFile(manifests, 'utf-8');
|
|
97
|
+
}
|
|
98
|
+
} catch (error: any) {
|
|
99
|
+
ui.error(`Failed to read manifests: ${error.message}`);
|
|
100
|
+
process.exit(1);
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
// Parse manifests to show what will be applied
|
|
104
|
+
const resources = parseManifests(manifestContent);
|
|
105
|
+
ui.newLine();
|
|
106
|
+
ui.print('Resources to apply:');
|
|
107
|
+
for (const resource of resources) {
|
|
108
|
+
ui.print(
|
|
109
|
+
` - ${resource.kind}/${resource.name}${resource.namespace ? ` (${resource.namespace})` : ''}`
|
|
110
|
+
);
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
// Dry run mode
|
|
114
|
+
if (options.dryRun) {
|
|
115
|
+
ui.newLine();
|
|
116
|
+
ui.startSpinner({ message: 'Running dry-run...' });
|
|
117
|
+
|
|
118
|
+
const result = await k8sClient.apply(manifestContent, {
|
|
119
|
+
namespace: options.namespace,
|
|
120
|
+
dryRun: true,
|
|
121
|
+
});
|
|
122
|
+
|
|
123
|
+
if (!result.success) {
|
|
124
|
+
ui.stopSpinnerFail('Dry-run failed');
|
|
125
|
+
ui.error(result.error || 'Unknown error');
|
|
126
|
+
process.exit(1);
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
ui.stopSpinnerSuccess('Dry-run successful');
|
|
130
|
+
ui.newLine();
|
|
131
|
+
ui.info('No changes applied (dry-run mode)');
|
|
132
|
+
|
|
133
|
+
if (result.output) {
|
|
134
|
+
ui.newLine();
|
|
135
|
+
ui.print(result.output);
|
|
136
|
+
}
|
|
137
|
+
|
|
138
|
+
return;
|
|
139
|
+
}
|
|
140
|
+
|
|
141
|
+
// Run safety checks if not skipped
|
|
142
|
+
if (!options.skipSafety) {
|
|
143
|
+
const resourceList = resources.map(r => `${r.kind}/${r.name}`);
|
|
144
|
+
const safetyResult = await runK8sSafetyChecks('apply', resourceList, options);
|
|
145
|
+
|
|
146
|
+
if (!safetyResult.passed) {
|
|
147
|
+
ui.newLine();
|
|
148
|
+
ui.error('Safety checks failed - operation blocked');
|
|
149
|
+
for (const blocker of safetyResult.blockers) {
|
|
150
|
+
ui.print(` ${ui.color('✗', 'red')} ${blocker.message}`);
|
|
151
|
+
}
|
|
152
|
+
process.exit(1);
|
|
153
|
+
}
|
|
154
|
+
|
|
155
|
+
// If safety requires approval, prompt for it
|
|
156
|
+
if (safetyResult.requiresApproval) {
|
|
157
|
+
const approvalResult = await promptForApproval({
|
|
158
|
+
title: 'Kubernetes Apply',
|
|
159
|
+
operation: 'kubectl apply',
|
|
160
|
+
risks: safetyResult.risks,
|
|
161
|
+
environment: options.environment,
|
|
162
|
+
affectedResources: resourceList,
|
|
163
|
+
});
|
|
164
|
+
|
|
165
|
+
if (!approvalResult.approved) {
|
|
166
|
+
ui.newLine();
|
|
167
|
+
ui.info(`Apply cancelled: ${approvalResult.reason || 'User declined'}`);
|
|
168
|
+
return;
|
|
169
|
+
}
|
|
170
|
+
} else {
|
|
171
|
+
// Show safety summary and simple confirm
|
|
172
|
+
displaySafetySummary({
|
|
173
|
+
operation: 'kubectl apply',
|
|
174
|
+
risks: safetyResult.risks,
|
|
175
|
+
passed: safetyResult.passed,
|
|
176
|
+
});
|
|
177
|
+
|
|
178
|
+
ui.newLine();
|
|
179
|
+
const proceed = await confirm({
|
|
180
|
+
message: `Apply ${resources.length} resource(s)?`,
|
|
181
|
+
defaultValue: true,
|
|
182
|
+
});
|
|
183
|
+
|
|
184
|
+
if (!proceed) {
|
|
185
|
+
ui.info('Apply cancelled');
|
|
186
|
+
return;
|
|
187
|
+
}
|
|
188
|
+
}
|
|
189
|
+
} else {
|
|
190
|
+
// Simple confirmation when safety is skipped
|
|
191
|
+
ui.newLine();
|
|
192
|
+
const proceed = await confirm({
|
|
193
|
+
message: `Apply ${resources.length} resource(s)?`,
|
|
194
|
+
defaultValue: true,
|
|
195
|
+
});
|
|
196
|
+
|
|
197
|
+
if (!proceed) {
|
|
198
|
+
ui.info('Apply cancelled');
|
|
199
|
+
return;
|
|
200
|
+
}
|
|
201
|
+
}
|
|
202
|
+
|
|
203
|
+
// Apply manifests
|
|
204
|
+
ui.newLine();
|
|
205
|
+
ui.startSpinner({ message: 'Applying manifests...' });
|
|
206
|
+
|
|
207
|
+
const result = await k8sClient.apply(manifestContent, {
|
|
208
|
+
namespace: options.namespace,
|
|
209
|
+
});
|
|
210
|
+
|
|
211
|
+
if (!result.success) {
|
|
212
|
+
ui.stopSpinnerFail('Apply failed');
|
|
213
|
+
ui.error(result.error || 'Unknown error');
|
|
214
|
+
process.exit(1);
|
|
215
|
+
}
|
|
216
|
+
|
|
217
|
+
ui.stopSpinnerSuccess('Apply complete!');
|
|
218
|
+
|
|
219
|
+
// Track successful k8s apply
|
|
220
|
+
try {
|
|
221
|
+
const { trackGeneration } = await import('../../telemetry');
|
|
222
|
+
trackGeneration('k8s-apply', ['kubernetes']);
|
|
223
|
+
} catch {
|
|
224
|
+
/* telemetry failure is non-critical */
|
|
225
|
+
}
|
|
226
|
+
|
|
227
|
+
// Display results
|
|
228
|
+
ui.newLine();
|
|
229
|
+
if (result.created?.length) {
|
|
230
|
+
ui.print('Created:');
|
|
231
|
+
for (const r of result.created) {
|
|
232
|
+
ui.print(` ${ui.color('+', 'green')} ${r}`);
|
|
233
|
+
}
|
|
234
|
+
}
|
|
235
|
+
if (result.configured?.length) {
|
|
236
|
+
ui.print('Configured:');
|
|
237
|
+
for (const r of result.configured) {
|
|
238
|
+
ui.print(` ${ui.color('~', 'yellow')} ${r}`);
|
|
239
|
+
}
|
|
240
|
+
}
|
|
241
|
+
|
|
242
|
+
// Wait for resources to be ready
|
|
243
|
+
if (options.wait) {
|
|
244
|
+
ui.newLine();
|
|
245
|
+
await waitForResources(resources, options.namespace);
|
|
246
|
+
}
|
|
247
|
+
}
|
|
248
|
+
|
|
249
|
+
/**
|
|
250
|
+
* Wait for rollout resources to be ready by polling kubectl rollout status
|
|
251
|
+
* via the k8s tools service. Only Deployments, StatefulSets, and DaemonSets
|
|
252
|
+
* are tracked; other resource kinds are skipped.
|
|
253
|
+
*/
|
|
254
|
+
async function waitForResources(
|
|
255
|
+
resources: Array<{ kind: string; name: string; namespace?: string }>,
|
|
256
|
+
defaultNamespace?: string
|
|
257
|
+
): Promise<void> {
|
|
258
|
+
const rolloutKinds = ['Deployment', 'StatefulSet', 'DaemonSet'];
|
|
259
|
+
const rolloutResources = resources.filter(r => rolloutKinds.includes(r.kind));
|
|
260
|
+
|
|
261
|
+
if (rolloutResources.length === 0) {
|
|
262
|
+
ui.success('All resources applied (no rollout resources to wait for)');
|
|
263
|
+
return;
|
|
264
|
+
}
|
|
265
|
+
|
|
266
|
+
ui.startSpinner({ message: `Waiting for ${rolloutResources.length} resource(s) to be ready...` });
|
|
267
|
+
|
|
268
|
+
const timeout = 120_000; // 120 seconds
|
|
269
|
+
const pollInterval = 2_000; // 2 seconds
|
|
270
|
+
const startTime = Date.now();
|
|
271
|
+
let readyCount = 0;
|
|
272
|
+
const readySet = new Set<string>();
|
|
273
|
+
|
|
274
|
+
while (readyCount < rolloutResources.length) {
|
|
275
|
+
if (Date.now() - startTime > timeout) {
|
|
276
|
+
const pending = rolloutResources
|
|
277
|
+
.filter(r => !readySet.has(`${r.kind}/${r.name}`))
|
|
278
|
+
.map(r => `${r.kind}/${r.name}`);
|
|
279
|
+
ui.stopSpinnerFail(`Timeout: ${pending.join(', ')} not ready after ${timeout / 1000}s`);
|
|
280
|
+
return;
|
|
281
|
+
}
|
|
282
|
+
|
|
283
|
+
for (const resource of rolloutResources) {
|
|
284
|
+
const key = `${resource.kind}/${resource.name}`;
|
|
285
|
+
if (readySet.has(key)) {
|
|
286
|
+
continue;
|
|
287
|
+
}
|
|
288
|
+
|
|
289
|
+
try {
|
|
290
|
+
const result = await k8sClient.rollout(
|
|
291
|
+
resource.kind.toLowerCase(),
|
|
292
|
+
resource.name,
|
|
293
|
+
'status',
|
|
294
|
+
{ namespace: resource.namespace || defaultNamespace }
|
|
295
|
+
);
|
|
296
|
+
|
|
297
|
+
if (result.success && result.output && result.output.includes('successfully rolled out')) {
|
|
298
|
+
readySet.add(key);
|
|
299
|
+
readyCount++;
|
|
300
|
+
}
|
|
301
|
+
} catch {
|
|
302
|
+
// Not ready yet, continue polling
|
|
303
|
+
}
|
|
304
|
+
}
|
|
305
|
+
|
|
306
|
+
if (readyCount < rolloutResources.length) {
|
|
307
|
+
ui.updateSpinner(`Waiting: ${readyCount}/${rolloutResources.length} resources ready...`);
|
|
308
|
+
await new Promise(resolve => setTimeout(resolve, pollInterval));
|
|
309
|
+
}
|
|
310
|
+
}
|
|
311
|
+
|
|
312
|
+
ui.stopSpinnerSuccess(`All ${rolloutResources.length} resource(s) ready`);
|
|
313
|
+
}
|
|
314
|
+
|
|
315
|
+
/**
|
|
316
|
+
* Apply using local kubectl CLI
|
|
317
|
+
*/
|
|
318
|
+
async function applyWithLocalCLI(options: ApplyK8sOptions): Promise<void> {
|
|
319
|
+
const { spawn } = await import('child_process');
|
|
320
|
+
|
|
321
|
+
const manifests = options.manifests || '.';
|
|
322
|
+
|
|
323
|
+
// Build kubectl command
|
|
324
|
+
const args = ['apply', '-f', manifests];
|
|
325
|
+
|
|
326
|
+
if (options.namespace) {
|
|
327
|
+
args.push('-n', options.namespace);
|
|
328
|
+
}
|
|
329
|
+
|
|
330
|
+
if (options.dryRun) {
|
|
331
|
+
args.push('--dry-run=client');
|
|
332
|
+
}
|
|
333
|
+
|
|
334
|
+
if (options.wait) {
|
|
335
|
+
args.push('--wait');
|
|
336
|
+
}
|
|
337
|
+
|
|
338
|
+
if (options.prune) {
|
|
339
|
+
args.push('--prune');
|
|
340
|
+
}
|
|
341
|
+
|
|
342
|
+
if (options.force) {
|
|
343
|
+
args.push('--force');
|
|
344
|
+
}
|
|
345
|
+
|
|
346
|
+
if (options.recursive) {
|
|
347
|
+
args.push('-R');
|
|
348
|
+
}
|
|
349
|
+
|
|
350
|
+
if (options.selector) {
|
|
351
|
+
args.push('-l', options.selector);
|
|
352
|
+
}
|
|
353
|
+
|
|
354
|
+
ui.info(`Running: kubectl ${args.join(' ')}`);
|
|
355
|
+
ui.newLine();
|
|
356
|
+
|
|
357
|
+
// Run kubectl
|
|
358
|
+
return new Promise((resolve, _reject) => {
|
|
359
|
+
const proc = spawn('kubectl', args, {
|
|
360
|
+
stdio: 'inherit',
|
|
361
|
+
});
|
|
362
|
+
|
|
363
|
+
proc.on('error', error => {
|
|
364
|
+
ui.error(`Failed to run kubectl: ${error.message}`);
|
|
365
|
+
ui.info('Make sure kubectl is installed and in your PATH');
|
|
366
|
+
process.exit(1);
|
|
367
|
+
});
|
|
368
|
+
|
|
369
|
+
proc.on('close', code => {
|
|
370
|
+
if (code === 0) {
|
|
371
|
+
ui.newLine();
|
|
372
|
+
ui.success('kubectl apply completed successfully');
|
|
373
|
+
|
|
374
|
+
// Track successful k8s apply
|
|
375
|
+
try {
|
|
376
|
+
// eslint-disable-next-line @typescript-eslint/no-var-requires
|
|
377
|
+
const { trackGeneration } = require('../../telemetry');
|
|
378
|
+
trackGeneration('k8s-apply', ['kubernetes']);
|
|
379
|
+
} catch {
|
|
380
|
+
/* telemetry failure is non-critical */
|
|
381
|
+
}
|
|
382
|
+
|
|
383
|
+
resolve();
|
|
384
|
+
} else {
|
|
385
|
+
ui.newLine();
|
|
386
|
+
ui.error(`kubectl apply failed with exit code ${code}`);
|
|
387
|
+
process.exit(code || 1);
|
|
388
|
+
}
|
|
389
|
+
});
|
|
390
|
+
});
|
|
391
|
+
}
|
|
392
|
+
|
|
393
|
+
/**
|
|
394
|
+
* Run safety checks for the operation
|
|
395
|
+
*/
|
|
396
|
+
async function runK8sSafetyChecks(
|
|
397
|
+
operation: string,
|
|
398
|
+
resources: string[],
|
|
399
|
+
options: ApplyK8sOptions
|
|
400
|
+
): Promise<SafetyCheckResult> {
|
|
401
|
+
const policy = loadSafetyPolicy();
|
|
402
|
+
|
|
403
|
+
const context: SafetyContext = {
|
|
404
|
+
operation,
|
|
405
|
+
type: 'kubernetes',
|
|
406
|
+
environment: options.environment,
|
|
407
|
+
resources,
|
|
408
|
+
metadata: {
|
|
409
|
+
manifests: options.manifests,
|
|
410
|
+
namespace: options.namespace,
|
|
411
|
+
},
|
|
412
|
+
};
|
|
413
|
+
|
|
414
|
+
return evaluateSafety(context, policy);
|
|
415
|
+
}
|
|
416
|
+
|
|
417
|
+
/**
|
|
418
|
+
* Parse manifests to extract resource information
|
|
419
|
+
*/
|
|
420
|
+
function parseManifests(content: string): Array<{
|
|
421
|
+
kind: string;
|
|
422
|
+
name: string;
|
|
423
|
+
namespace?: string;
|
|
424
|
+
}> {
|
|
425
|
+
const resources: Array<{ kind: string; name: string; namespace?: string }> = [];
|
|
426
|
+
|
|
427
|
+
// Split by document separator
|
|
428
|
+
const documents = content.split(/^---$/m);
|
|
429
|
+
|
|
430
|
+
for (const doc of documents) {
|
|
431
|
+
const trimmed = doc.trim();
|
|
432
|
+
if (!trimmed) {
|
|
433
|
+
continue;
|
|
434
|
+
}
|
|
435
|
+
|
|
436
|
+
// Simple YAML parsing for kind and metadata
|
|
437
|
+
const kindMatch = trimmed.match(/^kind:\s*(.+)$/m);
|
|
438
|
+
const nameMatch = trimmed.match(/^\s+name:\s*(.+)$/m);
|
|
439
|
+
const namespaceMatch = trimmed.match(/^\s+namespace:\s*(.+)$/m);
|
|
440
|
+
|
|
441
|
+
if (kindMatch && nameMatch) {
|
|
442
|
+
resources.push({
|
|
443
|
+
kind: kindMatch[1].trim(),
|
|
444
|
+
name: nameMatch[1].trim(),
|
|
445
|
+
namespace: namespaceMatch?.[1]?.trim(),
|
|
446
|
+
});
|
|
447
|
+
}
|
|
448
|
+
}
|
|
449
|
+
|
|
450
|
+
return resources;
|
|
451
|
+
}
|
|
452
|
+
|
|
453
|
+
// Export as default
|
|
454
|
+
export default applyK8sCommand;
|