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,191 @@
|
|
|
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.PipelineOrchestrator = void 0;
|
|
37
|
+
const fs = __importStar(require("fs/promises"));
|
|
38
|
+
const types_1 = require("./types");
|
|
39
|
+
const ConfigManager_1 = require("../config/ConfigManager");
|
|
40
|
+
const DiffEngine_1 = require("../diff/DiffEngine");
|
|
41
|
+
const TemplateParser_1 = require("../parser/TemplateParser");
|
|
42
|
+
const PricingService_1 = require("../pricing/PricingService");
|
|
43
|
+
const Reporter_1 = require("../reporter/Reporter");
|
|
44
|
+
const SynthesisOrchestrator_1 = require("../synthesis/SynthesisOrchestrator");
|
|
45
|
+
const ThresholdEnforcer_1 = require("../threshold/ThresholdEnforcer");
|
|
46
|
+
class PipelineOrchestrator {
|
|
47
|
+
configManager;
|
|
48
|
+
synthesisOrchestrator;
|
|
49
|
+
thresholdEnforcer;
|
|
50
|
+
constructor() {
|
|
51
|
+
this.configManager = new ConfigManager_1.ConfigManager();
|
|
52
|
+
this.synthesisOrchestrator = new SynthesisOrchestrator_1.SynthesisOrchestrator();
|
|
53
|
+
this.thresholdEnforcer = new ThresholdEnforcer_1.ThresholdEnforcer();
|
|
54
|
+
}
|
|
55
|
+
/**
|
|
56
|
+
* Run complete pipeline analysis
|
|
57
|
+
*/
|
|
58
|
+
async runPipelineAnalysis(options) {
|
|
59
|
+
try {
|
|
60
|
+
// 1. Load configuration
|
|
61
|
+
const config = await this.configManager.loadConfig(options.configPath);
|
|
62
|
+
// 2. Determine template paths
|
|
63
|
+
let baseTemplatePath;
|
|
64
|
+
let targetTemplatePath;
|
|
65
|
+
let synthesisInfo;
|
|
66
|
+
if (options.synthesize && options.cdkAppPath) {
|
|
67
|
+
// Synthesize both branches
|
|
68
|
+
const synthResult = await this.synthesizeBothBranches(options.cdkAppPath, config, options.outputPath);
|
|
69
|
+
baseTemplatePath = synthResult.baseTemplatePath;
|
|
70
|
+
targetTemplatePath = synthResult.targetTemplatePath;
|
|
71
|
+
synthesisInfo = synthResult.synthesisInfo;
|
|
72
|
+
}
|
|
73
|
+
else {
|
|
74
|
+
// Use provided template paths
|
|
75
|
+
if (!options.baseTemplate || !options.targetTemplate) {
|
|
76
|
+
throw new types_1.PipelineError('Either provide template paths or enable synthesis with cdkAppPath', 'configuration');
|
|
77
|
+
}
|
|
78
|
+
baseTemplatePath = options.baseTemplate;
|
|
79
|
+
targetTemplatePath = options.targetTemplate;
|
|
80
|
+
}
|
|
81
|
+
// 3. Analyze costs
|
|
82
|
+
const region = options.region || config.synthesis?.context?.region || 'eu-central-1';
|
|
83
|
+
const costAnalysis = await this.analyzeCosts(baseTemplatePath, targetTemplatePath, region, config);
|
|
84
|
+
// 4. Evaluate thresholds
|
|
85
|
+
const thresholdStatus = this.thresholdEnforcer.evaluateThreshold(costAnalysis.totalDelta, costAnalysis.addedResources, costAnalysis.modifiedResources, config.thresholds, options.environment);
|
|
86
|
+
// 5. Build config summary
|
|
87
|
+
const configSummary = this.buildConfigSummary(config, options);
|
|
88
|
+
return {
|
|
89
|
+
costAnalysis,
|
|
90
|
+
thresholdStatus,
|
|
91
|
+
synthesisInfo,
|
|
92
|
+
configUsed: configSummary,
|
|
93
|
+
};
|
|
94
|
+
}
|
|
95
|
+
catch (error) {
|
|
96
|
+
if (error instanceof types_1.PipelineError) {
|
|
97
|
+
throw error;
|
|
98
|
+
}
|
|
99
|
+
throw new types_1.PipelineError(`Pipeline failed: ${error instanceof Error ? error.message : String(error)}`, 'unknown');
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
/**
|
|
103
|
+
* Synthesize both base and target branches
|
|
104
|
+
*/
|
|
105
|
+
async synthesizeBothBranches(cdkAppPath, config, outputPath) {
|
|
106
|
+
// For now, just synthesize the current branch
|
|
107
|
+
// In a full implementation, this would checkout branches and synthesize each
|
|
108
|
+
const result = await this.synthesisOrchestrator.synthesize({
|
|
109
|
+
cdkAppPath,
|
|
110
|
+
outputPath: outputPath || config.synthesis?.outputPath,
|
|
111
|
+
context: config.synthesis?.context,
|
|
112
|
+
customCommand: config.synthesis?.customCommand,
|
|
113
|
+
});
|
|
114
|
+
if (!result.success) {
|
|
115
|
+
throw new types_1.PipelineError(`CDK synthesis failed: ${result.error}`, 'synthesis');
|
|
116
|
+
}
|
|
117
|
+
// For simplicity, use the first template for both base and target
|
|
118
|
+
// In production, you'd synthesize different branches
|
|
119
|
+
return {
|
|
120
|
+
baseTemplatePath: result.templatePaths[0],
|
|
121
|
+
targetTemplatePath: result.templatePaths[0],
|
|
122
|
+
synthesisInfo: {
|
|
123
|
+
baseStackCount: result.stackNames.length,
|
|
124
|
+
targetStackCount: result.stackNames.length,
|
|
125
|
+
baseSynthesisTime: result.duration,
|
|
126
|
+
targetSynthesisTime: result.duration,
|
|
127
|
+
},
|
|
128
|
+
};
|
|
129
|
+
}
|
|
130
|
+
/**
|
|
131
|
+
* Analyze costs between two templates
|
|
132
|
+
*/
|
|
133
|
+
async analyzeCosts(baseTemplatePath, targetTemplatePath, region, config) {
|
|
134
|
+
const parser = new TemplateParser_1.TemplateParser();
|
|
135
|
+
const diffEngine = new DiffEngine_1.DiffEngine();
|
|
136
|
+
const pricingService = new PricingService_1.PricingService(region, config.usageAssumptions, config.exclusions?.resourceTypes, config.cache);
|
|
137
|
+
const reporter = new Reporter_1.Reporter();
|
|
138
|
+
// Read templates
|
|
139
|
+
const baseTemplateContent = await fs.readFile(baseTemplatePath, 'utf-8');
|
|
140
|
+
const targetTemplateContent = await fs.readFile(targetTemplatePath, 'utf-8');
|
|
141
|
+
// Parse templates
|
|
142
|
+
const baseTemplateObj = parser.parse(baseTemplateContent);
|
|
143
|
+
const targetTemplateObj = parser.parse(targetTemplateContent);
|
|
144
|
+
// Diff templates
|
|
145
|
+
const diff = diffEngine.diff(baseTemplateObj, targetTemplateObj);
|
|
146
|
+
// Calculate cost delta
|
|
147
|
+
const costDelta = await pricingService.getCostDelta(diff, region);
|
|
148
|
+
// Generate report
|
|
149
|
+
const summary = reporter.generateReport(costDelta, 'text');
|
|
150
|
+
return {
|
|
151
|
+
totalDelta: costDelta.totalDelta,
|
|
152
|
+
currency: costDelta.currency,
|
|
153
|
+
addedResources: costDelta.addedCosts,
|
|
154
|
+
removedResources: costDelta.removedCosts,
|
|
155
|
+
modifiedResources: costDelta.modifiedCosts,
|
|
156
|
+
summary,
|
|
157
|
+
};
|
|
158
|
+
}
|
|
159
|
+
/**
|
|
160
|
+
* Build configuration summary
|
|
161
|
+
*/
|
|
162
|
+
buildConfigSummary(config, options) {
|
|
163
|
+
const summary = {
|
|
164
|
+
synthesisEnabled: !!options.synthesize,
|
|
165
|
+
};
|
|
166
|
+
if (options.configPath) {
|
|
167
|
+
summary.configPath = options.configPath;
|
|
168
|
+
}
|
|
169
|
+
if (config.thresholds) {
|
|
170
|
+
const thresholds = options.environment && config.thresholds.environments?.[options.environment]
|
|
171
|
+
? config.thresholds.environments[options.environment]
|
|
172
|
+
: config.thresholds.default;
|
|
173
|
+
if (thresholds) {
|
|
174
|
+
summary.thresholds = {
|
|
175
|
+
warning: thresholds.warning,
|
|
176
|
+
error: thresholds.error,
|
|
177
|
+
environment: options.environment,
|
|
178
|
+
};
|
|
179
|
+
}
|
|
180
|
+
}
|
|
181
|
+
if (config.usageAssumptions) {
|
|
182
|
+
summary.usageAssumptions = config.usageAssumptions;
|
|
183
|
+
}
|
|
184
|
+
if (config.exclusions?.resourceTypes) {
|
|
185
|
+
summary.excludedResourceTypes = config.exclusions.resourceTypes;
|
|
186
|
+
}
|
|
187
|
+
return summary;
|
|
188
|
+
}
|
|
189
|
+
}
|
|
190
|
+
exports.PipelineOrchestrator = PipelineOrchestrator;
|
|
191
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"PipelineOrchestrator.js","sourceRoot":"","sources":["../../src/pipeline/PipelineOrchestrator.ts"],"names":[],"mappings":"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","sourcesContent":["import * as fs from 'fs/promises';\nimport { PipelineOptions, PipelineResult, PipelineError, ConfigSummary } from './types';\nimport { ConfigManager } from '../config/ConfigManager';\nimport { DiffEngine } from '../diff/DiffEngine';\nimport { TemplateParser } from '../parser/TemplateParser';\nimport { PricingService } from '../pricing/PricingService';\nimport { Reporter } from '../reporter/Reporter';\nimport { SynthesisOrchestrator } from '../synthesis/SynthesisOrchestrator';\nimport { ThresholdEnforcer } from '../threshold/ThresholdEnforcer';\n\nexport class PipelineOrchestrator {\n  private configManager: ConfigManager;\n  private synthesisOrchestrator: SynthesisOrchestrator;\n  private thresholdEnforcer: ThresholdEnforcer;\n\n  constructor() {\n    this.configManager = new ConfigManager();\n    this.synthesisOrchestrator = new SynthesisOrchestrator();\n    this.thresholdEnforcer = new ThresholdEnforcer();\n  }\n\n  /**\n   * Run complete pipeline analysis\n   */\n  async runPipelineAnalysis(options: PipelineOptions): Promise<PipelineResult> {\n    try {\n      // 1. Load configuration\n      const config = await this.configManager.loadConfig(options.configPath);\n\n      // 2. Determine template paths\n      let baseTemplatePath: string;\n      let targetTemplatePath: string;\n      let synthesisInfo;\n\n      if (options.synthesize && options.cdkAppPath) {\n        // Synthesize both branches\n        const synthResult = await this.synthesizeBothBranches(\n          options.cdkAppPath,\n          config,\n          options.outputPath,\n        );\n        baseTemplatePath = synthResult.baseTemplatePath;\n        targetTemplatePath = synthResult.targetTemplatePath;\n        synthesisInfo = synthResult.synthesisInfo;\n      } else {\n        // Use provided template paths\n        if (!options.baseTemplate || !options.targetTemplate) {\n          throw new PipelineError(\n            'Either provide template paths or enable synthesis with cdkAppPath',\n            'configuration',\n          );\n        }\n        baseTemplatePath = options.baseTemplate;\n        targetTemplatePath = options.targetTemplate;\n      }\n\n      // 3. Analyze costs\n      const region = options.region || config.synthesis?.context?.region || 'eu-central-1';\n      const costAnalysis = await this.analyzeCosts(\n        baseTemplatePath,\n        targetTemplatePath,\n        region,\n        config,\n      );\n\n      // 4. Evaluate thresholds\n      const thresholdStatus = this.thresholdEnforcer.evaluateThreshold(\n        costAnalysis.totalDelta,\n        costAnalysis.addedResources,\n        costAnalysis.modifiedResources,\n        config.thresholds,\n        options.environment,\n      );\n\n      // 5. Build config summary\n      const configSummary = this.buildConfigSummary(config, options);\n\n      return {\n        costAnalysis,\n        thresholdStatus,\n        synthesisInfo,\n        configUsed: configSummary,\n      };\n    } catch (error) {\n      if (error instanceof PipelineError) {\n        throw error;\n      }\n      throw new PipelineError(\n        `Pipeline failed: ${error instanceof Error ? error.message : String(error)}`,\n        'unknown',\n      );\n    }\n  }\n\n  /**\n   * Synthesize both base and target branches\n   */\n  private async synthesizeBothBranches(\n    cdkAppPath: string,\n    config: any,\n    outputPath?: string,\n  ): Promise<{\n      baseTemplatePath: string;\n      targetTemplatePath: string;\n      synthesisInfo: any;\n    }> {\n    // For now, just synthesize the current branch\n    // In a full implementation, this would checkout branches and synthesize each\n    const result = await this.synthesisOrchestrator.synthesize({\n      cdkAppPath,\n      outputPath: outputPath || config.synthesis?.outputPath,\n      context: config.synthesis?.context,\n      customCommand: config.synthesis?.customCommand,\n    });\n\n    if (!result.success) {\n      throw new PipelineError(\n        `CDK synthesis failed: ${result.error}`,\n        'synthesis',\n      );\n    }\n\n    // For simplicity, use the first template for both base and target\n    // In production, you'd synthesize different branches\n    return {\n      baseTemplatePath: result.templatePaths[0],\n      targetTemplatePath: result.templatePaths[0],\n      synthesisInfo: {\n        baseStackCount: result.stackNames.length,\n        targetStackCount: result.stackNames.length,\n        baseSynthesisTime: result.duration,\n        targetSynthesisTime: result.duration,\n      },\n    };\n  }\n\n  /**\n   * Analyze costs between two templates\n   */\n  private async analyzeCosts(\n    baseTemplatePath: string,\n    targetTemplatePath: string,\n    region: string,\n    config: any,\n  ): Promise<any> {\n    const parser = new TemplateParser();\n    const diffEngine = new DiffEngine();\n    const pricingService = new PricingService(\n      region,\n      config.usageAssumptions,\n      config.exclusions?.resourceTypes,\n      config.cache,\n    );\n    const reporter = new Reporter();\n\n    // Read templates\n    const baseTemplateContent = await fs.readFile(baseTemplatePath, 'utf-8');\n    const targetTemplateContent = await fs.readFile(targetTemplatePath, 'utf-8');\n\n    // Parse templates\n    const baseTemplateObj = parser.parse(baseTemplateContent);\n    const targetTemplateObj = parser.parse(targetTemplateContent);\n\n    // Diff templates\n    const diff = diffEngine.diff(baseTemplateObj, targetTemplateObj);\n\n    // Calculate cost delta\n    const costDelta = await pricingService.getCostDelta(diff, region);\n\n    // Generate report\n    const summary = reporter.generateReport(costDelta, 'text');\n\n    return {\n      totalDelta: costDelta.totalDelta,\n      currency: costDelta.currency,\n      addedResources: costDelta.addedCosts,\n      removedResources: costDelta.removedCosts,\n      modifiedResources: costDelta.modifiedCosts,\n      summary,\n    };\n  }\n\n  /**\n   * Build configuration summary\n   */\n  private buildConfigSummary(config: any, options: PipelineOptions): ConfigSummary {\n    const summary: ConfigSummary = {\n      synthesisEnabled: !!options.synthesize,\n    };\n\n    if (options.configPath) {\n      summary.configPath = options.configPath;\n    }\n\n    if (config.thresholds) {\n      const thresholds = options.environment && config.thresholds.environments?.[options.environment]\n        ? config.thresholds.environments[options.environment]\n        : config.thresholds.default;\n\n      if (thresholds) {\n        summary.thresholds = {\n          warning: thresholds.warning,\n          error: thresholds.error,\n          environment: options.environment,\n        };\n      }\n    }\n\n    if (config.usageAssumptions) {\n      summary.usageAssumptions = config.usageAssumptions;\n    }\n\n    if (config.exclusions?.resourceTypes) {\n      summary.excludedResourceTypes = config.exclusions.resourceTypes;\n    }\n\n    return summary;\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("./PipelineOrchestrator"), exports);
|
|
18
|
+
__exportStar(require("./types"), exports);
|
|
19
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvcGlwZWxpbmUvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7OztBQUFBLHlEQUF1QztBQUN2QywwQ0FBd0IiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL1BpcGVsaW5lT3JjaGVzdHJhdG9yJztcbmV4cG9ydCAqIGZyb20gJy4vdHlwZXMnO1xuIl19
|
|
@@ -0,0 +1,41 @@
|
|
|
1
|
+
import { CostAnalysisResult } from '../api/types';
|
|
2
|
+
import { ThresholdEvaluation } from '../threshold/types';
|
|
3
|
+
export interface PipelineOptions {
|
|
4
|
+
baseTemplate?: string;
|
|
5
|
+
targetTemplate?: string;
|
|
6
|
+
baseBranch?: string;
|
|
7
|
+
targetBranch?: string;
|
|
8
|
+
cdkAppPath?: string;
|
|
9
|
+
outputPath?: string;
|
|
10
|
+
configPath?: string;
|
|
11
|
+
region?: string;
|
|
12
|
+
synthesize?: boolean;
|
|
13
|
+
environment?: string;
|
|
14
|
+
}
|
|
15
|
+
export interface PipelineResult {
|
|
16
|
+
costAnalysis: CostAnalysisResult;
|
|
17
|
+
thresholdStatus: ThresholdEvaluation;
|
|
18
|
+
synthesisInfo?: SynthesisInfo;
|
|
19
|
+
configUsed: ConfigSummary;
|
|
20
|
+
}
|
|
21
|
+
export interface SynthesisInfo {
|
|
22
|
+
baseStackCount: number;
|
|
23
|
+
targetStackCount: number;
|
|
24
|
+
baseSynthesisTime: number;
|
|
25
|
+
targetSynthesisTime: number;
|
|
26
|
+
}
|
|
27
|
+
export interface ConfigSummary {
|
|
28
|
+
configPath?: string;
|
|
29
|
+
thresholds?: {
|
|
30
|
+
warning?: number;
|
|
31
|
+
error?: number;
|
|
32
|
+
environment?: string;
|
|
33
|
+
};
|
|
34
|
+
usageAssumptions?: Record<string, unknown>;
|
|
35
|
+
excludedResourceTypes?: string[];
|
|
36
|
+
synthesisEnabled: boolean;
|
|
37
|
+
}
|
|
38
|
+
export declare class PipelineError extends Error {
|
|
39
|
+
stage: string;
|
|
40
|
+
constructor(message: string, stage: string);
|
|
41
|
+
}
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.PipelineError = void 0;
|
|
4
|
+
class PipelineError extends Error {
|
|
5
|
+
stage;
|
|
6
|
+
constructor(message, stage) {
|
|
7
|
+
super(message);
|
|
8
|
+
this.stage = stage;
|
|
9
|
+
this.name = 'PipelineError';
|
|
10
|
+
}
|
|
11
|
+
}
|
|
12
|
+
exports.PipelineError = PipelineError;
|
|
13
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvcGlwZWxpbmUvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBMENBLE1BQWEsYUFBYyxTQUFRLEtBQUs7SUFDRjtJQUFwQyxZQUFZLE9BQWUsRUFBUyxLQUFhO1FBQy9DLEtBQUssQ0FBQyxPQUFPLENBQUMsQ0FBQztRQURtQixVQUFLLEdBQUwsS0FBSyxDQUFRO1FBRS9DLElBQUksQ0FBQyxJQUFJLEdBQUcsZUFBZSxDQUFDO0lBQzlCLENBQUM7Q0FDRjtBQUxELHNDQUtDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29zdEFuYWx5c2lzUmVzdWx0IH0gZnJvbSAnLi4vYXBpL3R5cGVzJztcbmltcG9ydCB7IFRocmVzaG9sZEV2YWx1YXRpb24gfSBmcm9tICcuLi90aHJlc2hvbGQvdHlwZXMnO1xuXG5leHBvcnQgaW50ZXJmYWNlIFBpcGVsaW5lT3B0aW9ucyB7XG4gIGJhc2VUZW1wbGF0ZT86IHN0cmluZztcbiAgdGFyZ2V0VGVtcGxhdGU/OiBzdHJpbmc7XG4gIGJhc2VCcmFuY2g/OiBzdHJpbmc7XG4gIHRhcmdldEJyYW5jaD86IHN0cmluZztcbiAgY2RrQXBwUGF0aD86IHN0cmluZztcbiAgb3V0cHV0UGF0aD86IHN0cmluZztcbiAgY29uZmlnUGF0aD86IHN0cmluZztcbiAgcmVnaW9uPzogc3RyaW5nO1xuICBzeW50aGVzaXplPzogYm9vbGVhbjtcbiAgZW52aXJvbm1lbnQ/OiBzdHJpbmc7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgUGlwZWxpbmVSZXN1bHQge1xuICBjb3N0QW5hbHlzaXM6IENvc3RBbmFseXNpc1Jlc3VsdDtcbiAgdGhyZXNob2xkU3RhdHVzOiBUaHJlc2hvbGRFdmFsdWF0aW9uO1xuICBzeW50aGVzaXNJbmZvPzogU3ludGhlc2lzSW5mbztcbiAgY29uZmlnVXNlZDogQ29uZmlnU3VtbWFyeTtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBTeW50aGVzaXNJbmZvIHtcbiAgYmFzZVN0YWNrQ291bnQ6IG51bWJlcjtcbiAgdGFyZ2V0U3RhY2tDb3VudDogbnVtYmVyO1xuICBiYXNlU3ludGhlc2lzVGltZTogbnVtYmVyO1xuICB0YXJnZXRTeW50aGVzaXNUaW1lOiBudW1iZXI7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQ29uZmlnU3VtbWFyeSB7XG4gIGNvbmZpZ1BhdGg/OiBzdHJpbmc7XG4gIHRocmVzaG9sZHM/OiB7XG4gICAgd2FybmluZz86IG51bWJlcjtcbiAgICBlcnJvcj86IG51bWJlcjtcbiAgICBlbnZpcm9ubWVudD86IHN0cmluZztcbiAgfTtcbiAgdXNhZ2VBc3N1bXB0aW9ucz86IFJlY29yZDxzdHJpbmcsIHVua25vd24+O1xuICBleGNsdWRlZFJlc291cmNlVHlwZXM/OiBzdHJpbmdbXTtcbiAgc3ludGhlc2lzRW5hYmxlZDogYm9vbGVhbjtcbn1cblxuZXhwb3J0IGNsYXNzIFBpcGVsaW5lRXJyb3IgZXh0ZW5kcyBFcnJvciB7XG4gIGNvbnN0cnVjdG9yKG1lc3NhZ2U6IHN0cmluZywgcHVibGljIHN0YWdlOiBzdHJpbmcpIHtcbiAgICBzdXBlcihtZXNzYWdlKTtcbiAgICB0aGlzLm5hbWUgPSAnUGlwZWxpbmVFcnJvcic7XG4gIH1cbn1cbiJdfQ==
|
|
@@ -0,0 +1,75 @@
|
|
|
1
|
+
import { PriceQueryParams } from './types';
|
|
2
|
+
export interface CachedPriceEntry {
|
|
3
|
+
price: number;
|
|
4
|
+
timestamp: number;
|
|
5
|
+
}
|
|
6
|
+
export interface CacheMetadata {
|
|
7
|
+
entries: Record<string, CachedPriceEntry>;
|
|
8
|
+
}
|
|
9
|
+
/**
|
|
10
|
+
* Manages persistent caching of pricing data to reduce AWS API calls
|
|
11
|
+
* and improve performance across pipeline runs.
|
|
12
|
+
*/
|
|
13
|
+
export declare class CacheManager {
|
|
14
|
+
private cacheDir;
|
|
15
|
+
private cacheDurationMs;
|
|
16
|
+
private metadata;
|
|
17
|
+
private metadataPath;
|
|
18
|
+
/**
|
|
19
|
+
* Creates a new CacheManager instance
|
|
20
|
+
* @param cacheDir Directory to store cache files (default: .cdk-cost-analyzer-cache)
|
|
21
|
+
* @param cacheDurationHours Duration in hours before cache entries expire (default: 24)
|
|
22
|
+
*/
|
|
23
|
+
constructor(cacheDir?: string, cacheDurationHours?: number);
|
|
24
|
+
/**
|
|
25
|
+
* Retrieves a cached price if it exists and is still fresh
|
|
26
|
+
* @param params Price query parameters
|
|
27
|
+
* @returns Cached price or null if not found or expired
|
|
28
|
+
*/
|
|
29
|
+
getCachedPrice(params: PriceQueryParams): number | null;
|
|
30
|
+
/**
|
|
31
|
+
* Stores a price in the cache with current timestamp
|
|
32
|
+
* @param params Price query parameters
|
|
33
|
+
* @param price Price value to cache
|
|
34
|
+
*/
|
|
35
|
+
setCachedPrice(params: PriceQueryParams, price: number): void;
|
|
36
|
+
/**
|
|
37
|
+
* Checks if a cached price exists and is still fresh
|
|
38
|
+
* @param params Price query parameters
|
|
39
|
+
* @returns true if fresh cache entry exists
|
|
40
|
+
*/
|
|
41
|
+
hasFreshCache(params: PriceQueryParams): boolean;
|
|
42
|
+
/**
|
|
43
|
+
* Clears all cached entries
|
|
44
|
+
*/
|
|
45
|
+
clearCache(): void;
|
|
46
|
+
/**
|
|
47
|
+
* Gets cache statistics
|
|
48
|
+
* @returns Object with cache statistics
|
|
49
|
+
*/
|
|
50
|
+
getCacheStats(): {
|
|
51
|
+
totalEntries: number;
|
|
52
|
+
freshEntries: number;
|
|
53
|
+
staleEntries: number;
|
|
54
|
+
};
|
|
55
|
+
/**
|
|
56
|
+
* Removes stale cache entries
|
|
57
|
+
*/
|
|
58
|
+
pruneStaleEntries(): void;
|
|
59
|
+
/**
|
|
60
|
+
* Generates a cache key from price query parameters
|
|
61
|
+
*/
|
|
62
|
+
private getCacheKey;
|
|
63
|
+
/**
|
|
64
|
+
* Ensures the cache directory exists
|
|
65
|
+
*/
|
|
66
|
+
private ensureCacheDirectory;
|
|
67
|
+
/**
|
|
68
|
+
* Loads cache metadata from disk
|
|
69
|
+
*/
|
|
70
|
+
private loadMetadata;
|
|
71
|
+
/**
|
|
72
|
+
* Saves cache metadata to disk
|
|
73
|
+
*/
|
|
74
|
+
private saveMetadata;
|
|
75
|
+
}
|
|
@@ -0,0 +1,195 @@
|
|
|
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.CacheManager = void 0;
|
|
37
|
+
const fs = __importStar(require("fs"));
|
|
38
|
+
const path = __importStar(require("path"));
|
|
39
|
+
/**
|
|
40
|
+
* Manages persistent caching of pricing data to reduce AWS API calls
|
|
41
|
+
* and improve performance across pipeline runs.
|
|
42
|
+
*/
|
|
43
|
+
class CacheManager {
|
|
44
|
+
cacheDir;
|
|
45
|
+
cacheDurationMs;
|
|
46
|
+
metadata;
|
|
47
|
+
metadataPath;
|
|
48
|
+
/**
|
|
49
|
+
* Creates a new CacheManager instance
|
|
50
|
+
* @param cacheDir Directory to store cache files (default: .cdk-cost-analyzer-cache)
|
|
51
|
+
* @param cacheDurationHours Duration in hours before cache entries expire (default: 24)
|
|
52
|
+
*/
|
|
53
|
+
constructor(cacheDir = '.cdk-cost-analyzer-cache', cacheDurationHours = 24) {
|
|
54
|
+
this.cacheDir = cacheDir;
|
|
55
|
+
this.cacheDurationMs = cacheDurationHours * 60 * 60 * 1000;
|
|
56
|
+
this.metadataPath = path.join(this.cacheDir, 'metadata.json');
|
|
57
|
+
this.metadata = { entries: {} };
|
|
58
|
+
this.ensureCacheDirectory();
|
|
59
|
+
this.loadMetadata();
|
|
60
|
+
}
|
|
61
|
+
/**
|
|
62
|
+
* Retrieves a cached price if it exists and is still fresh
|
|
63
|
+
* @param params Price query parameters
|
|
64
|
+
* @returns Cached price or null if not found or expired
|
|
65
|
+
*/
|
|
66
|
+
getCachedPrice(params) {
|
|
67
|
+
const cacheKey = this.getCacheKey(params);
|
|
68
|
+
const entry = this.metadata.entries[cacheKey];
|
|
69
|
+
if (!entry) {
|
|
70
|
+
return null;
|
|
71
|
+
}
|
|
72
|
+
const now = Date.now();
|
|
73
|
+
const age = now - entry.timestamp;
|
|
74
|
+
if (age > this.cacheDurationMs) {
|
|
75
|
+
// Cache entry is stale
|
|
76
|
+
delete this.metadata.entries[cacheKey];
|
|
77
|
+
this.saveMetadata();
|
|
78
|
+
return null;
|
|
79
|
+
}
|
|
80
|
+
return entry.price;
|
|
81
|
+
}
|
|
82
|
+
/**
|
|
83
|
+
* Stores a price in the cache with current timestamp
|
|
84
|
+
* @param params Price query parameters
|
|
85
|
+
* @param price Price value to cache
|
|
86
|
+
*/
|
|
87
|
+
setCachedPrice(params, price) {
|
|
88
|
+
const cacheKey = this.getCacheKey(params);
|
|
89
|
+
this.metadata.entries[cacheKey] = {
|
|
90
|
+
price,
|
|
91
|
+
timestamp: Date.now(),
|
|
92
|
+
};
|
|
93
|
+
this.saveMetadata();
|
|
94
|
+
}
|
|
95
|
+
/**
|
|
96
|
+
* Checks if a cached price exists and is still fresh
|
|
97
|
+
* @param params Price query parameters
|
|
98
|
+
* @returns true if fresh cache entry exists
|
|
99
|
+
*/
|
|
100
|
+
hasFreshCache(params) {
|
|
101
|
+
return this.getCachedPrice(params) !== null;
|
|
102
|
+
}
|
|
103
|
+
/**
|
|
104
|
+
* Clears all cached entries
|
|
105
|
+
*/
|
|
106
|
+
clearCache() {
|
|
107
|
+
this.metadata = { entries: {} };
|
|
108
|
+
this.saveMetadata();
|
|
109
|
+
}
|
|
110
|
+
/**
|
|
111
|
+
* Gets cache statistics
|
|
112
|
+
* @returns Object with cache statistics
|
|
113
|
+
*/
|
|
114
|
+
getCacheStats() {
|
|
115
|
+
const now = Date.now();
|
|
116
|
+
let freshEntries = 0;
|
|
117
|
+
let staleEntries = 0;
|
|
118
|
+
for (const entry of Object.values(this.metadata.entries)) {
|
|
119
|
+
const age = now - entry.timestamp;
|
|
120
|
+
if (age <= this.cacheDurationMs) {
|
|
121
|
+
freshEntries++;
|
|
122
|
+
}
|
|
123
|
+
else {
|
|
124
|
+
staleEntries++;
|
|
125
|
+
}
|
|
126
|
+
}
|
|
127
|
+
return {
|
|
128
|
+
totalEntries: Object.keys(this.metadata.entries).length,
|
|
129
|
+
freshEntries,
|
|
130
|
+
staleEntries,
|
|
131
|
+
};
|
|
132
|
+
}
|
|
133
|
+
/**
|
|
134
|
+
* Removes stale cache entries
|
|
135
|
+
*/
|
|
136
|
+
pruneStaleEntries() {
|
|
137
|
+
const now = Date.now();
|
|
138
|
+
const freshEntries = {};
|
|
139
|
+
for (const [key, entry] of Object.entries(this.metadata.entries)) {
|
|
140
|
+
const age = now - entry.timestamp;
|
|
141
|
+
if (age <= this.cacheDurationMs) {
|
|
142
|
+
freshEntries[key] = entry;
|
|
143
|
+
}
|
|
144
|
+
}
|
|
145
|
+
this.metadata.entries = freshEntries;
|
|
146
|
+
this.saveMetadata();
|
|
147
|
+
}
|
|
148
|
+
/**
|
|
149
|
+
* Generates a cache key from price query parameters
|
|
150
|
+
*/
|
|
151
|
+
getCacheKey(params) {
|
|
152
|
+
const filterStr = params.filters
|
|
153
|
+
.map((f) => `${f.field}:${f.value}`)
|
|
154
|
+
.sort()
|
|
155
|
+
.join('|');
|
|
156
|
+
return `${params.serviceCode}:${params.region}:${filterStr}`;
|
|
157
|
+
}
|
|
158
|
+
/**
|
|
159
|
+
* Ensures the cache directory exists
|
|
160
|
+
*/
|
|
161
|
+
ensureCacheDirectory() {
|
|
162
|
+
if (!fs.existsSync(this.cacheDir)) {
|
|
163
|
+
fs.mkdirSync(this.cacheDir, { recursive: true });
|
|
164
|
+
}
|
|
165
|
+
}
|
|
166
|
+
/**
|
|
167
|
+
* Loads cache metadata from disk
|
|
168
|
+
*/
|
|
169
|
+
loadMetadata() {
|
|
170
|
+
try {
|
|
171
|
+
if (fs.existsSync(this.metadataPath)) {
|
|
172
|
+
const data = fs.readFileSync(this.metadataPath, 'utf-8');
|
|
173
|
+
this.metadata = JSON.parse(data);
|
|
174
|
+
}
|
|
175
|
+
}
|
|
176
|
+
catch (error) {
|
|
177
|
+
// If metadata is corrupted, start fresh
|
|
178
|
+
this.metadata = { entries: {} };
|
|
179
|
+
}
|
|
180
|
+
}
|
|
181
|
+
/**
|
|
182
|
+
* Saves cache metadata to disk
|
|
183
|
+
*/
|
|
184
|
+
saveMetadata() {
|
|
185
|
+
try {
|
|
186
|
+
fs.writeFileSync(this.metadataPath, JSON.stringify(this.metadata, null, 2), 'utf-8');
|
|
187
|
+
}
|
|
188
|
+
catch (error) {
|
|
189
|
+
// Silently fail if we can't write cache - not critical
|
|
190
|
+
console.warn(`Failed to save cache metadata: ${error instanceof Error ? error.message : String(error)}`);
|
|
191
|
+
}
|
|
192
|
+
}
|
|
193
|
+
}
|
|
194
|
+
exports.CacheManager = CacheManager;
|
|
195
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"CacheManager.js","sourceRoot":"","sources":["../../src/pricing/CacheManager.ts"],"names":[],"mappings":"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","sourcesContent":["import * as fs from 'fs';\nimport * as path from 'path';\nimport { PriceQueryParams } from './types';\n\nexport interface CachedPriceEntry {\n  price: number;\n  timestamp: number;\n}\n\nexport interface CacheMetadata {\n  entries: Record<string, CachedPriceEntry>;\n}\n\n/**\n * Manages persistent caching of pricing data to reduce AWS API calls\n * and improve performance across pipeline runs.\n */\nexport class CacheManager {\n  private cacheDir: string;\n  private cacheDurationMs: number;\n  private metadata: CacheMetadata;\n  private metadataPath: string;\n\n  /**\n   * Creates a new CacheManager instance\n   * @param cacheDir Directory to store cache files (default: .cdk-cost-analyzer-cache)\n   * @param cacheDurationHours Duration in hours before cache entries expire (default: 24)\n   */\n  constructor(\n    cacheDir: string = '.cdk-cost-analyzer-cache',\n    cacheDurationHours: number = 24,\n  ) {\n    this.cacheDir = cacheDir;\n    this.cacheDurationMs = cacheDurationHours * 60 * 60 * 1000;\n    this.metadataPath = path.join(this.cacheDir, 'metadata.json');\n    this.metadata = { entries: {} };\n    this.ensureCacheDirectory();\n    this.loadMetadata();\n  }\n\n  /**\n   * Retrieves a cached price if it exists and is still fresh\n   * @param params Price query parameters\n   * @returns Cached price or null if not found or expired\n   */\n  getCachedPrice(params: PriceQueryParams): number | null {\n    const cacheKey = this.getCacheKey(params);\n    const entry = this.metadata.entries[cacheKey];\n\n    if (!entry) {\n      return null;\n    }\n\n    const now = Date.now();\n    const age = now - entry.timestamp;\n\n    if (age > this.cacheDurationMs) {\n      // Cache entry is stale\n      delete this.metadata.entries[cacheKey];\n      this.saveMetadata();\n      return null;\n    }\n\n    return entry.price;\n  }\n\n  /**\n   * Stores a price in the cache with current timestamp\n   * @param params Price query parameters\n   * @param price Price value to cache\n   */\n  setCachedPrice(params: PriceQueryParams, price: number): void {\n    const cacheKey = this.getCacheKey(params);\n    this.metadata.entries[cacheKey] = {\n      price,\n      timestamp: Date.now(),\n    };\n    this.saveMetadata();\n  }\n\n  /**\n   * Checks if a cached price exists and is still fresh\n   * @param params Price query parameters\n   * @returns true if fresh cache entry exists\n   */\n  hasFreshCache(params: PriceQueryParams): boolean {\n    return this.getCachedPrice(params) !== null;\n  }\n\n  /**\n   * Clears all cached entries\n   */\n  clearCache(): void {\n    this.metadata = { entries: {} };\n    this.saveMetadata();\n  }\n\n  /**\n   * Gets cache statistics\n   * @returns Object with cache statistics\n   */\n  getCacheStats(): {\n    totalEntries: number;\n    freshEntries: number;\n    staleEntries: number;\n  } {\n    const now = Date.now();\n    let freshEntries = 0;\n    let staleEntries = 0;\n\n    for (const entry of Object.values(this.metadata.entries)) {\n      const age = now - entry.timestamp;\n      if (age <= this.cacheDurationMs) {\n        freshEntries++;\n      } else {\n        staleEntries++;\n      }\n    }\n\n    return {\n      totalEntries: Object.keys(this.metadata.entries).length,\n      freshEntries,\n      staleEntries,\n    };\n  }\n\n  /**\n   * Removes stale cache entries\n   */\n  pruneStaleEntries(): void {\n    const now = Date.now();\n    const freshEntries: Record<string, CachedPriceEntry> = {};\n\n    for (const [key, entry] of Object.entries(this.metadata.entries)) {\n      const age = now - entry.timestamp;\n      if (age <= this.cacheDurationMs) {\n        freshEntries[key] = entry;\n      }\n    }\n\n    this.metadata.entries = freshEntries;\n    this.saveMetadata();\n  }\n\n  /**\n   * Generates a cache key from price query parameters\n   */\n  private getCacheKey(params: PriceQueryParams): string {\n    const filterStr = params.filters\n      .map((f) => `${f.field}:${f.value}`)\n      .sort()\n      .join('|');\n    return `${params.serviceCode}:${params.region}:${filterStr}`;\n  }\n\n  /**\n   * Ensures the cache directory exists\n   */\n  private ensureCacheDirectory(): void {\n    if (!fs.existsSync(this.cacheDir)) {\n      fs.mkdirSync(this.cacheDir, { recursive: true });\n    }\n  }\n\n  /**\n   * Loads cache metadata from disk\n   */\n  private loadMetadata(): void {\n    try {\n      if (fs.existsSync(this.metadataPath)) {\n        const data = fs.readFileSync(this.metadataPath, 'utf-8');\n        this.metadata = JSON.parse(data);\n      }\n    } catch (error) {\n      // If metadata is corrupted, start fresh\n      this.metadata = { entries: {} };\n    }\n  }\n\n  /**\n   * Saves cache metadata to disk\n   */\n  private saveMetadata(): void {\n    try {\n      fs.writeFileSync(\n        this.metadataPath,\n        JSON.stringify(this.metadata, null, 2),\n        'utf-8',\n      );\n    } catch (error) {\n      // Silently fail if we can't write cache - not critical\n      console.warn(\n        `Failed to save cache metadata: ${error instanceof Error ? error.message : String(error)}`,\n      );\n    }\n  }\n}\n"]}
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
import { PricingClient as AWSPricingClient } from '@aws-sdk/client-pricing';
|
|
2
|
+
import { CacheManager } from './CacheManager';
|
|
3
|
+
import { PricingClient as IPricingClient, PriceQueryParams } from './types';
|
|
4
|
+
export declare class PricingClient implements IPricingClient {
|
|
5
|
+
private cache;
|
|
6
|
+
private client;
|
|
7
|
+
private cacheManager?;
|
|
8
|
+
constructor(region?: string, cacheManager?: CacheManager, awsClient?: AWSPricingClient);
|
|
9
|
+
/**
|
|
10
|
+
* Clean up resources and connections
|
|
11
|
+
*/
|
|
12
|
+
destroy(): void;
|
|
13
|
+
getPrice(params: PriceQueryParams): Promise<number | null>;
|
|
14
|
+
private fetchPriceWithRetry;
|
|
15
|
+
private fetchPrice;
|
|
16
|
+
private getCacheKey;
|
|
17
|
+
}
|