cdk-cost-analyzer 0.1.1

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 (228) hide show
  1. package/.cdk-cost-analyzer-cache/metadata.json +12 -0
  2. package/.gitlab-ci.yml +214 -0
  3. package/.husky/pre-commit +12 -0
  4. package/.kiro/hooks/accessibility-audit.kiro.hook +18 -0
  5. package/.kiro/hooks/api-schema-validation.kiro.hook +21 -0
  6. package/.kiro/hooks/auto-test-on-save.kiro.hook +19 -0
  7. package/.kiro/hooks/cdk-synth-on-change.kiro.hook +20 -0
  8. package/.kiro/hooks/code-coverage-check.kiro.hook +14 -0
  9. package/.kiro/hooks/commit-message-helper.kiro.hook +14 -0
  10. package/.kiro/hooks/dependency-update-check.kiro.hook +14 -0
  11. package/.kiro/hooks/env-file-validation.kiro.hook +18 -0
  12. package/.kiro/hooks/lint-and-format-on-save.kiro.hook +21 -0
  13. package/.kiro/hooks/mcp-config-validation.kiro.hook +17 -0
  14. package/.kiro/hooks/mcp-server-test.kiro.hook +14 -0
  15. package/.kiro/hooks/performance-analysis.kiro.hook +14 -0
  16. package/.kiro/hooks/readme-spell-check.kiro.hook +14 -0
  17. package/.kiro/hooks/security-scan-on-dependency-change.kiro.hook +21 -0
  18. package/.kiro/hooks/translation-update.kiro.hook +18 -0
  19. package/.kiro/hooks/update-documentation.kiro.hook +18 -0
  20. package/.kiro/settings/mcp.json +20 -0
  21. package/.kiro/specs/cdk-cost-analyzer/design.md +620 -0
  22. package/.kiro/specs/cdk-cost-analyzer/requirements.md +183 -0
  23. package/.kiro/specs/cdk-cost-analyzer/tasks.md +357 -0
  24. package/.kiro/specs/github-actions-ci/design.md +281 -0
  25. package/.kiro/specs/github-actions-ci/requirements.md +86 -0
  26. package/.kiro/specs/github-actions-ci/tasks.md +115 -0
  27. package/.kiro/specs/nlb-calculator-test-coverage/design.md +190 -0
  28. package/.kiro/specs/nlb-calculator-test-coverage/requirements.md +84 -0
  29. package/.kiro/specs/nlb-calculator-test-coverage/tasks.md +150 -0
  30. package/.kiro/specs/production-readiness/design.md +1213 -0
  31. package/.kiro/specs/production-readiness/requirements.md +312 -0
  32. package/.kiro/specs/production-readiness/tasks.md +269 -0
  33. package/.kiro/specs/repository-cleanup/design.md +283 -0
  34. package/.kiro/specs/repository-cleanup/requirements.md +74 -0
  35. package/.kiro/specs/repository-cleanup/tasks.md +64 -0
  36. package/.kiro/steering/aws-cli-best-practices.md +41 -0
  37. package/.kiro/steering/cdk-best-practices.md +49 -0
  38. package/.kiro/steering/development-standards.md +54 -0
  39. package/.kiro/steering/docker-best-practices.md +34 -0
  40. package/.kiro/steering/documentation-style.md +151 -0
  41. package/.kiro/steering/git-best-practices.md +37 -0
  42. package/.kiro/steering/mcp-best-practices.md +95 -0
  43. package/.kiro/steering/python-best-practices.md +48 -0
  44. package/.kiro/steering/react-best-practices.md +44 -0
  45. package/.kiro/steering/security-best-practices.md +41 -0
  46. package/.kiro/steering/testing-best-practices.md +59 -0
  47. package/.kiro/steering/typescript-best-practices.md +40 -0
  48. package/CHANGELOG.md +49 -0
  49. package/CONTRIBUTING.md +258 -0
  50. package/LICENSE +19 -0
  51. package/README.md +480 -0
  52. package/SECURITY.md +117 -0
  53. package/dist/api/index.d.ts +11 -0
  54. package/dist/api/index.js +65 -0
  55. package/dist/api/types.d.ts +15 -0
  56. package/dist/api/types.js +3 -0
  57. package/dist/cli/index.d.ts +2 -0
  58. package/dist/cli/index.js +262 -0
  59. package/dist/config/ConfigManager.d.ts +40 -0
  60. package/dist/config/ConfigManager.js +238 -0
  61. package/dist/config/index.d.ts +2 -0
  62. package/dist/config/index.js +19 -0
  63. package/dist/config/types.d.ts +72 -0
  64. package/dist/config/types.js +15 -0
  65. package/dist/diff/DiffEngine.d.ts +7 -0
  66. package/dist/diff/DiffEngine.js +73 -0
  67. package/dist/diff/index.d.ts +2 -0
  68. package/dist/diff/index.js +21 -0
  69. package/dist/diff/types.d.ts +20 -0
  70. package/dist/diff/types.js +3 -0
  71. package/dist/integrations/GitLabIntegration.d.ts +7 -0
  72. package/dist/integrations/GitLabIntegration.js +45 -0
  73. package/dist/integrations/index.d.ts +2 -0
  74. package/dist/integrations/index.js +21 -0
  75. package/dist/integrations/types.d.ts +11 -0
  76. package/dist/integrations/types.js +13 -0
  77. package/dist/parser/TemplateParser.d.ts +8 -0
  78. package/dist/parser/TemplateParser.js +75 -0
  79. package/dist/parser/index.d.ts +2 -0
  80. package/dist/parser/index.js +22 -0
  81. package/dist/parser/types.d.ts +30 -0
  82. package/dist/parser/types.js +3 -0
  83. package/dist/pipeline/PipelineOrchestrator.d.ts +23 -0
  84. package/dist/pipeline/PipelineOrchestrator.js +191 -0
  85. package/dist/pipeline/index.d.ts +2 -0
  86. package/dist/pipeline/index.js +19 -0
  87. package/dist/pipeline/types.d.ts +41 -0
  88. package/dist/pipeline/types.js +13 -0
  89. package/dist/pricing/CacheManager.d.ts +75 -0
  90. package/dist/pricing/CacheManager.js +195 -0
  91. package/dist/pricing/PricingClient.d.ts +17 -0
  92. package/dist/pricing/PricingClient.js +122 -0
  93. package/dist/pricing/PricingService.d.ts +16 -0
  94. package/dist/pricing/PricingService.js +149 -0
  95. package/dist/pricing/calculators/ALBCalculator.d.ts +16 -0
  96. package/dist/pricing/calculators/ALBCalculator.js +163 -0
  97. package/dist/pricing/calculators/APIGatewayCalculator.d.ts +10 -0
  98. package/dist/pricing/calculators/APIGatewayCalculator.js +177 -0
  99. package/dist/pricing/calculators/CloudFrontCalculator.d.ts +59 -0
  100. package/dist/pricing/calculators/CloudFrontCalculator.js +151 -0
  101. package/dist/pricing/calculators/DynamoDBCalculator.d.ts +9 -0
  102. package/dist/pricing/calculators/DynamoDBCalculator.js +146 -0
  103. package/dist/pricing/calculators/EC2Calculator.d.ts +7 -0
  104. package/dist/pricing/calculators/EC2Calculator.js +80 -0
  105. package/dist/pricing/calculators/ECSCalculator.d.ts +9 -0
  106. package/dist/pricing/calculators/ECSCalculator.js +116 -0
  107. package/dist/pricing/calculators/ElastiCacheCalculator.d.ts +8 -0
  108. package/dist/pricing/calculators/ElastiCacheCalculator.js +106 -0
  109. package/dist/pricing/calculators/LambdaCalculator.d.ts +13 -0
  110. package/dist/pricing/calculators/LambdaCalculator.js +111 -0
  111. package/dist/pricing/calculators/NLBCalculator.d.ts +16 -0
  112. package/dist/pricing/calculators/NLBCalculator.js +138 -0
  113. package/dist/pricing/calculators/NatGatewayCalculator.d.ts +12 -0
  114. package/dist/pricing/calculators/NatGatewayCalculator.js +116 -0
  115. package/dist/pricing/calculators/RDSCalculator.d.ts +9 -0
  116. package/dist/pricing/calculators/RDSCalculator.js +103 -0
  117. package/dist/pricing/calculators/S3Calculator.d.ts +8 -0
  118. package/dist/pricing/calculators/S3Calculator.js +68 -0
  119. package/dist/pricing/calculators/VPCEndpointCalculator.d.ts +12 -0
  120. package/dist/pricing/calculators/VPCEndpointCalculator.js +129 -0
  121. package/dist/pricing/index.d.ts +10 -0
  122. package/dist/pricing/index.js +37 -0
  123. package/dist/pricing/types.d.ts +53 -0
  124. package/dist/pricing/types.js +22 -0
  125. package/dist/releasetag.txt +1 -0
  126. package/dist/reporter/Reporter.d.ts +18 -0
  127. package/dist/reporter/Reporter.js +412 -0
  128. package/dist/reporter/index.d.ts +2 -0
  129. package/dist/reporter/index.js +21 -0
  130. package/dist/reporter/types.d.ts +72 -0
  131. package/dist/reporter/types.js +3 -0
  132. package/dist/synthesis/SynthesisOrchestrator.d.ts +26 -0
  133. package/dist/synthesis/SynthesisOrchestrator.js +243 -0
  134. package/dist/synthesis/index.d.ts +2 -0
  135. package/dist/synthesis/index.js +19 -0
  136. package/dist/synthesis/types.d.ts +17 -0
  137. package/dist/synthesis/types.js +13 -0
  138. package/dist/threshold/ThresholdEnforcer.d.ts +29 -0
  139. package/dist/threshold/ThresholdEnforcer.js +143 -0
  140. package/dist/threshold/index.d.ts +2 -0
  141. package/dist/threshold/index.js +19 -0
  142. package/dist/threshold/types.d.ts +15 -0
  143. package/dist/threshold/types.js +17 -0
  144. package/docs/CALCULATORS.md +820 -0
  145. package/docs/CI_CD.md +608 -0
  146. package/docs/CONFIGURATION.md +407 -0
  147. package/docs/DEVELOPMENT.md +387 -0
  148. package/docs/RELEASE.md +223 -0
  149. package/docs/TROUBLESHOOTING.md +847 -0
  150. package/examples/.cdk-cost-analyzer.yml +85 -0
  151. package/examples/.gitlab-ci.yml +125 -0
  152. package/examples/api-usage.js +26 -0
  153. package/examples/complex/base.json +16 -0
  154. package/examples/complex/target.json +29 -0
  155. package/examples/monorepo/.gitlab-ci.yml +251 -0
  156. package/examples/monorepo/README.md +341 -0
  157. package/examples/monorepo/package.json +27 -0
  158. package/examples/monorepo/packages/backend-infra/.cdk-cost-analyzer.yml +34 -0
  159. package/examples/monorepo/packages/backend-infra/bin/app.ts +16 -0
  160. package/examples/monorepo/packages/backend-infra/cdk.json +7 -0
  161. package/examples/monorepo/packages/backend-infra/lib/backend-stack.ts +128 -0
  162. package/examples/monorepo/packages/backend-infra/package.json +30 -0
  163. package/examples/monorepo/packages/backend-infra/tsconfig.json +11 -0
  164. package/examples/monorepo/packages/data-infra/.cdk-cost-analyzer.yml +38 -0
  165. package/examples/monorepo/packages/data-infra/bin/app.ts +16 -0
  166. package/examples/monorepo/packages/data-infra/cdk.json +7 -0
  167. package/examples/monorepo/packages/data-infra/lib/data-stack.ts +121 -0
  168. package/examples/monorepo/packages/data-infra/package.json +30 -0
  169. package/examples/monorepo/packages/data-infra/tsconfig.json +11 -0
  170. package/examples/monorepo/packages/frontend-infra/.cdk-cost-analyzer.yml +31 -0
  171. package/examples/monorepo/packages/frontend-infra/bin/app.ts +16 -0
  172. package/examples/monorepo/packages/frontend-infra/cdk.json +7 -0
  173. package/examples/monorepo/packages/frontend-infra/lib/frontend-stack.ts +60 -0
  174. package/examples/monorepo/packages/frontend-infra/package.json +30 -0
  175. package/examples/monorepo/packages/frontend-infra/tsconfig.json +11 -0
  176. package/examples/monorepo/tsconfig.json +35 -0
  177. package/examples/multi-stack/.cdk-cost-analyzer.yml +72 -0
  178. package/examples/multi-stack/.gitlab-ci.yml +184 -0
  179. package/examples/multi-stack/README.md +279 -0
  180. package/examples/multi-stack/bin/app.ts +36 -0
  181. package/examples/multi-stack/cdk.json +72 -0
  182. package/examples/multi-stack/lib/compute-stack.ts +128 -0
  183. package/examples/multi-stack/lib/networking-stack.ts +69 -0
  184. package/examples/multi-stack/lib/storage-stack.ts +141 -0
  185. package/examples/multi-stack/package-lock.json +4437 -0
  186. package/examples/multi-stack/package.json +42 -0
  187. package/examples/multi-stack/tsconfig.json +34 -0
  188. package/examples/simple/base.json +8 -0
  189. package/examples/simple/target.json +14 -0
  190. package/examples/single-stack/.NVP +0 -0
  191. package/examples/single-stack/.cdk-cost-analyzer.yml +52 -0
  192. package/examples/single-stack/.gitlab-ci.yml +126 -0
  193. package/examples/single-stack/README.md +184 -0
  194. package/examples/single-stack/UeK +0 -0
  195. package/examples/single-stack/bin/app.ts +16 -0
  196. package/examples/single-stack/cdk.json +72 -0
  197. package/examples/single-stack/lib/infrastructure-stack.ts +119 -0
  198. package/examples/single-stack/package-lock.json +4443 -0
  199. package/examples/single-stack/package.json +38 -0
  200. package/examples/single-stack/tsconfig.json +34 -0
  201. package/package.json +139 -0
  202. package/test-cdk-project/README-COMPUTE.md +141 -0
  203. package/test-cdk-project/README.md +95 -0
  204. package/test-cdk-project/app-with-compute.js +102 -0
  205. package/test-cdk-project/app.js +81 -0
  206. package/test-cdk-project/cdk-compute.json +3 -0
  207. package/test-cdk-project/cdk.context.json +7 -0
  208. package/test-cdk-project/cdk.json +3 -0
  209. package/test-cdk-project/cdk.out/TestStack.assets.json +21 -0
  210. package/test-cdk-project/cdk.out/TestStack.template.json +115 -0
  211. package/test-cdk-project/cdk.out/cdk.out +1 -0
  212. package/test-cdk-project/cdk.out/manifest.json +503 -0
  213. package/test-cdk-project/cdk.out/tree.json +1 -0
  214. package/test-cdk-project/cdk.out.base/TestStack.assets.json +21 -0
  215. package/test-cdk-project/cdk.out.base/TestStack.template.json +115 -0
  216. package/test-cdk-project/cdk.out.base/cdk.out +1 -0
  217. package/test-cdk-project/cdk.out.base/manifest.json +503 -0
  218. package/test-cdk-project/cdk.out.base/tree.json +1 -0
  219. package/test-cdk-project/cdk.out.target/TestStack.assets.json +21 -0
  220. package/test-cdk-project/cdk.out.target/TestStack.template.json +183 -0
  221. package/test-cdk-project/cdk.out.target/cdk.out +1 -0
  222. package/test-cdk-project/cdk.out.target/manifest.json +521 -0
  223. package/test-cdk-project/cdk.out.target/tree.json +1 -0
  224. package/test-cdk-project/package-lock.json +422 -0
  225. package/test-cdk-project/package.json +17 -0
  226. package/tools/workflows/README.md +102 -0
  227. package/tools/workflows/validate-workflows.js +109 -0
  228. package/tools/workflows/workflow-utils.ts +181 -0
