@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,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;