cdk-cost-analyzer 0.1.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (228) hide show
  1. package/.cdk-cost-analyzer-cache/metadata.json +12 -0
  2. package/.gitlab-ci.yml +214 -0
  3. package/.husky/pre-commit +12 -0
  4. package/.kiro/hooks/accessibility-audit.kiro.hook +18 -0
  5. package/.kiro/hooks/api-schema-validation.kiro.hook +21 -0
  6. package/.kiro/hooks/auto-test-on-save.kiro.hook +19 -0
  7. package/.kiro/hooks/cdk-synth-on-change.kiro.hook +20 -0
  8. package/.kiro/hooks/code-coverage-check.kiro.hook +14 -0
  9. package/.kiro/hooks/commit-message-helper.kiro.hook +14 -0
  10. package/.kiro/hooks/dependency-update-check.kiro.hook +14 -0
  11. package/.kiro/hooks/env-file-validation.kiro.hook +18 -0
  12. package/.kiro/hooks/lint-and-format-on-save.kiro.hook +21 -0
  13. package/.kiro/hooks/mcp-config-validation.kiro.hook +17 -0
  14. package/.kiro/hooks/mcp-server-test.kiro.hook +14 -0
  15. package/.kiro/hooks/performance-analysis.kiro.hook +14 -0
  16. package/.kiro/hooks/readme-spell-check.kiro.hook +14 -0
  17. package/.kiro/hooks/security-scan-on-dependency-change.kiro.hook +21 -0
  18. package/.kiro/hooks/translation-update.kiro.hook +18 -0
  19. package/.kiro/hooks/update-documentation.kiro.hook +18 -0
  20. package/.kiro/settings/mcp.json +20 -0
  21. package/.kiro/specs/cdk-cost-analyzer/design.md +620 -0
  22. package/.kiro/specs/cdk-cost-analyzer/requirements.md +183 -0
  23. package/.kiro/specs/cdk-cost-analyzer/tasks.md +357 -0
  24. package/.kiro/specs/github-actions-ci/design.md +281 -0
  25. package/.kiro/specs/github-actions-ci/requirements.md +86 -0
  26. package/.kiro/specs/github-actions-ci/tasks.md +115 -0
  27. package/.kiro/specs/nlb-calculator-test-coverage/design.md +190 -0
  28. package/.kiro/specs/nlb-calculator-test-coverage/requirements.md +84 -0
  29. package/.kiro/specs/nlb-calculator-test-coverage/tasks.md +150 -0
  30. package/.kiro/specs/production-readiness/design.md +1213 -0
  31. package/.kiro/specs/production-readiness/requirements.md +312 -0
  32. package/.kiro/specs/production-readiness/tasks.md +269 -0
  33. package/.kiro/specs/repository-cleanup/design.md +283 -0
  34. package/.kiro/specs/repository-cleanup/requirements.md +74 -0
  35. package/.kiro/specs/repository-cleanup/tasks.md +64 -0
  36. package/.kiro/steering/aws-cli-best-practices.md +41 -0
  37. package/.kiro/steering/cdk-best-practices.md +49 -0
  38. package/.kiro/steering/development-standards.md +54 -0
  39. package/.kiro/steering/docker-best-practices.md +34 -0
  40. package/.kiro/steering/documentation-style.md +151 -0
  41. package/.kiro/steering/git-best-practices.md +37 -0
  42. package/.kiro/steering/mcp-best-practices.md +95 -0
  43. package/.kiro/steering/python-best-practices.md +48 -0
  44. package/.kiro/steering/react-best-practices.md +44 -0
  45. package/.kiro/steering/security-best-practices.md +41 -0
  46. package/.kiro/steering/testing-best-practices.md +59 -0
  47. package/.kiro/steering/typescript-best-practices.md +40 -0
  48. package/CHANGELOG.md +49 -0
  49. package/CONTRIBUTING.md +258 -0
  50. package/LICENSE +19 -0
  51. package/README.md +480 -0
  52. package/SECURITY.md +117 -0
  53. package/dist/api/index.d.ts +11 -0
  54. package/dist/api/index.js +65 -0
  55. package/dist/api/types.d.ts +15 -0
  56. package/dist/api/types.js +3 -0
  57. package/dist/cli/index.d.ts +2 -0
  58. package/dist/cli/index.js +262 -0
  59. package/dist/config/ConfigManager.d.ts +40 -0
  60. package/dist/config/ConfigManager.js +238 -0
  61. package/dist/config/index.d.ts +2 -0
  62. package/dist/config/index.js +19 -0
  63. package/dist/config/types.d.ts +72 -0
  64. package/dist/config/types.js +15 -0
  65. package/dist/diff/DiffEngine.d.ts +7 -0
  66. package/dist/diff/DiffEngine.js +73 -0
  67. package/dist/diff/index.d.ts +2 -0
  68. package/dist/diff/index.js +21 -0
  69. package/dist/diff/types.d.ts +20 -0
  70. package/dist/diff/types.js +3 -0
  71. package/dist/integrations/GitLabIntegration.d.ts +7 -0
  72. package/dist/integrations/GitLabIntegration.js +45 -0
  73. package/dist/integrations/index.d.ts +2 -0
  74. package/dist/integrations/index.js +21 -0
  75. package/dist/integrations/types.d.ts +11 -0
  76. package/dist/integrations/types.js +13 -0
  77. package/dist/parser/TemplateParser.d.ts +8 -0
  78. package/dist/parser/TemplateParser.js +75 -0
  79. package/dist/parser/index.d.ts +2 -0
  80. package/dist/parser/index.js +22 -0
  81. package/dist/parser/types.d.ts +30 -0
  82. package/dist/parser/types.js +3 -0
  83. package/dist/pipeline/PipelineOrchestrator.d.ts +23 -0
  84. package/dist/pipeline/PipelineOrchestrator.js +191 -0
  85. package/dist/pipeline/index.d.ts +2 -0
  86. package/dist/pipeline/index.js +19 -0
  87. package/dist/pipeline/types.d.ts +41 -0
  88. package/dist/pipeline/types.js +13 -0
  89. package/dist/pricing/CacheManager.d.ts +75 -0
  90. package/dist/pricing/CacheManager.js +195 -0
  91. package/dist/pricing/PricingClient.d.ts +17 -0
  92. package/dist/pricing/PricingClient.js +122 -0
  93. package/dist/pricing/PricingService.d.ts +16 -0
  94. package/dist/pricing/PricingService.js +149 -0
  95. package/dist/pricing/calculators/ALBCalculator.d.ts +16 -0
  96. package/dist/pricing/calculators/ALBCalculator.js +163 -0
  97. package/dist/pricing/calculators/APIGatewayCalculator.d.ts +10 -0
  98. package/dist/pricing/calculators/APIGatewayCalculator.js +177 -0
  99. package/dist/pricing/calculators/CloudFrontCalculator.d.ts +59 -0
  100. package/dist/pricing/calculators/CloudFrontCalculator.js +151 -0
  101. package/dist/pricing/calculators/DynamoDBCalculator.d.ts +9 -0
  102. package/dist/pricing/calculators/DynamoDBCalculator.js +146 -0
  103. package/dist/pricing/calculators/EC2Calculator.d.ts +7 -0
  104. package/dist/pricing/calculators/EC2Calculator.js +80 -0
  105. package/dist/pricing/calculators/ECSCalculator.d.ts +9 -0
  106. package/dist/pricing/calculators/ECSCalculator.js +116 -0
  107. package/dist/pricing/calculators/ElastiCacheCalculator.d.ts +8 -0
  108. package/dist/pricing/calculators/ElastiCacheCalculator.js +106 -0
  109. package/dist/pricing/calculators/LambdaCalculator.d.ts +13 -0
  110. package/dist/pricing/calculators/LambdaCalculator.js +111 -0
  111. package/dist/pricing/calculators/NLBCalculator.d.ts +16 -0
  112. package/dist/pricing/calculators/NLBCalculator.js +138 -0
  113. package/dist/pricing/calculators/NatGatewayCalculator.d.ts +12 -0
  114. package/dist/pricing/calculators/NatGatewayCalculator.js +116 -0
  115. package/dist/pricing/calculators/RDSCalculator.d.ts +9 -0
  116. package/dist/pricing/calculators/RDSCalculator.js +103 -0
  117. package/dist/pricing/calculators/S3Calculator.d.ts +8 -0
  118. package/dist/pricing/calculators/S3Calculator.js +68 -0
  119. package/dist/pricing/calculators/VPCEndpointCalculator.d.ts +12 -0
  120. package/dist/pricing/calculators/VPCEndpointCalculator.js +129 -0
  121. package/dist/pricing/index.d.ts +10 -0
  122. package/dist/pricing/index.js +37 -0
  123. package/dist/pricing/types.d.ts +53 -0
  124. package/dist/pricing/types.js +22 -0
  125. package/dist/releasetag.txt +1 -0
  126. package/dist/reporter/Reporter.d.ts +18 -0
  127. package/dist/reporter/Reporter.js +412 -0
  128. package/dist/reporter/index.d.ts +2 -0
  129. package/dist/reporter/index.js +21 -0
  130. package/dist/reporter/types.d.ts +72 -0
  131. package/dist/reporter/types.js +3 -0
  132. package/dist/synthesis/SynthesisOrchestrator.d.ts +26 -0
  133. package/dist/synthesis/SynthesisOrchestrator.js +243 -0
  134. package/dist/synthesis/index.d.ts +2 -0
  135. package/dist/synthesis/index.js +19 -0
  136. package/dist/synthesis/types.d.ts +17 -0
  137. package/dist/synthesis/types.js +13 -0
  138. package/dist/threshold/ThresholdEnforcer.d.ts +29 -0
  139. package/dist/threshold/ThresholdEnforcer.js +143 -0
  140. package/dist/threshold/index.d.ts +2 -0
  141. package/dist/threshold/index.js +19 -0
  142. package/dist/threshold/types.d.ts +15 -0
  143. package/dist/threshold/types.js +17 -0
  144. package/docs/CALCULATORS.md +820 -0
  145. package/docs/CI_CD.md +608 -0
  146. package/docs/CONFIGURATION.md +407 -0
  147. package/docs/DEVELOPMENT.md +387 -0
  148. package/docs/RELEASE.md +223 -0
  149. package/docs/TROUBLESHOOTING.md +847 -0
  150. package/examples/.cdk-cost-analyzer.yml +85 -0
  151. package/examples/.gitlab-ci.yml +125 -0
  152. package/examples/api-usage.js +26 -0
  153. package/examples/complex/base.json +16 -0
  154. package/examples/complex/target.json +29 -0
  155. package/examples/monorepo/.gitlab-ci.yml +251 -0
  156. package/examples/monorepo/README.md +341 -0
  157. package/examples/monorepo/package.json +27 -0
  158. package/examples/monorepo/packages/backend-infra/.cdk-cost-analyzer.yml +34 -0
  159. package/examples/monorepo/packages/backend-infra/bin/app.ts +16 -0
  160. package/examples/monorepo/packages/backend-infra/cdk.json +7 -0
  161. package/examples/monorepo/packages/backend-infra/lib/backend-stack.ts +128 -0
  162. package/examples/monorepo/packages/backend-infra/package.json +30 -0
  163. package/examples/monorepo/packages/backend-infra/tsconfig.json +11 -0
  164. package/examples/monorepo/packages/data-infra/.cdk-cost-analyzer.yml +38 -0
  165. package/examples/monorepo/packages/data-infra/bin/app.ts +16 -0
  166. package/examples/monorepo/packages/data-infra/cdk.json +7 -0
  167. package/examples/monorepo/packages/data-infra/lib/data-stack.ts +121 -0
  168. package/examples/monorepo/packages/data-infra/package.json +30 -0
  169. package/examples/monorepo/packages/data-infra/tsconfig.json +11 -0
  170. package/examples/monorepo/packages/frontend-infra/.cdk-cost-analyzer.yml +31 -0
  171. package/examples/monorepo/packages/frontend-infra/bin/app.ts +16 -0
  172. package/examples/monorepo/packages/frontend-infra/cdk.json +7 -0
  173. package/examples/monorepo/packages/frontend-infra/lib/frontend-stack.ts +60 -0
  174. package/examples/monorepo/packages/frontend-infra/package.json +30 -0
  175. package/examples/monorepo/packages/frontend-infra/tsconfig.json +11 -0
  176. package/examples/monorepo/tsconfig.json +35 -0
  177. package/examples/multi-stack/.cdk-cost-analyzer.yml +72 -0
  178. package/examples/multi-stack/.gitlab-ci.yml +184 -0
  179. package/examples/multi-stack/README.md +279 -0
  180. package/examples/multi-stack/bin/app.ts +36 -0
  181. package/examples/multi-stack/cdk.json +72 -0
  182. package/examples/multi-stack/lib/compute-stack.ts +128 -0
  183. package/examples/multi-stack/lib/networking-stack.ts +69 -0
  184. package/examples/multi-stack/lib/storage-stack.ts +141 -0
  185. package/examples/multi-stack/package-lock.json +4437 -0
  186. package/examples/multi-stack/package.json +42 -0
  187. package/examples/multi-stack/tsconfig.json +34 -0
  188. package/examples/simple/base.json +8 -0
  189. package/examples/simple/target.json +14 -0
  190. package/examples/single-stack/.NVP +0 -0
  191. package/examples/single-stack/.cdk-cost-analyzer.yml +52 -0
  192. package/examples/single-stack/.gitlab-ci.yml +126 -0
  193. package/examples/single-stack/README.md +184 -0
  194. package/examples/single-stack/UeK +0 -0
  195. package/examples/single-stack/bin/app.ts +16 -0
  196. package/examples/single-stack/cdk.json +72 -0
  197. package/examples/single-stack/lib/infrastructure-stack.ts +119 -0
  198. package/examples/single-stack/package-lock.json +4443 -0
  199. package/examples/single-stack/package.json +38 -0
  200. package/examples/single-stack/tsconfig.json +34 -0
  201. package/package.json +139 -0
  202. package/test-cdk-project/README-COMPUTE.md +141 -0
  203. package/test-cdk-project/README.md +95 -0
  204. package/test-cdk-project/app-with-compute.js +102 -0
  205. package/test-cdk-project/app.js +81 -0
  206. package/test-cdk-project/cdk-compute.json +3 -0
  207. package/test-cdk-project/cdk.context.json +7 -0
  208. package/test-cdk-project/cdk.json +3 -0
  209. package/test-cdk-project/cdk.out/TestStack.assets.json +21 -0
  210. package/test-cdk-project/cdk.out/TestStack.template.json +115 -0
  211. package/test-cdk-project/cdk.out/cdk.out +1 -0
  212. package/test-cdk-project/cdk.out/manifest.json +503 -0
  213. package/test-cdk-project/cdk.out/tree.json +1 -0
  214. package/test-cdk-project/cdk.out.base/TestStack.assets.json +21 -0
  215. package/test-cdk-project/cdk.out.base/TestStack.template.json +115 -0
  216. package/test-cdk-project/cdk.out.base/cdk.out +1 -0
  217. package/test-cdk-project/cdk.out.base/manifest.json +503 -0
  218. package/test-cdk-project/cdk.out.base/tree.json +1 -0
  219. package/test-cdk-project/cdk.out.target/TestStack.assets.json +21 -0
  220. package/test-cdk-project/cdk.out.target/TestStack.template.json +183 -0
  221. package/test-cdk-project/cdk.out.target/cdk.out +1 -0
  222. package/test-cdk-project/cdk.out.target/manifest.json +521 -0
  223. package/test-cdk-project/cdk.out.target/tree.json +1 -0
  224. package/test-cdk-project/package-lock.json +422 -0
  225. package/test-cdk-project/package.json +17 -0
  226. package/tools/workflows/README.md +102 -0
  227. package/tools/workflows/validate-workflows.js +109 -0
  228. package/tools/workflows/workflow-utils.ts +181 -0
@@ -0,0 +1,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,2 @@
1
+ export * from './PipelineOrchestrator';
2
+ export * from './types';
@@ -0,0 +1,19 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __exportStar = (this && this.__exportStar) || function(m, exports) {
14
+ for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
15
+ };
16
+ Object.defineProperty(exports, "__esModule", { value: true });
17
+ __exportStar(require("./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
+ }