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,138 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.NLBCalculator = void 0;
4
+ class NLBCalculator {
5
+ customNewConnectionsPerSecond;
6
+ customActiveConnectionsPerMinute;
7
+ customProcessedBytesGB;
8
+ DEFAULT_NEW_CONNECTIONS_PER_SECOND = 25;
9
+ DEFAULT_ACTIVE_CONNECTIONS_PER_MINUTE = 3000;
10
+ DEFAULT_PROCESSED_BYTES_GB = 100;
11
+ HOURS_PER_MONTH = 730;
12
+ constructor(customNewConnectionsPerSecond, customActiveConnectionsPerMinute, customProcessedBytesGB) {
13
+ this.customNewConnectionsPerSecond = customNewConnectionsPerSecond;
14
+ this.customActiveConnectionsPerMinute = customActiveConnectionsPerMinute;
15
+ this.customProcessedBytesGB = customProcessedBytesGB;
16
+ }
17
+ supports(resourceType) {
18
+ return resourceType === 'AWS::ElasticLoadBalancingV2::LoadBalancer';
19
+ }
20
+ async calculateCost(resource, region, pricingClient) {
21
+ // Check if this is a Network Load Balancer
22
+ const loadBalancerType = resource.properties?.Type;
23
+ if (loadBalancerType && loadBalancerType !== 'network') {
24
+ return {
25
+ amount: 0,
26
+ currency: 'USD',
27
+ confidence: 'unknown',
28
+ assumptions: ['This calculator only supports Network Load Balancers'],
29
+ };
30
+ }
31
+ try {
32
+ // Get hourly rate
33
+ const hourlyRate = await pricingClient.getPrice({
34
+ serviceCode: 'AWSELB',
35
+ region: this.normalizeRegion(region),
36
+ filters: [
37
+ { field: 'productFamily', value: 'Load Balancer-Network' },
38
+ { field: 'usagetype', value: `${this.getRegionPrefix(region)}LoadBalancerUsage` },
39
+ ],
40
+ });
41
+ // Get NLCU rate
42
+ const nlcuRate = await pricingClient.getPrice({
43
+ serviceCode: 'AWSELB',
44
+ region: this.normalizeRegion(region),
45
+ filters: [
46
+ { field: 'productFamily', value: 'Load Balancer-Network' },
47
+ { field: 'usagetype', value: `${this.getRegionPrefix(region)}LCUUsage` },
48
+ ],
49
+ });
50
+ if (hourlyRate === null || nlcuRate === null) {
51
+ return {
52
+ amount: 0,
53
+ currency: 'USD',
54
+ confidence: 'unknown',
55
+ assumptions: [
56
+ `Pricing data not available for Network Load Balancer in region ${region}`,
57
+ ],
58
+ };
59
+ }
60
+ const newConnectionsPerSecond = this.customNewConnectionsPerSecond || this.DEFAULT_NEW_CONNECTIONS_PER_SECOND;
61
+ const activeConnectionsPerMinute = this.customActiveConnectionsPerMinute || this.DEFAULT_ACTIVE_CONNECTIONS_PER_MINUTE;
62
+ const processedBytesGB = this.customProcessedBytesGB || this.DEFAULT_PROCESSED_BYTES_GB;
63
+ // Calculate NLCU consumption
64
+ // 1 NLCU provides: 800 new connections/sec, 100,000 active connections/min, 1 GB processed/hour
65
+ const nlcuFromNewConnections = newConnectionsPerSecond / 800;
66
+ const nlcuFromActiveConnections = activeConnectionsPerMinute / 100000;
67
+ const gbPerHour = processedBytesGB / this.HOURS_PER_MONTH;
68
+ const nlcuFromProcessedBytes = gbPerHour;
69
+ // Use the highest NLCU consumption
70
+ const nlcuPerHour = Math.max(nlcuFromNewConnections, nlcuFromActiveConnections, nlcuFromProcessedBytes);
71
+ const hourlyCost = hourlyRate * this.HOURS_PER_MONTH;
72
+ const nlcuCost = nlcuRate * nlcuPerHour * this.HOURS_PER_MONTH;
73
+ const totalCost = hourlyCost + nlcuCost;
74
+ return {
75
+ amount: totalCost,
76
+ currency: 'USD',
77
+ confidence: 'medium',
78
+ assumptions: [
79
+ `Hourly rate: $${hourlyRate.toFixed(4)}/hour × ${this.HOURS_PER_MONTH} hours = $${hourlyCost.toFixed(2)}/month`,
80
+ `NLCU consumption: ${nlcuPerHour.toFixed(2)} NLCU/hour based on:`,
81
+ ` - New connections: ${newConnectionsPerSecond}/sec → ${nlcuFromNewConnections.toFixed(2)} NLCU`,
82
+ ` - Active connections: ${activeConnectionsPerMinute}/min → ${nlcuFromActiveConnections.toFixed(2)} NLCU`,
83
+ ` - Processed data: ${processedBytesGB} GB/month → ${nlcuFromProcessedBytes.toFixed(2)} NLCU`,
84
+ `NLCU cost: $${nlcuRate.toFixed(4)}/NLCU/hour × ${nlcuPerHour.toFixed(2)} NLCU × ${this.HOURS_PER_MONTH} hours = $${nlcuCost.toFixed(2)}/month`,
85
+ `Total: $${totalCost.toFixed(2)}/month`,
86
+ ],
87
+ };
88
+ }
89
+ catch (error) {
90
+ return {
91
+ amount: 0,
92
+ currency: 'USD',
93
+ confidence: 'unknown',
94
+ assumptions: [`Failed to fetch pricing: ${error instanceof Error ? error.message : String(error)}`],
95
+ };
96
+ }
97
+ }
98
+ normalizeRegion(region) {
99
+ const regionMap = {
100
+ 'us-east-1': 'US East (N. Virginia)',
101
+ 'us-east-2': 'US East (Ohio)',
102
+ 'us-west-1': 'US West (N. California)',
103
+ 'us-west-2': 'US West (Oregon)',
104
+ 'eu-west-1': 'EU (Ireland)',
105
+ 'eu-west-2': 'EU (London)',
106
+ 'eu-west-3': 'EU (Paris)',
107
+ 'eu-central-1': 'EU (Frankfurt)',
108
+ 'eu-north-1': 'EU (Stockholm)',
109
+ 'ap-south-1': 'Asia Pacific (Mumbai)',
110
+ 'ap-southeast-1': 'Asia Pacific (Singapore)',
111
+ 'ap-southeast-2': 'Asia Pacific (Sydney)',
112
+ 'ap-northeast-1': 'Asia Pacific (Tokyo)',
113
+ 'ap-northeast-2': 'Asia Pacific (Seoul)',
114
+ };
115
+ return regionMap[region] || region;
116
+ }
117
+ getRegionPrefix(region) {
118
+ const prefixMap = {
119
+ 'us-east-1': 'USE1',
120
+ 'us-east-2': 'USE2',
121
+ 'us-west-1': 'USW1',
122
+ 'us-west-2': 'USW2',
123
+ 'eu-west-1': 'EUW1',
124
+ 'eu-west-2': 'EUW2',
125
+ 'eu-west-3': 'EUW3',
126
+ 'eu-central-1': 'EUC1',
127
+ 'eu-north-1': 'EUN1',
128
+ 'ap-south-1': 'APS1',
129
+ 'ap-southeast-1': 'APS2',
130
+ 'ap-southeast-2': 'APS3',
131
+ 'ap-northeast-1': 'APN1',
132
+ 'ap-northeast-2': 'APN2',
133
+ };
134
+ return prefixMap[region] || '';
135
+ }
136
+ }
137
+ exports.NLBCalculator = NLBCalculator;
138
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"NLBCalculator.js","sourceRoot":"","sources":["../../../src/pricing/calculators/NLBCalculator.ts"],"names":[],"mappings":"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","sourcesContent":["import { ResourceWithId } from '../../diff/types';\nimport { ResourceCostCalculator, MonthlyCost, PricingClient } from '../types';\n\nexport class NLBCalculator implements ResourceCostCalculator {\n  private readonly DEFAULT_NEW_CONNECTIONS_PER_SECOND = 25;\n  private readonly DEFAULT_ACTIVE_CONNECTIONS_PER_MINUTE = 3000;\n  private readonly DEFAULT_PROCESSED_BYTES_GB = 100;\n  private readonly HOURS_PER_MONTH = 730;\n\n  constructor(\n    private customNewConnectionsPerSecond?: number,\n    private customActiveConnectionsPerMinute?: number,\n    private customProcessedBytesGB?: number,\n  ) {}\n\n  supports(resourceType: string): boolean {\n    return resourceType === 'AWS::ElasticLoadBalancingV2::LoadBalancer';\n  }\n\n  async calculateCost(\n    resource: ResourceWithId,\n    region: string,\n    pricingClient: PricingClient,\n  ): Promise<MonthlyCost> {\n    // Check if this is a Network Load Balancer\n    const loadBalancerType = resource.properties?.Type;\n    if (loadBalancerType && loadBalancerType !== 'network') {\n      return {\n        amount: 0,\n        currency: 'USD',\n        confidence: 'unknown',\n        assumptions: ['This calculator only supports Network Load Balancers'],\n      };\n    }\n\n    try {\n      // Get hourly rate\n      const hourlyRate = await pricingClient.getPrice({\n        serviceCode: 'AWSELB',\n        region: this.normalizeRegion(region),\n        filters: [\n          { field: 'productFamily', value: 'Load Balancer-Network' },\n          { field: 'usagetype', value: `${this.getRegionPrefix(region)}LoadBalancerUsage` },\n        ],\n      });\n\n      // Get NLCU rate\n      const nlcuRate = await pricingClient.getPrice({\n        serviceCode: 'AWSELB',\n        region: this.normalizeRegion(region),\n        filters: [\n          { field: 'productFamily', value: 'Load Balancer-Network' },\n          { field: 'usagetype', value: `${this.getRegionPrefix(region)}LCUUsage` },\n        ],\n      });\n\n      if (hourlyRate === null || nlcuRate === null) {\n        return {\n          amount: 0,\n          currency: 'USD',\n          confidence: 'unknown',\n          assumptions: [\n            `Pricing data not available for Network Load Balancer in region ${region}`,\n          ],\n        };\n      }\n\n      const newConnectionsPerSecond =\n        this.customNewConnectionsPerSecond || this.DEFAULT_NEW_CONNECTIONS_PER_SECOND;\n      const activeConnectionsPerMinute =\n        this.customActiveConnectionsPerMinute || this.DEFAULT_ACTIVE_CONNECTIONS_PER_MINUTE;\n      const processedBytesGB =\n        this.customProcessedBytesGB || this.DEFAULT_PROCESSED_BYTES_GB;\n\n      // Calculate NLCU consumption\n      // 1 NLCU provides: 800 new connections/sec, 100,000 active connections/min, 1 GB processed/hour\n      const nlcuFromNewConnections = newConnectionsPerSecond / 800;\n      const nlcuFromActiveConnections = activeConnectionsPerMinute / 100000;\n      const gbPerHour = processedBytesGB / this.HOURS_PER_MONTH;\n      const nlcuFromProcessedBytes = gbPerHour;\n\n      // Use the highest NLCU consumption\n      const nlcuPerHour = Math.max(\n        nlcuFromNewConnections,\n        nlcuFromActiveConnections,\n        nlcuFromProcessedBytes,\n      );\n\n      const hourlyCost = hourlyRate * this.HOURS_PER_MONTH;\n      const nlcuCost = nlcuRate * nlcuPerHour * this.HOURS_PER_MONTH;\n      const totalCost = hourlyCost + nlcuCost;\n\n      return {\n        amount: totalCost,\n        currency: 'USD',\n        confidence: 'medium',\n        assumptions: [\n          `Hourly rate: $${hourlyRate.toFixed(4)}/hour × ${this.HOURS_PER_MONTH} hours = $${hourlyCost.toFixed(2)}/month`,\n          `NLCU consumption: ${nlcuPerHour.toFixed(2)} NLCU/hour based on:`,\n          `  - New connections: ${newConnectionsPerSecond}/sec → ${nlcuFromNewConnections.toFixed(2)} NLCU`,\n          `  - Active connections: ${activeConnectionsPerMinute}/min → ${nlcuFromActiveConnections.toFixed(2)} NLCU`,\n          `  - Processed data: ${processedBytesGB} GB/month → ${nlcuFromProcessedBytes.toFixed(2)} NLCU`,\n          `NLCU cost: $${nlcuRate.toFixed(4)}/NLCU/hour × ${nlcuPerHour.toFixed(2)} NLCU × ${this.HOURS_PER_MONTH} hours = $${nlcuCost.toFixed(2)}/month`,\n          `Total: $${totalCost.toFixed(2)}/month`,\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 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  private getRegionPrefix(region: string): string {\n    const prefixMap: Record<string, string> = {\n      'us-east-1': 'USE1',\n      'us-east-2': 'USE2',\n      'us-west-1': 'USW1',\n      'us-west-2': 'USW2',\n      'eu-west-1': 'EUW1',\n      'eu-west-2': 'EUW2',\n      'eu-west-3': 'EUW3',\n      'eu-central-1': 'EUC1',\n      'eu-north-1': 'EUN1',\n      'ap-south-1': 'APS1',\n      'ap-southeast-1': 'APS2',\n      'ap-southeast-2': 'APS3',\n      'ap-northeast-1': 'APN1',\n      'ap-northeast-2': 'APN2',\n    };\n\n    return prefixMap[region] || '';\n  }\n}\n"]}
@@ -0,0 +1,12 @@
1
+ import { ResourceWithId } from '../../diff/types';
2
+ import { ResourceCostCalculator, MonthlyCost, PricingClient } from '../types';
3
+ export declare class NatGatewayCalculator implements ResourceCostCalculator {
4
+ private customDataProcessedGB?;
5
+ private readonly DEFAULT_DATA_PROCESSED_GB;
6
+ private readonly HOURS_PER_MONTH;
7
+ constructor(customDataProcessedGB?: number | undefined);
8
+ supports(resourceType: string): boolean;
9
+ calculateCost(_resource: ResourceWithId, region: string, pricingClient: PricingClient): Promise<MonthlyCost>;
10
+ private normalizeRegion;
11
+ private getRegionPrefix;
12
+ }
@@ -0,0 +1,116 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.NatGatewayCalculator = void 0;
4
+ class NatGatewayCalculator {
5
+ customDataProcessedGB;
6
+ DEFAULT_DATA_PROCESSED_GB = 100;
7
+ HOURS_PER_MONTH = 730;
8
+ constructor(customDataProcessedGB) {
9
+ this.customDataProcessedGB = customDataProcessedGB;
10
+ }
11
+ supports(resourceType) {
12
+ return resourceType === 'AWS::EC2::NatGateway';
13
+ }
14
+ async calculateCost(_resource, region, pricingClient) {
15
+ try {
16
+ // Get hourly rate
17
+ const hourlyRate = await pricingClient.getPrice({
18
+ serviceCode: 'AmazonEC2',
19
+ region: this.normalizeRegion(region),
20
+ filters: [
21
+ { field: 'productFamily', value: 'NAT Gateway' },
22
+ { field: 'usagetype', value: `${this.getRegionPrefix(region)}NatGateway-Hours` },
23
+ ],
24
+ });
25
+ // Get data processing rate
26
+ const dataProcessingRate = await pricingClient.getPrice({
27
+ serviceCode: 'AmazonEC2',
28
+ region: this.normalizeRegion(region),
29
+ filters: [
30
+ { field: 'productFamily', value: 'NAT Gateway' },
31
+ { field: 'usagetype', value: `${this.getRegionPrefix(region)}NatGateway-Bytes` },
32
+ ],
33
+ });
34
+ if (hourlyRate === null || dataProcessingRate === null) {
35
+ const dataProcessedGB = this.customDataProcessedGB || this.DEFAULT_DATA_PROCESSED_GB;
36
+ const assumptions = [
37
+ `Pricing data not available for NAT Gateway in region ${region}`,
38
+ `Would assume ${dataProcessedGB} GB of data processing per month`,
39
+ `Would assume ${this.HOURS_PER_MONTH} hours per month`,
40
+ ];
41
+ if (this.customDataProcessedGB !== undefined) {
42
+ assumptions.push('Using custom data processing assumption from configuration');
43
+ }
44
+ return {
45
+ amount: 0,
46
+ currency: 'USD',
47
+ confidence: 'unknown',
48
+ assumptions,
49
+ };
50
+ }
51
+ const dataProcessedGB = this.customDataProcessedGB || this.DEFAULT_DATA_PROCESSED_GB;
52
+ const hourlyCost = hourlyRate * this.HOURS_PER_MONTH;
53
+ const dataProcessingCost = dataProcessingRate * dataProcessedGB;
54
+ const totalCost = hourlyCost + dataProcessingCost;
55
+ return {
56
+ amount: totalCost,
57
+ currency: 'USD',
58
+ confidence: 'medium',
59
+ assumptions: [
60
+ `Hourly rate: $${hourlyRate.toFixed(4)}/hour × ${this.HOURS_PER_MONTH} hours = $${hourlyCost.toFixed(2)}/month`,
61
+ `Data processing: $${dataProcessingRate.toFixed(4)}/GB × ${dataProcessedGB} GB = $${dataProcessingCost.toFixed(2)}/month`,
62
+ `Total: $${totalCost.toFixed(2)}/month`,
63
+ ],
64
+ };
65
+ }
66
+ catch (error) {
67
+ return {
68
+ amount: 0,
69
+ currency: 'USD',
70
+ confidence: 'unknown',
71
+ assumptions: [`Failed to fetch pricing: ${error instanceof Error ? error.message : String(error)}`],
72
+ };
73
+ }
74
+ }
75
+ normalizeRegion(region) {
76
+ const regionMap = {
77
+ 'us-east-1': 'US East (N. Virginia)',
78
+ 'us-east-2': 'US East (Ohio)',
79
+ 'us-west-1': 'US West (N. California)',
80
+ 'us-west-2': 'US West (Oregon)',
81
+ 'eu-west-1': 'EU (Ireland)',
82
+ 'eu-west-2': 'EU (London)',
83
+ 'eu-west-3': 'EU (Paris)',
84
+ 'eu-central-1': 'EU (Frankfurt)',
85
+ 'eu-north-1': 'EU (Stockholm)',
86
+ 'ap-south-1': 'Asia Pacific (Mumbai)',
87
+ 'ap-southeast-1': 'Asia Pacific (Singapore)',
88
+ 'ap-southeast-2': 'Asia Pacific (Sydney)',
89
+ 'ap-northeast-1': 'Asia Pacific (Tokyo)',
90
+ 'ap-northeast-2': 'Asia Pacific (Seoul)',
91
+ };
92
+ return regionMap[region] || region;
93
+ }
94
+ getRegionPrefix(region) {
95
+ // AWS uses region prefixes in usage types (e.g., USE1 for us-east-1)
96
+ const prefixMap = {
97
+ 'us-east-1': 'USE1',
98
+ 'us-east-2': 'USE2',
99
+ 'us-west-1': 'USW1',
100
+ 'us-west-2': 'USW2',
101
+ 'eu-west-1': 'EUW1',
102
+ 'eu-west-2': 'EUW2',
103
+ 'eu-west-3': 'EUW3',
104
+ 'eu-central-1': 'EUC1',
105
+ 'eu-north-1': 'EUN1',
106
+ 'ap-south-1': 'APS1',
107
+ 'ap-southeast-1': 'APS2',
108
+ 'ap-southeast-2': 'APS3',
109
+ 'ap-northeast-1': 'APN1',
110
+ 'ap-northeast-2': 'APN2',
111
+ };
112
+ return prefixMap[region] || '';
113
+ }
114
+ }
115
+ exports.NatGatewayCalculator = NatGatewayCalculator;
116
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"NatGatewayCalculator.js","sourceRoot":"","sources":["../../../src/pricing/calculators/NatGatewayCalculator.ts"],"names":[],"mappings":"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","sourcesContent":["import { ResourceWithId } from '../../diff/types';\nimport { ResourceCostCalculator, MonthlyCost, PricingClient } from '../types';\n\nexport class NatGatewayCalculator implements ResourceCostCalculator {\n  private readonly DEFAULT_DATA_PROCESSED_GB = 100;\n  private readonly HOURS_PER_MONTH = 730;\n\n  constructor(private customDataProcessedGB?: number) {}\n\n  supports(resourceType: string): boolean {\n    return resourceType === 'AWS::EC2::NatGateway';\n  }\n\n  async calculateCost(\n    _resource: ResourceWithId,\n    region: string,\n    pricingClient: PricingClient,\n  ): Promise<MonthlyCost> {\n    try {\n      // Get hourly rate\n      const hourlyRate = await pricingClient.getPrice({\n        serviceCode: 'AmazonEC2',\n        region: this.normalizeRegion(region),\n        filters: [\n          { field: 'productFamily', value: 'NAT Gateway' },\n          { field: 'usagetype', value: `${this.getRegionPrefix(region)}NatGateway-Hours` },\n        ],\n      });\n\n      // Get data processing rate\n      const dataProcessingRate = await pricingClient.getPrice({\n        serviceCode: 'AmazonEC2',\n        region: this.normalizeRegion(region),\n        filters: [\n          { field: 'productFamily', value: 'NAT Gateway' },\n          { field: 'usagetype', value: `${this.getRegionPrefix(region)}NatGateway-Bytes` },\n        ],\n      });\n\n      if (hourlyRate === null || dataProcessingRate === null) {\n        const dataProcessedGB = this.customDataProcessedGB || this.DEFAULT_DATA_PROCESSED_GB;\n        const assumptions = [\n          `Pricing data not available for NAT Gateway in region ${region}`,\n          `Would assume ${dataProcessedGB} GB of data processing per month`,\n          `Would assume ${this.HOURS_PER_MONTH} hours per month`,\n        ];\n\n        if (this.customDataProcessedGB !== undefined) {\n          assumptions.push('Using custom data processing assumption from configuration');\n        }\n\n        return {\n          amount: 0,\n          currency: 'USD',\n          confidence: 'unknown',\n          assumptions,\n        };\n      }\n\n      const dataProcessedGB = this.customDataProcessedGB || this.DEFAULT_DATA_PROCESSED_GB;\n      const hourlyCost = hourlyRate * this.HOURS_PER_MONTH;\n      const dataProcessingCost = dataProcessingRate * dataProcessedGB;\n      const totalCost = hourlyCost + dataProcessingCost;\n\n      return {\n        amount: totalCost,\n        currency: 'USD',\n        confidence: 'medium',\n        assumptions: [\n          `Hourly rate: $${hourlyRate.toFixed(4)}/hour × ${this.HOURS_PER_MONTH} hours = $${hourlyCost.toFixed(2)}/month`,\n          `Data processing: $${dataProcessingRate.toFixed(4)}/GB × ${dataProcessedGB} GB = $${dataProcessingCost.toFixed(2)}/month`,\n          `Total: $${totalCost.toFixed(2)}/month`,\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 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  private getRegionPrefix(region: string): string {\n    // AWS uses region prefixes in usage types (e.g., USE1 for us-east-1)\n    const prefixMap: Record<string, string> = {\n      'us-east-1': 'USE1',\n      'us-east-2': 'USE2',\n      'us-west-1': 'USW1',\n      'us-west-2': 'USW2',\n      'eu-west-1': 'EUW1',\n      'eu-west-2': 'EUW2',\n      'eu-west-3': 'EUW3',\n      'eu-central-1': 'EUC1',\n      'eu-north-1': 'EUN1',\n      'ap-south-1': 'APS1',\n      'ap-southeast-1': 'APS2',\n      'ap-southeast-2': 'APS3',\n      'ap-northeast-1': 'APN1',\n      'ap-northeast-2': 'APN2',\n    };\n\n    return prefixMap[region] || '';\n  }\n}\n"]}
@@ -0,0 +1,9 @@
1
+ import { ResourceWithId } from '../../diff/types';
2
+ import { ResourceCostCalculator, MonthlyCost, PricingClient } from '../types';
3
+ export declare class RDSCalculator implements ResourceCostCalculator {
4
+ private readonly DEFAULT_STORAGE_GB;
5
+ supports(resourceType: string): boolean;
6
+ calculateCost(resource: ResourceWithId, region: string, pricingClient: PricingClient): Promise<MonthlyCost>;
7
+ private normalizeEngine;
8
+ private normalizeRegion;
9
+ }
@@ -0,0 +1,103 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.RDSCalculator = void 0;
4
+ class RDSCalculator {
5
+ DEFAULT_STORAGE_GB = 100;
6
+ supports(resourceType) {
7
+ return resourceType === 'AWS::RDS::DBInstance';
8
+ }
9
+ async calculateCost(resource, region, pricingClient) {
10
+ const instanceClass = resource.properties.DBInstanceClass;
11
+ const engine = resource.properties.Engine;
12
+ if (!instanceClass || !engine) {
13
+ return {
14
+ amount: 0,
15
+ currency: 'USD',
16
+ confidence: 'unknown',
17
+ assumptions: ['DB instance class or engine not specified'],
18
+ };
19
+ }
20
+ try {
21
+ const hourlyRate = await pricingClient.getPrice({
22
+ serviceCode: 'AmazonRDS',
23
+ region: this.normalizeRegion(region),
24
+ filters: [
25
+ { field: 'instanceType', value: instanceClass },
26
+ { field: 'databaseEngine', value: this.normalizeEngine(engine) },
27
+ { field: 'deploymentOption', value: 'Single-AZ' },
28
+ ],
29
+ });
30
+ const storagePrice = await pricingClient.getPrice({
31
+ serviceCode: 'AmazonRDS',
32
+ region: this.normalizeRegion(region),
33
+ filters: [
34
+ { field: 'volumeType', value: 'General Purpose' },
35
+ { field: 'databaseEngine', value: this.normalizeEngine(engine) },
36
+ ],
37
+ });
38
+ if (hourlyRate === null) {
39
+ return {
40
+ amount: 0,
41
+ currency: 'USD',
42
+ confidence: 'unknown',
43
+ assumptions: [`Pricing data not available for instance class ${instanceClass} in region ${region}`],
44
+ };
45
+ }
46
+ const monthlyHours = 730;
47
+ const instanceCost = hourlyRate * monthlyHours;
48
+ const storageCost = (storagePrice || 0) * this.DEFAULT_STORAGE_GB;
49
+ const totalCost = instanceCost + storageCost;
50
+ return {
51
+ amount: totalCost,
52
+ currency: 'USD',
53
+ confidence: 'high',
54
+ assumptions: [
55
+ `Assumes ${monthlyHours} hours per month (24/7 operation)`,
56
+ `Assumes ${this.DEFAULT_STORAGE_GB} GB of General Purpose (gp2) storage`,
57
+ 'Assumes Single-AZ deployment',
58
+ ],
59
+ };
60
+ }
61
+ catch (error) {
62
+ return {
63
+ amount: 0,
64
+ currency: 'USD',
65
+ confidence: 'unknown',
66
+ assumptions: [`Failed to fetch pricing: ${error instanceof Error ? error.message : String(error)}`],
67
+ };
68
+ }
69
+ }
70
+ normalizeEngine(engine) {
71
+ const engineMap = {
72
+ 'mysql': 'MySQL',
73
+ 'postgres': 'PostgreSQL',
74
+ 'mariadb': 'MariaDB',
75
+ 'oracle-se2': 'Oracle',
76
+ 'sqlserver-ex': 'SQL Server',
77
+ 'aurora-mysql': 'Aurora MySQL',
78
+ 'aurora-postgresql': 'Aurora PostgreSQL',
79
+ };
80
+ return engineMap[engine.toLowerCase()] || engine;
81
+ }
82
+ normalizeRegion(region) {
83
+ const regionMap = {
84
+ 'us-east-1': 'US East (N. Virginia)',
85
+ 'us-east-2': 'US East (Ohio)',
86
+ 'us-west-1': 'US West (N. California)',
87
+ 'us-west-2': 'US West (Oregon)',
88
+ 'eu-west-1': 'EU (Ireland)',
89
+ 'eu-west-2': 'EU (London)',
90
+ 'eu-west-3': 'EU (Paris)',
91
+ 'eu-central-1': 'EU (Frankfurt)',
92
+ 'eu-north-1': 'EU (Stockholm)',
93
+ 'ap-south-1': 'Asia Pacific (Mumbai)',
94
+ 'ap-southeast-1': 'Asia Pacific (Singapore)',
95
+ 'ap-southeast-2': 'Asia Pacific (Sydney)',
96
+ 'ap-northeast-1': 'Asia Pacific (Tokyo)',
97
+ 'ap-northeast-2': 'Asia Pacific (Seoul)',
98
+ };
99
+ return regionMap[region] || region;
100
+ }
101
+ }
102
+ exports.RDSCalculator = RDSCalculator;
103
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"RDSCalculator.js","sourceRoot":"","sources":["../../../src/pricing/calculators/RDSCalculator.ts"],"names":[],"mappings":";;;AAGA,MAAa,aAAa;IACP,kBAAkB,GAAG,GAAG,CAAC;IAE1C,QAAQ,CAAC,YAAoB;QAC3B,OAAO,YAAY,KAAK,sBAAsB,CAAC;IACjD,CAAC;IAED,KAAK,CAAC,aAAa,CACjB,QAAwB,EACxB,MAAc,EACd,aAA4B;QAE5B,MAAM,aAAa,GAAG,QAAQ,CAAC,UAAU,CAAC,eAAyB,CAAC;QACpE,MAAM,MAAM,GAAG,QAAQ,CAAC,UAAU,CAAC,MAAgB,CAAC;QAEpD,IAAI,CAAC,aAAa,IAAI,CAAC,MAAM,EAAE,CAAC;YAC9B,OAAO;gBACL,MAAM,EAAE,CAAC;gBACT,QAAQ,EAAE,KAAK;gBACf,UAAU,EAAE,SAAS;gBACrB,WAAW,EAAE,CAAC,2CAA2C,CAAC;aAC3D,CAAC;QACJ,CAAC;QAED,IAAI,CAAC;YACH,MAAM,UAAU,GAAG,MAAM,aAAa,CAAC,QAAQ,CAAC;gBAC9C,WAAW,EAAE,WAAW;gBACxB,MAAM,EAAE,IAAI,CAAC,eAAe,CAAC,MAAM,CAAC;gBACpC,OAAO,EAAE;oBACP,EAAE,KAAK,EAAE,cAAc,EAAE,KAAK,EAAE,aAAa,EAAE;oBAC/C,EAAE,KAAK,EAAE,gBAAgB,EAAE,KAAK,EAAE,IAAI,CAAC,eAAe,CAAC,MAAM,CAAC,EAAE;oBAChE,EAAE,KAAK,EAAE,kBAAkB,EAAE,KAAK,EAAE,WAAW,EAAE;iBAClD;aACF,CAAC,CAAC;YAEH,MAAM,YAAY,GAAG,MAAM,aAAa,CAAC,QAAQ,CAAC;gBAChD,WAAW,EAAE,WAAW;gBACxB,MAAM,EAAE,IAAI,CAAC,eAAe,CAAC,MAAM,CAAC;gBACpC,OAAO,EAAE;oBACP,EAAE,KAAK,EAAE,YAAY,EAAE,KAAK,EAAE,iBAAiB,EAAE;oBACjD,EAAE,KAAK,EAAE,gBAAgB,EAAE,KAAK,EAAE,IAAI,CAAC,eAAe,CAAC,MAAM,CAAC,EAAE;iBACjE;aACF,CAAC,CAAC;YAEH,IAAI,UAAU,KAAK,IAAI,EAAE,CAAC;gBACxB,OAAO;oBACL,MAAM,EAAE,CAAC;oBACT,QAAQ,EAAE,KAAK;oBACf,UAAU,EAAE,SAAS;oBACrB,WAAW,EAAE,CAAC,iDAAiD,aAAa,cAAc,MAAM,EAAE,CAAC;iBACpG,CAAC;YACJ,CAAC;YAED,MAAM,YAAY,GAAG,GAAG,CAAC;YACzB,MAAM,YAAY,GAAG,UAAU,GAAG,YAAY,CAAC;YAC/C,MAAM,WAAW,GAAG,CAAC,YAAY,IAAI,CAAC,CAAC,GAAG,IAAI,CAAC,kBAAkB,CAAC;YAClE,MAAM,SAAS,GAAG,YAAY,GAAG,WAAW,CAAC;YAE7C,OAAO;gBACL,MAAM,EAAE,SAAS;gBACjB,QAAQ,EAAE,KAAK;gBACf,UAAU,EAAE,MAAM;gBAClB,WAAW,EAAE;oBACX,WAAW,YAAY,mCAAmC;oBAC1D,WAAW,IAAI,CAAC,kBAAkB,sCAAsC;oBACxE,8BAA8B;iBAC/B;aACF,CAAC;QACJ,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO;gBACL,MAAM,EAAE,CAAC;gBACT,QAAQ,EAAE,KAAK;gBACf,UAAU,EAAE,SAAS;gBACrB,WAAW,EAAE,CAAC,4BAA4B,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC;aACpG,CAAC;QACJ,CAAC;IACH,CAAC;IAEO,eAAe,CAAC,MAAc;QACpC,MAAM,SAAS,GAA2B;YACxC,OAAO,EAAE,OAAO;YAChB,UAAU,EAAE,YAAY;YACxB,SAAS,EAAE,SAAS;YACpB,YAAY,EAAE,QAAQ;YACtB,cAAc,EAAE,YAAY;YAC5B,cAAc,EAAE,cAAc;YAC9B,mBAAmB,EAAE,mBAAmB;SACzC,CAAC;QAEF,OAAO,SAAS,CAAC,MAAM,CAAC,WAAW,EAAE,CAAC,IAAI,MAAM,CAAC;IACnD,CAAC;IAEO,eAAe,CAAC,MAAc;QACpC,MAAM,SAAS,GAA2B;YACxC,WAAW,EAAE,uBAAuB;YACpC,WAAW,EAAE,gBAAgB;YAC7B,WAAW,EAAE,yBAAyB;YACtC,WAAW,EAAE,kBAAkB;YAC/B,WAAW,EAAE,cAAc;YAC3B,WAAW,EAAE,aAAa;YAC1B,WAAW,EAAE,YAAY;YACzB,cAAc,EAAE,gBAAgB;YAChC,YAAY,EAAE,gBAAgB;YAC9B,YAAY,EAAE,uBAAuB;YACrC,gBAAgB,EAAE,0BAA0B;YAC5C,gBAAgB,EAAE,uBAAuB;YACzC,gBAAgB,EAAE,sBAAsB;YACxC,gBAAgB,EAAE,sBAAsB;SACzC,CAAC;QAEF,OAAO,SAAS,CAAC,MAAM,CAAC,IAAI,MAAM,CAAC;IACrC,CAAC;CACF;AAhHD,sCAgHC","sourcesContent":["import { ResourceWithId } from '../../diff/types';\nimport { ResourceCostCalculator, MonthlyCost, PricingClient } from '../types';\n\nexport class RDSCalculator implements ResourceCostCalculator {\n  private readonly DEFAULT_STORAGE_GB = 100;\n\n  supports(resourceType: string): boolean {\n    return resourceType === 'AWS::RDS::DBInstance';\n  }\n\n  async calculateCost(\n    resource: ResourceWithId,\n    region: string,\n    pricingClient: PricingClient,\n  ): Promise<MonthlyCost> {\n    const instanceClass = resource.properties.DBInstanceClass as string;\n    const engine = resource.properties.Engine as string;\n\n    if (!instanceClass || !engine) {\n      return {\n        amount: 0,\n        currency: 'USD',\n        confidence: 'unknown',\n        assumptions: ['DB instance class or engine not specified'],\n      };\n    }\n\n    try {\n      const hourlyRate = await pricingClient.getPrice({\n        serviceCode: 'AmazonRDS',\n        region: this.normalizeRegion(region),\n        filters: [\n          { field: 'instanceType', value: instanceClass },\n          { field: 'databaseEngine', value: this.normalizeEngine(engine) },\n          { field: 'deploymentOption', value: 'Single-AZ' },\n        ],\n      });\n\n      const storagePrice = await pricingClient.getPrice({\n        serviceCode: 'AmazonRDS',\n        region: this.normalizeRegion(region),\n        filters: [\n          { field: 'volumeType', value: 'General Purpose' },\n          { field: 'databaseEngine', value: this.normalizeEngine(engine) },\n        ],\n      });\n\n      if (hourlyRate === null) {\n        return {\n          amount: 0,\n          currency: 'USD',\n          confidence: 'unknown',\n          assumptions: [`Pricing data not available for instance class ${instanceClass} in region ${region}`],\n        };\n      }\n\n      const monthlyHours = 730;\n      const instanceCost = hourlyRate * monthlyHours;\n      const storageCost = (storagePrice || 0) * this.DEFAULT_STORAGE_GB;\n      const totalCost = instanceCost + storageCost;\n\n      return {\n        amount: totalCost,\n        currency: 'USD',\n        confidence: 'high',\n        assumptions: [\n          `Assumes ${monthlyHours} hours per month (24/7 operation)`,\n          `Assumes ${this.DEFAULT_STORAGE_GB} GB of General Purpose (gp2) storage`,\n          'Assumes Single-AZ deployment',\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 normalizeEngine(engine: string): string {\n    const engineMap: Record<string, string> = {\n      'mysql': 'MySQL',\n      'postgres': 'PostgreSQL',\n      'mariadb': 'MariaDB',\n      'oracle-se2': 'Oracle',\n      'sqlserver-ex': 'SQL Server',\n      'aurora-mysql': 'Aurora MySQL',\n      'aurora-postgresql': 'Aurora PostgreSQL',\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,8 @@
1
+ import { ResourceWithId } from '../../diff/types';
2
+ import { ResourceCostCalculator, MonthlyCost, PricingClient } from '../types';
3
+ export declare class S3Calculator implements ResourceCostCalculator {
4
+ private readonly DEFAULT_STORAGE_GB;
5
+ supports(resourceType: string): boolean;
6
+ calculateCost(_resource: ResourceWithId, region: string, pricingClient: PricingClient): Promise<MonthlyCost>;
7
+ private normalizeRegion;
8
+ }
@@ -0,0 +1,68 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.S3Calculator = void 0;
4
+ class S3Calculator {
5
+ DEFAULT_STORAGE_GB = 100;
6
+ supports(resourceType) {
7
+ return resourceType === 'AWS::S3::Bucket';
8
+ }
9
+ async calculateCost(_resource, region, pricingClient) {
10
+ try {
11
+ const pricePerGB = await pricingClient.getPrice({
12
+ serviceCode: 'AmazonS3',
13
+ region: this.normalizeRegion(region),
14
+ filters: [
15
+ { field: 'storageClass', value: 'General Purpose' },
16
+ { field: 'volumeType', value: 'Standard' },
17
+ ],
18
+ });
19
+ if (pricePerGB === null) {
20
+ return {
21
+ amount: 0,
22
+ currency: 'USD',
23
+ confidence: 'unknown',
24
+ assumptions: [`Pricing data not available for S3 in region ${region}`],
25
+ };
26
+ }
27
+ const monthlyCost = pricePerGB * this.DEFAULT_STORAGE_GB;
28
+ return {
29
+ amount: monthlyCost,
30
+ currency: 'USD',
31
+ confidence: 'medium',
32
+ assumptions: [
33
+ `Assumes ${this.DEFAULT_STORAGE_GB} GB of standard storage`,
34
+ 'Does not include request costs or data transfer',
35
+ ],
36
+ };
37
+ }
38
+ catch (error) {
39
+ return {
40
+ amount: 0,
41
+ currency: 'USD',
42
+ confidence: 'unknown',
43
+ assumptions: [`Failed to fetch pricing: ${error instanceof Error ? error.message : String(error)}`],
44
+ };
45
+ }
46
+ }
47
+ normalizeRegion(region) {
48
+ const regionMap = {
49
+ 'us-east-1': 'US East (N. Virginia)',
50
+ 'us-east-2': 'US East (Ohio)',
51
+ 'us-west-1': 'US West (N. California)',
52
+ 'us-west-2': 'US West (Oregon)',
53
+ 'eu-west-1': 'EU (Ireland)',
54
+ 'eu-west-2': 'EU (London)',
55
+ 'eu-west-3': 'EU (Paris)',
56
+ 'eu-central-1': 'EU (Frankfurt)',
57
+ 'eu-north-1': 'EU (Stockholm)',
58
+ 'ap-south-1': 'Asia Pacific (Mumbai)',
59
+ 'ap-southeast-1': 'Asia Pacific (Singapore)',
60
+ 'ap-southeast-2': 'Asia Pacific (Sydney)',
61
+ 'ap-northeast-1': 'Asia Pacific (Tokyo)',
62
+ 'ap-northeast-2': 'Asia Pacific (Seoul)',
63
+ };
64
+ return regionMap[region] || region;
65
+ }
66
+ }
67
+ exports.S3Calculator = S3Calculator;
68
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiUzNDYWxjdWxhdG9yLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL3ByaWNpbmcvY2FsY3VsYXRvcnMvUzNDYWxjdWxhdG9yLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUdBLE1BQWEsWUFBWTtJQUNOLGtCQUFrQixHQUFHLEdBQUcsQ0FBQztJQUUxQyxRQUFRLENBQUMsWUFBb0I7UUFDM0IsT0FBTyxZQUFZLEtBQUssaUJBQWlCLENBQUM7SUFDNUMsQ0FBQztJQUVELEtBQUssQ0FBQyxhQUFhLENBQ2pCLFNBQXlCLEVBQ3pCLE1BQWMsRUFDZCxhQUE0QjtRQUU1QixJQUFJLENBQUM7WUFDSCxNQUFNLFVBQVUsR0FBRyxNQUFNLGFBQWEsQ0FBQyxRQUFRLENBQUM7Z0JBQzlDLFdBQVcsRUFBRSxVQUFVO2dCQUN2QixNQUFNLEVBQUUsSUFBSSxDQUFDLGVBQWUsQ0FBQyxNQUFNLENBQUM7Z0JBQ3BDLE9BQU8sRUFBRTtvQkFDUCxFQUFFLEtBQUssRUFBRSxjQUFjLEVBQUUsS0FBSyxFQUFFLGlCQUFpQixFQUFFO29CQUNuRCxFQUFFLEtBQUssRUFBRSxZQUFZLEVBQUUsS0FBSyxFQUFFLFVBQVUsRUFBRTtpQkFDM0M7YUFDRixDQUFDLENBQUM7WUFFSCxJQUFJLFVBQVUsS0FBSyxJQUFJLEVBQUUsQ0FBQztnQkFDeEIsT0FBTztvQkFDTCxNQUFNLEVBQUUsQ0FBQztvQkFDVCxRQUFRLEVBQUUsS0FBSztvQkFDZixVQUFVLEVBQUUsU0FBUztvQkFDckIsV0FBVyxFQUFFLENBQUMsK0NBQStDLE1BQU0sRUFBRSxDQUFDO2lCQUN2RSxDQUFDO1lBQ0osQ0FBQztZQUVELE1BQU0sV0FBVyxHQUFHLFVBQVUsR0FBRyxJQUFJLENBQUMsa0JBQWtCLENBQUM7WUFFekQsT0FBTztnQkFDTCxNQUFNLEVBQUUsV0FBVztnQkFDbkIsUUFBUSxFQUFFLEtBQUs7Z0JBQ2YsVUFBVSxFQUFFLFFBQVE7Z0JBQ3BCLFdBQVcsRUFBRTtvQkFDWCxXQUFXLElBQUksQ0FBQyxrQkFBa0IseUJBQXlCO29CQUMzRCxpREFBaUQ7aUJBQ2xEO2FBQ0YsQ0FBQztRQUNKLENBQUM7UUFBQyxPQUFPLEtBQUssRUFBRSxDQUFDO1lBQ2YsT0FBTztnQkFDTCxNQUFNLEVBQUUsQ0FBQztnQkFDVCxRQUFRLEVBQUUsS0FBSztnQkFDZixVQUFVLEVBQUUsU0FBUztnQkFDckIsV0FBVyxFQUFFLENBQUMsNEJBQTRCLEtBQUssWUFBWSxLQUFLLENBQUMsQ0FBQyxDQUFDLEtBQUssQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFDLE1BQU0sQ0FBQyxLQUFLLENBQUMsRUFBRSxDQUFDO2FBQ3BHLENBQUM7UUFDSixDQUFDO0lBQ0gsQ0FBQztJQUVPLGVBQWUsQ0FBQyxNQUFjO1FBQ3BDLE1BQU0sU0FBUyxHQUEyQjtZQUN4QyxXQUFXLEVBQUUsdUJBQXVCO1lBQ3BDLFdBQVcsRUFBRSxnQkFBZ0I7WUFDN0IsV0FBVyxFQUFFLHlCQUF5QjtZQUN0QyxXQUFXLEVBQUUsa0JBQWtCO1lBQy9CLFdBQVcsRUFBRSxjQUFjO1lBQzNCLFdBQVcsRUFBRSxhQUFhO1lBQzFCLFdBQVcsRUFBRSxZQUFZO1lBQ3pCLGNBQWMsRUFBRSxnQkFBZ0I7WUFDaEMsWUFBWSxFQUFFLGdCQUFnQjtZQUM5QixZQUFZLEVBQUUsdUJBQXVCO1lBQ3JDLGdCQUFnQixFQUFFLDBCQUEwQjtZQUM1QyxnQkFBZ0IsRUFBRSx1QkFBdUI7WUFDekMsZ0JBQWdCLEVBQUUsc0JBQXNCO1lBQ3hDLGdCQUFnQixFQUFFLHNCQUFzQjtTQUN6QyxDQUFDO1FBRUYsT0FBTyxTQUFTLENBQUMsTUFBTSxDQUFDLElBQUksTUFBTSxDQUFDO0lBQ3JDLENBQUM7Q0FDRjtBQXhFRCxvQ0F3RUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBSZXNvdXJjZVdpdGhJZCB9IGZyb20gJy4uLy4uL2RpZmYvdHlwZXMnO1xuaW1wb3J0IHsgUmVzb3VyY2VDb3N0Q2FsY3VsYXRvciwgTW9udGhseUNvc3QsIFByaWNpbmdDbGllbnQgfSBmcm9tICcuLi90eXBlcyc7XG5cbmV4cG9ydCBjbGFzcyBTM0NhbGN1bGF0b3IgaW1wbGVtZW50cyBSZXNvdXJjZUNvc3RDYWxjdWxhdG9yIHtcbiAgcHJpdmF0ZSByZWFkb25seSBERUZBVUxUX1NUT1JBR0VfR0IgPSAxMDA7XG5cbiAgc3VwcG9ydHMocmVzb3VyY2VUeXBlOiBzdHJpbmcpOiBib29sZWFuIHtcbiAgICByZXR1cm4gcmVzb3VyY2VUeXBlID09PSAnQVdTOjpTMzo6QnVja2V0JztcbiAgfVxuXG4gIGFzeW5jIGNhbGN1bGF0ZUNvc3QoXG4gICAgX3Jlc291cmNlOiBSZXNvdXJjZVdpdGhJZCxcbiAgICByZWdpb246IHN0cmluZyxcbiAgICBwcmljaW5nQ2xpZW50OiBQcmljaW5nQ2xpZW50LFxuICApOiBQcm9taXNlPE1vbnRobHlDb3N0PiB7XG4gICAgdHJ5IHtcbiAgICAgIGNvbnN0IHByaWNlUGVyR0IgPSBhd2FpdCBwcmljaW5nQ2xpZW50LmdldFByaWNlKHtcbiAgICAgICAgc2VydmljZUNvZGU6ICdBbWF6b25TMycsXG4gICAgICAgIHJlZ2lvbjogdGhpcy5ub3JtYWxpemVSZWdpb24ocmVnaW9uKSxcbiAgICAgICAgZmlsdGVyczogW1xuICAgICAgICAgIHsgZmllbGQ6ICdzdG9yYWdlQ2xhc3MnLCB2YWx1ZTogJ0dlbmVyYWwgUHVycG9zZScgfSxcbiAgICAgICAgICB7IGZpZWxkOiAndm9sdW1lVHlwZScsIHZhbHVlOiAnU3RhbmRhcmQnIH0sXG4gICAgICAgIF0sXG4gICAgICB9KTtcblxuICAgICAgaWYgKHByaWNlUGVyR0IgPT09IG51bGwpIHtcbiAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICBhbW91bnQ6IDAsXG4gICAgICAgICAgY3VycmVuY3k6ICdVU0QnLFxuICAgICAgICAgIGNvbmZpZGVuY2U6ICd1bmtub3duJyxcbiAgICAgICAgICBhc3N1bXB0aW9uczogW2BQcmljaW5nIGRhdGEgbm90IGF2YWlsYWJsZSBmb3IgUzMgaW4gcmVnaW9uICR7cmVnaW9ufWBdLFxuICAgICAgICB9O1xuICAgICAgfVxuXG4gICAgICBjb25zdCBtb250aGx5Q29zdCA9IHByaWNlUGVyR0IgKiB0aGlzLkRFRkFVTFRfU1RPUkFHRV9HQjtcblxuICAgICAgcmV0dXJuIHtcbiAgICAgICAgYW1vdW50OiBtb250aGx5Q29zdCxcbiAgICAgICAgY3VycmVuY3k6ICdVU0QnLFxuICAgICAgICBjb25maWRlbmNlOiAnbWVkaXVtJyxcbiAgICAgICAgYXNzdW1wdGlvbnM6IFtcbiAgICAgICAgICBgQXNzdW1lcyAke3RoaXMuREVGQVVMVF9TVE9SQUdFX0dCfSBHQiBvZiBzdGFuZGFyZCBzdG9yYWdlYCxcbiAgICAgICAgICAnRG9lcyBub3QgaW5jbHVkZSByZXF1ZXN0IGNvc3RzIG9yIGRhdGEgdHJhbnNmZXInLFxuICAgICAgICBdLFxuICAgICAgfTtcbiAgICB9IGNhdGNoIChlcnJvcikge1xuICAgICAgcmV0dXJuIHtcbiAgICAgICAgYW1vdW50OiAwLFxuICAgICAgICBjdXJyZW5jeTogJ1VTRCcsXG4gICAgICAgIGNvbmZpZGVuY2U6ICd1bmtub3duJyxcbiAgICAgICAgYXNzdW1wdGlvbnM6IFtgRmFpbGVkIHRvIGZldGNoIHByaWNpbmc6ICR7ZXJyb3IgaW5zdGFuY2VvZiBFcnJvciA/IGVycm9yLm1lc3NhZ2UgOiBTdHJpbmcoZXJyb3IpfWBdLFxuICAgICAgfTtcbiAgICB9XG4gIH1cblxuICBwcml2YXRlIG5vcm1hbGl6ZVJlZ2lvbihyZWdpb246IHN0cmluZyk6IHN0cmluZyB7XG4gICAgY29uc3QgcmVnaW9uTWFwOiBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+ID0ge1xuICAgICAgJ3VzLWVhc3QtMSc6ICdVUyBFYXN0IChOLiBWaXJnaW5pYSknLFxuICAgICAgJ3VzLWVhc3QtMic6ICdVUyBFYXN0IChPaGlvKScsXG4gICAgICAndXMtd2VzdC0xJzogJ1VTIFdlc3QgKE4uIENhbGlmb3JuaWEpJyxcbiAgICAgICd1cy13ZXN0LTInOiAnVVMgV2VzdCAoT3JlZ29uKScsXG4gICAgICAnZXUtd2VzdC0xJzogJ0VVIChJcmVsYW5kKScsXG4gICAgICAnZXUtd2VzdC0yJzogJ0VVIChMb25kb24pJyxcbiAgICAgICdldS13ZXN0LTMnOiAnRVUgKFBhcmlzKScsXG4gICAgICAnZXUtY2VudHJhbC0xJzogJ0VVIChGcmFua2Z1cnQpJyxcbiAgICAgICdldS1ub3J0aC0xJzogJ0VVIChTdG9ja2hvbG0pJyxcbiAgICAgICdhcC1zb3V0aC0xJzogJ0FzaWEgUGFjaWZpYyAoTXVtYmFpKScsXG4gICAgICAnYXAtc291dGhlYXN0LTEnOiAnQXNpYSBQYWNpZmljIChTaW5nYXBvcmUpJyxcbiAgICAgICdhcC1zb3V0aGVhc3QtMic6ICdBc2lhIFBhY2lmaWMgKFN5ZG5leSknLFxuICAgICAgJ2FwLW5vcnRoZWFzdC0xJzogJ0FzaWEgUGFjaWZpYyAoVG9reW8pJyxcbiAgICAgICdhcC1ub3J0aGVhc3QtMic6ICdBc2lhIFBhY2lmaWMgKFNlb3VsKScsXG4gICAgfTtcblxuICAgIHJldHVybiByZWdpb25NYXBbcmVnaW9uXSB8fCByZWdpb247XG4gIH1cbn1cbiJdfQ==
@@ -0,0 +1,12 @@
1
+ import { ResourceWithId } from '../../diff/types';
2
+ import { ResourceCostCalculator, MonthlyCost, PricingClient } from '../types';
3
+ export declare class VPCEndpointCalculator implements ResourceCostCalculator {
4
+ private customDataProcessedGB?;
5
+ private readonly DEFAULT_DATA_PROCESSED_GB;
6
+ private readonly HOURS_PER_MONTH;
7
+ constructor(customDataProcessedGB?: number | undefined);
8
+ supports(resourceType: string): boolean;
9
+ calculateCost(resource: ResourceWithId, region: string, pricingClient: PricingClient): Promise<MonthlyCost>;
10
+ private normalizeRegion;
11
+ private getRegionPrefix;
12
+ }