@@ -0,0 +1,243 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
35
+ Object.defineProperty(exports, "__esModule", { value: true });
36
+ exports.SynthesisOrchestrator = void 0;
37
+ const child_process_1 = require("child_process");
38
+ const fs = __importStar(require("fs/promises"));
39
+ const path = __importStar(require("path"));
40
+ const types_1 = require("./types");
41
+ class SynthesisOrchestrator {
42
+ DEFAULT_OUTPUT_PATH = 'cdk.out';
43
+ /**
44
+ * Execute CDK synthesis
45
+ */
46
+ async synthesize(options) {
47
+ const startTime = Date.now();
48
+ try {
49
+ const outputPath = options.outputPath || this.DEFAULT_OUTPUT_PATH;
50
+ const command = options.customCommand || 'npx cdk synth';
51
+ await this.executeSynthesis(command, options.cdkAppPath, options.context, outputPath);
52
+ const fullOutputPath = path.join(options.cdkAppPath, outputPath);
53
+ const { templatePaths, stackNames } = await this.findTemplates(fullOutputPath);
54
+ const duration = Date.now() - startTime;
55
+ return {
56
+ success: true,
57
+ templatePaths,
58
+ stackNames,
59
+ duration,
60
+ };
61
+ }
62
+ catch (error) {
63
+ const duration = Date.now() - startTime;
64
+ const errorMessage = error instanceof Error ? error.message : String(error);
65
+ if (error instanceof types_1.SynthesisError) {
66
+ return {
67
+ success: false,
68
+ templatePaths: [],
69
+ stackNames: [],
70
+ error: errorMessage,
71
+ duration,
72
+ };
73
+ }
74
+ return {
75
+ success: false,
76
+ templatePaths: [],
77
+ stackNames: [],
78
+ error: `Synthesis failed: ${errorMessage}`,
79
+ duration,
80
+ };
81
+ }
82
+ }
83
+ /**
84
+ * Execute synthesis command
85
+ *
86
+ * Uses shell: false for security to prevent command injection attacks.
87
+ * Arguments are passed as an array to avoid shell interpretation.
88
+ *
89
+ * Implements a 15-second timeout to prevent hanging processes in CI:
90
+ * - Sends SIGTERM for graceful termination
91
+ * - Follows up with SIGKILL after 1 second if process doesn't exit
92
+ * - Prevents duplicate resolution using isResolved flag
93
+ * - Ensures all event listeners are cleaned up
94
+ * - Uses process.kill as fallback for stubborn processes
95
+ */
96
+ async executeSynthesis(command, cdkAppPath, context, outputPath) {
97
+ return new Promise((resolve, reject) => {
98
+ const [cmd, ...args] = command.split(' ');
99
+ // Add context arguments
100
+ const allArgs = [...args];
101
+ if (context) {
102
+ for (const [key, value] of Object.entries(context)) {
103
+ allArgs.push('-c', `${key}=${value}`);
104
+ }
105
+ }
106
+ // Add output path if specified
107
+ if (outputPath) {
108
+ allArgs.push('--output', outputPath);
109
+ }
110
+ // Use shell: false for security and pass arguments properly
111
+ const proc = (0, child_process_1.spawn)(cmd, allArgs, {
112
+ cwd: cdkAppPath,
113
+ shell: false,
114
+ stdio: ['ignore', 'pipe', 'pipe'],
115
+ detached: false, // Ensure process is part of the same process group
116
+ });
117
+ let stdout = '';
118
+ let stderr = '';
119
+ let isResolved = false;
120
+ let killTimeout = null;
121
+ const cleanup = () => {
122
+ if (killTimeout) {
123
+ clearTimeout(killTimeout);
124
+ killTimeout = null;
125
+ }
126
+ // Remove all listeners to prevent memory leaks
127
+ proc.removeAllListeners();
128
+ proc.stdout?.removeAllListeners();
129
+ proc.stderr?.removeAllListeners();
130
+ };
131
+ const forceKill = () => {
132
+ try {
133
+ // Try multiple kill methods for stubborn processes
134
+ if (proc.pid && !proc.killed) {
135
+ proc.kill('SIGKILL');
136
+ // Also try process.kill as fallback
137
+ try {
138
+ process.kill(proc.pid, 'SIGKILL');
139
+ }
140
+ catch (killError) {
141
+ // Ignore kill errors - process might already be dead
142
+ }
143
+ }
144
+ }
145
+ catch (error) {
146
+ // Ignore errors during force kill
147
+ }
148
+ };
149
+ // Set up timeout to prevent hanging processes
150
+ const timeout = setTimeout(() => {
151
+ if (!isResolved) {
152
+ isResolved = true;
153
+ // First try graceful termination
154
+ proc.kill('SIGTERM');
155
+ // Force kill after 1 second if still running
156
+ killTimeout = setTimeout(() => {
157
+ forceKill();
158
+ }, 1000);
159
+ cleanup();
160
+ reject(new types_1.SynthesisError('CDK synthesis timed out after 15 seconds', stderr || stdout || 'No output captured'));
161
+ }
162
+ }, 15000); // Reduced to 15 second timeout
163
+ proc.stdout?.on('data', (data) => {
164
+ stdout += data.toString();
165
+ });
166
+ proc.stderr?.on('data', (data) => {
167
+ stderr += data.toString();
168
+ });
169
+ proc.on('error', (error) => {
170
+ if (!isResolved) {
171
+ isResolved = true;
172
+ clearTimeout(timeout);
173
+ cleanup();
174
+ reject(new types_1.SynthesisError(`Failed to execute synthesis command: ${error.message}`, stderr || 'No error output captured'));
175
+ }
176
+ });
177
+ proc.on('close', (code) => {
178
+ if (!isResolved) {
179
+ isResolved = true;
180
+ clearTimeout(timeout);
181
+ cleanup();
182
+ if (code !== 0) {
183
+ reject(new types_1.SynthesisError(`CDK synthesis failed with exit code ${code}`, stderr || stdout || 'No output captured'));
184
+ }
185
+ else {
186
+ resolve();
187
+ }
188
+ }
189
+ });
190
+ // Additional safety: handle process exit
191
+ proc.on('exit', (code, signal) => {
192
+ if (!isResolved) {
193
+ isResolved = true;
194
+ clearTimeout(timeout);
195
+ cleanup();
196
+ if (signal) {
197
+ reject(new types_1.SynthesisError(`CDK synthesis terminated by signal ${signal}`, stderr || stdout || 'No output captured'));
198
+ }
199
+ else if (code !== 0) {
200
+ reject(new types_1.SynthesisError(`CDK synthesis failed with exit code ${code}`, stderr || stdout || 'No output captured'));
201
+ }
202
+ else {
203
+ resolve();
204
+ }
205
+ }
206
+ });
207
+ });
208
+ }
209
+ /**
210
+ * Find all CloudFormation templates in output directory
211
+ */
212
+ async findTemplates(outputPath) {
213
+ try {
214
+ const files = await fs.readdir(outputPath);
215
+ const templatePaths = [];
216
+ const stackNames = [];
217
+ for (const file of files) {
218
+ // Match CloudFormation template files (stack-name.template.json or stack-name.template.yaml)
219
+ if (file.endsWith('.template.json') ||
220
+ file.endsWith('.template.yaml') ||
221
+ file.endsWith('.template.yml')) {
222
+ const fullPath = path.join(outputPath, file);
223
+ templatePaths.push(fullPath);
224
+ // Extract stack name from filename
225
+ const stackName = file
226
+ .replace('.template.json', '')
227
+ .replace('.template.yaml', '')
228
+ .replace('.template.yml', '');
229
+ stackNames.push(stackName);
230
+ }
231
+ }
232
+ if (templatePaths.length === 0) {
233
+ throw new Error('No CloudFormation templates found in output directory');
234
+ }
235
+ return { templatePaths, stackNames };
236
+ }
237
+ catch (error) {
238
+ throw new Error(`Failed to find templates: ${error instanceof Error ? error.message : String(error)}`);
239
+ }
240
+ }
241
+ }
242
+ exports.SynthesisOrchestrator = SynthesisOrchestrator;
243
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"SynthesisOrchestrator.js","sourceRoot":"","sources":["../../src/synthesis/SynthesisOrchestrator.ts"],"names":[],"mappings":"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","sourcesContent":["import { spawn } from 'child_process';\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\nimport { SynthesisOptions, SynthesisResult, SynthesisError } from './types';\n\nexport class SynthesisOrchestrator {\n  private readonly DEFAULT_OUTPUT_PATH = 'cdk.out';\n\n  /**\n   * Execute CDK synthesis\n   */\n  async synthesize(options: SynthesisOptions): Promise<SynthesisResult> {\n    const startTime = Date.now();\n\n    try {\n      const outputPath = options.outputPath || this.DEFAULT_OUTPUT_PATH;\n      const command = options.customCommand || 'npx cdk synth';\n\n      await this.executeSynthesis(command, options.cdkAppPath, options.context, outputPath);\n\n      const fullOutputPath = path.join(options.cdkAppPath, outputPath);\n      const { templatePaths, stackNames } = await this.findTemplates(fullOutputPath);\n\n      const duration = Date.now() - startTime;\n\n      return {\n        success: true,\n        templatePaths,\n        stackNames,\n        duration,\n      };\n    } catch (error) {\n      const duration = Date.now() - startTime;\n      const errorMessage = error instanceof Error ? error.message : String(error);\n\n      if (error instanceof SynthesisError) {\n        return {\n          success: false,\n          templatePaths: [],\n          stackNames: [],\n          error: errorMessage,\n          duration,\n        };\n      }\n\n      return {\n        success: false,\n        templatePaths: [],\n        stackNames: [],\n        error: `Synthesis failed: ${errorMessage}`,\n        duration,\n      };\n    }\n  }\n\n  /**\n   * Execute synthesis command\n   *\n   * Uses shell: false for security to prevent command injection attacks.\n   * Arguments are passed as an array to avoid shell interpretation.\n   * \n   * Implements a 15-second timeout to prevent hanging processes in CI:\n   * - Sends SIGTERM for graceful termination\n   * - Follows up with SIGKILL after 1 second if process doesn't exit\n   * - Prevents duplicate resolution using isResolved flag\n   * - Ensures all event listeners are cleaned up\n   * - Uses process.kill as fallback for stubborn processes\n   */\n  private async executeSynthesis(\n    command: string,\n    cdkAppPath: string,\n    context?: Record<string, string>,\n    outputPath?: string,\n  ): Promise<void> {\n    return new Promise((resolve, reject) => {\n      const [cmd, ...args] = command.split(' ');\n\n      // Add context arguments\n      const allArgs = [...args];\n      if (context) {\n        for (const [key, value] of Object.entries(context)) {\n          allArgs.push('-c', `${key}=${value}`);\n        }\n      }\n\n      // Add output path if specified\n      if (outputPath) {\n        allArgs.push('--output', outputPath);\n      }\n\n      // Use shell: false for security and pass arguments properly\n      const proc = spawn(cmd, allArgs, {\n        cwd: cdkAppPath,\n        shell: false,\n        stdio: ['ignore', 'pipe', 'pipe'],\n        detached: false, // Ensure process is part of the same process group\n      });\n\n      let stdout = '';\n      let stderr = '';\n      let isResolved = false;\n      let killTimeout: NodeJS.Timeout | null = null;\n\n      const cleanup = (): void => {\n        if (killTimeout) {\n          clearTimeout(killTimeout);\n          killTimeout = null;\n        }\n        // Remove all listeners to prevent memory leaks\n        proc.removeAllListeners();\n        proc.stdout?.removeAllListeners();\n        proc.stderr?.removeAllListeners();\n      };\n\n      const forceKill = (): void => {\n        try {\n          // Try multiple kill methods for stubborn processes\n          if (proc.pid && !proc.killed) {\n            proc.kill('SIGKILL');\n            // Also try process.kill as fallback\n            try {\n              process.kill(proc.pid, 'SIGKILL');\n            } catch (killError) {\n              // Ignore kill errors - process might already be dead\n            }\n          }\n        } catch (error) {\n          // Ignore errors during force kill\n        }\n      };\n\n      // Set up timeout to prevent hanging processes\n      const timeout = setTimeout(() => {\n        if (!isResolved) {\n          isResolved = true;\n          \n          // First try graceful termination\n          proc.kill('SIGTERM');\n          \n          // Force kill after 1 second if still running\n          killTimeout = setTimeout(() => {\n            forceKill();\n          }, 1000);\n          \n          cleanup();\n          reject(\n            new SynthesisError(\n              'CDK synthesis timed out after 15 seconds',\n              stderr || stdout || 'No output captured',\n            ),\n          );\n        }\n      }, 15000); // Reduced to 15 second timeout\n\n      proc.stdout?.on('data', (data: Buffer) => {\n        stdout += data.toString();\n      });\n\n      proc.stderr?.on('data', (data: Buffer) => {\n        stderr += data.toString();\n      });\n\n      proc.on('error', (error: Error) => {\n        if (!isResolved) {\n          isResolved = true;\n          clearTimeout(timeout);\n          cleanup();\n          reject(\n            new SynthesisError(\n              `Failed to execute synthesis command: ${error.message}`,\n              stderr || 'No error output captured',\n            ),\n          );\n        }\n      });\n\n      proc.on('close', (code: number | null) => {\n        if (!isResolved) {\n          isResolved = true;\n          clearTimeout(timeout);\n          cleanup();\n          if (code !== 0) {\n            reject(\n              new SynthesisError(\n                `CDK synthesis failed with exit code ${code}`,\n                stderr || stdout || 'No output captured',\n              ),\n            );\n          } else {\n            resolve();\n          }\n        }\n      });\n\n      // Additional safety: handle process exit\n      proc.on('exit', (code: number | null, signal: string | null) => {\n        if (!isResolved) {\n          isResolved = true;\n          clearTimeout(timeout);\n          cleanup();\n          if (signal) {\n            reject(\n              new SynthesisError(\n                `CDK synthesis terminated by signal ${signal}`,\n                stderr || stdout || 'No output captured',\n              ),\n            );\n          } else if (code !== 0) {\n            reject(\n              new SynthesisError(\n                `CDK synthesis failed with exit code ${code}`,\n                stderr || stdout || 'No output captured',\n              ),\n            );\n          } else {\n            resolve();\n          }\n        }\n      });\n    });\n  }\n\n  /**\n   * Find all CloudFormation templates in output directory\n   */\n  private async findTemplates(\n    outputPath: string,\n  ): Promise<{ templatePaths: string[]; stackNames: string[] }> {\n    try {\n      const files = await fs.readdir(outputPath);\n\n      const templatePaths: string[] = [];\n      const stackNames: string[] = [];\n\n      for (const file of files) {\n        // Match CloudFormation template files (stack-name.template.json or stack-name.template.yaml)\n        if (\n          file.endsWith('.template.json') ||\n          file.endsWith('.template.yaml') ||\n          file.endsWith('.template.yml')\n        ) {\n          const fullPath = path.join(outputPath, file);\n          templatePaths.push(fullPath);\n\n          // Extract stack name from filename\n          const stackName = file\n            .replace('.template.json', '')\n            .replace('.template.yaml', '')\n            .replace('.template.yml', '');\n          stackNames.push(stackName);\n        }\n      }\n\n      if (templatePaths.length === 0) {\n        throw new Error('No CloudFormation templates found in output directory');\n      }\n\n      return { templatePaths, stackNames };\n    } catch (error) {\n      throw new Error(\n        `Failed to find templates: ${error instanceof Error ? error.message : String(error)}`,\n      );\n    }\n  }\n}\n"]}
@@ -0,0 +1,2 @@
1
+ export * from './SynthesisOrchestrator';
2
+ export * from './types';
@@ -0,0 +1,19 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __exportStar = (this && this.__exportStar) || function(m, exports) {
14
+ for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
15
+ };
16
+ Object.defineProperty(exports, "__esModule", { value: true });
17
+ __exportStar(require("./SynthesisOrchestrator"), exports);
18
+ __exportStar(require("./types"), exports);
19
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvc3ludGhlc2lzL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7QUFBQSwwREFBd0M7QUFDeEMsMENBQXdCIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9TeW50aGVzaXNPcmNoZXN0cmF0b3InO1xuZXhwb3J0ICogZnJvbSAnLi90eXBlcyc7XG4iXX0=
@@ -0,0 +1,17 @@
1
+ export interface SynthesisOptions {
2
+ cdkAppPath: string;
3
+ outputPath?: string;
4
+ context?: Record<string, string>;
5
+ customCommand?: string;
6
+ }
7
+ export interface SynthesisResult {
8
+ success: boolean;
9
+ templatePaths: string[];
10
+ stackNames: string[];
11
+ error?: string;
12
+ duration: number;
13
+ }
14
+ export declare class SynthesisError extends Error {
15
+ cdkOutput: string;
16
+ constructor(message: string, cdkOutput: string);
17
+ }
@@ -0,0 +1,13 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.SynthesisError = void 0;
4
+ class SynthesisError extends Error {
5
+ cdkOutput;
6
+ constructor(message, cdkOutput) {
7
+ super(message);
8
+ this.cdkOutput = cdkOutput;
9
+ this.name = 'SynthesisError';
10
+ }
11
+ }
12
+ exports.SynthesisError = SynthesisError;
13
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvc3ludGhlc2lzL3R5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQWVBLE1BQWEsY0FBZSxTQUFRLEtBQUs7SUFHOUI7SUFGVCxZQUNFLE9BQWUsRUFDUixTQUFpQjtRQUV4QixLQUFLLENBQUMsT0FBTyxDQUFDLENBQUM7UUFGUixjQUFTLEdBQVQsU0FBUyxDQUFRO1FBR3hCLElBQUksQ0FBQyxJQUFJLEdBQUcsZ0JBQWdCLENBQUM7SUFDL0IsQ0FBQztDQUNGO0FBUkQsd0NBUUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgaW50ZXJmYWNlIFN5bnRoZXNpc09wdGlvbnMge1xuICBjZGtBcHBQYXRoOiBzdHJpbmc7XG4gIG91dHB1dFBhdGg/OiBzdHJpbmc7XG4gIGNvbnRleHQ/OiBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+O1xuICBjdXN0b21Db21tYW5kPzogc3RyaW5nO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFN5bnRoZXNpc1Jlc3VsdCB7XG4gIHN1Y2Nlc3M6IGJvb2xlYW47XG4gIHRlbXBsYXRlUGF0aHM6IHN0cmluZ1tdO1xuICBzdGFja05hbWVzOiBzdHJpbmdbXTtcbiAgZXJyb3I/OiBzdHJpbmc7XG4gIGR1cmF0aW9uOiBudW1iZXI7XG59XG5cbmV4cG9ydCBjbGFzcyBTeW50aGVzaXNFcnJvciBleHRlbmRzIEVycm9yIHtcbiAgY29uc3RydWN0b3IoXG4gICAgbWVzc2FnZTogc3RyaW5nLFxuICAgIHB1YmxpYyBjZGtPdXRwdXQ6IHN0cmluZyxcbiAgKSB7XG4gICAgc3VwZXIobWVzc2FnZSk7XG4gICAgdGhpcy5uYW1lID0gJ1N5bnRoZXNpc0Vycm9yJztcbiAgfVxufVxuIl19
@@ -0,0 +1,29 @@
1
+ import { ThresholdEvaluation } from './types';
2
+ import { ThresholdConfig } from '../config/types';
3
+ import { ResourceCost, ModifiedResourceCost } from '../pricing/types';
4
+ export declare class ThresholdEnforcer {
5
+ /**
6
+ * Evaluate cost delta against configured thresholds
7
+ */
8
+ evaluateThreshold(costDelta: number, addedResources: ResourceCost[], modifiedResources: ModifiedResourceCost[], config?: ThresholdConfig, environment?: string): ThresholdEvaluation;
9
+ /**
10
+ * Select appropriate threshold based on environment
11
+ */
12
+ private selectThresholds;
13
+ /**
14
+ * Get top cost contributors sorted by impact
15
+ */
16
+ private getTopContributors;
17
+ /**
18
+ * Format error threshold message
19
+ */
20
+ private formatErrorMessage;
21
+ /**
22
+ * Format warning threshold message
23
+ */
24
+ private formatWarningMessage;
25
+ /**
26
+ * Get recommendations based on threshold level and contributors
27
+ */
28
+ private getRecommendations;
29
+ }
@@ -0,0 +1,143 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.ThresholdEnforcer = void 0;
4
+ class ThresholdEnforcer {
5
+ /**
6
+ * Evaluate cost delta against configured thresholds
7
+ */
8
+ evaluateThreshold(costDelta, addedResources, modifiedResources, config, environment) {
9
+ if (!config) {
10
+ return {
11
+ passed: true,
12
+ level: 'none',
13
+ delta: costDelta,
14
+ message: 'No thresholds configured',
15
+ recommendations: [],
16
+ };
17
+ }
18
+ const thresholds = this.selectThresholds(config, environment);
19
+ if (!thresholds) {
20
+ return {
21
+ passed: true,
22
+ level: 'none',
23
+ delta: costDelta,
24
+ message: 'No thresholds configured',
25
+ recommendations: [],
26
+ };
27
+ }
28
+ // Check error threshold first
29
+ if (thresholds.error !== undefined && costDelta > thresholds.error) {
30
+ const topContributors = this.getTopContributors(addedResources, modifiedResources, 5);
31
+ return {
32
+ passed: false,
33
+ level: 'error',
34
+ threshold: thresholds.error,
35
+ delta: costDelta,
36
+ message: this.formatErrorMessage(costDelta, thresholds.error),
37
+ recommendations: this.getRecommendations('error', topContributors),
38
+ };
39
+ }
40
+ // Check warning threshold
41
+ if (thresholds.warning !== undefined && costDelta > thresholds.warning) {
42
+ const topContributors = this.getTopContributors(addedResources, modifiedResources, 5);
43
+ return {
44
+ passed: true,
45
+ level: 'warning',
46
+ threshold: thresholds.warning,
47
+ delta: costDelta,
48
+ message: this.formatWarningMessage(costDelta, thresholds.warning),
49
+ recommendations: this.getRecommendations('warning', topContributors),
50
+ };
51
+ }
52
+ return {
53
+ passed: true,
54
+ level: 'none',
55
+ delta: costDelta,
56
+ message: `Cost delta $${costDelta.toFixed(2)}/month is within thresholds`,
57
+ recommendations: [],
58
+ };
59
+ }
60
+ /**
61
+ * Select appropriate threshold based on environment
62
+ */
63
+ selectThresholds(config, environment) {
64
+ if (environment && config.environments?.[environment]) {
65
+ return config.environments[environment];
66
+ }
67
+ return config.default;
68
+ }
69
+ /**
70
+ * Get top cost contributors sorted by impact
71
+ */
72
+ getTopContributors(addedResources, modifiedResources, limit) {
73
+ const allContributors = [
74
+ ...addedResources,
75
+ ...modifiedResources.map((r) => ({
76
+ logicalId: r.logicalId,
77
+ type: r.type,
78
+ monthlyCost: {
79
+ amount: r.costDelta,
80
+ currency: r.newMonthlyCost.currency,
81
+ confidence: r.newMonthlyCost.confidence,
82
+ assumptions: r.newMonthlyCost.assumptions,
83
+ },
84
+ })),
85
+ ];
86
+ return allContributors
87
+ .sort((a, b) => b.monthlyCost.amount - a.monthlyCost.amount)
88
+ .slice(0, limit);
89
+ }
90
+ /**
91
+ * Format error threshold message
92
+ */
93
+ formatErrorMessage(delta, threshold) {
94
+ const exceededBy = delta - threshold;
95
+ const percentage = ((exceededBy / threshold) * 100).toFixed(1);
96
+ return `Cost increase of $${delta.toFixed(2)}/month exceeds error threshold of $${threshold.toFixed(2)}/month by $${exceededBy.toFixed(2)} (${percentage}%)`;
97
+ }
98
+ /**
99
+ * Format warning threshold message
100
+ */
101
+ formatWarningMessage(delta, threshold) {
102
+ const exceededBy = delta - threshold;
103
+ const percentage = ((exceededBy / threshold) * 100).toFixed(1);
104
+ return `Cost increase of $${delta.toFixed(2)}/month exceeds warning threshold of $${threshold.toFixed(2)}/month by $${exceededBy.toFixed(2)} (${percentage}%)`;
105
+ }
106
+ /**
107
+ * Get recommendations based on threshold level and contributors
108
+ */
109
+ getRecommendations(level, topContributors) {
110
+ const recommendations = [];
111
+ if (level === 'error') {
112
+ recommendations.push('This change cannot be merged without approval due to cost impact.');
113
+ recommendations.push('Review the cost breakdown and consider optimizations before proceeding.');
114
+ recommendations.push('Contact your FinOps team for threshold override approval if this cost increase is necessary.');
115
+ }
116
+ else {
117
+ recommendations.push('Review this cost increase with your team before merging.');
118
+ recommendations.push('Consider whether all resources in this change are necessary.');
119
+ }
120
+ if (topContributors.length > 0) {
121
+ recommendations.push(`Top cost contributors: ${topContributors
122
+ .map((r) => `${r.type} (${r.logicalId}): $${r.monthlyCost.amount.toFixed(2)}/month`)
123
+ .join(', ')}`);
124
+ // Specific recommendations based on resource types
125
+ const resourceTypes = new Set(topContributors.map((r) => r.type));
126
+ if (resourceTypes.has('AWS::RDS::DBInstance')) {
127
+ recommendations.push('Consider using smaller RDS instance types or Aurora Serverless for lower costs.');
128
+ }
129
+ if (resourceTypes.has('AWS::EC2::Instance')) {
130
+ recommendations.push('Consider using smaller EC2 instance types, Spot instances, or Savings Plans.');
131
+ }
132
+ if (resourceTypes.has('AWS::EC2::NatGateway')) {
133
+ recommendations.push('NAT Gateways have high data processing costs. Consider using VPC endpoints or consolidating NAT Gateways.');
134
+ }
135
+ if (resourceTypes.has('AWS::ElasticLoadBalancingV2::LoadBalancer')) {
136
+ recommendations.push('Load Balancers have hourly costs. Consider sharing load balancers across services if possible.');
137
+ }
138
+ }
139
+ return recommendations;
140
+ }
141
+ }
142
+ exports.ThresholdEnforcer = ThresholdEnforcer;
143
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ThresholdEnforcer.js","sourceRoot":"","sources":["../../src/threshold/ThresholdEnforcer.ts"],"names":[],"mappings":"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","sourcesContent":["import { ThresholdEvaluation } from './types';\nimport { ThresholdConfig, ThresholdLevels } from '../config/types';\nimport { ResourceCost, ModifiedResourceCost } from '../pricing/types';\n\nexport class ThresholdEnforcer {\n  /**\n   * Evaluate cost delta against configured thresholds\n   */\n  evaluateThreshold(\n    costDelta: number,\n    addedResources: ResourceCost[],\n    modifiedResources: ModifiedResourceCost[],\n    config?: ThresholdConfig,\n    environment?: string,\n  ): ThresholdEvaluation {\n    if (!config) {\n      return {\n        passed: true,\n        level: 'none',\n        delta: costDelta,\n        message: 'No thresholds configured',\n        recommendations: [],\n      };\n    }\n\n    const thresholds = this.selectThresholds(config, environment);\n\n    if (!thresholds) {\n      return {\n        passed: true,\n        level: 'none',\n        delta: costDelta,\n        message: 'No thresholds configured',\n        recommendations: [],\n      };\n    }\n\n    // Check error threshold first\n    if (thresholds.error !== undefined && costDelta > thresholds.error) {\n      const topContributors = this.getTopContributors(\n        addedResources,\n        modifiedResources,\n        5,\n      );\n\n      return {\n        passed: false,\n        level: 'error',\n        threshold: thresholds.error,\n        delta: costDelta,\n        message: this.formatErrorMessage(costDelta, thresholds.error),\n        recommendations: this.getRecommendations('error', topContributors),\n      };\n    }\n\n    // Check warning threshold\n    if (thresholds.warning !== undefined && costDelta > thresholds.warning) {\n      const topContributors = this.getTopContributors(\n        addedResources,\n        modifiedResources,\n        5,\n      );\n\n      return {\n        passed: true,\n        level: 'warning',\n        threshold: thresholds.warning,\n        delta: costDelta,\n        message: this.formatWarningMessage(costDelta, thresholds.warning),\n        recommendations: this.getRecommendations('warning', topContributors),\n      };\n    }\n\n    return {\n      passed: true,\n      level: 'none',\n      delta: costDelta,\n      message: `Cost delta $${costDelta.toFixed(2)}/month is within thresholds`,\n      recommendations: [],\n    };\n  }\n\n  /**\n   * Select appropriate threshold based on environment\n   */\n  private selectThresholds(\n    config: ThresholdConfig,\n    environment?: string,\n  ): ThresholdLevels | undefined {\n    if (environment && config.environments?.[environment]) {\n      return config.environments[environment];\n    }\n    return config.default;\n  }\n\n  /**\n   * Get top cost contributors sorted by impact\n   */\n  private getTopContributors(\n    addedResources: ResourceCost[],\n    modifiedResources: ModifiedResourceCost[],\n    limit: number,\n  ): ResourceCost[] {\n    const allContributors: ResourceCost[] = [\n      ...addedResources,\n      ...modifiedResources.map((r) => ({\n        logicalId: r.logicalId,\n        type: r.type,\n        monthlyCost: {\n          amount: r.costDelta,\n          currency: r.newMonthlyCost.currency,\n          confidence: r.newMonthlyCost.confidence,\n          assumptions: r.newMonthlyCost.assumptions,\n        },\n      })),\n    ];\n\n    return allContributors\n      .sort((a, b) => b.monthlyCost.amount - a.monthlyCost.amount)\n      .slice(0, limit);\n  }\n\n  /**\n   * Format error threshold message\n   */\n  private formatErrorMessage(delta: number, threshold: number): string {\n    const exceededBy = delta - threshold;\n    const percentage = ((exceededBy / threshold) * 100).toFixed(1);\n\n    return `Cost increase of $${delta.toFixed(2)}/month exceeds error threshold of $${threshold.toFixed(2)}/month by $${exceededBy.toFixed(2)} (${percentage}%)`;\n  }\n\n  /**\n   * Format warning threshold message\n   */\n  private formatWarningMessage(delta: number, threshold: number): string {\n    const exceededBy = delta - threshold;\n    const percentage = ((exceededBy / threshold) * 100).toFixed(1);\n\n    return `Cost increase of $${delta.toFixed(2)}/month exceeds warning threshold of $${threshold.toFixed(2)}/month by $${exceededBy.toFixed(2)} (${percentage}%)`;\n  }\n\n  /**\n   * Get recommendations based on threshold level and contributors\n   */\n  private getRecommendations(\n    level: 'warning' | 'error',\n    topContributors: ResourceCost[],\n  ): string[] {\n    const recommendations: string[] = [];\n\n    if (level === 'error') {\n      recommendations.push(\n        'This change cannot be merged without approval due to cost impact.',\n      );\n      recommendations.push(\n        'Review the cost breakdown and consider optimizations before proceeding.',\n      );\n      recommendations.push(\n        'Contact your FinOps team for threshold override approval if this cost increase is necessary.',\n      );\n    } else {\n      recommendations.push(\n        'Review this cost increase with your team before merging.',\n      );\n      recommendations.push(\n        'Consider whether all resources in this change are necessary.',\n      );\n    }\n\n    if (topContributors.length > 0) {\n      recommendations.push(\n        `Top cost contributors: ${topContributors\n          .map((r) => `${r.type} (${r.logicalId}): $${r.monthlyCost.amount.toFixed(2)}/month`)\n          .join(', ')}`,\n      );\n\n      // Specific recommendations based on resource types\n      const resourceTypes = new Set(topContributors.map((r) => r.type));\n\n      if (resourceTypes.has('AWS::RDS::DBInstance')) {\n        recommendations.push(\n          'Consider using smaller RDS instance types or Aurora Serverless for lower costs.',\n        );\n      }\n\n      if (resourceTypes.has('AWS::EC2::Instance')) {\n        recommendations.push(\n          'Consider using smaller EC2 instance types, Spot instances, or Savings Plans.',\n        );\n      }\n\n      if (resourceTypes.has('AWS::EC2::NatGateway')) {\n        recommendations.push(\n          'NAT Gateways have high data processing costs. Consider using VPC endpoints or consolidating NAT Gateways.',\n        );\n      }\n\n      if (\n        resourceTypes.has('AWS::ElasticLoadBalancingV2::LoadBalancer')\n      ) {\n        recommendations.push(\n          'Load Balancers have hourly costs. Consider sharing load balancers across services if possible.',\n        );\n      }\n    }\n\n    return recommendations;\n  }\n}\n"]}
@@ -0,0 +1,2 @@
1
+ export * from './ThresholdEnforcer';
2
+ export * from './types';
@@ -0,0 +1,19 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __exportStar = (this && this.__exportStar) || function(m, exports) {
14
+ for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
15
+ };
16
+ Object.defineProperty(exports, "__esModule", { value: true });
17
+ __exportStar(require("./ThresholdEnforcer"), exports);
18
+ __exportStar(require("./types"), exports);
19
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvdGhyZXNob2xkL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7QUFBQSxzREFBb0M7QUFDcEMsMENBQXdCIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9UaHJlc2hvbGRFbmZvcmNlcic7XG5leHBvcnQgKiBmcm9tICcuL3R5cGVzJztcbiJdfQ==
@@ -0,0 +1,15 @@
1
+ import { ResourceCost } from '../pricing/types';
2
+ export interface ThresholdEvaluation {
3
+ passed: boolean;
4
+ level: 'none' | 'warning' | 'error';
5
+ threshold?: number;
6
+ delta: number;
7
+ message: string;
8
+ recommendations: string[];
9
+ }
10
+ export declare class ThresholdExceededError extends Error {
11
+ threshold: number;
12
+ actualDelta: number;
13
+ topContributors: ResourceCost[];
14
+ constructor(message: string, threshold: number, actualDelta: number, topContributors: ResourceCost[]);
15
+ }
@@ -0,0 +1,17 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.ThresholdExceededError = void 0;
4
+ class ThresholdExceededError extends Error {
5
+ threshold;
6
+ actualDelta;
7
+ topContributors;
8
+ constructor(message, threshold, actualDelta, topContributors) {
9
+ super(message);
10
+ this.threshold = threshold;
11
+ this.actualDelta = actualDelta;
12
+ this.topContributors = topContributors;
13
+ this.name = 'ThresholdExceededError';
14
+ }
15
+ }
16
+ exports.ThresholdExceededError = ThresholdExceededError;
17
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvdGhyZXNob2xkL3R5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQVdBLE1BQWEsc0JBQXVCLFNBQVEsS0FBSztJQUd0QztJQUNBO0lBQ0E7SUFKVCxZQUNFLE9BQWUsRUFDUixTQUFpQixFQUNqQixXQUFtQixFQUNuQixlQUErQjtRQUV0QyxLQUFLLENBQUMsT0FBTyxDQUFDLENBQUM7UUFKUixjQUFTLEdBQVQsU0FBUyxDQUFRO1FBQ2pCLGdCQUFXLEdBQVgsV0FBVyxDQUFRO1FBQ25CLG9CQUFlLEdBQWYsZUFBZSxDQUFnQjtRQUd0QyxJQUFJLENBQUMsSUFBSSxHQUFHLHdCQUF3QixDQUFDO0lBQ3ZDLENBQUM7Q0FDRjtBQVZELHdEQVVDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgUmVzb3VyY2VDb3N0IH0gZnJvbSAnLi4vcHJpY2luZy90eXBlcyc7XG5cbmV4cG9ydCBpbnRlcmZhY2UgVGhyZXNob2xkRXZhbHVhdGlvbiB7XG4gIHBhc3NlZDogYm9vbGVhbjtcbiAgbGV2ZWw6ICdub25lJyB8ICd3YXJuaW5nJyB8ICdlcnJvcic7XG4gIHRocmVzaG9sZD86IG51bWJlcjtcbiAgZGVsdGE6IG51bWJlcjtcbiAgbWVzc2FnZTogc3RyaW5nO1xuICByZWNvbW1lbmRhdGlvbnM6IHN0cmluZ1tdO1xufVxuXG5leHBvcnQgY2xhc3MgVGhyZXNob2xkRXhjZWVkZWRFcnJvciBleHRlbmRzIEVycm9yIHtcbiAgY29uc3RydWN0b3IoXG4gICAgbWVzc2FnZTogc3RyaW5nLFxuICAgIHB1YmxpYyB0aHJlc2hvbGQ6IG51bWJlcixcbiAgICBwdWJsaWMgYWN0dWFsRGVsdGE6IG51bWJlcixcbiAgICBwdWJsaWMgdG9wQ29udHJpYnV0b3JzOiBSZXNvdXJjZUNvc3RbXSxcbiAgKSB7XG4gICAgc3VwZXIobWVzc2FnZSk7XG4gICAgdGhpcy5uYW1lID0gJ1RocmVzaG9sZEV4Y2VlZGVkRXJyb3InO1xuICB9XG59XG4iXX0=