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,3 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvYXBpL3R5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBSZXNvdXJjZUNvc3QsIE1vZGlmaWVkUmVzb3VyY2VDb3N0IH0gZnJvbSAnLi4vcHJpY2luZy90eXBlcyc7XG5cbmV4cG9ydCBpbnRlcmZhY2UgQW5hbHl6ZU9wdGlvbnMge1xuICBiYXNlVGVtcGxhdGU6IHN0cmluZztcbiAgdGFyZ2V0VGVtcGxhdGU6IHN0cmluZztcbiAgcmVnaW9uPzogc3RyaW5nO1xuICBmb3JtYXQ/OiAndGV4dCcgfCAnanNvbicgfCAnbWFya2Rvd24nO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIENvc3RBbmFseXNpc1Jlc3VsdCB7XG4gIHRvdGFsRGVsdGE6IG51bWJlcjtcbiAgY3VycmVuY3k6IHN0cmluZztcbiAgYWRkZWRSZXNvdXJjZXM6IFJlc291cmNlQ29zdFtdO1xuICByZW1vdmVkUmVzb3VyY2VzOiBSZXNvdXJjZUNvc3RbXTtcbiAgbW9kaWZpZWRSZXNvdXJjZXM6IE1vZGlmaWVkUmVzb3VyY2VDb3N0W107XG4gIHN1bW1hcnk6IHN0cmluZztcbn1cbiJdfQ==
|
|
@@ -0,0 +1,262 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
"use strict";
|
|
3
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
4
|
+
if (k2 === undefined) k2 = k;
|
|
5
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
6
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
7
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
8
|
+
}
|
|
9
|
+
Object.defineProperty(o, k2, desc);
|
|
10
|
+
}) : (function(o, m, k, k2) {
|
|
11
|
+
if (k2 === undefined) k2 = k;
|
|
12
|
+
o[k2] = m[k];
|
|
13
|
+
}));
|
|
14
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
15
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
16
|
+
}) : function(o, v) {
|
|
17
|
+
o["default"] = v;
|
|
18
|
+
});
|
|
19
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
20
|
+
var ownKeys = function(o) {
|
|
21
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
22
|
+
var ar = [];
|
|
23
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
24
|
+
return ar;
|
|
25
|
+
};
|
|
26
|
+
return ownKeys(o);
|
|
27
|
+
};
|
|
28
|
+
return function (mod) {
|
|
29
|
+
if (mod && mod.__esModule) return mod;
|
|
30
|
+
var result = {};
|
|
31
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
32
|
+
__setModuleDefault(result, mod);
|
|
33
|
+
return result;
|
|
34
|
+
};
|
|
35
|
+
})();
|
|
36
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
37
|
+
const fs = __importStar(require("fs"));
|
|
38
|
+
const commander_1 = require("commander");
|
|
39
|
+
const api_1 = require("../api");
|
|
40
|
+
const GitLabIntegration_1 = require("../integrations/GitLabIntegration");
|
|
41
|
+
const PipelineOrchestrator_1 = require("../pipeline/PipelineOrchestrator");
|
|
42
|
+
const program = new commander_1.Command();
|
|
43
|
+
program
|
|
44
|
+
.name('cdk-cost-analyzer')
|
|
45
|
+
.description('Analyze AWS CDK infrastructure changes and provide cost impact summaries')
|
|
46
|
+
.version('1.0.0');
|
|
47
|
+
// Original command for direct template comparison
|
|
48
|
+
program
|
|
49
|
+
.command('compare')
|
|
50
|
+
.description('Compare two CloudFormation templates')
|
|
51
|
+
.argument('<base>', 'Path to base CloudFormation template')
|
|
52
|
+
.argument('<target>', 'Path to target CloudFormation template')
|
|
53
|
+
.option('--region <region>', 'AWS region', 'eu-central-1')
|
|
54
|
+
.option('--format <format>', 'Output format: text|json|markdown', 'text')
|
|
55
|
+
.option('--config <path>', 'Path to configuration file')
|
|
56
|
+
.action(async (basePath, targetPath, options) => {
|
|
57
|
+
try {
|
|
58
|
+
if (!fs.existsSync(basePath)) {
|
|
59
|
+
console.error(`Error: Base template file not found: ${basePath}`);
|
|
60
|
+
process.exit(1);
|
|
61
|
+
}
|
|
62
|
+
if (!fs.existsSync(targetPath)) {
|
|
63
|
+
console.error(`Error: Target template file not found: ${targetPath}`);
|
|
64
|
+
process.exit(1);
|
|
65
|
+
}
|
|
66
|
+
const baseTemplate = fs.readFileSync(basePath, 'utf-8');
|
|
67
|
+
const targetTemplate = fs.readFileSync(targetPath, 'utf-8');
|
|
68
|
+
const result = await (0, api_1.analyzeCosts)({
|
|
69
|
+
baseTemplate,
|
|
70
|
+
targetTemplate,
|
|
71
|
+
region: options.region,
|
|
72
|
+
format: options.format,
|
|
73
|
+
});
|
|
74
|
+
if (options.format === 'json') {
|
|
75
|
+
console.log(JSON.stringify(result, null, 2));
|
|
76
|
+
}
|
|
77
|
+
else {
|
|
78
|
+
console.log(result.summary);
|
|
79
|
+
}
|
|
80
|
+
process.exit(0);
|
|
81
|
+
}
|
|
82
|
+
catch (error) {
|
|
83
|
+
if (error instanceof Error) {
|
|
84
|
+
console.error(`Error: ${error.message}`);
|
|
85
|
+
}
|
|
86
|
+
else {
|
|
87
|
+
console.error(`Error: ${String(error)}`);
|
|
88
|
+
}
|
|
89
|
+
process.exit(1);
|
|
90
|
+
}
|
|
91
|
+
});
|
|
92
|
+
// New pipeline command for CI/CD integration with synthesis
|
|
93
|
+
program
|
|
94
|
+
.command('pipeline')
|
|
95
|
+
.description('Run cost analysis in CI/CD pipeline with automatic synthesis')
|
|
96
|
+
.option('--base <path>', 'Path to base template (if not using synthesis)')
|
|
97
|
+
.option('--target <path>', 'Path to target template (if not using synthesis)')
|
|
98
|
+
.option('--synth', 'Enable automatic CDK synthesis')
|
|
99
|
+
.option('--cdk-app-path <path>', 'Path to CDK application directory')
|
|
100
|
+
.option('--region <region>', 'AWS region', 'eu-central-1')
|
|
101
|
+
.option('--config <path>', 'Path to configuration file')
|
|
102
|
+
.option('--environment <env>', 'Environment name for threshold selection')
|
|
103
|
+
.option('--format <format>', 'Output format: text|json|markdown', 'text')
|
|
104
|
+
.option('--post-to-gitlab', 'Post results to GitLab merge request')
|
|
105
|
+
.action(async (options) => {
|
|
106
|
+
try {
|
|
107
|
+
// Check AWS credentials
|
|
108
|
+
if (!process.env.AWS_ACCESS_KEY_ID && !process.env.AWS_PROFILE) {
|
|
109
|
+
console.error('Error: AWS credentials not configured');
|
|
110
|
+
console.error('');
|
|
111
|
+
console.error('Please set AWS credentials using one of:');
|
|
112
|
+
console.error(' - AWS_ACCESS_KEY_ID and AWS_SECRET_ACCESS_KEY environment variables');
|
|
113
|
+
console.error(' - AWS_PROFILE environment variable');
|
|
114
|
+
console.error(' - AWS credentials file (~/.aws/credentials)');
|
|
115
|
+
console.error('');
|
|
116
|
+
console.error('For GitLab CI, configure AWS credentials in CI/CD variables:');
|
|
117
|
+
console.error(' Settings > CI/CD > Variables');
|
|
118
|
+
console.error(' Add: AWS_ACCESS_KEY_ID, AWS_SECRET_ACCESS_KEY, AWS_REGION');
|
|
119
|
+
process.exit(1);
|
|
120
|
+
}
|
|
121
|
+
const orchestrator = new PipelineOrchestrator_1.PipelineOrchestrator();
|
|
122
|
+
const result = await orchestrator.runPipelineAnalysis({
|
|
123
|
+
baseTemplate: options.base,
|
|
124
|
+
targetTemplate: options.target,
|
|
125
|
+
synthesize: options.synth,
|
|
126
|
+
cdkAppPath: options.cdkAppPath,
|
|
127
|
+
region: options.region,
|
|
128
|
+
configPath: options.config,
|
|
129
|
+
environment: options.environment,
|
|
130
|
+
});
|
|
131
|
+
// Display results
|
|
132
|
+
if (options.format === 'json') {
|
|
133
|
+
console.log(JSON.stringify(result, null, 2));
|
|
134
|
+
}
|
|
135
|
+
else {
|
|
136
|
+
console.log(result.costAnalysis.summary);
|
|
137
|
+
console.log('');
|
|
138
|
+
// Display threshold status
|
|
139
|
+
if (result.thresholdStatus.level !== 'none') {
|
|
140
|
+
console.log('='.repeat(80));
|
|
141
|
+
console.log('THRESHOLD STATUS');
|
|
142
|
+
console.log('='.repeat(80));
|
|
143
|
+
console.log(result.thresholdStatus.message);
|
|
144
|
+
console.log('');
|
|
145
|
+
if (result.thresholdStatus.recommendations.length > 0) {
|
|
146
|
+
console.log('Recommendations:');
|
|
147
|
+
result.thresholdStatus.recommendations.forEach((rec, i) => {
|
|
148
|
+
console.log(`${i + 1}. ${rec}`);
|
|
149
|
+
});
|
|
150
|
+
console.log('');
|
|
151
|
+
}
|
|
152
|
+
}
|
|
153
|
+
// Display config summary
|
|
154
|
+
if (result.configUsed.configPath || result.configUsed.thresholds) {
|
|
155
|
+
console.log('='.repeat(80));
|
|
156
|
+
console.log('CONFIGURATION');
|
|
157
|
+
console.log('='.repeat(80));
|
|
158
|
+
if (result.configUsed.configPath) {
|
|
159
|
+
console.log(`Config file: ${result.configUsed.configPath}`);
|
|
160
|
+
}
|
|
161
|
+
if (result.configUsed.thresholds) {
|
|
162
|
+
console.log(`Thresholds: warning=$${result.configUsed.thresholds.warning || 'none'}, error=$${result.configUsed.thresholds.error || 'none'}`);
|
|
163
|
+
if (result.configUsed.thresholds.environment) {
|
|
164
|
+
console.log(`Environment: ${result.configUsed.thresholds.environment}`);
|
|
165
|
+
}
|
|
166
|
+
}
|
|
167
|
+
if (result.configUsed.excludedResourceTypes?.length) {
|
|
168
|
+
console.log(`Excluded resources: ${result.configUsed.excludedResourceTypes.join(', ')}`);
|
|
169
|
+
}
|
|
170
|
+
console.log('');
|
|
171
|
+
}
|
|
172
|
+
}
|
|
173
|
+
// Post to GitLab if requested
|
|
174
|
+
if (options.postToGitlab) {
|
|
175
|
+
const gitlabToken = process.env.GITLAB_TOKEN || process.env.CI_JOB_TOKEN;
|
|
176
|
+
const projectId = process.env.CI_PROJECT_ID;
|
|
177
|
+
const mergeRequestIid = process.env.CI_MERGE_REQUEST_IID;
|
|
178
|
+
const gitlabUrl = process.env.CI_SERVER_URL;
|
|
179
|
+
if (!gitlabToken || !projectId || !mergeRequestIid || !gitlabUrl) {
|
|
180
|
+
console.error('Warning: GitLab environment variables not found. Skipping GitLab post.');
|
|
181
|
+
console.error('Required: CI_JOB_TOKEN, CI_PROJECT_ID, CI_MERGE_REQUEST_IID, CI_SERVER_URL');
|
|
182
|
+
}
|
|
183
|
+
else {
|
|
184
|
+
try {
|
|
185
|
+
const gitlab = new GitLabIntegration_1.GitLabIntegration({
|
|
186
|
+
token: gitlabToken,
|
|
187
|
+
apiUrl: gitlabUrl,
|
|
188
|
+
});
|
|
189
|
+
await gitlab.postMergeRequestComment(projectId, mergeRequestIid, result.costAnalysis.summary);
|
|
190
|
+
console.log('Results posted to GitLab merge request');
|
|
191
|
+
}
|
|
192
|
+
catch (error) {
|
|
193
|
+
console.error(`Warning: Failed to post to GitLab: ${error instanceof Error ? error.message : String(error)}`);
|
|
194
|
+
}
|
|
195
|
+
}
|
|
196
|
+
}
|
|
197
|
+
// Exit with appropriate code based on threshold
|
|
198
|
+
if (result.thresholdStatus.level === 'error' && !result.thresholdStatus.passed) {
|
|
199
|
+
console.error('Pipeline failed: Cost threshold exceeded');
|
|
200
|
+
process.exit(2);
|
|
201
|
+
}
|
|
202
|
+
process.exit(0);
|
|
203
|
+
}
|
|
204
|
+
catch (error) {
|
|
205
|
+
if (error instanceof Error) {
|
|
206
|
+
console.error(`Error: ${error.message}`);
|
|
207
|
+
}
|
|
208
|
+
else {
|
|
209
|
+
console.error(`Error: ${String(error)}`);
|
|
210
|
+
}
|
|
211
|
+
process.exit(1);
|
|
212
|
+
}
|
|
213
|
+
});
|
|
214
|
+
// Default to compare command for backward compatibility
|
|
215
|
+
program
|
|
216
|
+
.argument('[base]', 'Path to base CloudFormation template')
|
|
217
|
+
.argument('[target]', 'Path to target CloudFormation template')
|
|
218
|
+
.option('--region <region>', 'AWS region', 'eu-central-1')
|
|
219
|
+
.option('--format <format>', 'Output format: text|json|markdown', 'text')
|
|
220
|
+
.option('--config <path>', 'Path to configuration file')
|
|
221
|
+
.action(async (basePath, targetPath, options) => {
|
|
222
|
+
if (!basePath || !targetPath) {
|
|
223
|
+
program.help();
|
|
224
|
+
return;
|
|
225
|
+
}
|
|
226
|
+
try {
|
|
227
|
+
if (!fs.existsSync(basePath)) {
|
|
228
|
+
console.error(`Error: Base template file not found: ${basePath}`);
|
|
229
|
+
process.exit(1);
|
|
230
|
+
}
|
|
231
|
+
if (!fs.existsSync(targetPath)) {
|
|
232
|
+
console.error(`Error: Target template file not found: ${targetPath}`);
|
|
233
|
+
process.exit(1);
|
|
234
|
+
}
|
|
235
|
+
const baseTemplate = fs.readFileSync(basePath, 'utf-8');
|
|
236
|
+
const targetTemplate = fs.readFileSync(targetPath, 'utf-8');
|
|
237
|
+
const result = await (0, api_1.analyzeCosts)({
|
|
238
|
+
baseTemplate,
|
|
239
|
+
targetTemplate,
|
|
240
|
+
region: options?.region || 'eu-central-1',
|
|
241
|
+
format: options?.format || 'text',
|
|
242
|
+
});
|
|
243
|
+
if (options?.format === 'json') {
|
|
244
|
+
console.log(JSON.stringify(result, null, 2));
|
|
245
|
+
}
|
|
246
|
+
else {
|
|
247
|
+
console.log(result.summary);
|
|
248
|
+
}
|
|
249
|
+
process.exit(0);
|
|
250
|
+
}
|
|
251
|
+
catch (error) {
|
|
252
|
+
if (error instanceof Error) {
|
|
253
|
+
console.error(`Error: ${error.message}`);
|
|
254
|
+
}
|
|
255
|
+
else {
|
|
256
|
+
console.error(`Error: ${String(error)}`);
|
|
257
|
+
}
|
|
258
|
+
process.exit(1);
|
|
259
|
+
}
|
|
260
|
+
});
|
|
261
|
+
program.parse();
|
|
262
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/cli/index.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAEA,uCAAyB;AACzB,yCAAoC;AACpC,gCAAsC;AACtC,yEAAsE;AACtE,2EAAwE;AAExE,MAAM,OAAO,GAAG,IAAI,mBAAO,EAAE,CAAC;AAE9B,OAAO;KACJ,IAAI,CAAC,mBAAmB,CAAC;KACzB,WAAW,CAAC,0EAA0E,CAAC;KACvF,OAAO,CAAC,OAAO,CAAC,CAAC;AAEpB,kDAAkD;AAClD,OAAO;KACJ,OAAO,CAAC,SAAS,CAAC;KAClB,WAAW,CAAC,sCAAsC,CAAC;KACnD,QAAQ,CAAC,QAAQ,EAAE,sCAAsC,CAAC;KAC1D,QAAQ,CAAC,UAAU,EAAE,wCAAwC,CAAC;KAC9D,MAAM,CAAC,mBAAmB,EAAE,YAAY,EAAE,cAAc,CAAC;KACzD,MAAM,CAAC,mBAAmB,EAAE,mCAAmC,EAAE,MAAM,CAAC;KACxE,MAAM,CAAC,iBAAiB,EAAE,4BAA4B,CAAC;KACvD,MAAM,CAAC,KAAK,EAAE,QAAgB,EAAE,UAAkB,EAAE,OAA4D,EAAE,EAAE;IACnH,IAAI,CAAC;QACH,IAAI,CAAC,EAAE,CAAC,UAAU,CAAC,QAAQ,CAAC,EAAE,CAAC;YAC7B,OAAO,CAAC,KAAK,CAAC,wCAAwC,QAAQ,EAAE,CAAC,CAAC;YAClE,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QAClB,CAAC;QAED,IAAI,CAAC,EAAE,CAAC,UAAU,CAAC,UAAU,CAAC,EAAE,CAAC;YAC/B,OAAO,CAAC,KAAK,CAAC,0CAA0C,UAAU,EAAE,CAAC,CAAC;YACtE,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QAClB,CAAC;QAED,MAAM,YAAY,GAAG,EAAE,CAAC,YAAY,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC;QACxD,MAAM,cAAc,GAAG,EAAE,CAAC,YAAY,CAAC,UAAU,EAAE,OAAO,CAAC,CAAC;QAE5D,MAAM,MAAM,GAAG,MAAM,IAAA,kBAAY,EAAC;YAChC,YAAY;YACZ,cAAc;YACd,MAAM,EAAE,OAAO,CAAC,MAAM;YACtB,MAAM,EAAE,OAAO,CAAC,MAAsC;SACvD,CAAC,CAAC;QAEH,IAAI,OAAO,CAAC,MAAM,KAAK,MAAM,EAAE,CAAC;YAC9B,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;QAC/C,CAAC;aAAM,CAAC;YACN,OAAO,CAAC,GAAG,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;QAC9B,CAAC;QAED,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,IAAI,KAAK,YAAY,KAAK,EAAE,CAAC;YAC3B,OAAO,CAAC,KAAK,CAAC,UAAU,KAAK,CAAC,OAAO,EAAE,CAAC,CAAC;QAC3C,CAAC;aAAM,CAAC;YACN,OAAO,CAAC,KAAK,CAAC,UAAU,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;QAC3C,CAAC;QACD,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;AACH,CAAC,CAAC,CAAC;AAEL,4DAA4D;AAC5D,OAAO;KACJ,OAAO,CAAC,UAAU,CAAC;KACnB,WAAW,CAAC,8DAA8D,CAAC;KAC3E,MAAM,CAAC,eAAe,EAAE,gDAAgD,CAAC;KACzE,MAAM,CAAC,iBAAiB,EAAE,kDAAkD,CAAC;KAC7E,MAAM,CAAC,SAAS,EAAE,gCAAgC,CAAC;KACnD,MAAM,CAAC,uBAAuB,EAAE,mCAAmC,CAAC;KACpE,MAAM,CAAC,mBAAmB,EAAE,YAAY,EAAE,cAAc,CAAC;KACzD,MAAM,CAAC,iBAAiB,EAAE,4BAA4B,CAAC;KACvD,MAAM,CAAC,qBAAqB,EAAE,0CAA0C,CAAC;KACzE,MAAM,CAAC,mBAAmB,EAAE,mCAAmC,EAAE,MAAM,CAAC;KACxE,MAAM,CAAC,kBAAkB,EAAE,sCAAsC,CAAC;KAClE,MAAM,CAAC,KAAK,EAAE,OAUd,EAAE,EAAE;IACH,IAAI,CAAC;QACH,wBAAwB;QACxB,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,iBAAiB,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,WAAW,EAAE,CAAC;YAC/D,OAAO,CAAC,KAAK,CAAC,uCAAuC,CAAC,CAAC;YACvD,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;YAClB,OAAO,CAAC,KAAK,CAAC,0CAA0C,CAAC,CAAC;YAC1D,OAAO,CAAC,KAAK,CAAC,uEAAuE,CAAC,CAAC;YACvF,OAAO,CAAC,KAAK,CAAC,sCAAsC,CAAC,CAAC;YACtD,OAAO,CAAC,KAAK,CAAC,+CAA+C,CAAC,CAAC;YAC/D,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;YAClB,OAAO,CAAC,KAAK,CAAC,8DAA8D,CAAC,CAAC;YAC9E,OAAO,CAAC,KAAK,CAAC,gCAAgC,CAAC,CAAC;YAChD,OAAO,CAAC,KAAK,CAAC,6DAA6D,CAAC,CAAC;YAC7E,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QAClB,CAAC;QAED,MAAM,YAAY,GAAG,IAAI,2CAAoB,EAAE,CAAC;QAEhD,MAAM,MAAM,GAAG,MAAM,YAAY,CAAC,mBAAmB,CAAC;YACpD,YAAY,EAAE,OAAO,CAAC,IAAI;YAC1B,cAAc,EAAE,OAAO,CAAC,MAAM;YAC9B,UAAU,EAAE,OAAO,CAAC,KAAK;YACzB,UAAU,EAAE,OAAO,CAAC,UAAU;YAC9B,MAAM,EAAE,OAAO,CAAC,MAAM;YACtB,UAAU,EAAE,OAAO,CAAC,MAAM;YAC1B,WAAW,EAAE,OAAO,CAAC,WAAW;SACjC,CAAC,CAAC;QAEH,kBAAkB;QAClB,IAAI,OAAO,CAAC,MAAM,KAAK,MAAM,EAAE,CAAC;YAC9B,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;QAC/C,CAAC;aAAM,CAAC;YACN,OAAO,CAAC,GAAG,CAAC,MAAM,CAAC,YAAY,CAAC,OAAO,CAAC,CAAC;YACzC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;YAEhB,2BAA2B;YAC3B,IAAI,MAAM,CAAC,eAAe,CAAC,KAAK,KAAK,MAAM,EAAE,CAAC;gBAC5C,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC;gBAC5B,OAAO,CAAC,GAAG,CAAC,kBAAkB,CAAC,CAAC;gBAChC,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC;gBAC5B,OAAO,CAAC,GAAG,CAAC,MAAM,CAAC,eAAe,CAAC,OAAO,CAAC,CAAC;gBAC5C,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;gBAEhB,IAAI,MAAM,CAAC,eAAe,CAAC,eAAe,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;oBACtD,OAAO,CAAC,GAAG,CAAC,kBAAkB,CAAC,CAAC;oBAChC,MAAM,CAAC,eAAe,CAAC,eAAe,CAAC,OAAO,CAAC,CAAC,GAAG,EAAE,CAAC,EAAE,EAAE;wBACxD,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,GAAG,CAAC,KAAK,GAAG,EAAE,CAAC,CAAC;oBAClC,CAAC,CAAC,CAAC;oBACH,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;gBAClB,CAAC;YACH,CAAC;YAED,yBAAyB;YACzB,IAAI,MAAM,CAAC,UAAU,CAAC,UAAU,IAAI,MAAM,CAAC,UAAU,CAAC,UAAU,EAAE,CAAC;gBACjE,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC;gBAC5B,OAAO,CAAC,GAAG,CAAC,eAAe,CAAC,CAAC;gBAC7B,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC;gBAC5B,IAAI,MAAM,CAAC,UAAU,CAAC,UAAU,EAAE,CAAC;oBACjC,OAAO,CAAC,GAAG,CAAC,gBAAgB,MAAM,CAAC,UAAU,CAAC,UAAU,EAAE,CAAC,CAAC;gBAC9D,CAAC;gBACD,IAAI,MAAM,CAAC,UAAU,CAAC,UAAU,EAAE,CAAC;oBACjC,OAAO,CAAC,GAAG,CAAC,wBAAwB,MAAM,CAAC,UAAU,CAAC,UAAU,CAAC,OAAO,IAAI,MAAM,YAAY,MAAM,CAAC,UAAU,CAAC,UAAU,CAAC,KAAK,IAAI,MAAM,EAAE,CAAC,CAAC;oBAC9I,IAAI,MAAM,CAAC,UAAU,CAAC,UAAU,CAAC,WAAW,EAAE,CAAC;wBAC7C,OAAO,CAAC,GAAG,CAAC,gBAAgB,MAAM,CAAC,UAAU,CAAC,UAAU,CAAC,WAAW,EAAE,CAAC,CAAC;oBAC1E,CAAC;gBACH,CAAC;gBACD,IAAI,MAAM,CAAC,UAAU,CAAC,qBAAqB,EAAE,MAAM,EAAE,CAAC;oBACpD,OAAO,CAAC,GAAG,CAAC,uBAAuB,MAAM,CAAC,UAAU,CAAC,qBAAqB,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;gBAC3F,CAAC;gBACD,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;YAClB,CAAC;QACH,CAAC;QAED,8BAA8B;QAC9B,IAAI,OAAO,CAAC,YAAY,EAAE,CAAC;YACzB,MAAM,WAAW,GAAG,OAAO,CAAC,GAAG,CAAC,YAAY,IAAI,OAAO,CAAC,GAAG,CAAC,YAAY,CAAC;YACzE,MAAM,SAAS,GAAG,OAAO,CAAC,GAAG,CAAC,aAAa,CAAC;YAC5C,MAAM,eAAe,GAAG,OAAO,CAAC,GAAG,CAAC,oBAAoB,CAAC;YACzD,MAAM,SAAS,GAAG,OAAO,CAAC,GAAG,CAAC,aAAa,CAAC;YAE5C,IAAI,CAAC,WAAW,IAAI,CAAC,SAAS,IAAI,CAAC,eAAe,IAAI,CAAC,SAAS,EAAE,CAAC;gBACjE,OAAO,CAAC,KAAK,CAAC,wEAAwE,CAAC,CAAC;gBACxF,OAAO,CAAC,KAAK,CAAC,4EAA4E,CAAC,CAAC;YAC9F,CAAC;iBAAM,CAAC;gBACN,IAAI,CAAC;oBACH,MAAM,MAAM,GAAG,IAAI,qCAAiB,CAAC;wBACnC,KAAK,EAAE,WAAW;wBAClB,MAAM,EAAE,SAAS;qBAClB,CAAC,CAAC;oBACH,MAAM,MAAM,CAAC,uBAAuB,CAClC,SAAS,EACT,eAAe,EACf,MAAM,CAAC,YAAY,CAAC,OAAO,CAC5B,CAAC;oBACF,OAAO,CAAC,GAAG,CAAC,wCAAwC,CAAC,CAAC;gBACxD,CAAC;gBAAC,OAAO,KAAK,EAAE,CAAC;oBACf,OAAO,CAAC,KAAK,CAAC,sCAAsC,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;gBAChH,CAAC;YACH,CAAC;QACH,CAAC;QAED,gDAAgD;QAChD,IAAI,MAAM,CAAC,eAAe,CAAC,KAAK,KAAK,OAAO,IAAI,CAAC,MAAM,CAAC,eAAe,CAAC,MAAM,EAAE,CAAC;YAC/E,OAAO,CAAC,KAAK,CAAC,0CAA0C,CAAC,CAAC;YAC1D,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QAClB,CAAC;QAED,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,IAAI,KAAK,YAAY,KAAK,EAAE,CAAC;YAC3B,OAAO,CAAC,KAAK,CAAC,UAAU,KAAK,CAAC,OAAO,EAAE,CAAC,CAAC;QAC3C,CAAC;aAAM,CAAC;YACN,OAAO,CAAC,KAAK,CAAC,UAAU,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;QAC3C,CAAC;QACD,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;AACH,CAAC,CAAC,CAAC;AAEL,wDAAwD;AACxD,OAAO;KACJ,QAAQ,CAAC,QAAQ,EAAE,sCAAsC,CAAC;KAC1D,QAAQ,CAAC,UAAU,EAAE,wCAAwC,CAAC;KAC9D,MAAM,CAAC,mBAAmB,EAAE,YAAY,EAAE,cAAc,CAAC;KACzD,MAAM,CAAC,mBAAmB,EAAE,mCAAmC,EAAE,MAAM,CAAC;KACxE,MAAM,CAAC,iBAAiB,EAAE,4BAA4B,CAAC;KACvD,MAAM,CAAC,KAAK,EAAE,QAAiB,EAAE,UAAmB,EAAE,OAA6D,EAAE,EAAE;IACtH,IAAI,CAAC,QAAQ,IAAI,CAAC,UAAU,EAAE,CAAC;QAC7B,OAAO,CAAC,IAAI,EAAE,CAAC;QACf,OAAO;IACT,CAAC;IAED,IAAI,CAAC;QACH,IAAI,CAAC,EAAE,CAAC,UAAU,CAAC,QAAQ,CAAC,EAAE,CAAC;YAC7B,OAAO,CAAC,KAAK,CAAC,wCAAwC,QAAQ,EAAE,CAAC,CAAC;YAClE,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QAClB,CAAC;QAED,IAAI,CAAC,EAAE,CAAC,UAAU,CAAC,UAAU,CAAC,EAAE,CAAC;YAC/B,OAAO,CAAC,KAAK,CAAC,0CAA0C,UAAU,EAAE,CAAC,CAAC;YACtE,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QAClB,CAAC;QAED,MAAM,YAAY,GAAG,EAAE,CAAC,YAAY,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC;QACxD,MAAM,cAAc,GAAG,EAAE,CAAC,YAAY,CAAC,UAAU,EAAE,OAAO,CAAC,CAAC;QAE5D,MAAM,MAAM,GAAG,MAAM,IAAA,kBAAY,EAAC;YAChC,YAAY;YACZ,cAAc;YACd,MAAM,EAAE,OAAO,EAAE,MAAM,IAAI,cAAc;YACzC,MAAM,EAAG,OAAO,EAAE,MAAuC,IAAI,MAAM;SACpE,CAAC,CAAC;QAEH,IAAI,OAAO,EAAE,MAAM,KAAK,MAAM,EAAE,CAAC;YAC/B,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;QAC/C,CAAC;aAAM,CAAC;YACN,OAAO,CAAC,GAAG,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;QAC9B,CAAC;QAED,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,IAAI,KAAK,YAAY,KAAK,EAAE,CAAC;YAC3B,OAAO,CAAC,KAAK,CAAC,UAAU,KAAK,CAAC,OAAO,EAAE,CAAC,CAAC;QAC3C,CAAC;aAAM,CAAC;YACN,OAAO,CAAC,KAAK,CAAC,UAAU,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;QAC3C,CAAC;QACD,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;AACH,CAAC,CAAC,CAAC;AAEL,OAAO,CAAC,KAAK,EAAE,CAAC","sourcesContent":["#!/usr/bin/env node\n\nimport * as fs from 'fs';\nimport { Command } from 'commander';\nimport { analyzeCosts } from '../api';\nimport { GitLabIntegration } from '../integrations/GitLabIntegration';\nimport { PipelineOrchestrator } from '../pipeline/PipelineOrchestrator';\n\nconst program = new Command();\n\nprogram\n  .name('cdk-cost-analyzer')\n  .description('Analyze AWS CDK infrastructure changes and provide cost impact summaries')\n  .version('1.0.0');\n\n// Original command for direct template comparison\nprogram\n  .command('compare')\n  .description('Compare two CloudFormation templates')\n  .argument('<base>', 'Path to base CloudFormation template')\n  .argument('<target>', 'Path to target CloudFormation template')\n  .option('--region <region>', 'AWS region', 'eu-central-1')\n  .option('--format <format>', 'Output format: text|json|markdown', 'text')\n  .option('--config <path>', 'Path to configuration file')\n  .action(async (basePath: string, targetPath: string, options: { region: string; format: string; config?: string }) => {\n    try {\n      if (!fs.existsSync(basePath)) {\n        console.error(`Error: Base template file not found: ${basePath}`);\n        process.exit(1);\n      }\n\n      if (!fs.existsSync(targetPath)) {\n        console.error(`Error: Target template file not found: ${targetPath}`);\n        process.exit(1);\n      }\n\n      const baseTemplate = fs.readFileSync(basePath, 'utf-8');\n      const targetTemplate = fs.readFileSync(targetPath, 'utf-8');\n\n      const result = await analyzeCosts({\n        baseTemplate,\n        targetTemplate,\n        region: options.region,\n        format: options.format as 'text' | 'json' | 'markdown',\n      });\n\n      if (options.format === 'json') {\n        console.log(JSON.stringify(result, null, 2));\n      } else {\n        console.log(result.summary);\n      }\n\n      process.exit(0);\n    } catch (error) {\n      if (error instanceof Error) {\n        console.error(`Error: ${error.message}`);\n      } else {\n        console.error(`Error: ${String(error)}`);\n      }\n      process.exit(1);\n    }\n  });\n\n// New pipeline command for CI/CD integration with synthesis\nprogram\n  .command('pipeline')\n  .description('Run cost analysis in CI/CD pipeline with automatic synthesis')\n  .option('--base <path>', 'Path to base template (if not using synthesis)')\n  .option('--target <path>', 'Path to target template (if not using synthesis)')\n  .option('--synth', 'Enable automatic CDK synthesis')\n  .option('--cdk-app-path <path>', 'Path to CDK application directory')\n  .option('--region <region>', 'AWS region', 'eu-central-1')\n  .option('--config <path>', 'Path to configuration file')\n  .option('--environment <env>', 'Environment name for threshold selection')\n  .option('--format <format>', 'Output format: text|json|markdown', 'text')\n  .option('--post-to-gitlab', 'Post results to GitLab merge request')\n  .action(async (options: {\n    base?: string;\n    target?: string;\n    synth?: boolean;\n    cdkAppPath?: string;\n    region: string;\n    config?: string;\n    environment?: string;\n    format: string;\n    postToGitlab?: boolean;\n  }) => {\n    try {\n      // Check AWS credentials\n      if (!process.env.AWS_ACCESS_KEY_ID && !process.env.AWS_PROFILE) {\n        console.error('Error: AWS credentials not configured');\n        console.error('');\n        console.error('Please set AWS credentials using one of:');\n        console.error('  - AWS_ACCESS_KEY_ID and AWS_SECRET_ACCESS_KEY environment variables');\n        console.error('  - AWS_PROFILE environment variable');\n        console.error('  - AWS credentials file (~/.aws/credentials)');\n        console.error('');\n        console.error('For GitLab CI, configure AWS credentials in CI/CD variables:');\n        console.error('  Settings > CI/CD > Variables');\n        console.error('  Add: AWS_ACCESS_KEY_ID, AWS_SECRET_ACCESS_KEY, AWS_REGION');\n        process.exit(1);\n      }\n\n      const orchestrator = new PipelineOrchestrator();\n\n      const result = await orchestrator.runPipelineAnalysis({\n        baseTemplate: options.base,\n        targetTemplate: options.target,\n        synthesize: options.synth,\n        cdkAppPath: options.cdkAppPath,\n        region: options.region,\n        configPath: options.config,\n        environment: options.environment,\n      });\n\n      // Display results\n      if (options.format === 'json') {\n        console.log(JSON.stringify(result, null, 2));\n      } else {\n        console.log(result.costAnalysis.summary);\n        console.log('');\n\n        // Display threshold status\n        if (result.thresholdStatus.level !== 'none') {\n          console.log('='.repeat(80));\n          console.log('THRESHOLD STATUS');\n          console.log('='.repeat(80));\n          console.log(result.thresholdStatus.message);\n          console.log('');\n\n          if (result.thresholdStatus.recommendations.length > 0) {\n            console.log('Recommendations:');\n            result.thresholdStatus.recommendations.forEach((rec, i) => {\n              console.log(`${i + 1}. ${rec}`);\n            });\n            console.log('');\n          }\n        }\n\n        // Display config summary\n        if (result.configUsed.configPath || result.configUsed.thresholds) {\n          console.log('='.repeat(80));\n          console.log('CONFIGURATION');\n          console.log('='.repeat(80));\n          if (result.configUsed.configPath) {\n            console.log(`Config file: ${result.configUsed.configPath}`);\n          }\n          if (result.configUsed.thresholds) {\n            console.log(`Thresholds: warning=$${result.configUsed.thresholds.warning || 'none'}, error=$${result.configUsed.thresholds.error || 'none'}`);\n            if (result.configUsed.thresholds.environment) {\n              console.log(`Environment: ${result.configUsed.thresholds.environment}`);\n            }\n          }\n          if (result.configUsed.excludedResourceTypes?.length) {\n            console.log(`Excluded resources: ${result.configUsed.excludedResourceTypes.join(', ')}`);\n          }\n          console.log('');\n        }\n      }\n\n      // Post to GitLab if requested\n      if (options.postToGitlab) {\n        const gitlabToken = process.env.GITLAB_TOKEN || process.env.CI_JOB_TOKEN;\n        const projectId = process.env.CI_PROJECT_ID;\n        const mergeRequestIid = process.env.CI_MERGE_REQUEST_IID;\n        const gitlabUrl = process.env.CI_SERVER_URL;\n\n        if (!gitlabToken || !projectId || !mergeRequestIid || !gitlabUrl) {\n          console.error('Warning: GitLab environment variables not found. Skipping GitLab post.');\n          console.error('Required: CI_JOB_TOKEN, CI_PROJECT_ID, CI_MERGE_REQUEST_IID, CI_SERVER_URL');\n        } else {\n          try {\n            const gitlab = new GitLabIntegration({\n              token: gitlabToken,\n              apiUrl: gitlabUrl,\n            });\n            await gitlab.postMergeRequestComment(\n              projectId,\n              mergeRequestIid,\n              result.costAnalysis.summary,\n            );\n            console.log('Results posted to GitLab merge request');\n          } catch (error) {\n            console.error(`Warning: Failed to post to GitLab: ${error instanceof Error ? error.message : String(error)}`);\n          }\n        }\n      }\n\n      // Exit with appropriate code based on threshold\n      if (result.thresholdStatus.level === 'error' && !result.thresholdStatus.passed) {\n        console.error('Pipeline failed: Cost threshold exceeded');\n        process.exit(2);\n      }\n\n      process.exit(0);\n    } catch (error) {\n      if (error instanceof Error) {\n        console.error(`Error: ${error.message}`);\n      } else {\n        console.error(`Error: ${String(error)}`);\n      }\n      process.exit(1);\n    }\n  });\n\n// Default to compare command for backward compatibility\nprogram\n  .argument('[base]', 'Path to base CloudFormation template')\n  .argument('[target]', 'Path to target CloudFormation template')\n  .option('--region <region>', 'AWS region', 'eu-central-1')\n  .option('--format <format>', 'Output format: text|json|markdown', 'text')\n  .option('--config <path>', 'Path to configuration file')\n  .action(async (basePath?: string, targetPath?: string, options?: { region: string; format: string; config?: string }) => {\n    if (!basePath || !targetPath) {\n      program.help();\n      return;\n    }\n\n    try {\n      if (!fs.existsSync(basePath)) {\n        console.error(`Error: Base template file not found: ${basePath}`);\n        process.exit(1);\n      }\n\n      if (!fs.existsSync(targetPath)) {\n        console.error(`Error: Target template file not found: ${targetPath}`);\n        process.exit(1);\n      }\n\n      const baseTemplate = fs.readFileSync(basePath, 'utf-8');\n      const targetTemplate = fs.readFileSync(targetPath, 'utf-8');\n\n      const result = await analyzeCosts({\n        baseTemplate,\n        targetTemplate,\n        region: options?.region || 'eu-central-1',\n        format: (options?.format as 'text' | 'json' | 'markdown') || 'text',\n      });\n\n      if (options?.format === 'json') {\n        console.log(JSON.stringify(result, null, 2));\n      } else {\n        console.log(result.summary);\n      }\n\n      process.exit(0);\n    } catch (error) {\n      if (error instanceof Error) {\n        console.error(`Error: ${error.message}`);\n      } else {\n        console.error(`Error: ${String(error)}`);\n      }\n      process.exit(1);\n    }\n  });\n\nprogram.parse();\n"]}
|
|
@@ -0,0 +1,40 @@
|
|
|
1
|
+
import { CostAnalyzerConfig, ValidationResult } from './types';
|
|
2
|
+
export declare class ConfigManager {
|
|
3
|
+
private static readonly CONFIG_FILE_NAMES;
|
|
4
|
+
/**
|
|
5
|
+
* Load configuration from file or return default configuration
|
|
6
|
+
*/
|
|
7
|
+
loadConfig(configPath?: string): Promise<CostAnalyzerConfig>;
|
|
8
|
+
/**
|
|
9
|
+
* Validate configuration structure and values
|
|
10
|
+
*/
|
|
11
|
+
validateConfig(config: CostAnalyzerConfig): ValidationResult;
|
|
12
|
+
/**
|
|
13
|
+
* Resolve configuration file path using search order
|
|
14
|
+
*/
|
|
15
|
+
private resolveConfigPath;
|
|
16
|
+
/**
|
|
17
|
+
* Read and parse configuration file (YAML or JSON)
|
|
18
|
+
*/
|
|
19
|
+
private readConfigFile;
|
|
20
|
+
/**
|
|
21
|
+
* Check if file exists
|
|
22
|
+
*/
|
|
23
|
+
private fileExists;
|
|
24
|
+
/**
|
|
25
|
+
* Validate threshold configuration
|
|
26
|
+
*/
|
|
27
|
+
private validateThresholds;
|
|
28
|
+
/**
|
|
29
|
+
* Validate usage assumptions
|
|
30
|
+
*/
|
|
31
|
+
private validateUsageAssumptions;
|
|
32
|
+
/**
|
|
33
|
+
* Get default configuration
|
|
34
|
+
*/
|
|
35
|
+
private getDefaultConfig;
|
|
36
|
+
/**
|
|
37
|
+
* Merge user configuration with defaults
|
|
38
|
+
*/
|
|
39
|
+
private mergeWithDefaults;
|
|
40
|
+
}
|
|
@@ -0,0 +1,238 @@
|
|
|
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.ConfigManager = void 0;
|
|
37
|
+
const fs = __importStar(require("fs/promises"));
|
|
38
|
+
const path = __importStar(require("path"));
|
|
39
|
+
const yaml = __importStar(require("js-yaml"));
|
|
40
|
+
const types_1 = require("./types");
|
|
41
|
+
class ConfigManager {
|
|
42
|
+
static CONFIG_FILE_NAMES = [
|
|
43
|
+
'.cdk-cost-analyzer.yml',
|
|
44
|
+
'.cdk-cost-analyzer.yaml',
|
|
45
|
+
'.cdk-cost-analyzer.json',
|
|
46
|
+
];
|
|
47
|
+
/**
|
|
48
|
+
* Load configuration from file or return default configuration
|
|
49
|
+
*/
|
|
50
|
+
async loadConfig(configPath) {
|
|
51
|
+
try {
|
|
52
|
+
const resolvedPath = await this.resolveConfigPath(configPath);
|
|
53
|
+
if (!resolvedPath) {
|
|
54
|
+
return this.getDefaultConfig();
|
|
55
|
+
}
|
|
56
|
+
const config = await this.readConfigFile(resolvedPath);
|
|
57
|
+
const validation = this.validateConfig(config);
|
|
58
|
+
if (!validation.valid) {
|
|
59
|
+
throw new types_1.ConfigurationError('Invalid configuration', resolvedPath, validation.errors);
|
|
60
|
+
}
|
|
61
|
+
return this.mergeWithDefaults(config);
|
|
62
|
+
}
|
|
63
|
+
catch (error) {
|
|
64
|
+
if (error instanceof types_1.ConfigurationError) {
|
|
65
|
+
throw error;
|
|
66
|
+
}
|
|
67
|
+
throw new types_1.ConfigurationError(`Failed to load configuration: ${error instanceof Error ? error.message : String(error)}`, configPath || 'unknown', []);
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
/**
|
|
71
|
+
* Validate configuration structure and values
|
|
72
|
+
*/
|
|
73
|
+
validateConfig(config) {
|
|
74
|
+
const errors = [];
|
|
75
|
+
const warnings = [];
|
|
76
|
+
if (config.thresholds) {
|
|
77
|
+
this.validateThresholds(config.thresholds, errors, warnings);
|
|
78
|
+
}
|
|
79
|
+
if (config.usageAssumptions) {
|
|
80
|
+
this.validateUsageAssumptions(config.usageAssumptions, errors);
|
|
81
|
+
}
|
|
82
|
+
if (config.cache?.durationHours !== undefined) {
|
|
83
|
+
if (config.cache.durationHours <= 0) {
|
|
84
|
+
errors.push('cache.durationHours must be positive');
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
return {
|
|
88
|
+
valid: errors.length === 0,
|
|
89
|
+
errors,
|
|
90
|
+
warnings,
|
|
91
|
+
};
|
|
92
|
+
}
|
|
93
|
+
/**
|
|
94
|
+
* Resolve configuration file path using search order
|
|
95
|
+
*/
|
|
96
|
+
async resolveConfigPath(configPath) {
|
|
97
|
+
if (configPath) {
|
|
98
|
+
const exists = await this.fileExists(configPath);
|
|
99
|
+
if (!exists) {
|
|
100
|
+
throw new Error(`Configuration file not found: ${configPath}`);
|
|
101
|
+
}
|
|
102
|
+
return configPath;
|
|
103
|
+
}
|
|
104
|
+
// Search in current directory
|
|
105
|
+
for (const fileName of ConfigManager.CONFIG_FILE_NAMES) {
|
|
106
|
+
const filePath = path.join(process.cwd(), fileName);
|
|
107
|
+
if (await this.fileExists(filePath)) {
|
|
108
|
+
return filePath;
|
|
109
|
+
}
|
|
110
|
+
}
|
|
111
|
+
return null;
|
|
112
|
+
}
|
|
113
|
+
/**
|
|
114
|
+
* Read and parse configuration file (YAML or JSON)
|
|
115
|
+
*/
|
|
116
|
+
async readConfigFile(filePath) {
|
|
117
|
+
const content = await fs.readFile(filePath, 'utf-8');
|
|
118
|
+
const ext = path.extname(filePath).toLowerCase();
|
|
119
|
+
try {
|
|
120
|
+
if (ext === '.json') {
|
|
121
|
+
return JSON.parse(content);
|
|
122
|
+
}
|
|
123
|
+
else {
|
|
124
|
+
return yaml.load(content);
|
|
125
|
+
}
|
|
126
|
+
}
|
|
127
|
+
catch (error) {
|
|
128
|
+
throw new Error(`Failed to parse configuration file: ${error instanceof Error ? error.message : String(error)}`);
|
|
129
|
+
}
|
|
130
|
+
}
|
|
131
|
+
/**
|
|
132
|
+
* Check if file exists
|
|
133
|
+
*/
|
|
134
|
+
async fileExists(filePath) {
|
|
135
|
+
try {
|
|
136
|
+
await fs.access(filePath);
|
|
137
|
+
return true;
|
|
138
|
+
}
|
|
139
|
+
catch {
|
|
140
|
+
return false;
|
|
141
|
+
}
|
|
142
|
+
}
|
|
143
|
+
/**
|
|
144
|
+
* Validate threshold configuration
|
|
145
|
+
*/
|
|
146
|
+
validateThresholds(thresholds, errors, warnings) {
|
|
147
|
+
const validateLevels = (levels, prefix) => {
|
|
148
|
+
if (levels.warning !== undefined && levels.warning < 0) {
|
|
149
|
+
errors.push(`${prefix}.warning must be non-negative`);
|
|
150
|
+
}
|
|
151
|
+
if (levels.error !== undefined && levels.error < 0) {
|
|
152
|
+
errors.push(`${prefix}.error must be non-negative`);
|
|
153
|
+
}
|
|
154
|
+
if (levels.warning !== undefined &&
|
|
155
|
+
levels.error !== undefined &&
|
|
156
|
+
levels.warning > levels.error) {
|
|
157
|
+
warnings.push(`${prefix}.warning (${levels.warning}) is greater than ${prefix}.error (${levels.error})`);
|
|
158
|
+
}
|
|
159
|
+
};
|
|
160
|
+
if (thresholds.default) {
|
|
161
|
+
validateLevels(thresholds.default, 'thresholds.default');
|
|
162
|
+
}
|
|
163
|
+
if (thresholds.environments) {
|
|
164
|
+
for (const [env, levels] of Object.entries(thresholds.environments)) {
|
|
165
|
+
validateLevels(levels, `thresholds.environments.${env}`);
|
|
166
|
+
}
|
|
167
|
+
}
|
|
168
|
+
}
|
|
169
|
+
/**
|
|
170
|
+
* Validate usage assumptions
|
|
171
|
+
*/
|
|
172
|
+
validateUsageAssumptions(assumptions, errors) {
|
|
173
|
+
const validatePositive = (value, configPath) => {
|
|
174
|
+
if (value !== undefined && value < 0) {
|
|
175
|
+
errors.push(`${configPath} must be non-negative`);
|
|
176
|
+
}
|
|
177
|
+
};
|
|
178
|
+
if (assumptions.s3) {
|
|
179
|
+
validatePositive(assumptions.s3.storageGB, 'usageAssumptions.s3.storageGB');
|
|
180
|
+
validatePositive(assumptions.s3.getRequests, 'usageAssumptions.s3.getRequests');
|
|
181
|
+
validatePositive(assumptions.s3.putRequests, 'usageAssumptions.s3.putRequests');
|
|
182
|
+
}
|
|
183
|
+
if (assumptions.lambda) {
|
|
184
|
+
validatePositive(assumptions.lambda.invocationsPerMonth, 'usageAssumptions.lambda.invocationsPerMonth');
|
|
185
|
+
validatePositive(assumptions.lambda.averageDurationMs, 'usageAssumptions.lambda.averageDurationMs');
|
|
186
|
+
}
|
|
187
|
+
if (assumptions.natGateway) {
|
|
188
|
+
validatePositive(assumptions.natGateway.dataProcessedGB, 'usageAssumptions.natGateway.dataProcessedGB');
|
|
189
|
+
}
|
|
190
|
+
if (assumptions.alb) {
|
|
191
|
+
validatePositive(assumptions.alb.newConnectionsPerSecond, 'usageAssumptions.alb.newConnectionsPerSecond');
|
|
192
|
+
validatePositive(assumptions.alb.activeConnectionsPerMinute, 'usageAssumptions.alb.activeConnectionsPerMinute');
|
|
193
|
+
validatePositive(assumptions.alb.processedBytesGB, 'usageAssumptions.alb.processedBytesGB');
|
|
194
|
+
}
|
|
195
|
+
if (assumptions.nlb) {
|
|
196
|
+
validatePositive(assumptions.nlb.newConnectionsPerSecond, 'usageAssumptions.nlb.newConnectionsPerSecond');
|
|
197
|
+
validatePositive(assumptions.nlb.activeConnectionsPerMinute, 'usageAssumptions.nlb.activeConnectionsPerMinute');
|
|
198
|
+
validatePositive(assumptions.nlb.processedBytesGB, 'usageAssumptions.nlb.processedBytesGB');
|
|
199
|
+
}
|
|
200
|
+
if (assumptions.cloudfront) {
|
|
201
|
+
validatePositive(assumptions.cloudfront.dataTransferGB, 'usageAssumptions.cloudfront.dataTransferGB');
|
|
202
|
+
validatePositive(assumptions.cloudfront.requests, 'usageAssumptions.cloudfront.requests');
|
|
203
|
+
}
|
|
204
|
+
if (assumptions.apiGateway) {
|
|
205
|
+
validatePositive(assumptions.apiGateway.requestsPerMonth, 'usageAssumptions.apiGateway.requestsPerMonth');
|
|
206
|
+
}
|
|
207
|
+
if (assumptions.vpcEndpoint) {
|
|
208
|
+
validatePositive(assumptions.vpcEndpoint.dataProcessedGB, 'usageAssumptions.vpcEndpoint.dataProcessedGB');
|
|
209
|
+
}
|
|
210
|
+
}
|
|
211
|
+
/**
|
|
212
|
+
* Get default configuration
|
|
213
|
+
*/
|
|
214
|
+
getDefaultConfig() {
|
|
215
|
+
return {
|
|
216
|
+
cache: {
|
|
217
|
+
enabled: true,
|
|
218
|
+
durationHours: 24,
|
|
219
|
+
},
|
|
220
|
+
};
|
|
221
|
+
}
|
|
222
|
+
/**
|
|
223
|
+
* Merge user configuration with defaults
|
|
224
|
+
*/
|
|
225
|
+
mergeWithDefaults(config) {
|
|
226
|
+
const defaults = this.getDefaultConfig();
|
|
227
|
+
return {
|
|
228
|
+
...defaults,
|
|
229
|
+
...config,
|
|
230
|
+
cache: {
|
|
231
|
+
...defaults.cache,
|
|
232
|
+
...config.cache,
|
|
233
|
+
},
|
|
234
|
+
};
|
|
235
|
+
}
|
|
236
|
+
}
|
|
237
|
+
exports.ConfigManager = ConfigManager;
|
|
238
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ConfigManager.js","sourceRoot":"","sources":["../../src/config/ConfigManager.ts"],"names":[],"mappings":"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","sourcesContent":["import * as fs from 'fs/promises';\nimport * as path from 'path';\nimport * as yaml from 'js-yaml';\nimport {\n  CostAnalyzerConfig,\n  ValidationResult,\n  ConfigurationError,\n  ThresholdLevels,\n} from './types';\n\nexport class ConfigManager {\n  private static readonly CONFIG_FILE_NAMES = [\n    '.cdk-cost-analyzer.yml',\n    '.cdk-cost-analyzer.yaml',\n    '.cdk-cost-analyzer.json',\n  ];\n\n  /**\n   * Load configuration from file or return default configuration\n   */\n  async loadConfig(configPath?: string): Promise<CostAnalyzerConfig> {\n    try {\n      const resolvedPath = await this.resolveConfigPath(configPath);\n\n      if (!resolvedPath) {\n        return this.getDefaultConfig();\n      }\n\n      const config = await this.readConfigFile(resolvedPath);\n      const validation = this.validateConfig(config);\n\n      if (!validation.valid) {\n        throw new ConfigurationError(\n          'Invalid configuration',\n          resolvedPath,\n          validation.errors,\n        );\n      }\n\n      return this.mergeWithDefaults(config);\n    } catch (error) {\n      if (error instanceof ConfigurationError) {\n        throw error;\n      }\n      throw new ConfigurationError(\n        `Failed to load configuration: ${error instanceof Error ? error.message : String(error)}`,\n        configPath || 'unknown',\n        [],\n      );\n    }\n  }\n\n  /**\n   * Validate configuration structure and values\n   */\n  validateConfig(config: CostAnalyzerConfig): ValidationResult {\n    const errors: string[] = [];\n    const warnings: string[] = [];\n\n    if (config.thresholds) {\n      this.validateThresholds(config.thresholds, errors, warnings);\n    }\n\n    if (config.usageAssumptions) {\n      this.validateUsageAssumptions(config.usageAssumptions, errors);\n    }\n\n    if (config.cache?.durationHours !== undefined) {\n      if (config.cache.durationHours <= 0) {\n        errors.push('cache.durationHours must be positive');\n      }\n    }\n\n    return {\n      valid: errors.length === 0,\n      errors,\n      warnings,\n    };\n  }\n\n  /**\n   * Resolve configuration file path using search order\n   */\n  private async resolveConfigPath(configPath?: string): Promise<string | null> {\n    if (configPath) {\n      const exists = await this.fileExists(configPath);\n      if (!exists) {\n        throw new Error(`Configuration file not found: ${configPath}`);\n      }\n      return configPath;\n    }\n\n    // Search in current directory\n    for (const fileName of ConfigManager.CONFIG_FILE_NAMES) {\n      const filePath = path.join(process.cwd(), fileName);\n      if (await this.fileExists(filePath)) {\n        return filePath;\n      }\n    }\n\n    return null;\n  }\n\n  /**\n   * Read and parse configuration file (YAML or JSON)\n   */\n  private async readConfigFile(filePath: string): Promise<CostAnalyzerConfig> {\n    const content = await fs.readFile(filePath, 'utf-8');\n    const ext = path.extname(filePath).toLowerCase();\n\n    try {\n      if (ext === '.json') {\n        return JSON.parse(content);\n      } else {\n        return yaml.load(content) as CostAnalyzerConfig;\n      }\n    } catch (error) {\n      throw new Error(\n        `Failed to parse configuration file: ${error instanceof Error ? error.message : String(error)}`,\n      );\n    }\n  }\n\n  /**\n   * Check if file exists\n   */\n  private async fileExists(filePath: string): Promise<boolean> {\n    try {\n      await fs.access(filePath);\n      return true;\n    } catch {\n      return false;\n    }\n  }\n\n  /**\n   * Validate threshold configuration\n   */\n  private validateThresholds(\n    thresholds: NonNullable<CostAnalyzerConfig['thresholds']>,\n    errors: string[],\n    warnings: string[],\n  ): void {\n    const validateLevels = (levels: ThresholdLevels, prefix: string) => {\n      if (levels.warning !== undefined && levels.warning < 0) {\n        errors.push(`${prefix}.warning must be non-negative`);\n      }\n      if (levels.error !== undefined && levels.error < 0) {\n        errors.push(`${prefix}.error must be non-negative`);\n      }\n      if (\n        levels.warning !== undefined &&\n        levels.error !== undefined &&\n        levels.warning > levels.error\n      ) {\n        warnings.push(\n          `${prefix}.warning (${levels.warning}) is greater than ${prefix}.error (${levels.error})`,\n        );\n      }\n    };\n\n    if (thresholds.default) {\n      validateLevels(thresholds.default, 'thresholds.default');\n    }\n\n    if (thresholds.environments) {\n      for (const [env, levels] of Object.entries(thresholds.environments)) {\n        validateLevels(levels, `thresholds.environments.${env}`);\n      }\n    }\n  }\n\n  /**\n   * Validate usage assumptions\n   */\n  private validateUsageAssumptions(\n    assumptions: NonNullable<CostAnalyzerConfig['usageAssumptions']>,\n    errors: string[],\n  ): void {\n    const validatePositive = (value: number | undefined, configPath: string) => {\n      if (value !== undefined && value < 0) {\n        errors.push(`${configPath} must be non-negative`);\n      }\n    };\n\n    if (assumptions.s3) {\n      validatePositive(assumptions.s3.storageGB, 'usageAssumptions.s3.storageGB');\n      validatePositive(assumptions.s3.getRequests, 'usageAssumptions.s3.getRequests');\n      validatePositive(assumptions.s3.putRequests, 'usageAssumptions.s3.putRequests');\n    }\n\n    if (assumptions.lambda) {\n      validatePositive(\n        assumptions.lambda.invocationsPerMonth,\n        'usageAssumptions.lambda.invocationsPerMonth',\n      );\n      validatePositive(\n        assumptions.lambda.averageDurationMs,\n        'usageAssumptions.lambda.averageDurationMs',\n      );\n    }\n\n    if (assumptions.natGateway) {\n      validatePositive(\n        assumptions.natGateway.dataProcessedGB,\n        'usageAssumptions.natGateway.dataProcessedGB',\n      );\n    }\n\n    if (assumptions.alb) {\n      validatePositive(\n        assumptions.alb.newConnectionsPerSecond,\n        'usageAssumptions.alb.newConnectionsPerSecond',\n      );\n      validatePositive(\n        assumptions.alb.activeConnectionsPerMinute,\n        'usageAssumptions.alb.activeConnectionsPerMinute',\n      );\n      validatePositive(\n        assumptions.alb.processedBytesGB,\n        'usageAssumptions.alb.processedBytesGB',\n      );\n    }\n\n    if (assumptions.nlb) {\n      validatePositive(\n        assumptions.nlb.newConnectionsPerSecond,\n        'usageAssumptions.nlb.newConnectionsPerSecond',\n      );\n      validatePositive(\n        assumptions.nlb.activeConnectionsPerMinute,\n        'usageAssumptions.nlb.activeConnectionsPerMinute',\n      );\n      validatePositive(\n        assumptions.nlb.processedBytesGB,\n        'usageAssumptions.nlb.processedBytesGB',\n      );\n    }\n\n    if (assumptions.cloudfront) {\n      validatePositive(\n        assumptions.cloudfront.dataTransferGB,\n        'usageAssumptions.cloudfront.dataTransferGB',\n      );\n      validatePositive(\n        assumptions.cloudfront.requests,\n        'usageAssumptions.cloudfront.requests',\n      );\n    }\n\n    if (assumptions.apiGateway) {\n      validatePositive(\n        assumptions.apiGateway.requestsPerMonth,\n        'usageAssumptions.apiGateway.requestsPerMonth',\n      );\n    }\n\n    if (assumptions.vpcEndpoint) {\n      validatePositive(\n        assumptions.vpcEndpoint.dataProcessedGB,\n        'usageAssumptions.vpcEndpoint.dataProcessedGB',\n      );\n    }\n  }\n\n  /**\n   * Get default configuration\n   */\n  private getDefaultConfig(): CostAnalyzerConfig {\n    return {\n      cache: {\n        enabled: true,\n        durationHours: 24,\n      },\n    };\n  }\n\n  /**\n   * Merge user configuration with defaults\n   */\n  private mergeWithDefaults(config: CostAnalyzerConfig): CostAnalyzerConfig {\n    const defaults = this.getDefaultConfig();\n    return {\n      ...defaults,\n      ...config,\n      cache: {\n        ...defaults.cache,\n        ...config.cache,\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("./ConfigManager"), exports);
|
|
18
|
+
__exportStar(require("./types"), exports);
|
|
19
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvY29uZmlnL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7QUFBQSxrREFBZ0M7QUFDaEMsMENBQXdCIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9Db25maWdNYW5hZ2VyJztcbmV4cG9ydCAqIGZyb20gJy4vdHlwZXMnO1xuIl19
|