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,80 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.EC2Calculator = void 0;
4
+ class EC2Calculator {
5
+ supports(resourceType) {
6
+ return resourceType === 'AWS::EC2::Instance';
7
+ }
8
+ async calculateCost(resource, region, pricingClient) {
9
+ const instanceType = resource.properties.InstanceType;
10
+ if (!instanceType) {
11
+ return {
12
+ amount: 0,
13
+ currency: 'USD',
14
+ confidence: 'unknown',
15
+ assumptions: ['Instance type not specified'],
16
+ };
17
+ }
18
+ try {
19
+ const hourlyRate = await pricingClient.getPrice({
20
+ serviceCode: 'AmazonEC2',
21
+ region: this.normalizeRegion(region),
22
+ filters: [
23
+ { field: 'instanceType', value: instanceType },
24
+ { field: 'operatingSystem', value: 'Linux' },
25
+ { field: 'tenancy', value: 'Shared' },
26
+ { field: 'preInstalledSw', value: 'NA' },
27
+ { field: 'capacitystatus', value: 'Used' },
28
+ ],
29
+ });
30
+ if (hourlyRate === null) {
31
+ return {
32
+ amount: 0,
33
+ currency: 'USD',
34
+ confidence: 'unknown',
35
+ assumptions: [`Pricing data not available for instance type ${instanceType} in region ${region}`],
36
+ };
37
+ }
38
+ const monthlyHours = 730;
39
+ const monthlyCost = hourlyRate * monthlyHours;
40
+ return {
41
+ amount: monthlyCost,
42
+ currency: 'USD',
43
+ confidence: 'high',
44
+ assumptions: [
45
+ `Assumes ${monthlyHours} hours per month (24/7 operation)`,
46
+ 'Assumes Linux OS, shared tenancy, on-demand pricing',
47
+ ],
48
+ };
49
+ }
50
+ catch (error) {
51
+ return {
52
+ amount: 0,
53
+ currency: 'USD',
54
+ confidence: 'unknown',
55
+ assumptions: [`Failed to fetch pricing: ${error instanceof Error ? error.message : String(error)}`],
56
+ };
57
+ }
58
+ }
59
+ normalizeRegion(region) {
60
+ const regionMap = {
61
+ 'us-east-1': 'US East (N. Virginia)',
62
+ 'us-east-2': 'US East (Ohio)',
63
+ 'us-west-1': 'US West (N. California)',
64
+ 'us-west-2': 'US West (Oregon)',
65
+ 'eu-west-1': 'EU (Ireland)',
66
+ 'eu-west-2': 'EU (London)',
67
+ 'eu-west-3': 'EU (Paris)',
68
+ 'eu-central-1': 'EU (Frankfurt)',
69
+ 'eu-north-1': 'EU (Stockholm)',
70
+ 'ap-south-1': 'Asia Pacific (Mumbai)',
71
+ 'ap-southeast-1': 'Asia Pacific (Singapore)',
72
+ 'ap-southeast-2': 'Asia Pacific (Sydney)',
73
+ 'ap-northeast-1': 'Asia Pacific (Tokyo)',
74
+ 'ap-northeast-2': 'Asia Pacific (Seoul)',
75
+ };
76
+ return regionMap[region] || region;
77
+ }
78
+ }
79
+ exports.EC2Calculator = EC2Calculator;
80
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiRUMyQ2FsY3VsYXRvci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9wcmljaW5nL2NhbGN1bGF0b3JzL0VDMkNhbGN1bGF0b3IudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBR0EsTUFBYSxhQUFhO0lBQ3hCLFFBQVEsQ0FBQyxZQUFvQjtRQUMzQixPQUFPLFlBQVksS0FBSyxvQkFBb0IsQ0FBQztJQUMvQyxDQUFDO0lBRUQsS0FBSyxDQUFDLGFBQWEsQ0FDakIsUUFBd0IsRUFDeEIsTUFBYyxFQUNkLGFBQTRCO1FBRTVCLE1BQU0sWUFBWSxHQUFHLFFBQVEsQ0FBQyxVQUFVLENBQUMsWUFBc0IsQ0FBQztRQUVoRSxJQUFJLENBQUMsWUFBWSxFQUFFLENBQUM7WUFDbEIsT0FBTztnQkFDTCxNQUFNLEVBQUUsQ0FBQztnQkFDVCxRQUFRLEVBQUUsS0FBSztnQkFDZixVQUFVLEVBQUUsU0FBUztnQkFDckIsV0FBVyxFQUFFLENBQUMsNkJBQTZCLENBQUM7YUFDN0MsQ0FBQztRQUNKLENBQUM7UUFFRCxJQUFJLENBQUM7WUFDSCxNQUFNLFVBQVUsR0FBRyxNQUFNLGFBQWEsQ0FBQyxRQUFRLENBQUM7Z0JBQzlDLFdBQVcsRUFBRSxXQUFXO2dCQUN4QixNQUFNLEVBQUUsSUFBSSxDQUFDLGVBQWUsQ0FBQyxNQUFNLENBQUM7Z0JBQ3BDLE9BQU8sRUFBRTtvQkFDUCxFQUFFLEtBQUssRUFBRSxjQUFjLEVBQUUsS0FBSyxFQUFFLFlBQVksRUFBRTtvQkFDOUMsRUFBRSxLQUFLLEVBQUUsaUJBQWlCLEVBQUUsS0FBSyxFQUFFLE9BQU8sRUFBRTtvQkFDNUMsRUFBRSxLQUFLLEVBQUUsU0FBUyxFQUFFLEtBQUssRUFBRSxRQUFRLEVBQUU7b0JBQ3JDLEVBQUUsS0FBSyxFQUFFLGdCQUFnQixFQUFFLEtBQUssRUFBRSxJQUFJLEVBQUU7b0JBQ3hDLEVBQUUsS0FBSyxFQUFFLGdCQUFnQixFQUFFLEtBQUssRUFBRSxNQUFNLEVBQUU7aUJBQzNDO2FBQ0YsQ0FBQyxDQUFDO1lBRUgsSUFBSSxVQUFVLEtBQUssSUFBSSxFQUFFLENBQUM7Z0JBQ3hCLE9BQU87b0JBQ0wsTUFBTSxFQUFFLENBQUM7b0JBQ1QsUUFBUSxFQUFFLEtBQUs7b0JBQ2YsVUFBVSxFQUFFLFNBQVM7b0JBQ3JCLFdBQVcsRUFBRSxDQUFDLGdEQUFnRCxZQUFZLGNBQWMsTUFBTSxFQUFFLENBQUM7aUJBQ2xHLENBQUM7WUFDSixDQUFDO1lBRUQsTUFBTSxZQUFZLEdBQUcsR0FBRyxDQUFDO1lBQ3pCLE1BQU0sV0FBVyxHQUFHLFVBQVUsR0FBRyxZQUFZLENBQUM7WUFFOUMsT0FBTztnQkFDTCxNQUFNLEVBQUUsV0FBVztnQkFDbkIsUUFBUSxFQUFFLEtBQUs7Z0JBQ2YsVUFBVSxFQUFFLE1BQU07Z0JBQ2xCLFdBQVcsRUFBRTtvQkFDWCxXQUFXLFlBQVksbUNBQW1DO29CQUMxRCxxREFBcUQ7aUJBQ3REO2FBQ0YsQ0FBQztRQUNKLENBQUM7UUFBQyxPQUFPLEtBQUssRUFBRSxDQUFDO1lBQ2YsT0FBTztnQkFDTCxNQUFNLEVBQUUsQ0FBQztnQkFDVCxRQUFRLEVBQUUsS0FBSztnQkFDZixVQUFVLEVBQUUsU0FBUztnQkFDckIsV0FBVyxFQUFFLENBQUMsNEJBQTRCLEtBQUssWUFBWSxLQUFLLENBQUMsQ0FBQyxDQUFDLEtBQUssQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFDLE1BQU0sQ0FBQyxLQUFLLENBQUMsRUFBRSxDQUFDO2FBQ3BHLENBQUM7UUFDSixDQUFDO0lBQ0gsQ0FBQztJQUVPLGVBQWUsQ0FBQyxNQUFjO1FBQ3BDLE1BQU0sU0FBUyxHQUEyQjtZQUN4QyxXQUFXLEVBQUUsdUJBQXVCO1lBQ3BDLFdBQVcsRUFBRSxnQkFBZ0I7WUFDN0IsV0FBVyxFQUFFLHlCQUF5QjtZQUN0QyxXQUFXLEVBQUUsa0JBQWtCO1lBQy9CLFdBQVcsRUFBRSxjQUFjO1lBQzNCLFdBQVcsRUFBRSxhQUFhO1lBQzFCLFdBQVcsRUFBRSxZQUFZO1lBQ3pCLGNBQWMsRUFBRSxnQkFBZ0I7WUFDaEMsWUFBWSxFQUFFLGdCQUFnQjtZQUM5QixZQUFZLEVBQUUsdUJBQXVCO1lBQ3JDLGdCQUFnQixFQUFFLDBCQUEwQjtZQUM1QyxnQkFBZ0IsRUFBRSx1QkFBdUI7WUFDekMsZ0JBQWdCLEVBQUUsc0JBQXNCO1lBQ3hDLGdCQUFnQixFQUFFLHNCQUFzQjtTQUN6QyxDQUFDO1FBRUYsT0FBTyxTQUFTLENBQUMsTUFBTSxDQUFDLElBQUksTUFBTSxDQUFDO0lBQ3JDLENBQUM7Q0FDRjtBQXJGRCxzQ0FxRkMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBSZXNvdXJjZVdpdGhJZCB9IGZyb20gJy4uLy4uL2RpZmYvdHlwZXMnO1xuaW1wb3J0IHsgUmVzb3VyY2VDb3N0Q2FsY3VsYXRvciwgTW9udGhseUNvc3QsIFByaWNpbmdDbGllbnQgfSBmcm9tICcuLi90eXBlcyc7XG5cbmV4cG9ydCBjbGFzcyBFQzJDYWxjdWxhdG9yIGltcGxlbWVudHMgUmVzb3VyY2VDb3N0Q2FsY3VsYXRvciB7XG4gIHN1cHBvcnRzKHJlc291cmNlVHlwZTogc3RyaW5nKTogYm9vbGVhbiB7XG4gICAgcmV0dXJuIHJlc291cmNlVHlwZSA9PT0gJ0FXUzo6RUMyOjpJbnN0YW5jZSc7XG4gIH1cblxuICBhc3luYyBjYWxjdWxhdGVDb3N0KFxuICAgIHJlc291cmNlOiBSZXNvdXJjZVdpdGhJZCxcbiAgICByZWdpb246IHN0cmluZyxcbiAgICBwcmljaW5nQ2xpZW50OiBQcmljaW5nQ2xpZW50LFxuICApOiBQcm9taXNlPE1vbnRobHlDb3N0PiB7XG4gICAgY29uc3QgaW5zdGFuY2VUeXBlID0gcmVzb3VyY2UucHJvcGVydGllcy5JbnN0YW5jZVR5cGUgYXMgc3RyaW5nO1xuXG4gICAgaWYgKCFpbnN0YW5jZVR5cGUpIHtcbiAgICAgIHJldHVybiB7XG4gICAgICAgIGFtb3VudDogMCxcbiAgICAgICAgY3VycmVuY3k6ICdVU0QnLFxuICAgICAgICBjb25maWRlbmNlOiAndW5rbm93bicsXG4gICAgICAgIGFzc3VtcHRpb25zOiBbJ0luc3RhbmNlIHR5cGUgbm90IHNwZWNpZmllZCddLFxuICAgICAgfTtcbiAgICB9XG5cbiAgICB0cnkge1xuICAgICAgY29uc3QgaG91cmx5UmF0ZSA9IGF3YWl0IHByaWNpbmdDbGllbnQuZ2V0UHJpY2Uoe1xuICAgICAgICBzZXJ2aWNlQ29kZTogJ0FtYXpvbkVDMicsXG4gICAgICAgIHJlZ2lvbjogdGhpcy5ub3JtYWxpemVSZWdpb24ocmVnaW9uKSxcbiAgICAgICAgZmlsdGVyczogW1xuICAgICAgICAgIHsgZmllbGQ6ICdpbnN0YW5jZVR5cGUnLCB2YWx1ZTogaW5zdGFuY2VUeXBlIH0sXG4gICAgICAgICAgeyBmaWVsZDogJ29wZXJhdGluZ1N5c3RlbScsIHZhbHVlOiAnTGludXgnIH0sXG4gICAgICAgICAgeyBmaWVsZDogJ3RlbmFuY3knLCB2YWx1ZTogJ1NoYXJlZCcgfSxcbiAgICAgICAgICB7IGZpZWxkOiAncHJlSW5zdGFsbGVkU3cnLCB2YWx1ZTogJ05BJyB9LFxuICAgICAgICAgIHsgZmllbGQ6ICdjYXBhY2l0eXN0YXR1cycsIHZhbHVlOiAnVXNlZCcgfSxcbiAgICAgICAgXSxcbiAgICAgIH0pO1xuXG4gICAgICBpZiAoaG91cmx5UmF0ZSA9PT0gbnVsbCkge1xuICAgICAgICByZXR1cm4ge1xuICAgICAgICAgIGFtb3VudDogMCxcbiAgICAgICAgICBjdXJyZW5jeTogJ1VTRCcsXG4gICAgICAgICAgY29uZmlkZW5jZTogJ3Vua25vd24nLFxuICAgICAgICAgIGFzc3VtcHRpb25zOiBbYFByaWNpbmcgZGF0YSBub3QgYXZhaWxhYmxlIGZvciBpbnN0YW5jZSB0eXBlICR7aW5zdGFuY2VUeXBlfSBpbiByZWdpb24gJHtyZWdpb259YF0sXG4gICAgICAgIH07XG4gICAgICB9XG5cbiAgICAgIGNvbnN0IG1vbnRobHlIb3VycyA9IDczMDtcbiAgICAgIGNvbnN0IG1vbnRobHlDb3N0ID0gaG91cmx5UmF0ZSAqIG1vbnRobHlIb3VycztcblxuICAgICAgcmV0dXJuIHtcbiAgICAgICAgYW1vdW50OiBtb250aGx5Q29zdCxcbiAgICAgICAgY3VycmVuY3k6ICdVU0QnLFxuICAgICAgICBjb25maWRlbmNlOiAnaGlnaCcsXG4gICAgICAgIGFzc3VtcHRpb25zOiBbXG4gICAgICAgICAgYEFzc3VtZXMgJHttb250aGx5SG91cnN9IGhvdXJzIHBlciBtb250aCAoMjQvNyBvcGVyYXRpb24pYCxcbiAgICAgICAgICAnQXNzdW1lcyBMaW51eCBPUywgc2hhcmVkIHRlbmFuY3ksIG9uLWRlbWFuZCBwcmljaW5nJyxcbiAgICAgICAgXSxcbiAgICAgIH07XG4gICAgfSBjYXRjaCAoZXJyb3IpIHtcbiAgICAgIHJldHVybiB7XG4gICAgICAgIGFtb3VudDogMCxcbiAgICAgICAgY3VycmVuY3k6ICdVU0QnLFxuICAgICAgICBjb25maWRlbmNlOiAndW5rbm93bicsXG4gICAgICAgIGFzc3VtcHRpb25zOiBbYEZhaWxlZCB0byBmZXRjaCBwcmljaW5nOiAke2Vycm9yIGluc3RhbmNlb2YgRXJyb3IgPyBlcnJvci5tZXNzYWdlIDogU3RyaW5nKGVycm9yKX1gXSxcbiAgICAgIH07XG4gICAgfVxuICB9XG5cbiAgcHJpdmF0ZSBub3JtYWxpemVSZWdpb24ocmVnaW9uOiBzdHJpbmcpOiBzdHJpbmcge1xuICAgIGNvbnN0IHJlZ2lvbk1hcDogUmVjb3JkPHN0cmluZywgc3RyaW5nPiA9IHtcbiAgICAgICd1cy1lYXN0LTEnOiAnVVMgRWFzdCAoTi4gVmlyZ2luaWEpJyxcbiAgICAgICd1cy1lYXN0LTInOiAnVVMgRWFzdCAoT2hpbyknLFxuICAgICAgJ3VzLXdlc3QtMSc6ICdVUyBXZXN0IChOLiBDYWxpZm9ybmlhKScsXG4gICAgICAndXMtd2VzdC0yJzogJ1VTIFdlc3QgKE9yZWdvbiknLFxuICAgICAgJ2V1LXdlc3QtMSc6ICdFVSAoSXJlbGFuZCknLFxuICAgICAgJ2V1LXdlc3QtMic6ICdFVSAoTG9uZG9uKScsXG4gICAgICAnZXUtd2VzdC0zJzogJ0VVIChQYXJpcyknLFxuICAgICAgJ2V1LWNlbnRyYWwtMSc6ICdFVSAoRnJhbmtmdXJ0KScsXG4gICAgICAnZXUtbm9ydGgtMSc6ICdFVSAoU3RvY2tob2xtKScsXG4gICAgICAnYXAtc291dGgtMSc6ICdBc2lhIFBhY2lmaWMgKE11bWJhaSknLFxuICAgICAgJ2FwLXNvdXRoZWFzdC0xJzogJ0FzaWEgUGFjaWZpYyAoU2luZ2Fwb3JlKScsXG4gICAgICAnYXAtc291dGhlYXN0LTInOiAnQXNpYSBQYWNpZmljIChTeWRuZXkpJyxcbiAgICAgICdhcC1ub3J0aGVhc3QtMSc6ICdBc2lhIFBhY2lmaWMgKFRva3lvKScsXG4gICAgICAnYXAtbm9ydGhlYXN0LTInOiAnQXNpYSBQYWNpZmljIChTZW91bCknLFxuICAgIH07XG5cbiAgICByZXR1cm4gcmVnaW9uTWFwW3JlZ2lvbl0gfHwgcmVnaW9uO1xuICB9XG59XG4iXX0=
@@ -0,0 +1,9 @@
1
+ import { ResourceWithId } from '../../diff/types';
2
+ import { ResourceCostCalculator, MonthlyCost, PricingClient } from '../types';
3
+ export declare class ECSCalculator implements ResourceCostCalculator {
4
+ supports(resourceType: string): boolean;
5
+ calculateCost(resource: ResourceWithId, region: string, pricingClient: PricingClient): Promise<MonthlyCost>;
6
+ private calculateFargateCost;
7
+ private calculateEC2Cost;
8
+ private normalizeRegion;
9
+ }
@@ -0,0 +1,116 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.ECSCalculator = void 0;
4
+ class ECSCalculator {
5
+ supports(resourceType) {
6
+ return resourceType === 'AWS::ECS::Service';
7
+ }
8
+ async calculateCost(resource, region, pricingClient) {
9
+ const desiredCount = resource.properties.DesiredCount || 1;
10
+ const launchType = resource.properties.LaunchType || 'FARGATE';
11
+ if (launchType === 'FARGATE') {
12
+ return this.calculateFargateCost(resource, desiredCount, region, pricingClient);
13
+ }
14
+ else if (launchType === 'EC2') {
15
+ return this.calculateEC2Cost(desiredCount);
16
+ }
17
+ else {
18
+ return {
19
+ amount: 0,
20
+ currency: 'USD',
21
+ confidence: 'unknown',
22
+ assumptions: [`Unsupported launch type: ${launchType}`],
23
+ };
24
+ }
25
+ }
26
+ async calculateFargateCost(_resource, desiredCount, region, pricingClient) {
27
+ try {
28
+ // Default assumptions for Fargate task
29
+ const assumedVCpu = 0.25; // 0.25 vCPU
30
+ const assumedMemoryGB = 0.5; // 0.5 GB
31
+ const vCpuCostPerHour = await pricingClient.getPrice({
32
+ serviceCode: 'AmazonECS',
33
+ region: this.normalizeRegion(region),
34
+ filters: [
35
+ { field: 'productFamily', value: 'Compute' },
36
+ { field: 'usagetype', value: `${region}-Fargate-vCPU-Hours:perCPU` },
37
+ ],
38
+ });
39
+ const memoryCostPerHour = await pricingClient.getPrice({
40
+ serviceCode: 'AmazonECS',
41
+ region: this.normalizeRegion(region),
42
+ filters: [
43
+ { field: 'productFamily', value: 'Compute' },
44
+ { field: 'usagetype', value: `${region}-Fargate-GB-Hours:perGB` },
45
+ ],
46
+ });
47
+ if (vCpuCostPerHour === null || memoryCostPerHour === null) {
48
+ return {
49
+ amount: 0,
50
+ currency: 'USD',
51
+ confidence: 'unknown',
52
+ assumptions: ['Pricing data not available for ECS Fargate'],
53
+ };
54
+ }
55
+ const hoursPerMonth = 730;
56
+ const vCpuCost = assumedVCpu * vCpuCostPerHour * hoursPerMonth * desiredCount;
57
+ const memoryCost = assumedMemoryGB * memoryCostPerHour * hoursPerMonth * desiredCount;
58
+ const monthlyCost = vCpuCost + memoryCost;
59
+ return {
60
+ amount: monthlyCost,
61
+ currency: 'USD',
62
+ confidence: 'medium',
63
+ assumptions: [
64
+ `${desiredCount} task(s) running`,
65
+ `Assumes ${assumedVCpu} vCPU per task`,
66
+ `Assumes ${assumedMemoryGB} GB memory per task`,
67
+ `Assumes ${hoursPerMonth} hours per month (24/7 operation)`,
68
+ 'Fargate launch type',
69
+ 'Does not include data transfer or storage costs',
70
+ ],
71
+ };
72
+ }
73
+ catch (error) {
74
+ return {
75
+ amount: 0,
76
+ currency: 'USD',
77
+ confidence: 'unknown',
78
+ assumptions: [`Failed to fetch pricing: ${error instanceof Error ? error.message : String(error)}`],
79
+ };
80
+ }
81
+ }
82
+ calculateEC2Cost(desiredCount) {
83
+ return {
84
+ amount: 0,
85
+ currency: 'USD',
86
+ confidence: 'low',
87
+ assumptions: [
88
+ `${desiredCount} task(s) running on EC2 launch type`,
89
+ 'EC2 launch type costs depend on underlying EC2 instances',
90
+ 'Refer to EC2 instance costs for actual pricing',
91
+ 'Does not include ECS task-specific costs (minimal for EC2 launch type)',
92
+ ],
93
+ };
94
+ }
95
+ normalizeRegion(region) {
96
+ const regionMap = {
97
+ 'us-east-1': 'US East (N. Virginia)',
98
+ 'us-east-2': 'US East (Ohio)',
99
+ 'us-west-1': 'US West (N. California)',
100
+ 'us-west-2': 'US West (Oregon)',
101
+ 'eu-west-1': 'EU (Ireland)',
102
+ 'eu-west-2': 'EU (London)',
103
+ 'eu-west-3': 'EU (Paris)',
104
+ 'eu-central-1': 'EU (Frankfurt)',
105
+ 'eu-north-1': 'EU (Stockholm)',
106
+ 'ap-south-1': 'Asia Pacific (Mumbai)',
107
+ 'ap-southeast-1': 'Asia Pacific (Singapore)',
108
+ 'ap-southeast-2': 'Asia Pacific (Sydney)',
109
+ 'ap-northeast-1': 'Asia Pacific (Tokyo)',
110
+ 'ap-northeast-2': 'Asia Pacific (Seoul)',
111
+ };
112
+ return regionMap[region] || region;
113
+ }
114
+ }
115
+ exports.ECSCalculator = ECSCalculator;
116
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ECSCalculator.js","sourceRoot":"","sources":["../../../src/pricing/calculators/ECSCalculator.ts"],"names":[],"mappings":"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","sourcesContent":["import { ResourceWithId } from '../../diff/types';\nimport { ResourceCostCalculator, MonthlyCost, PricingClient } from '../types';\n\nexport class ECSCalculator implements ResourceCostCalculator {\n  supports(resourceType: string): boolean {\n    return resourceType === 'AWS::ECS::Service';\n  }\n\n  async calculateCost(\n    resource: ResourceWithId,\n    region: string,\n    pricingClient: PricingClient,\n  ): Promise<MonthlyCost> {\n    const desiredCount = (resource.properties.DesiredCount as number) || 1;\n    const launchType = (resource.properties.LaunchType as string) || 'FARGATE';\n\n    if (launchType === 'FARGATE') {\n      return this.calculateFargateCost(resource, desiredCount, region, pricingClient);\n    } else if (launchType === 'EC2') {\n      return this.calculateEC2Cost(desiredCount);\n    } else {\n      return {\n        amount: 0,\n        currency: 'USD',\n        confidence: 'unknown',\n        assumptions: [`Unsupported launch type: ${launchType}`],\n      };\n    }\n  }\n\n  private async calculateFargateCost(\n    _resource: ResourceWithId,\n    desiredCount: number,\n    region: string,\n    pricingClient: PricingClient,\n  ): Promise<MonthlyCost> {\n    try {\n      // Default assumptions for Fargate task\n      const assumedVCpu = 0.25; // 0.25 vCPU\n      const assumedMemoryGB = 0.5; // 0.5 GB\n\n      const vCpuCostPerHour = await pricingClient.getPrice({\n        serviceCode: 'AmazonECS',\n        region: this.normalizeRegion(region),\n        filters: [\n          { field: 'productFamily', value: 'Compute' },\n          { field: 'usagetype', value: `${region}-Fargate-vCPU-Hours:perCPU` },\n        ],\n      });\n\n      const memoryCostPerHour = await pricingClient.getPrice({\n        serviceCode: 'AmazonECS',\n        region: this.normalizeRegion(region),\n        filters: [\n          { field: 'productFamily', value: 'Compute' },\n          { field: 'usagetype', value: `${region}-Fargate-GB-Hours:perGB` },\n        ],\n      });\n\n      if (vCpuCostPerHour === null || memoryCostPerHour === null) {\n        return {\n          amount: 0,\n          currency: 'USD',\n          confidence: 'unknown',\n          assumptions: ['Pricing data not available for ECS Fargate'],\n        };\n      }\n\n      const hoursPerMonth = 730;\n      const vCpuCost = assumedVCpu * vCpuCostPerHour * hoursPerMonth * desiredCount;\n      const memoryCost = assumedMemoryGB * memoryCostPerHour * hoursPerMonth * desiredCount;\n      const monthlyCost = vCpuCost + memoryCost;\n\n      return {\n        amount: monthlyCost,\n        currency: 'USD',\n        confidence: 'medium',\n        assumptions: [\n          `${desiredCount} task(s) running`,\n          `Assumes ${assumedVCpu} vCPU per task`,\n          `Assumes ${assumedMemoryGB} GB memory per task`,\n          `Assumes ${hoursPerMonth} hours per month (24/7 operation)`,\n          'Fargate launch type',\n          'Does not include data transfer or storage costs',\n        ],\n      };\n    } catch (error) {\n      return {\n        amount: 0,\n        currency: 'USD',\n        confidence: 'unknown',\n        assumptions: [`Failed to fetch pricing: ${error instanceof Error ? error.message : String(error)}`],\n      };\n    }\n  }\n\n  private calculateEC2Cost(desiredCount: number): MonthlyCost {\n    return {\n      amount: 0,\n      currency: 'USD',\n      confidence: 'low',\n      assumptions: [\n        `${desiredCount} task(s) running on EC2 launch type`,\n        'EC2 launch type costs depend on underlying EC2 instances',\n        'Refer to EC2 instance costs for actual pricing',\n        'Does not include ECS task-specific costs (minimal for EC2 launch type)',\n      ],\n    };\n  }\n\n  private normalizeRegion(region: string): string {\n    const regionMap: Record<string, string> = {\n      'us-east-1': 'US East (N. Virginia)',\n      'us-east-2': 'US East (Ohio)',\n      'us-west-1': 'US West (N. California)',\n      'us-west-2': 'US West (Oregon)',\n      'eu-west-1': 'EU (Ireland)',\n      'eu-west-2': 'EU (London)',\n      'eu-west-3': 'EU (Paris)',\n      'eu-central-1': 'EU (Frankfurt)',\n      'eu-north-1': 'EU (Stockholm)',\n      'ap-south-1': 'Asia Pacific (Mumbai)',\n      'ap-southeast-1': 'Asia Pacific (Singapore)',\n      'ap-southeast-2': 'Asia Pacific (Sydney)',\n      'ap-northeast-1': 'Asia Pacific (Tokyo)',\n      'ap-northeast-2': 'Asia Pacific (Seoul)',\n    };\n\n    return regionMap[region] || region;\n  }\n}\n"]}
@@ -0,0 +1,8 @@
1
+ import { ResourceWithId } from '../../diff/types';
2
+ import { ResourceCostCalculator, MonthlyCost, PricingClient } from '../types';
3
+ export declare class ElastiCacheCalculator implements ResourceCostCalculator {
4
+ supports(resourceType: string): boolean;
5
+ calculateCost(resource: ResourceWithId, region: string, pricingClient: PricingClient): Promise<MonthlyCost>;
6
+ private normalizeEngine;
7
+ private normalizeRegion;
8
+ }
@@ -0,0 +1,106 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.ElastiCacheCalculator = void 0;
4
+ class ElastiCacheCalculator {
5
+ supports(resourceType) {
6
+ return resourceType === 'AWS::ElastiCache::CacheCluster';
7
+ }
8
+ async calculateCost(resource, region, pricingClient) {
9
+ const cacheNodeType = resource.properties.CacheNodeType;
10
+ const engine = resource.properties.Engine;
11
+ const numCacheNodes = resource.properties.NumCacheNodes || 1;
12
+ const azMode = resource.properties.AZMode;
13
+ if (!cacheNodeType || !engine) {
14
+ return {
15
+ amount: 0,
16
+ currency: 'USD',
17
+ confidence: 'unknown',
18
+ assumptions: ['Cache node type or engine not specified'],
19
+ };
20
+ }
21
+ try {
22
+ // Query AWS Pricing API for node hourly rates
23
+ const hourlyRate = await pricingClient.getPrice({
24
+ serviceCode: 'AmazonElastiCache',
25
+ region: this.normalizeRegion(region),
26
+ filters: [
27
+ { field: 'instanceType', value: cacheNodeType },
28
+ { field: 'cacheEngine', value: this.normalizeEngine(engine) },
29
+ ],
30
+ });
31
+ if (hourlyRate === null) {
32
+ return {
33
+ amount: 0,
34
+ currency: 'USD',
35
+ confidence: 'unknown',
36
+ assumptions: [
37
+ `Pricing data not available for node type ${cacheNodeType} with engine ${engine} in region ${region}`,
38
+ ],
39
+ };
40
+ }
41
+ const monthlyHours = 730;
42
+ const perNodeCost = hourlyRate * monthlyHours;
43
+ let totalCost = perNodeCost * numCacheNodes;
44
+ const assumptions = [
45
+ `Assumes ${monthlyHours} hours per month (24/7 operation)`,
46
+ `Node type: ${cacheNodeType}`,
47
+ `Engine: ${engine}`,
48
+ `Number of cache nodes: ${numCacheNodes}`,
49
+ `Per-node monthly cost: $${perNodeCost.toFixed(2)}`,
50
+ ];
51
+ // Account for multi-AZ replica costs when configured
52
+ if (azMode === 'cross-az') {
53
+ // Multi-AZ adds replica nodes (typically doubles the cost)
54
+ totalCost *= 2;
55
+ assumptions.push('Multi-AZ deployment with replica nodes (cost doubled)');
56
+ }
57
+ else {
58
+ assumptions.push('Single-AZ deployment');
59
+ }
60
+ return {
61
+ amount: totalCost,
62
+ currency: 'USD',
63
+ confidence: 'high',
64
+ assumptions,
65
+ };
66
+ }
67
+ catch (error) {
68
+ return {
69
+ amount: 0,
70
+ currency: 'USD',
71
+ confidence: 'unknown',
72
+ assumptions: [
73
+ `Failed to fetch pricing: ${error instanceof Error ? error.message : String(error)}`,
74
+ ],
75
+ };
76
+ }
77
+ }
78
+ normalizeEngine(engine) {
79
+ const engineMap = {
80
+ redis: 'Redis',
81
+ memcached: 'Memcached',
82
+ };
83
+ return engineMap[engine.toLowerCase()] || engine;
84
+ }
85
+ normalizeRegion(region) {
86
+ const regionMap = {
87
+ 'us-east-1': 'US East (N. Virginia)',
88
+ 'us-east-2': 'US East (Ohio)',
89
+ 'us-west-1': 'US West (N. California)',
90
+ 'us-west-2': 'US West (Oregon)',
91
+ 'eu-west-1': 'EU (Ireland)',
92
+ 'eu-west-2': 'EU (London)',
93
+ 'eu-west-3': 'EU (Paris)',
94
+ 'eu-central-1': 'EU (Frankfurt)',
95
+ 'eu-north-1': 'EU (Stockholm)',
96
+ 'ap-south-1': 'Asia Pacific (Mumbai)',
97
+ 'ap-southeast-1': 'Asia Pacific (Singapore)',
98
+ 'ap-southeast-2': 'Asia Pacific (Sydney)',
99
+ 'ap-northeast-1': 'Asia Pacific (Tokyo)',
100
+ 'ap-northeast-2': 'Asia Pacific (Seoul)',
101
+ };
102
+ return regionMap[region] || region;
103
+ }
104
+ }
105
+ exports.ElastiCacheCalculator = ElastiCacheCalculator;
106
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ElastiCacheCalculator.js","sourceRoot":"","sources":["../../../src/pricing/calculators/ElastiCacheCalculator.ts"],"names":[],"mappings":"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","sourcesContent":["import { ResourceWithId } from '../../diff/types';\nimport { ResourceCostCalculator, MonthlyCost, PricingClient } from '../types';\n\nexport class ElastiCacheCalculator implements ResourceCostCalculator {\n  supports(resourceType: string): boolean {\n    return resourceType === 'AWS::ElastiCache::CacheCluster';\n  }\n\n  async calculateCost(\n    resource: ResourceWithId,\n    region: string,\n    pricingClient: PricingClient,\n  ): Promise<MonthlyCost> {\n    const cacheNodeType = resource.properties.CacheNodeType as string;\n    const engine = resource.properties.Engine as string;\n    const numCacheNodes = (resource.properties.NumCacheNodes as number) || 1;\n    const azMode = resource.properties.AZMode as string;\n\n    if (!cacheNodeType || !engine) {\n      return {\n        amount: 0,\n        currency: 'USD',\n        confidence: 'unknown',\n        assumptions: ['Cache node type or engine not specified'],\n      };\n    }\n\n    try {\n      // Query AWS Pricing API for node hourly rates\n      const hourlyRate = await pricingClient.getPrice({\n        serviceCode: 'AmazonElastiCache',\n        region: this.normalizeRegion(region),\n        filters: [\n          { field: 'instanceType', value: cacheNodeType },\n          { field: 'cacheEngine', value: this.normalizeEngine(engine) },\n        ],\n      });\n\n      if (hourlyRate === null) {\n        return {\n          amount: 0,\n          currency: 'USD',\n          confidence: 'unknown',\n          assumptions: [\n            `Pricing data not available for node type ${cacheNodeType} with engine ${engine} in region ${region}`,\n          ],\n        };\n      }\n\n      const monthlyHours = 730;\n      const perNodeCost = hourlyRate * monthlyHours;\n      let totalCost = perNodeCost * numCacheNodes;\n\n      const assumptions = [\n        `Assumes ${monthlyHours} hours per month (24/7 operation)`,\n        `Node type: ${cacheNodeType}`,\n        `Engine: ${engine}`,\n        `Number of cache nodes: ${numCacheNodes}`,\n        `Per-node monthly cost: $${perNodeCost.toFixed(2)}`,\n      ];\n\n      // Account for multi-AZ replica costs when configured\n      if (azMode === 'cross-az') {\n        // Multi-AZ adds replica nodes (typically doubles the cost)\n        totalCost *= 2;\n        assumptions.push('Multi-AZ deployment with replica nodes (cost doubled)');\n      } else {\n        assumptions.push('Single-AZ deployment');\n      }\n\n      return {\n        amount: totalCost,\n        currency: 'USD',\n        confidence: 'high',\n        assumptions,\n      };\n    } catch (error) {\n      return {\n        amount: 0,\n        currency: 'USD',\n        confidence: 'unknown',\n        assumptions: [\n          `Failed to fetch pricing: ${error instanceof Error ? error.message : String(error)}`,\n        ],\n      };\n    }\n  }\n\n  private normalizeEngine(engine: string): string {\n    const engineMap: Record<string, string> = {\n      redis: 'Redis',\n      memcached: 'Memcached',\n    };\n\n    return engineMap[engine.toLowerCase()] || engine;\n  }\n\n  private normalizeRegion(region: string): string {\n    const regionMap: Record<string, string> = {\n      'us-east-1': 'US East (N. Virginia)',\n      'us-east-2': 'US East (Ohio)',\n      'us-west-1': 'US West (N. California)',\n      'us-west-2': 'US West (Oregon)',\n      'eu-west-1': 'EU (Ireland)',\n      'eu-west-2': 'EU (London)',\n      'eu-west-3': 'EU (Paris)',\n      'eu-central-1': 'EU (Frankfurt)',\n      'eu-north-1': 'EU (Stockholm)',\n      'ap-south-1': 'Asia Pacific (Mumbai)',\n      'ap-southeast-1': 'Asia Pacific (Singapore)',\n      'ap-southeast-2': 'Asia Pacific (Sydney)',\n      'ap-northeast-1': 'Asia Pacific (Tokyo)',\n      'ap-northeast-2': 'Asia Pacific (Seoul)',\n    };\n\n    return regionMap[region] || region;\n  }\n}\n"]}
@@ -0,0 +1,13 @@
1
+ import { ResourceWithId } from '../../diff/types';
2
+ import { ResourceCostCalculator, MonthlyCost, PricingClient } from '../types';
3
+ export declare class LambdaCalculator implements ResourceCostCalculator {
4
+ private readonly customInvocationsPerMonth?;
5
+ private readonly customAverageDurationMs?;
6
+ private readonly DEFAULT_INVOCATIONS;
7
+ private readonly DEFAULT_DURATION_MS;
8
+ private readonly DEFAULT_MEMORY_MB;
9
+ constructor(customInvocationsPerMonth?: number | undefined, customAverageDurationMs?: number | undefined);
10
+ supports(resourceType: string): boolean;
11
+ calculateCost(resource: ResourceWithId, region: string, pricingClient: PricingClient): Promise<MonthlyCost>;
12
+ private normalizeRegion;
13
+ }
@@ -0,0 +1,111 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.LambdaCalculator = void 0;
4
+ class LambdaCalculator {
5
+ customInvocationsPerMonth;
6
+ customAverageDurationMs;
7
+ DEFAULT_INVOCATIONS = 1000000;
8
+ DEFAULT_DURATION_MS = 1000;
9
+ DEFAULT_MEMORY_MB = 128;
10
+ constructor(customInvocationsPerMonth, customAverageDurationMs) {
11
+ this.customInvocationsPerMonth = customInvocationsPerMonth;
12
+ this.customAverageDurationMs = customAverageDurationMs;
13
+ }
14
+ supports(resourceType) {
15
+ return resourceType === 'AWS::Lambda::Function';
16
+ }
17
+ async calculateCost(resource, region, pricingClient) {
18
+ const memorySize = resource.properties.MemorySize || this.DEFAULT_MEMORY_MB;
19
+ const invocations = this.customInvocationsPerMonth ?? this.DEFAULT_INVOCATIONS;
20
+ const durationMs = this.customAverageDurationMs ?? this.DEFAULT_DURATION_MS;
21
+ try {
22
+ const requestPrice = await pricingClient.getPrice({
23
+ serviceCode: 'AWSLambda',
24
+ region: this.normalizeRegion(region),
25
+ filters: [
26
+ { field: 'group', value: 'AWS-Lambda-Requests' },
27
+ ],
28
+ });
29
+ const computePrice = await pricingClient.getPrice({
30
+ serviceCode: 'AWSLambda',
31
+ region: this.normalizeRegion(region),
32
+ filters: [
33
+ { field: 'group', value: 'AWS-Lambda-Duration' },
34
+ ],
35
+ });
36
+ const assumptions = [
37
+ `Assumes ${invocations.toLocaleString()} invocations per month`,
38
+ `Assumes ${durationMs}ms average execution time`,
39
+ `Assumes ${memorySize}MB memory allocation`,
40
+ ];
41
+ if (this.customInvocationsPerMonth !== undefined) {
42
+ assumptions.push('Using custom invocation count assumption from configuration');
43
+ }
44
+ if (this.customAverageDurationMs !== undefined) {
45
+ assumptions.push('Using custom duration assumption from configuration');
46
+ }
47
+ if (requestPrice === null || computePrice === null) {
48
+ return {
49
+ amount: 0,
50
+ currency: 'USD',
51
+ confidence: 'unknown',
52
+ assumptions: [
53
+ `Pricing data not available for Lambda in region ${region}`,
54
+ ...assumptions,
55
+ ],
56
+ };
57
+ }
58
+ const requestCost = (invocations / 1000000) * requestPrice;
59
+ const gbSeconds = (memorySize / 1024) * (durationMs / 1000) * invocations;
60
+ const computeCost = gbSeconds * computePrice;
61
+ const totalCost = requestCost + computeCost;
62
+ return {
63
+ amount: totalCost,
64
+ currency: 'USD',
65
+ confidence: 'medium',
66
+ assumptions,
67
+ };
68
+ }
69
+ catch (error) {
70
+ const assumptions = [
71
+ `Failed to fetch pricing: ${error instanceof Error ? error.message : String(error)}`,
72
+ `Would assume ${invocations.toLocaleString()} invocations per month`,
73
+ `Would assume ${durationMs}ms average execution time`,
74
+ `Would assume ${memorySize}MB memory allocation`,
75
+ ];
76
+ if (this.customInvocationsPerMonth !== undefined) {
77
+ assumptions.push('Using custom invocation count assumption from configuration');
78
+ }
79
+ if (this.customAverageDurationMs !== undefined) {
80
+ assumptions.push('Using custom duration assumption from configuration');
81
+ }
82
+ return {
83
+ amount: 0,
84
+ currency: 'USD',
85
+ confidence: 'unknown',
86
+ assumptions,
87
+ };
88
+ }
89
+ }
90
+ normalizeRegion(region) {
91
+ const regionMap = {
92
+ 'us-east-1': 'US East (N. Virginia)',
93
+ 'us-east-2': 'US East (Ohio)',
94
+ 'us-west-1': 'US West (N. California)',
95
+ 'us-west-2': 'US West (Oregon)',
96
+ 'eu-west-1': 'EU (Ireland)',
97
+ 'eu-west-2': 'EU (London)',
98
+ 'eu-west-3': 'EU (Paris)',
99
+ 'eu-central-1': 'EU (Frankfurt)',
100
+ 'eu-north-1': 'EU (Stockholm)',
101
+ 'ap-south-1': 'Asia Pacific (Mumbai)',
102
+ 'ap-southeast-1': 'Asia Pacific (Singapore)',
103
+ 'ap-southeast-2': 'Asia Pacific (Sydney)',
104
+ 'ap-northeast-1': 'Asia Pacific (Tokyo)',
105
+ 'ap-northeast-2': 'Asia Pacific (Seoul)',
106
+ };
107
+ return regionMap[region] || region;
108
+ }
109
+ }
110
+ exports.LambdaCalculator = LambdaCalculator;
111
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"LambdaCalculator.js","sourceRoot":"","sources":["../../../src/pricing/calculators/LambdaCalculator.ts"],"names":[],"mappings":"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","sourcesContent":["import { ResourceWithId } from '../../diff/types';\nimport { ResourceCostCalculator, MonthlyCost, PricingClient } from '../types';\n\nexport class LambdaCalculator implements ResourceCostCalculator {\n  private readonly DEFAULT_INVOCATIONS = 1000000;\n  private readonly DEFAULT_DURATION_MS = 1000;\n  private readonly DEFAULT_MEMORY_MB = 128;\n\n  constructor(\n    private readonly customInvocationsPerMonth?: number,\n    private readonly customAverageDurationMs?: number,\n  ) {}\n\n  supports(resourceType: string): boolean {\n    return resourceType === 'AWS::Lambda::Function';\n  }\n\n  async calculateCost(\n    resource: ResourceWithId,\n    region: string,\n    pricingClient: PricingClient,\n  ): Promise<MonthlyCost> {\n    const memorySize = (resource.properties.MemorySize as number) || this.DEFAULT_MEMORY_MB;\n    const invocations = this.customInvocationsPerMonth ?? this.DEFAULT_INVOCATIONS;\n    const durationMs = this.customAverageDurationMs ?? this.DEFAULT_DURATION_MS;\n\n    try {\n      const requestPrice = await pricingClient.getPrice({\n        serviceCode: 'AWSLambda',\n        region: this.normalizeRegion(region),\n        filters: [\n          { field: 'group', value: 'AWS-Lambda-Requests' },\n        ],\n      });\n\n      const computePrice = await pricingClient.getPrice({\n        serviceCode: 'AWSLambda',\n        region: this.normalizeRegion(region),\n        filters: [\n          { field: 'group', value: 'AWS-Lambda-Duration' },\n        ],\n      });\n\n      const assumptions = [\n        `Assumes ${invocations.toLocaleString()} invocations per month`,\n        `Assumes ${durationMs}ms average execution time`,\n        `Assumes ${memorySize}MB memory allocation`,\n      ];\n\n      if (this.customInvocationsPerMonth !== undefined) {\n        assumptions.push('Using custom invocation count assumption from configuration');\n      }\n      if (this.customAverageDurationMs !== undefined) {\n        assumptions.push('Using custom duration assumption from configuration');\n      }\n\n      if (requestPrice === null || computePrice === null) {\n        return {\n          amount: 0,\n          currency: 'USD',\n          confidence: 'unknown',\n          assumptions: [\n            `Pricing data not available for Lambda in region ${region}`,\n            ...assumptions,\n          ],\n        };\n      }\n\n      const requestCost = (invocations / 1000000) * requestPrice;\n\n      const gbSeconds = (memorySize / 1024) * (durationMs / 1000) * invocations;\n      const computeCost = gbSeconds * computePrice;\n\n      const totalCost = requestCost + computeCost;\n\n      return {\n        amount: totalCost,\n        currency: 'USD',\n        confidence: 'medium',\n        assumptions,\n      };\n    } catch (error) {\n      const assumptions = [\n        `Failed to fetch pricing: ${error instanceof Error ? error.message : String(error)}`,\n        `Would assume ${invocations.toLocaleString()} invocations per month`,\n        `Would assume ${durationMs}ms average execution time`,\n        `Would assume ${memorySize}MB memory allocation`,\n      ];\n\n      if (this.customInvocationsPerMonth !== undefined) {\n        assumptions.push('Using custom invocation count assumption from configuration');\n      }\n      if (this.customAverageDurationMs !== undefined) {\n        assumptions.push('Using custom duration assumption from configuration');\n      }\n\n      return {\n        amount: 0,\n        currency: 'USD',\n        confidence: 'unknown',\n        assumptions,\n      };\n    }\n  }\n\n  private normalizeRegion(region: string): string {\n    const regionMap: Record<string, string> = {\n      'us-east-1': 'US East (N. Virginia)',\n      'us-east-2': 'US East (Ohio)',\n      'us-west-1': 'US West (N. California)',\n      'us-west-2': 'US West (Oregon)',\n      'eu-west-1': 'EU (Ireland)',\n      'eu-west-2': 'EU (London)',\n      'eu-west-3': 'EU (Paris)',\n      'eu-central-1': 'EU (Frankfurt)',\n      'eu-north-1': 'EU (Stockholm)',\n      'ap-south-1': 'Asia Pacific (Mumbai)',\n      'ap-southeast-1': 'Asia Pacific (Singapore)',\n      'ap-southeast-2': 'Asia Pacific (Sydney)',\n      'ap-northeast-1': 'Asia Pacific (Tokyo)',\n      'ap-northeast-2': 'Asia Pacific (Seoul)',\n    };\n\n    return regionMap[region] || region;\n  }\n}\n"]}
@@ -0,0 +1,16 @@
1
+ import { ResourceWithId } from '../../diff/types';
2
+ import { ResourceCostCalculator, MonthlyCost, PricingClient } from '../types';
3
+ export declare class NLBCalculator implements ResourceCostCalculator {
4
+ private customNewConnectionsPerSecond?;
5
+ private customActiveConnectionsPerMinute?;
6
+ private customProcessedBytesGB?;
7
+ private readonly DEFAULT_NEW_CONNECTIONS_PER_SECOND;
8
+ private readonly DEFAULT_ACTIVE_CONNECTIONS_PER_MINUTE;
9
+ private readonly DEFAULT_PROCESSED_BYTES_GB;
10
+ private readonly HOURS_PER_MONTH;
11
+ constructor(customNewConnectionsPerSecond?: number | undefined, customActiveConnectionsPerMinute?: number | undefined, customProcessedBytesGB?: number | undefined);
12
+ supports(resourceType: string): boolean;
13
+ calculateCost(resource: ResourceWithId, region: string, pricingClient: PricingClient): Promise<MonthlyCost>;
14
+ private normalizeRegion;
15
+ private getRegionPrefix;
16
+ }