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.
- package/.cdk-cost-analyzer-cache/metadata.json +12 -0
- package/.gitlab-ci.yml +214 -0
- package/.husky/pre-commit +12 -0
- package/.kiro/hooks/accessibility-audit.kiro.hook +18 -0
- package/.kiro/hooks/api-schema-validation.kiro.hook +21 -0
- package/.kiro/hooks/auto-test-on-save.kiro.hook +19 -0
- package/.kiro/hooks/cdk-synth-on-change.kiro.hook +20 -0
- package/.kiro/hooks/code-coverage-check.kiro.hook +14 -0
- package/.kiro/hooks/commit-message-helper.kiro.hook +14 -0
- package/.kiro/hooks/dependency-update-check.kiro.hook +14 -0
- package/.kiro/hooks/env-file-validation.kiro.hook +18 -0
- package/.kiro/hooks/lint-and-format-on-save.kiro.hook +21 -0
- package/.kiro/hooks/mcp-config-validation.kiro.hook +17 -0
- package/.kiro/hooks/mcp-server-test.kiro.hook +14 -0
- package/.kiro/hooks/performance-analysis.kiro.hook +14 -0
- package/.kiro/hooks/readme-spell-check.kiro.hook +14 -0
- package/.kiro/hooks/security-scan-on-dependency-change.kiro.hook +21 -0
- package/.kiro/hooks/translation-update.kiro.hook +18 -0
- package/.kiro/hooks/update-documentation.kiro.hook +18 -0
- package/.kiro/settings/mcp.json +20 -0
- package/.kiro/specs/cdk-cost-analyzer/design.md +620 -0
- package/.kiro/specs/cdk-cost-analyzer/requirements.md +183 -0
- package/.kiro/specs/cdk-cost-analyzer/tasks.md +357 -0
- package/.kiro/specs/github-actions-ci/design.md +281 -0
- package/.kiro/specs/github-actions-ci/requirements.md +86 -0
- package/.kiro/specs/github-actions-ci/tasks.md +115 -0
- package/.kiro/specs/nlb-calculator-test-coverage/design.md +190 -0
- package/.kiro/specs/nlb-calculator-test-coverage/requirements.md +84 -0
- package/.kiro/specs/nlb-calculator-test-coverage/tasks.md +150 -0
- package/.kiro/specs/production-readiness/design.md +1213 -0
- package/.kiro/specs/production-readiness/requirements.md +312 -0
- package/.kiro/specs/production-readiness/tasks.md +269 -0
- package/.kiro/specs/repository-cleanup/design.md +283 -0
- package/.kiro/specs/repository-cleanup/requirements.md +74 -0
- package/.kiro/specs/repository-cleanup/tasks.md +64 -0
- package/.kiro/steering/aws-cli-best-practices.md +41 -0
- package/.kiro/steering/cdk-best-practices.md +49 -0
- package/.kiro/steering/development-standards.md +54 -0
- package/.kiro/steering/docker-best-practices.md +34 -0
- package/.kiro/steering/documentation-style.md +151 -0
- package/.kiro/steering/git-best-practices.md +37 -0
- package/.kiro/steering/mcp-best-practices.md +95 -0
- package/.kiro/steering/python-best-practices.md +48 -0
- package/.kiro/steering/react-best-practices.md +44 -0
- package/.kiro/steering/security-best-practices.md +41 -0
- package/.kiro/steering/testing-best-practices.md +59 -0
- package/.kiro/steering/typescript-best-practices.md +40 -0
- package/CHANGELOG.md +49 -0
- package/CONTRIBUTING.md +258 -0
- package/LICENSE +19 -0
- package/README.md +480 -0
- package/SECURITY.md +117 -0
- package/dist/api/index.d.ts +11 -0
- package/dist/api/index.js +65 -0
- package/dist/api/types.d.ts +15 -0
- package/dist/api/types.js +3 -0
- package/dist/cli/index.d.ts +2 -0
- package/dist/cli/index.js +262 -0
- package/dist/config/ConfigManager.d.ts +40 -0
- package/dist/config/ConfigManager.js +238 -0
- package/dist/config/index.d.ts +2 -0
- package/dist/config/index.js +19 -0
- package/dist/config/types.d.ts +72 -0
- package/dist/config/types.js +15 -0
- package/dist/diff/DiffEngine.d.ts +7 -0
- package/dist/diff/DiffEngine.js +73 -0
- package/dist/diff/index.d.ts +2 -0
- package/dist/diff/index.js +21 -0
- package/dist/diff/types.d.ts +20 -0
- package/dist/diff/types.js +3 -0
- package/dist/integrations/GitLabIntegration.d.ts +7 -0
- package/dist/integrations/GitLabIntegration.js +45 -0
- package/dist/integrations/index.d.ts +2 -0
- package/dist/integrations/index.js +21 -0
- package/dist/integrations/types.d.ts +11 -0
- package/dist/integrations/types.js +13 -0
- package/dist/parser/TemplateParser.d.ts +8 -0
- package/dist/parser/TemplateParser.js +75 -0
- package/dist/parser/index.d.ts +2 -0
- package/dist/parser/index.js +22 -0
- package/dist/parser/types.d.ts +30 -0
- package/dist/parser/types.js +3 -0
- package/dist/pipeline/PipelineOrchestrator.d.ts +23 -0
- package/dist/pipeline/PipelineOrchestrator.js +191 -0
- package/dist/pipeline/index.d.ts +2 -0
- package/dist/pipeline/index.js +19 -0
- package/dist/pipeline/types.d.ts +41 -0
- package/dist/pipeline/types.js +13 -0
- package/dist/pricing/CacheManager.d.ts +75 -0
- package/dist/pricing/CacheManager.js +195 -0
- package/dist/pricing/PricingClient.d.ts +17 -0
- package/dist/pricing/PricingClient.js +122 -0
- package/dist/pricing/PricingService.d.ts +16 -0
- package/dist/pricing/PricingService.js +149 -0
- package/dist/pricing/calculators/ALBCalculator.d.ts +16 -0
- package/dist/pricing/calculators/ALBCalculator.js +163 -0
- package/dist/pricing/calculators/APIGatewayCalculator.d.ts +10 -0
- package/dist/pricing/calculators/APIGatewayCalculator.js +177 -0
- package/dist/pricing/calculators/CloudFrontCalculator.d.ts +59 -0
- package/dist/pricing/calculators/CloudFrontCalculator.js +151 -0
- package/dist/pricing/calculators/DynamoDBCalculator.d.ts +9 -0
- package/dist/pricing/calculators/DynamoDBCalculator.js +146 -0
- package/dist/pricing/calculators/EC2Calculator.d.ts +7 -0
- package/dist/pricing/calculators/EC2Calculator.js +80 -0
- package/dist/pricing/calculators/ECSCalculator.d.ts +9 -0
- package/dist/pricing/calculators/ECSCalculator.js +116 -0
- package/dist/pricing/calculators/ElastiCacheCalculator.d.ts +8 -0
- package/dist/pricing/calculators/ElastiCacheCalculator.js +106 -0
- package/dist/pricing/calculators/LambdaCalculator.d.ts +13 -0
- package/dist/pricing/calculators/LambdaCalculator.js +111 -0
- package/dist/pricing/calculators/NLBCalculator.d.ts +16 -0
- package/dist/pricing/calculators/NLBCalculator.js +138 -0
- package/dist/pricing/calculators/NatGatewayCalculator.d.ts +12 -0
- package/dist/pricing/calculators/NatGatewayCalculator.js +116 -0
- package/dist/pricing/calculators/RDSCalculator.d.ts +9 -0
- package/dist/pricing/calculators/RDSCalculator.js +103 -0
- package/dist/pricing/calculators/S3Calculator.d.ts +8 -0
- package/dist/pricing/calculators/S3Calculator.js +68 -0
- package/dist/pricing/calculators/VPCEndpointCalculator.d.ts +12 -0
- package/dist/pricing/calculators/VPCEndpointCalculator.js +129 -0
- package/dist/pricing/index.d.ts +10 -0
- package/dist/pricing/index.js +37 -0
- package/dist/pricing/types.d.ts +53 -0
- package/dist/pricing/types.js +22 -0
- package/dist/releasetag.txt +1 -0
- package/dist/reporter/Reporter.d.ts +18 -0
- package/dist/reporter/Reporter.js +412 -0
- package/dist/reporter/index.d.ts +2 -0
- package/dist/reporter/index.js +21 -0
- package/dist/reporter/types.d.ts +72 -0
- package/dist/reporter/types.js +3 -0
- package/dist/synthesis/SynthesisOrchestrator.d.ts +26 -0
- package/dist/synthesis/SynthesisOrchestrator.js +243 -0
- package/dist/synthesis/index.d.ts +2 -0
- package/dist/synthesis/index.js +19 -0
- package/dist/synthesis/types.d.ts +17 -0
- package/dist/synthesis/types.js +13 -0
- package/dist/threshold/ThresholdEnforcer.d.ts +29 -0
- package/dist/threshold/ThresholdEnforcer.js +143 -0
- package/dist/threshold/index.d.ts +2 -0
- package/dist/threshold/index.js +19 -0
- package/dist/threshold/types.d.ts +15 -0
- package/dist/threshold/types.js +17 -0
- package/docs/CALCULATORS.md +820 -0
- package/docs/CI_CD.md +608 -0
- package/docs/CONFIGURATION.md +407 -0
- package/docs/DEVELOPMENT.md +387 -0
- package/docs/RELEASE.md +223 -0
- package/docs/TROUBLESHOOTING.md +847 -0
- package/examples/.cdk-cost-analyzer.yml +85 -0
- package/examples/.gitlab-ci.yml +125 -0
- package/examples/api-usage.js +26 -0
- package/examples/complex/base.json +16 -0
- package/examples/complex/target.json +29 -0
- package/examples/monorepo/.gitlab-ci.yml +251 -0
- package/examples/monorepo/README.md +341 -0
- package/examples/monorepo/package.json +27 -0
- package/examples/monorepo/packages/backend-infra/.cdk-cost-analyzer.yml +34 -0
- package/examples/monorepo/packages/backend-infra/bin/app.ts +16 -0
- package/examples/monorepo/packages/backend-infra/cdk.json +7 -0
- package/examples/monorepo/packages/backend-infra/lib/backend-stack.ts +128 -0
- package/examples/monorepo/packages/backend-infra/package.json +30 -0
- package/examples/monorepo/packages/backend-infra/tsconfig.json +11 -0
- package/examples/monorepo/packages/data-infra/.cdk-cost-analyzer.yml +38 -0
- package/examples/monorepo/packages/data-infra/bin/app.ts +16 -0
- package/examples/monorepo/packages/data-infra/cdk.json +7 -0
- package/examples/monorepo/packages/data-infra/lib/data-stack.ts +121 -0
- package/examples/monorepo/packages/data-infra/package.json +30 -0
- package/examples/monorepo/packages/data-infra/tsconfig.json +11 -0
- package/examples/monorepo/packages/frontend-infra/.cdk-cost-analyzer.yml +31 -0
- package/examples/monorepo/packages/frontend-infra/bin/app.ts +16 -0
- package/examples/monorepo/packages/frontend-infra/cdk.json +7 -0
- package/examples/monorepo/packages/frontend-infra/lib/frontend-stack.ts +60 -0
- package/examples/monorepo/packages/frontend-infra/package.json +30 -0
- package/examples/monorepo/packages/frontend-infra/tsconfig.json +11 -0
- package/examples/monorepo/tsconfig.json +35 -0
- package/examples/multi-stack/.cdk-cost-analyzer.yml +72 -0
- package/examples/multi-stack/.gitlab-ci.yml +184 -0
- package/examples/multi-stack/README.md +279 -0
- package/examples/multi-stack/bin/app.ts +36 -0
- package/examples/multi-stack/cdk.json +72 -0
- package/examples/multi-stack/lib/compute-stack.ts +128 -0
- package/examples/multi-stack/lib/networking-stack.ts +69 -0
- package/examples/multi-stack/lib/storage-stack.ts +141 -0
- package/examples/multi-stack/package-lock.json +4437 -0
- package/examples/multi-stack/package.json +42 -0
- package/examples/multi-stack/tsconfig.json +34 -0
- package/examples/simple/base.json +8 -0
- package/examples/simple/target.json +14 -0
- package/examples/single-stack/.NVP +0 -0
- package/examples/single-stack/.cdk-cost-analyzer.yml +52 -0
- package/examples/single-stack/.gitlab-ci.yml +126 -0
- package/examples/single-stack/README.md +184 -0
- package/examples/single-stack/UeK +0 -0
- package/examples/single-stack/bin/app.ts +16 -0
- package/examples/single-stack/cdk.json +72 -0
- package/examples/single-stack/lib/infrastructure-stack.ts +119 -0
- package/examples/single-stack/package-lock.json +4443 -0
- package/examples/single-stack/package.json +38 -0
- package/examples/single-stack/tsconfig.json +34 -0
- package/package.json +139 -0
- package/test-cdk-project/README-COMPUTE.md +141 -0
- package/test-cdk-project/README.md +95 -0
- package/test-cdk-project/app-with-compute.js +102 -0
- package/test-cdk-project/app.js +81 -0
- package/test-cdk-project/cdk-compute.json +3 -0
- package/test-cdk-project/cdk.context.json +7 -0
- package/test-cdk-project/cdk.json +3 -0
- package/test-cdk-project/cdk.out/TestStack.assets.json +21 -0
- package/test-cdk-project/cdk.out/TestStack.template.json +115 -0
- package/test-cdk-project/cdk.out/cdk.out +1 -0
- package/test-cdk-project/cdk.out/manifest.json +503 -0
- package/test-cdk-project/cdk.out/tree.json +1 -0
- package/test-cdk-project/cdk.out.base/TestStack.assets.json +21 -0
- package/test-cdk-project/cdk.out.base/TestStack.template.json +115 -0
- package/test-cdk-project/cdk.out.base/cdk.out +1 -0
- package/test-cdk-project/cdk.out.base/manifest.json +503 -0
- package/test-cdk-project/cdk.out.base/tree.json +1 -0
- package/test-cdk-project/cdk.out.target/TestStack.assets.json +21 -0
- package/test-cdk-project/cdk.out.target/TestStack.template.json +183 -0
- package/test-cdk-project/cdk.out.target/cdk.out +1 -0
- package/test-cdk-project/cdk.out.target/manifest.json +521 -0
- package/test-cdk-project/cdk.out.target/tree.json +1 -0
- package/test-cdk-project/package-lock.json +422 -0
- package/test-cdk-project/package.json +17 -0
- package/tools/workflows/README.md +102 -0
- package/tools/workflows/validate-workflows.js +109 -0
- 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,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,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=
|