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,122 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.PricingClient = void 0;
4
+ const client_pricing_1 = require("@aws-sdk/client-pricing");
5
+ const types_1 = require("./types");
6
+ class PricingClient {
7
+ cache = new Map();
8
+ client;
9
+ cacheManager;
10
+ constructor(region = 'us-east-1', cacheManager, awsClient) {
11
+ this.client = awsClient || new client_pricing_1.PricingClient({ region });
12
+ this.cacheManager = cacheManager;
13
+ }
14
+ /**
15
+ * Clean up resources and connections
16
+ */
17
+ destroy() {
18
+ this.cache.clear();
19
+ // AWS SDK v3 clients automatically clean up their connections
20
+ // but we can help by clearing references
21
+ if (this.client && typeof this.client.destroy === 'function') {
22
+ this.client.destroy();
23
+ }
24
+ }
25
+ async getPrice(params) {
26
+ const cacheKey = this.getCacheKey(params);
27
+ // Check in-memory cache first
28
+ if (this.cache.has(cacheKey)) {
29
+ return this.cache.get(cacheKey);
30
+ }
31
+ // Check persistent cache if cache manager is available
32
+ if (this.cacheManager) {
33
+ const cachedPrice = this.cacheManager.getCachedPrice(params);
34
+ if (cachedPrice !== null) {
35
+ // Store in memory cache for faster subsequent access
36
+ this.cache.set(cacheKey, cachedPrice);
37
+ return cachedPrice;
38
+ }
39
+ }
40
+ try {
41
+ const price = await this.fetchPriceWithRetry(params);
42
+ if (price !== null) {
43
+ // Store in both memory and persistent cache
44
+ this.cache.set(cacheKey, price);
45
+ if (this.cacheManager) {
46
+ this.cacheManager.setCachedPrice(params, price);
47
+ }
48
+ }
49
+ return price;
50
+ }
51
+ catch (error) {
52
+ // On API failure, try to use cached data even if stale
53
+ if (this.cache.has(cacheKey)) {
54
+ return this.cache.get(cacheKey);
55
+ }
56
+ if (this.cacheManager) {
57
+ const cachedPrice = this.cacheManager.getCachedPrice(params);
58
+ if (cachedPrice !== null) {
59
+ return cachedPrice;
60
+ }
61
+ }
62
+ return null;
63
+ }
64
+ }
65
+ async fetchPriceWithRetry(params, maxRetries = 3) {
66
+ let lastError = null;
67
+ for (let attempt = 0; attempt < maxRetries; attempt++) {
68
+ try {
69
+ return await this.fetchPrice(params);
70
+ }
71
+ catch (error) {
72
+ lastError = error;
73
+ if (attempt < maxRetries - 1) {
74
+ const delay = Math.pow(2, attempt) * 1000;
75
+ await new Promise(resolve => setTimeout(resolve, delay));
76
+ }
77
+ }
78
+ }
79
+ throw new types_1.PricingAPIError(`Failed to fetch pricing after ${maxRetries} attempts: ${lastError?.message}`, false);
80
+ }
81
+ async fetchPrice(params) {
82
+ const filters = params.filters.map(f => ({
83
+ Type: f.type || 'TERM_MATCH',
84
+ Field: f.field,
85
+ Value: f.value,
86
+ }));
87
+ const command = new client_pricing_1.GetProductsCommand({
88
+ ServiceCode: params.serviceCode,
89
+ Filters: filters,
90
+ MaxResults: 1,
91
+ });
92
+ const response = await this.client.send(command);
93
+ if (!response.PriceList || response.PriceList.length === 0) {
94
+ return null;
95
+ }
96
+ const priceItem = JSON.parse(response.PriceList[0]);
97
+ const onDemand = priceItem?.terms?.OnDemand;
98
+ if (!onDemand) {
99
+ return null;
100
+ }
101
+ const termKey = Object.keys(onDemand)[0];
102
+ const priceDimensions = onDemand[termKey]?.priceDimensions;
103
+ if (!priceDimensions) {
104
+ return null;
105
+ }
106
+ const dimensionKey = Object.keys(priceDimensions)[0];
107
+ const pricePerUnit = priceDimensions[dimensionKey]?.pricePerUnit?.USD;
108
+ if (!pricePerUnit) {
109
+ return null;
110
+ }
111
+ return parseFloat(pricePerUnit);
112
+ }
113
+ getCacheKey(params) {
114
+ const filterStr = params.filters
115
+ .map(f => `${f.field}:${f.value}`)
116
+ .sort()
117
+ .join('|');
118
+ return `${params.serviceCode}:${params.region}:${filterStr}`;
119
+ }
120
+ }
121
+ exports.PricingClient = PricingClient;
122
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"PricingClient.js","sourceRoot":"","sources":["../../src/pricing/PricingClient.ts"],"names":[],"mappings":"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","sourcesContent":["import { PricingClient as AWSPricingClient, GetProductsCommand } from '@aws-sdk/client-pricing';\nimport { CacheManager } from './CacheManager';\nimport { PricingClient as IPricingClient, PriceQueryParams, PricingAPIError } from './types';\n\nexport class PricingClient implements IPricingClient {\n  private cache: Map<string, number> = new Map();\n  private client: AWSPricingClient;\n  private cacheManager?: CacheManager;\n\n  constructor(\n    region: string = 'us-east-1',\n    cacheManager?: CacheManager,\n    awsClient?: AWSPricingClient,\n  ) {\n    this.client = awsClient || new AWSPricingClient({ region });\n    this.cacheManager = cacheManager;\n  }\n\n  /**\n   * Clean up resources and connections\n   */\n  destroy(): void {\n    this.cache.clear();\n    // AWS SDK v3 clients automatically clean up their connections\n    // but we can help by clearing references\n    if (this.client && typeof (this.client as any).destroy === 'function') {\n      (this.client as any).destroy();\n    }\n  }\n\n  async getPrice(params: PriceQueryParams): Promise<number | null> {\n    const cacheKey = this.getCacheKey(params);\n\n    // Check in-memory cache first\n    if (this.cache.has(cacheKey)) {\n      return this.cache.get(cacheKey)!;\n    }\n\n    // Check persistent cache if cache manager is available\n    if (this.cacheManager) {\n      const cachedPrice = this.cacheManager.getCachedPrice(params);\n      if (cachedPrice !== null) {\n        // Store in memory cache for faster subsequent access\n        this.cache.set(cacheKey, cachedPrice);\n        return cachedPrice;\n      }\n    }\n\n    try {\n      const price = await this.fetchPriceWithRetry(params);\n      if (price !== null) {\n        // Store in both memory and persistent cache\n        this.cache.set(cacheKey, price);\n        if (this.cacheManager) {\n          this.cacheManager.setCachedPrice(params, price);\n        }\n      }\n      return price;\n    } catch (error) {\n      // On API failure, try to use cached data even if stale\n      if (this.cache.has(cacheKey)) {\n        return this.cache.get(cacheKey)!;\n      }\n      if (this.cacheManager) {\n        const cachedPrice = this.cacheManager.getCachedPrice(params);\n        if (cachedPrice !== null) {\n          return cachedPrice;\n        }\n      }\n      return null;\n    }\n  }\n\n  private async fetchPriceWithRetry(params: PriceQueryParams, maxRetries: number = 3): Promise<number | null> {\n    let lastError: Error | null = null;\n\n    for (let attempt = 0; attempt < maxRetries; attempt++) {\n      try {\n        return await this.fetchPrice(params);\n      } catch (error) {\n        lastError = error as Error;\n        if (attempt < maxRetries - 1) {\n          const delay = Math.pow(2, attempt) * 1000;\n          await new Promise(resolve => setTimeout(resolve, delay));\n        }\n      }\n    }\n\n    throw new PricingAPIError(\n      `Failed to fetch pricing after ${maxRetries} attempts: ${lastError?.message}`,\n      false,\n    );\n  }\n\n  private async fetchPrice(params: PriceQueryParams): Promise<number | null> {\n    const filters = params.filters.map(f => ({\n      Type: f.type || 'TERM_MATCH',\n      Field: f.field,\n      Value: f.value,\n    }));\n\n    const command = new GetProductsCommand({\n      ServiceCode: params.serviceCode,\n      Filters: filters,\n      MaxResults: 1,\n    });\n\n    const response = await this.client.send(command);\n\n    if (!response.PriceList || response.PriceList.length === 0) {\n      return null;\n    }\n\n    const priceItem = JSON.parse(response.PriceList[0]);\n    const onDemand = priceItem?.terms?.OnDemand;\n\n    if (!onDemand) {\n      return null;\n    }\n\n    const termKey = Object.keys(onDemand)[0];\n    const priceDimensions = onDemand[termKey]?.priceDimensions;\n\n    if (!priceDimensions) {\n      return null;\n    }\n\n    const dimensionKey = Object.keys(priceDimensions)[0];\n    const pricePerUnit = priceDimensions[dimensionKey]?.pricePerUnit?.USD;\n\n    if (!pricePerUnit) {\n      return null;\n    }\n\n    return parseFloat(pricePerUnit);\n  }\n\n  private getCacheKey(params: PriceQueryParams): string {\n    const filterStr = params.filters\n      .map(f => `${f.field}:${f.value}`)\n      .sort()\n      .join('|');\n    return `${params.serviceCode}:${params.region}:${filterStr}`;\n  }\n}\n"]}
@@ -0,0 +1,16 @@
1
+ import { PricingClient } from './PricingClient';
2
+ import { PricingService as IPricingService, MonthlyCost, CostDelta } from './types';
3
+ import { UsageAssumptionsConfig, CacheConfig } from '../config/types';
4
+ import { ResourceWithId, ResourceDiff } from '../diff/types';
5
+ export declare class PricingService implements IPricingService {
6
+ private calculators;
7
+ private pricingClient;
8
+ private excludedResourceTypes;
9
+ constructor(region?: string, usageAssumptions?: UsageAssumptionsConfig, excludedResourceTypes?: string[], cacheConfig?: CacheConfig, pricingClient?: PricingClient);
10
+ getResourceCost(resource: ResourceWithId, region: string): Promise<MonthlyCost>;
11
+ getCostDelta(diff: ResourceDiff, region: string): Promise<CostDelta>;
12
+ /**
13
+ * Clean up resources and connections
14
+ */
15
+ destroy(): void;
16
+ }
@@ -0,0 +1,149 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.PricingService = void 0;
4
+ const CacheManager_1 = require("./CacheManager");
5
+ const ALBCalculator_1 = require("./calculators/ALBCalculator");
6
+ const APIGatewayCalculator_1 = require("./calculators/APIGatewayCalculator");
7
+ const CloudFrontCalculator_1 = require("./calculators/CloudFrontCalculator");
8
+ const DynamoDBCalculator_1 = require("./calculators/DynamoDBCalculator");
9
+ const EC2Calculator_1 = require("./calculators/EC2Calculator");
10
+ const ECSCalculator_1 = require("./calculators/ECSCalculator");
11
+ const ElastiCacheCalculator_1 = require("./calculators/ElastiCacheCalculator");
12
+ const LambdaCalculator_1 = require("./calculators/LambdaCalculator");
13
+ const NatGatewayCalculator_1 = require("./calculators/NatGatewayCalculator");
14
+ const NLBCalculator_1 = require("./calculators/NLBCalculator");
15
+ const RDSCalculator_1 = require("./calculators/RDSCalculator");
16
+ const S3Calculator_1 = require("./calculators/S3Calculator");
17
+ const VPCEndpointCalculator_1 = require("./calculators/VPCEndpointCalculator");
18
+ const PricingClient_1 = require("./PricingClient");
19
+ class PricingService {
20
+ calculators;
21
+ pricingClient;
22
+ excludedResourceTypes;
23
+ constructor(region = 'us-east-1', usageAssumptions, excludedResourceTypes, cacheConfig, pricingClient) {
24
+ // Use provided pricing client or create a new one
25
+ if (pricingClient) {
26
+ this.pricingClient = pricingClient;
27
+ }
28
+ else {
29
+ // Initialize cache manager if caching is enabled
30
+ let cacheManager;
31
+ if (cacheConfig?.enabled !== false) {
32
+ const cacheDuration = cacheConfig?.durationHours ?? 24;
33
+ cacheManager = new CacheManager_1.CacheManager('.cdk-cost-analyzer-cache', cacheDuration);
34
+ }
35
+ this.pricingClient = new PricingClient_1.PricingClient(region, cacheManager);
36
+ }
37
+ this.excludedResourceTypes = new Set(excludedResourceTypes || []);
38
+ this.calculators = [
39
+ new EC2Calculator_1.EC2Calculator(),
40
+ new S3Calculator_1.S3Calculator(),
41
+ new LambdaCalculator_1.LambdaCalculator(usageAssumptions?.lambda?.invocationsPerMonth, usageAssumptions?.lambda?.averageDurationMs),
42
+ new RDSCalculator_1.RDSCalculator(),
43
+ new DynamoDBCalculator_1.DynamoDBCalculator(),
44
+ new ECSCalculator_1.ECSCalculator(),
45
+ new APIGatewayCalculator_1.APIGatewayCalculator(),
46
+ new NatGatewayCalculator_1.NatGatewayCalculator(usageAssumptions?.natGateway?.dataProcessedGB),
47
+ new ALBCalculator_1.ALBCalculator(usageAssumptions?.alb?.newConnectionsPerSecond, usageAssumptions?.alb?.activeConnectionsPerMinute, usageAssumptions?.alb?.processedBytesGB),
48
+ new NLBCalculator_1.NLBCalculator(usageAssumptions?.nlb?.newConnectionsPerSecond, usageAssumptions?.nlb?.activeConnectionsPerMinute, usageAssumptions?.nlb?.processedBytesGB),
49
+ new VPCEndpointCalculator_1.VPCEndpointCalculator(usageAssumptions?.vpcEndpoint?.dataProcessedGB),
50
+ new CloudFrontCalculator_1.CloudFrontCalculator(usageAssumptions?.cloudfront?.dataTransferGB, usageAssumptions?.cloudfront?.requests),
51
+ new ElastiCacheCalculator_1.ElastiCacheCalculator(),
52
+ ];
53
+ }
54
+ async getResourceCost(resource, region) {
55
+ // Check if resource type is excluded
56
+ if (this.excludedResourceTypes.has(resource.type)) {
57
+ return {
58
+ amount: 0,
59
+ currency: 'USD',
60
+ confidence: 'high',
61
+ assumptions: [`Resource type ${resource.type} is excluded from cost analysis`],
62
+ };
63
+ }
64
+ const calculator = this.calculators.find(calc => calc.supports(resource.type));
65
+ if (!calculator) {
66
+ return {
67
+ amount: 0,
68
+ currency: 'USD',
69
+ confidence: 'unknown',
70
+ assumptions: [`Resource type ${resource.type} is not supported`],
71
+ };
72
+ }
73
+ try {
74
+ return await calculator.calculateCost(resource, region, this.pricingClient);
75
+ }
76
+ catch (error) {
77
+ return {
78
+ amount: 0,
79
+ currency: 'USD',
80
+ confidence: 'unknown',
81
+ assumptions: [`Failed to calculate cost: ${error instanceof Error ? error.message : String(error)}`],
82
+ };
83
+ }
84
+ }
85
+ async getCostDelta(diff, region) {
86
+ // Filter out excluded resources before processing
87
+ const filteredAdded = diff.added.filter(r => !this.excludedResourceTypes.has(r.type));
88
+ const filteredRemoved = diff.removed.filter(r => !this.excludedResourceTypes.has(r.type));
89
+ const filteredModified = diff.modified.filter(r => !this.excludedResourceTypes.has(r.type));
90
+ const addedCosts = await Promise.all(filteredAdded.map(async (resource) => {
91
+ const monthlyCost = await this.getResourceCost(resource, region);
92
+ return {
93
+ logicalId: resource.logicalId,
94
+ type: resource.type,
95
+ monthlyCost,
96
+ };
97
+ }));
98
+ const removedCosts = await Promise.all(filteredRemoved.map(async (resource) => {
99
+ const monthlyCost = await this.getResourceCost(resource, region);
100
+ return {
101
+ logicalId: resource.logicalId,
102
+ type: resource.type,
103
+ monthlyCost,
104
+ };
105
+ }));
106
+ const modifiedCosts = await Promise.all(filteredModified.map(async (resource) => {
107
+ const oldResource = {
108
+ logicalId: resource.logicalId,
109
+ type: resource.type,
110
+ properties: resource.oldProperties,
111
+ };
112
+ const newResource = {
113
+ logicalId: resource.logicalId,
114
+ type: resource.type,
115
+ properties: resource.newProperties,
116
+ };
117
+ const oldMonthlyCost = await this.getResourceCost(oldResource, region);
118
+ const newMonthlyCost = await this.getResourceCost(newResource, region);
119
+ const costDelta = newMonthlyCost.amount - oldMonthlyCost.amount;
120
+ return {
121
+ logicalId: resource.logicalId,
122
+ type: resource.type,
123
+ monthlyCost: newMonthlyCost,
124
+ oldMonthlyCost,
125
+ newMonthlyCost,
126
+ costDelta,
127
+ };
128
+ }));
129
+ const totalAddedCost = addedCosts.reduce((sum, r) => sum + r.monthlyCost.amount, 0);
130
+ const totalRemovedCost = removedCosts.reduce((sum, r) => sum + r.monthlyCost.amount, 0);
131
+ const totalModifiedDelta = modifiedCosts.reduce((sum, r) => sum + r.costDelta, 0);
132
+ const totalDelta = totalAddedCost - totalRemovedCost + totalModifiedDelta;
133
+ return {
134
+ totalDelta,
135
+ currency: 'USD',
136
+ addedCosts,
137
+ removedCosts,
138
+ modifiedCosts,
139
+ };
140
+ }
141
+ /**
142
+ * Clean up resources and connections
143
+ */
144
+ destroy() {
145
+ this.pricingClient.destroy();
146
+ }
147
+ }
148
+ exports.PricingService = PricingService;
149
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"PricingService.js","sourceRoot":"","sources":["../../src/pricing/PricingService.ts"],"names":[],"mappings":";;;AAAA,iDAA8C;AAC9C,+DAA4D;AAC5D,6EAA0E;AAC1E,6EAA0E;AAC1E,yEAAsE;AACtE,+DAA4D;AAC5D,+DAA4D;AAC5D,+EAA4E;AAC5E,qEAAkE;AAClE,6EAA0E;AAC1E,+DAA4D;AAC5D,+DAA4D;AAC5D,6DAA0D;AAC1D,+EAA4E;AAC5E,mDAAgD;AAUhD,MAAa,cAAc;IACjB,WAAW,CAA2B;IACtC,aAAa,CAAgB;IAC7B,qBAAqB,CAAc;IAE3C,YACE,SAAiB,WAAW,EAC5B,gBAAyC,EACzC,qBAAgC,EAChC,WAAyB,EACzB,aAA6B;QAE7B,kDAAkD;QAClD,IAAI,aAAa,EAAE,CAAC;YAClB,IAAI,CAAC,aAAa,GAAG,aAAa,CAAC;QACrC,CAAC;aAAM,CAAC;YACN,iDAAiD;YACjD,IAAI,YAAsC,CAAC;YAC3C,IAAI,WAAW,EAAE,OAAO,KAAK,KAAK,EAAE,CAAC;gBACnC,MAAM,aAAa,GAAG,WAAW,EAAE,aAAa,IAAI,EAAE,CAAC;gBACvD,YAAY,GAAG,IAAI,2BAAY,CAAC,0BAA0B,EAAE,aAAa,CAAC,CAAC;YAC7E,CAAC;YAED,IAAI,CAAC,aAAa,GAAG,IAAI,6BAAa,CAAC,MAAM,EAAE,YAAY,CAAC,CAAC;QAC/D,CAAC;QACD,IAAI,CAAC,qBAAqB,GAAG,IAAI,GAAG,CAAC,qBAAqB,IAAI,EAAE,CAAC,CAAC;QAClE,IAAI,CAAC,WAAW,GAAG;YACjB,IAAI,6BAAa,EAAE;YACnB,IAAI,2BAAY,EAAE;YAClB,IAAI,mCAAgB,CAClB,gBAAgB,EAAE,MAAM,EAAE,mBAAmB,EAC7C,gBAAgB,EAAE,MAAM,EAAE,iBAAiB,CAC5C;YACD,IAAI,6BAAa,EAAE;YACnB,IAAI,uCAAkB,EAAE;YACxB,IAAI,6BAAa,EAAE;YACnB,IAAI,2CAAoB,EAAE;YAC1B,IAAI,2CAAoB,CAAC,gBAAgB,EAAE,UAAU,EAAE,eAAe,CAAC;YACvE,IAAI,6BAAa,CACf,gBAAgB,EAAE,GAAG,EAAE,uBAAuB,EAC9C,gBAAgB,EAAE,GAAG,EAAE,0BAA0B,EACjD,gBAAgB,EAAE,GAAG,EAAE,gBAAgB,CACxC;YACD,IAAI,6BAAa,CACf,gBAAgB,EAAE,GAAG,EAAE,uBAAuB,EAC9C,gBAAgB,EAAE,GAAG,EAAE,0BAA0B,EACjD,gBAAgB,EAAE,GAAG,EAAE,gBAAgB,CACxC;YACD,IAAI,6CAAqB,CAAC,gBAAgB,EAAE,WAAW,EAAE,eAAe,CAAC;YACzE,IAAI,2CAAoB,CACtB,gBAAgB,EAAE,UAAU,EAAE,cAAc,EAC5C,gBAAgB,EAAE,UAAU,EAAE,QAAQ,CACvC;YACD,IAAI,6CAAqB,EAAE;SAC5B,CAAC;IACJ,CAAC;IAED,KAAK,CAAC,eAAe,CAAC,QAAwB,EAAE,MAAc;QAC5D,qCAAqC;QACrC,IAAI,IAAI,CAAC,qBAAqB,CAAC,GAAG,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC;YAClD,OAAO;gBACL,MAAM,EAAE,CAAC;gBACT,QAAQ,EAAE,KAAK;gBACf,UAAU,EAAE,MAAM;gBAClB,WAAW,EAAE,CAAC,iBAAiB,QAAQ,CAAC,IAAI,iCAAiC,CAAC;aAC/E,CAAC;QACJ,CAAC;QAED,MAAM,UAAU,GAAG,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC;QAE/E,IAAI,CAAC,UAAU,EAAE,CAAC;YAChB,OAAO;gBACL,MAAM,EAAE,CAAC;gBACT,QAAQ,EAAE,KAAK;gBACf,UAAU,EAAE,SAAS;gBACrB,WAAW,EAAE,CAAC,iBAAiB,QAAQ,CAAC,IAAI,mBAAmB,CAAC;aACjE,CAAC;QACJ,CAAC;QAED,IAAI,CAAC;YACH,OAAO,MAAM,UAAU,CAAC,aAAa,CAAC,QAAQ,EAAE,MAAM,EAAE,IAAI,CAAC,aAAa,CAAC,CAAC;QAC9E,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO;gBACL,MAAM,EAAE,CAAC;gBACT,QAAQ,EAAE,KAAK;gBACf,UAAU,EAAE,SAAS;gBACrB,WAAW,EAAE,CAAC,6BAA6B,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC;aACrG,CAAC;QACJ,CAAC;IACH,CAAC;IAED,KAAK,CAAC,YAAY,CAAC,IAAkB,EAAE,MAAc;QACnD,kDAAkD;QAClD,MAAM,aAAa,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,IAAI,CAAC,qBAAqB,CAAC,GAAG,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC;QACtF,MAAM,eAAe,GAAG,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,IAAI,CAAC,qBAAqB,CAAC,GAAG,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC;QAC1F,MAAM,gBAAgB,GAAG,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,IAAI,CAAC,qBAAqB,CAAC,GAAG,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC;QAE5F,MAAM,UAAU,GAAG,MAAM,OAAO,CAAC,GAAG,CAClC,aAAa,CAAC,GAAG,CAAC,KAAK,EAAE,QAAQ,EAAE,EAAE;YACnC,MAAM,WAAW,GAAG,MAAM,IAAI,CAAC,eAAe,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC;YACjE,OAAO;gBACL,SAAS,EAAE,QAAQ,CAAC,SAAS;gBAC7B,IAAI,EAAE,QAAQ,CAAC,IAAI;gBACnB,WAAW;aACZ,CAAC;QACJ,CAAC,CAAC,CACH,CAAC;QAEF,MAAM,YAAY,GAAG,MAAM,OAAO,CAAC,GAAG,CACpC,eAAe,CAAC,GAAG,CAAC,KAAK,EAAE,QAAQ,EAAE,EAAE;YACrC,MAAM,WAAW,GAAG,MAAM,IAAI,CAAC,eAAe,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC;YACjE,OAAO;gBACL,SAAS,EAAE,QAAQ,CAAC,SAAS;gBAC7B,IAAI,EAAE,QAAQ,CAAC,IAAI;gBACnB,WAAW;aACZ,CAAC;QACJ,CAAC,CAAC,CACH,CAAC;QAEF,MAAM,aAAa,GAAG,MAAM,OAAO,CAAC,GAAG,CACrC,gBAAgB,CAAC,GAAG,CAAC,KAAK,EAAE,QAAQ,EAAE,EAAE;YACtC,MAAM,WAAW,GAAmB;gBAClC,SAAS,EAAE,QAAQ,CAAC,SAAS;gBAC7B,IAAI,EAAE,QAAQ,CAAC,IAAI;gBACnB,UAAU,EAAE,QAAQ,CAAC,aAAa;aACnC,CAAC;YACF,MAAM,WAAW,GAAmB;gBAClC,SAAS,EAAE,QAAQ,CAAC,SAAS;gBAC7B,IAAI,EAAE,QAAQ,CAAC,IAAI;gBACnB,UAAU,EAAE,QAAQ,CAAC,aAAa;aACnC,CAAC;YAEF,MAAM,cAAc,GAAG,MAAM,IAAI,CAAC,eAAe,CAAC,WAAW,EAAE,MAAM,CAAC,CAAC;YACvE,MAAM,cAAc,GAAG,MAAM,IAAI,CAAC,eAAe,CAAC,WAAW,EAAE,MAAM,CAAC,CAAC;YACvE,MAAM,SAAS,GAAG,cAAc,CAAC,MAAM,GAAG,cAAc,CAAC,MAAM,CAAC;YAEhE,OAAO;gBACL,SAAS,EAAE,QAAQ,CAAC,SAAS;gBAC7B,IAAI,EAAE,QAAQ,CAAC,IAAI;gBACnB,WAAW,EAAE,cAAc;gBAC3B,cAAc;gBACd,cAAc;gBACd,SAAS;aACV,CAAC;QACJ,CAAC,CAAC,CACH,CAAC;QAEF,MAAM,cAAc,GAAG,UAAU,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC,GAAG,GAAG,CAAC,CAAC,WAAW,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;QACpF,MAAM,gBAAgB,GAAG,YAAY,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC,GAAG,GAAG,CAAC,CAAC,WAAW,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;QACxF,MAAM,kBAAkB,GAAG,aAAa,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC,GAAG,GAAG,CAAC,CAAC,SAAS,EAAE,CAAC,CAAC,CAAC;QAElF,MAAM,UAAU,GAAG,cAAc,GAAG,gBAAgB,GAAG,kBAAkB,CAAC;QAE1E,OAAO;YACL,UAAU;YACV,QAAQ,EAAE,KAAK;YACf,UAAU;YACV,YAAY;YACZ,aAAa;SACd,CAAC;IACJ,CAAC;IAED;;OAEG;IACH,OAAO;QACL,IAAI,CAAC,aAAa,CAAC,OAAO,EAAE,CAAC;IAC/B,CAAC;CACF;AAxKD,wCAwKC","sourcesContent":["import { CacheManager } from './CacheManager';\nimport { ALBCalculator } from './calculators/ALBCalculator';\nimport { APIGatewayCalculator } from './calculators/APIGatewayCalculator';\nimport { CloudFrontCalculator } from './calculators/CloudFrontCalculator';\nimport { DynamoDBCalculator } from './calculators/DynamoDBCalculator';\nimport { EC2Calculator } from './calculators/EC2Calculator';\nimport { ECSCalculator } from './calculators/ECSCalculator';\nimport { ElastiCacheCalculator } from './calculators/ElastiCacheCalculator';\nimport { LambdaCalculator } from './calculators/LambdaCalculator';\nimport { NatGatewayCalculator } from './calculators/NatGatewayCalculator';\nimport { NLBCalculator } from './calculators/NLBCalculator';\nimport { RDSCalculator } from './calculators/RDSCalculator';\nimport { S3Calculator } from './calculators/S3Calculator';\nimport { VPCEndpointCalculator } from './calculators/VPCEndpointCalculator';\nimport { PricingClient } from './PricingClient';\nimport {\n  PricingService as IPricingService,\n  MonthlyCost,\n  CostDelta,\n  ResourceCostCalculator,\n} from './types';\nimport { UsageAssumptionsConfig, CacheConfig } from '../config/types';\nimport { ResourceWithId, ResourceDiff } from '../diff/types';\n\nexport class PricingService implements IPricingService {\n  private calculators: ResourceCostCalculator[];\n  private pricingClient: PricingClient;\n  private excludedResourceTypes: Set<string>;\n\n  constructor(\n    region: string = 'us-east-1',\n    usageAssumptions?: UsageAssumptionsConfig,\n    excludedResourceTypes?: string[],\n    cacheConfig?: CacheConfig,\n    pricingClient?: PricingClient,\n  ) {\n    // Use provided pricing client or create a new one\n    if (pricingClient) {\n      this.pricingClient = pricingClient;\n    } else {\n      // Initialize cache manager if caching is enabled\n      let cacheManager: CacheManager | undefined;\n      if (cacheConfig?.enabled !== false) {\n        const cacheDuration = cacheConfig?.durationHours ?? 24;\n        cacheManager = new CacheManager('.cdk-cost-analyzer-cache', cacheDuration);\n      }\n\n      this.pricingClient = new PricingClient(region, cacheManager);\n    }\n    this.excludedResourceTypes = new Set(excludedResourceTypes || []);\n    this.calculators = [\n      new EC2Calculator(),\n      new S3Calculator(),\n      new LambdaCalculator(\n        usageAssumptions?.lambda?.invocationsPerMonth,\n        usageAssumptions?.lambda?.averageDurationMs,\n      ),\n      new RDSCalculator(),\n      new DynamoDBCalculator(),\n      new ECSCalculator(),\n      new APIGatewayCalculator(),\n      new NatGatewayCalculator(usageAssumptions?.natGateway?.dataProcessedGB),\n      new ALBCalculator(\n        usageAssumptions?.alb?.newConnectionsPerSecond,\n        usageAssumptions?.alb?.activeConnectionsPerMinute,\n        usageAssumptions?.alb?.processedBytesGB,\n      ),\n      new NLBCalculator(\n        usageAssumptions?.nlb?.newConnectionsPerSecond,\n        usageAssumptions?.nlb?.activeConnectionsPerMinute,\n        usageAssumptions?.nlb?.processedBytesGB,\n      ),\n      new VPCEndpointCalculator(usageAssumptions?.vpcEndpoint?.dataProcessedGB),\n      new CloudFrontCalculator(\n        usageAssumptions?.cloudfront?.dataTransferGB,\n        usageAssumptions?.cloudfront?.requests,\n      ),\n      new ElastiCacheCalculator(),\n    ];\n  }\n\n  async getResourceCost(resource: ResourceWithId, region: string): Promise<MonthlyCost> {\n    // Check if resource type is excluded\n    if (this.excludedResourceTypes.has(resource.type)) {\n      return {\n        amount: 0,\n        currency: 'USD',\n        confidence: 'high',\n        assumptions: [`Resource type ${resource.type} is excluded from cost analysis`],\n      };\n    }\n\n    const calculator = this.calculators.find(calc => calc.supports(resource.type));\n\n    if (!calculator) {\n      return {\n        amount: 0,\n        currency: 'USD',\n        confidence: 'unknown',\n        assumptions: [`Resource type ${resource.type} is not supported`],\n      };\n    }\n\n    try {\n      return await calculator.calculateCost(resource, region, this.pricingClient);\n    } catch (error) {\n      return {\n        amount: 0,\n        currency: 'USD',\n        confidence: 'unknown',\n        assumptions: [`Failed to calculate cost: ${error instanceof Error ? error.message : String(error)}`],\n      };\n    }\n  }\n\n  async getCostDelta(diff: ResourceDiff, region: string): Promise<CostDelta> {\n    // Filter out excluded resources before processing\n    const filteredAdded = diff.added.filter(r => !this.excludedResourceTypes.has(r.type));\n    const filteredRemoved = diff.removed.filter(r => !this.excludedResourceTypes.has(r.type));\n    const filteredModified = diff.modified.filter(r => !this.excludedResourceTypes.has(r.type));\n\n    const addedCosts = await Promise.all(\n      filteredAdded.map(async (resource) => {\n        const monthlyCost = await this.getResourceCost(resource, region);\n        return {\n          logicalId: resource.logicalId,\n          type: resource.type,\n          monthlyCost,\n        };\n      }),\n    );\n\n    const removedCosts = await Promise.all(\n      filteredRemoved.map(async (resource) => {\n        const monthlyCost = await this.getResourceCost(resource, region);\n        return {\n          logicalId: resource.logicalId,\n          type: resource.type,\n          monthlyCost,\n        };\n      }),\n    );\n\n    const modifiedCosts = await Promise.all(\n      filteredModified.map(async (resource) => {\n        const oldResource: ResourceWithId = {\n          logicalId: resource.logicalId,\n          type: resource.type,\n          properties: resource.oldProperties,\n        };\n        const newResource: ResourceWithId = {\n          logicalId: resource.logicalId,\n          type: resource.type,\n          properties: resource.newProperties,\n        };\n\n        const oldMonthlyCost = await this.getResourceCost(oldResource, region);\n        const newMonthlyCost = await this.getResourceCost(newResource, region);\n        const costDelta = newMonthlyCost.amount - oldMonthlyCost.amount;\n\n        return {\n          logicalId: resource.logicalId,\n          type: resource.type,\n          monthlyCost: newMonthlyCost,\n          oldMonthlyCost,\n          newMonthlyCost,\n          costDelta,\n        };\n      }),\n    );\n\n    const totalAddedCost = addedCosts.reduce((sum, r) => sum + r.monthlyCost.amount, 0);\n    const totalRemovedCost = removedCosts.reduce((sum, r) => sum + r.monthlyCost.amount, 0);\n    const totalModifiedDelta = modifiedCosts.reduce((sum, r) => sum + r.costDelta, 0);\n\n    const totalDelta = totalAddedCost - totalRemovedCost + totalModifiedDelta;\n\n    return {\n      totalDelta,\n      currency: 'USD',\n      addedCosts,\n      removedCosts,\n      modifiedCosts,\n    };\n  }\n\n  /**\n   * Clean up resources and connections\n   */\n  destroy(): void {\n    this.pricingClient.destroy();\n  }\n}\n"]}
@@ -0,0 +1,16 @@
1
+ import { ResourceWithId } from '../../diff/types';
2
+ import { ResourceCostCalculator, MonthlyCost, PricingClient } from '../types';
3
+ export declare class ALBCalculator 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
+ }
@@ -0,0 +1,163 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.ALBCalculator = void 0;
4
+ class ALBCalculator {
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 an Application Load Balancer (not Network Load Balancer)
22
+ const loadBalancerType = resource.properties?.Type;
23
+ if (loadBalancerType && loadBalancerType !== 'application') {
24
+ return {
25
+ amount: 0,
26
+ currency: 'USD',
27
+ confidence: 'unknown',
28
+ assumptions: ['This calculator only supports Application 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-Application' },
38
+ { field: 'usagetype', value: `${this.getRegionPrefix(region)}LoadBalancerUsage` },
39
+ ],
40
+ });
41
+ // Get LCU rate
42
+ const lcuRate = await pricingClient.getPrice({
43
+ serviceCode: 'AWSELB',
44
+ region: this.normalizeRegion(region),
45
+ filters: [
46
+ { field: 'productFamily', value: 'Load Balancer-Application' },
47
+ { field: 'usagetype', value: `${this.getRegionPrefix(region)}LCUUsage` },
48
+ ],
49
+ });
50
+ const newConnectionsPerSecond = this.customNewConnectionsPerSecond || this.DEFAULT_NEW_CONNECTIONS_PER_SECOND;
51
+ const activeConnectionsPerMinute = this.customActiveConnectionsPerMinute || this.DEFAULT_ACTIVE_CONNECTIONS_PER_MINUTE;
52
+ const processedBytesGB = this.customProcessedBytesGB || this.DEFAULT_PROCESSED_BYTES_GB;
53
+ if (hourlyRate === null || lcuRate === null) {
54
+ // Calculate LCU consumption for assumptions even when pricing fails
55
+ const lcuFromNewConnections = newConnectionsPerSecond / 25;
56
+ const lcuFromActiveConnections = activeConnectionsPerMinute / 3000;
57
+ const gbPerHour = processedBytesGB / this.HOURS_PER_MONTH;
58
+ const lcuFromProcessedBytes = gbPerHour;
59
+ const lcuPerHour = Math.max(lcuFromNewConnections, lcuFromActiveConnections, lcuFromProcessedBytes);
60
+ return {
61
+ amount: 0,
62
+ currency: 'USD',
63
+ confidence: 'unknown',
64
+ assumptions: [
65
+ `Pricing data not available for Application Load Balancer in region ${region}`,
66
+ `Would use LCU consumption: ${lcuPerHour.toFixed(2)} LCU/hour based on:`,
67
+ ` - New connections: ${newConnectionsPerSecond}/sec → ${lcuFromNewConnections.toFixed(2)} LCU`,
68
+ ` - Active connections: ${activeConnectionsPerMinute}/min → ${lcuFromActiveConnections.toFixed(2)} LCU`,
69
+ ` - Processed data: ${processedBytesGB} GB/month → ${lcuFromProcessedBytes.toFixed(2)} LCU`,
70
+ ],
71
+ };
72
+ }
73
+ // Calculate LCU consumption
74
+ // 1 LCU provides: 25 new connections/sec, 3000 active connections/min, 1 GB processed/hour, 1000 rule evaluations/sec
75
+ const lcuFromNewConnections = newConnectionsPerSecond / 25;
76
+ const lcuFromActiveConnections = activeConnectionsPerMinute / 3000;
77
+ const gbPerHour = processedBytesGB / this.HOURS_PER_MONTH;
78
+ const lcuFromProcessedBytes = gbPerHour;
79
+ // Use the highest LCU consumption
80
+ const lcuPerHour = Math.max(lcuFromNewConnections, lcuFromActiveConnections, lcuFromProcessedBytes);
81
+ const hourlyCost = hourlyRate * this.HOURS_PER_MONTH;
82
+ const lcuCost = lcuRate * lcuPerHour * this.HOURS_PER_MONTH;
83
+ const totalCost = hourlyCost + lcuCost;
84
+ return {
85
+ amount: totalCost,
86
+ currency: 'USD',
87
+ confidence: 'medium',
88
+ assumptions: [
89
+ `Hourly rate: $${hourlyRate.toFixed(4)}/hour × ${this.HOURS_PER_MONTH} hours = $${hourlyCost.toFixed(2)}/month`,
90
+ `LCU consumption: ${lcuPerHour.toFixed(2)} LCU/hour based on:`,
91
+ ` - New connections: ${newConnectionsPerSecond}/sec → ${lcuFromNewConnections.toFixed(2)} LCU`,
92
+ ` - Active connections: ${activeConnectionsPerMinute}/min → ${lcuFromActiveConnections.toFixed(2)} LCU`,
93
+ ` - Processed data: ${processedBytesGB} GB/month → ${lcuFromProcessedBytes.toFixed(2)} LCU`,
94
+ `LCU cost: $${lcuRate.toFixed(4)}/LCU/hour × ${lcuPerHour.toFixed(2)} LCU × ${this.HOURS_PER_MONTH} hours = $${lcuCost.toFixed(2)}/month`,
95
+ `Total: $${totalCost.toFixed(2)}/month`,
96
+ ],
97
+ };
98
+ }
99
+ catch (error) {
100
+ const newConnectionsPerSecond = this.customNewConnectionsPerSecond || this.DEFAULT_NEW_CONNECTIONS_PER_SECOND;
101
+ const activeConnectionsPerMinute = this.customActiveConnectionsPerMinute || this.DEFAULT_ACTIVE_CONNECTIONS_PER_MINUTE;
102
+ const processedBytesGB = this.customProcessedBytesGB || this.DEFAULT_PROCESSED_BYTES_GB;
103
+ // Calculate LCU consumption for assumptions even when error occurs
104
+ const lcuFromNewConnections = newConnectionsPerSecond / 25;
105
+ const lcuFromActiveConnections = activeConnectionsPerMinute / 3000;
106
+ const gbPerHour = processedBytesGB / this.HOURS_PER_MONTH;
107
+ const lcuFromProcessedBytes = gbPerHour;
108
+ const lcuPerHour = Math.max(lcuFromNewConnections, lcuFromActiveConnections, lcuFromProcessedBytes);
109
+ return {
110
+ amount: 0,
111
+ currency: 'USD',
112
+ confidence: 'unknown',
113
+ assumptions: [
114
+ `Failed to fetch pricing: ${error instanceof Error ? error.message : String(error)}`,
115
+ `Would use LCU consumption: ${lcuPerHour.toFixed(2)} LCU/hour based on:`,
116
+ ` - New connections: ${newConnectionsPerSecond}/sec → ${lcuFromNewConnections.toFixed(2)} LCU`,
117
+ ` - Active connections: ${activeConnectionsPerMinute}/min → ${lcuFromActiveConnections.toFixed(2)} LCU`,
118
+ ` - Processed data: ${processedBytesGB} GB/month → ${lcuFromProcessedBytes.toFixed(2)} LCU`,
119
+ ],
120
+ };
121
+ }
122
+ }
123
+ normalizeRegion(region) {
124
+ const regionMap = {
125
+ 'us-east-1': 'US East (N. Virginia)',
126
+ 'us-east-2': 'US East (Ohio)',
127
+ 'us-west-1': 'US West (N. California)',
128
+ 'us-west-2': 'US West (Oregon)',
129
+ 'eu-west-1': 'EU (Ireland)',
130
+ 'eu-west-2': 'EU (London)',
131
+ 'eu-west-3': 'EU (Paris)',
132
+ 'eu-central-1': 'EU (Frankfurt)',
133
+ 'eu-north-1': 'EU (Stockholm)',
134
+ 'ap-south-1': 'Asia Pacific (Mumbai)',
135
+ 'ap-southeast-1': 'Asia Pacific (Singapore)',
136
+ 'ap-southeast-2': 'Asia Pacific (Sydney)',
137
+ 'ap-northeast-1': 'Asia Pacific (Tokyo)',
138
+ 'ap-northeast-2': 'Asia Pacific (Seoul)',
139
+ };
140
+ return regionMap[region] || region;
141
+ }
142
+ getRegionPrefix(region) {
143
+ const prefixMap = {
144
+ 'us-east-1': 'USE1',
145
+ 'us-east-2': 'USE2',
146
+ 'us-west-1': 'USW1',
147
+ 'us-west-2': 'USW2',
148
+ 'eu-west-1': 'EUW1',
149
+ 'eu-west-2': 'EUW2',
150
+ 'eu-west-3': 'EUW3',
151
+ 'eu-central-1': 'EUC1',
152
+ 'eu-north-1': 'EUN1',
153
+ 'ap-south-1': 'APS1',
154
+ 'ap-southeast-1': 'APS2',
155
+ 'ap-southeast-2': 'APS3',
156
+ 'ap-northeast-1': 'APN1',
157
+ 'ap-northeast-2': 'APN2',
158
+ };
159
+ return prefixMap[region] || '';
160
+ }
161
+ }
162
+ exports.ALBCalculator = ALBCalculator;
163
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ALBCalculator.js","sourceRoot":"","sources":["../../../src/pricing/calculators/ALBCalculator.ts"],"names":[],"mappings":"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","sourcesContent":["import { ResourceWithId } from '../../diff/types';\nimport { ResourceCostCalculator, MonthlyCost, PricingClient } from '../types';\n\nexport class ALBCalculator 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 an Application Load Balancer (not Network Load Balancer)\n    const loadBalancerType = resource.properties?.Type;\n    if (loadBalancerType && loadBalancerType !== 'application') {\n      return {\n        amount: 0,\n        currency: 'USD',\n        confidence: 'unknown',\n        assumptions: ['This calculator only supports Application 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-Application' },\n          { field: 'usagetype', value: `${this.getRegionPrefix(region)}LoadBalancerUsage` },\n        ],\n      });\n\n      // Get LCU rate\n      const lcuRate = await pricingClient.getPrice({\n        serviceCode: 'AWSELB',\n        region: this.normalizeRegion(region),\n        filters: [\n          { field: 'productFamily', value: 'Load Balancer-Application' },\n          { field: 'usagetype', value: `${this.getRegionPrefix(region)}LCUUsage` },\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      if (hourlyRate === null || lcuRate === null) {\n        // Calculate LCU consumption for assumptions even when pricing fails\n        const lcuFromNewConnections = newConnectionsPerSecond / 25;\n        const lcuFromActiveConnections = activeConnectionsPerMinute / 3000;\n        const gbPerHour = processedBytesGB / this.HOURS_PER_MONTH;\n        const lcuFromProcessedBytes = gbPerHour;\n        const lcuPerHour = Math.max(\n          lcuFromNewConnections,\n          lcuFromActiveConnections,\n          lcuFromProcessedBytes,\n        );\n\n        return {\n          amount: 0,\n          currency: 'USD',\n          confidence: 'unknown',\n          assumptions: [\n            `Pricing data not available for Application Load Balancer in region ${region}`,\n            `Would use LCU consumption: ${lcuPerHour.toFixed(2)} LCU/hour based on:`,\n            `  - New connections: ${newConnectionsPerSecond}/sec → ${lcuFromNewConnections.toFixed(2)} LCU`,\n            `  - Active connections: ${activeConnectionsPerMinute}/min → ${lcuFromActiveConnections.toFixed(2)} LCU`,\n            `  - Processed data: ${processedBytesGB} GB/month → ${lcuFromProcessedBytes.toFixed(2)} LCU`,\n          ],\n        };\n      }\n\n      // Calculate LCU consumption\n      // 1 LCU provides: 25 new connections/sec, 3000 active connections/min, 1 GB processed/hour, 1000 rule evaluations/sec\n      const lcuFromNewConnections = newConnectionsPerSecond / 25;\n      const lcuFromActiveConnections = activeConnectionsPerMinute / 3000;\n      const gbPerHour = processedBytesGB / this.HOURS_PER_MONTH;\n      const lcuFromProcessedBytes = gbPerHour;\n\n      // Use the highest LCU consumption\n      const lcuPerHour = Math.max(\n        lcuFromNewConnections,\n        lcuFromActiveConnections,\n        lcuFromProcessedBytes,\n      );\n\n      const hourlyCost = hourlyRate * this.HOURS_PER_MONTH;\n      const lcuCost = lcuRate * lcuPerHour * this.HOURS_PER_MONTH;\n      const totalCost = hourlyCost + lcuCost;\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          `LCU consumption: ${lcuPerHour.toFixed(2)} LCU/hour based on:`,\n          `  - New connections: ${newConnectionsPerSecond}/sec → ${lcuFromNewConnections.toFixed(2)} LCU`,\n          `  - Active connections: ${activeConnectionsPerMinute}/min → ${lcuFromActiveConnections.toFixed(2)} LCU`,\n          `  - Processed data: ${processedBytesGB} GB/month → ${lcuFromProcessedBytes.toFixed(2)} LCU`,\n          `LCU cost: $${lcuRate.toFixed(4)}/LCU/hour × ${lcuPerHour.toFixed(2)} LCU × ${this.HOURS_PER_MONTH} hours = $${lcuCost.toFixed(2)}/month`,\n          `Total: $${totalCost.toFixed(2)}/month`,\n        ],\n      };\n    } catch (error) {\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 LCU consumption for assumptions even when error occurs\n      const lcuFromNewConnections = newConnectionsPerSecond / 25;\n      const lcuFromActiveConnections = activeConnectionsPerMinute / 3000;\n      const gbPerHour = processedBytesGB / this.HOURS_PER_MONTH;\n      const lcuFromProcessedBytes = gbPerHour;\n      const lcuPerHour = Math.max(\n        lcuFromNewConnections,\n        lcuFromActiveConnections,\n        lcuFromProcessedBytes,\n      );\n\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          `Would use LCU consumption: ${lcuPerHour.toFixed(2)} LCU/hour based on:`,\n          `  - New connections: ${newConnectionsPerSecond}/sec → ${lcuFromNewConnections.toFixed(2)} LCU`,\n          `  - Active connections: ${activeConnectionsPerMinute}/min → ${lcuFromActiveConnections.toFixed(2)} LCU`,\n          `  - Processed data: ${processedBytesGB} GB/month → ${lcuFromProcessedBytes.toFixed(2)} LCU`,\n        ],\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,10 @@
1
+ import { ResourceWithId } from '../../diff/types';
2
+ import { ResourceCostCalculator, MonthlyCost, PricingClient } from '../types';
3
+ export declare class APIGatewayCalculator implements ResourceCostCalculator {
4
+ supports(resourceType: string): boolean;
5
+ calculateCost(resource: ResourceWithId, region: string, pricingClient: PricingClient): Promise<MonthlyCost>;
6
+ private calculateRestApiCost;
7
+ private calculateHttpApiCost;
8
+ private calculateWebSocketCost;
9
+ private normalizeRegion;
10
+ }