cdk-cost-analyzer 0.1.39 → 0.1.41

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 (35) hide show
  1. package/.cdk-cost-analyzer-cache/metadata.json +8 -8
  2. package/.husky/pre-commit +6 -0
  3. package/dist/action/136.index.js +1 -1
  4. package/dist/action/762.index.js +1 -1
  5. package/dist/action/998.index.js +1 -1
  6. package/dist/action/config/types.d.ts +50 -0
  7. package/dist/action/index.js +3 -3
  8. package/dist/action/pricing/calculators/AuroraServerlessCalculator.d.ts +20 -0
  9. package/dist/action/pricing/calculators/EKSCalculator.d.ts +8 -0
  10. package/dist/action/pricing/calculators/KinesisCalculator.d.ts +27 -0
  11. package/dist/action/pricing/calculators/Route53Calculator.d.ts +18 -0
  12. package/dist/action/pricing/calculators/TransitGatewayCalculator.d.ts +16 -0
  13. package/dist/action/reporter/index.d.ts +1 -0
  14. package/dist/action/reporter/markdownUtils.d.ts +30 -0
  15. package/dist/config/types.d.ts +50 -0
  16. package/dist/config/types.js +1 -1
  17. package/dist/pricing/PricingService.js +11 -1
  18. package/dist/pricing/calculators/AuroraServerlessCalculator.d.ts +20 -0
  19. package/dist/pricing/calculators/AuroraServerlessCalculator.js +92 -0
  20. package/dist/pricing/calculators/EKSCalculator.d.ts +8 -0
  21. package/dist/pricing/calculators/EKSCalculator.js +46 -0
  22. package/dist/pricing/calculators/KinesisCalculator.d.ts +27 -0
  23. package/dist/pricing/calculators/KinesisCalculator.js +140 -0
  24. package/dist/pricing/calculators/Route53Calculator.d.ts +18 -0
  25. package/dist/pricing/calculators/Route53Calculator.js +99 -0
  26. package/dist/pricing/calculators/TransitGatewayCalculator.d.ts +16 -0
  27. package/dist/pricing/calculators/TransitGatewayCalculator.js +97 -0
  28. package/dist/releasetag.txt +1 -1
  29. package/dist/reporter/Reporter.js +94 -56
  30. package/dist/reporter/SingleTemplateReporter.js +49 -21
  31. package/dist/reporter/index.d.ts +1 -0
  32. package/dist/reporter/index.js +8 -2
  33. package/dist/reporter/markdownUtils.d.ts +30 -0
  34. package/dist/reporter/markdownUtils.js +93 -0
  35. package/package.json +2 -2
@@ -0,0 +1,140 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.KinesisCalculator = void 0;
4
+ const RegionMapper_1 = require("../RegionMapper");
5
+ class KinesisCalculator {
6
+ customShardCount;
7
+ customIngestionGB;
8
+ customRetrievalGB;
9
+ customFirehoseGB;
10
+ customKPUs;
11
+ HOURS_PER_MONTH = 730;
12
+ // Data Streams - Provisioned
13
+ DEFAULT_SHARD_COUNT = 2;
14
+ FALLBACK_SHARD_PRICE = 0.015;
15
+ // Data Streams - On-Demand
16
+ DEFAULT_INGESTION_GB = 1000;
17
+ DEFAULT_RETRIEVAL_GB = 2000;
18
+ FALLBACK_INGESTION_PRICE = 0.040;
19
+ FALLBACK_RETRIEVAL_PRICE = 0.015;
20
+ // Firehose
21
+ DEFAULT_FIREHOSE_GB = 1000;
22
+ FALLBACK_FIREHOSE_PRICE = 0.029;
23
+ // Analytics
24
+ DEFAULT_KPUS = 2;
25
+ FALLBACK_KPU_PRICE = 0.11;
26
+ constructor(customShardCount, customIngestionGB, customRetrievalGB, customFirehoseGB, customKPUs) {
27
+ this.customShardCount = customShardCount;
28
+ this.customIngestionGB = customIngestionGB;
29
+ this.customRetrievalGB = customRetrievalGB;
30
+ this.customFirehoseGB = customFirehoseGB;
31
+ this.customKPUs = customKPUs;
32
+ }
33
+ supports(resourceType) {
34
+ return (resourceType === 'AWS::Kinesis::Stream' ||
35
+ resourceType === 'AWS::KinesisFirehose::DeliveryStream' ||
36
+ resourceType === 'AWS::KinesisAnalyticsV2::Application');
37
+ }
38
+ async calculateCost(resource, region, pricingClient) {
39
+ try {
40
+ switch (resource.type) {
41
+ case 'AWS::Kinesis::Stream':
42
+ return await this.calculateDataStreamCost(resource, region, pricingClient);
43
+ case 'AWS::KinesisFirehose::DeliveryStream':
44
+ return this.calculateFirehoseCost();
45
+ case 'AWS::KinesisAnalyticsV2::Application':
46
+ return this.calculateAnalyticsCost();
47
+ default:
48
+ return {
49
+ amount: 0,
50
+ currency: 'USD',
51
+ confidence: 'unknown',
52
+ assumptions: [`Unsupported Kinesis resource type: ${resource.type}`],
53
+ };
54
+ }
55
+ }
56
+ catch (error) {
57
+ return {
58
+ amount: 0,
59
+ currency: 'USD',
60
+ confidence: 'unknown',
61
+ assumptions: [`Failed to fetch pricing: ${error instanceof Error ? error.message : String(error)}`],
62
+ };
63
+ }
64
+ }
65
+ async calculateDataStreamCost(resource, region, pricingClient) {
66
+ const props = resource.properties || {};
67
+ const streamModeDetails = props.StreamModeDetails;
68
+ const streamMode = streamModeDetails?.StreamMode || 'PROVISIONED';
69
+ if (streamMode === 'ON_DEMAND') {
70
+ return this.calculateOnDemandStreamCost();
71
+ }
72
+ const shardCount = this.customShardCount
73
+ ?? props.ShardCount
74
+ ?? this.DEFAULT_SHARD_COUNT;
75
+ const shardPrice = await pricingClient.getPrice({
76
+ serviceCode: 'AmazonKinesis',
77
+ region: (0, RegionMapper_1.normalizeRegion)(region),
78
+ filters: [
79
+ { field: 'productFamily', value: 'Kinesis Streams' },
80
+ { field: 'usagetype', value: 'shardHour-Provisioned' },
81
+ ],
82
+ });
83
+ const rate = shardPrice ?? this.FALLBACK_SHARD_PRICE;
84
+ const monthlyCost = shardCount * rate * this.HOURS_PER_MONTH;
85
+ return {
86
+ amount: monthlyCost,
87
+ currency: 'USD',
88
+ confidence: shardPrice !== null ? 'high' : 'medium',
89
+ assumptions: [
90
+ `Provisioned mode: ${shardCount} shards × $${rate.toFixed(4)}/shard-hour × ${this.HOURS_PER_MONTH}h = $${monthlyCost.toFixed(2)}/month`,
91
+ ...(shardPrice === null ? [`Using fallback pricing (API data not available for region ${region})`] : []),
92
+ ],
93
+ };
94
+ }
95
+ calculateOnDemandStreamCost() {
96
+ const ingestionGB = this.customIngestionGB ?? this.DEFAULT_INGESTION_GB;
97
+ const retrievalGB = this.customRetrievalGB ?? this.DEFAULT_RETRIEVAL_GB;
98
+ const ingestionCost = ingestionGB * this.FALLBACK_INGESTION_PRICE;
99
+ const retrievalCost = retrievalGB * this.FALLBACK_RETRIEVAL_PRICE;
100
+ const totalCost = ingestionCost + retrievalCost;
101
+ return {
102
+ amount: totalCost,
103
+ currency: 'USD',
104
+ confidence: 'medium',
105
+ assumptions: [
106
+ `On-demand mode`,
107
+ `Ingestion: ${ingestionGB} GB × $${this.FALLBACK_INGESTION_PRICE}/GB = $${ingestionCost.toFixed(2)}/month`,
108
+ `Retrieval: ${retrievalGB} GB × $${this.FALLBACK_RETRIEVAL_PRICE}/GB = $${retrievalCost.toFixed(2)}/month`,
109
+ ],
110
+ };
111
+ }
112
+ calculateFirehoseCost() {
113
+ const dataGB = this.customFirehoseGB ?? this.DEFAULT_FIREHOSE_GB;
114
+ const monthlyCost = dataGB * this.FALLBACK_FIREHOSE_PRICE;
115
+ return {
116
+ amount: monthlyCost,
117
+ currency: 'USD',
118
+ confidence: 'medium',
119
+ assumptions: [
120
+ `Firehose ingestion: ${dataGB} GB × $${this.FALLBACK_FIREHOSE_PRICE}/GB = $${monthlyCost.toFixed(2)}/month`,
121
+ 'Does not include format conversion or VPC delivery costs',
122
+ ],
123
+ };
124
+ }
125
+ calculateAnalyticsCost() {
126
+ const kpus = this.customKPUs ?? this.DEFAULT_KPUS;
127
+ const monthlyCost = kpus * this.FALLBACK_KPU_PRICE * this.HOURS_PER_MONTH;
128
+ return {
129
+ amount: monthlyCost,
130
+ currency: 'USD',
131
+ confidence: 'medium',
132
+ assumptions: [
133
+ `Kinesis Analytics: ${kpus} KPUs × $${this.FALLBACK_KPU_PRICE}/KPU-hour × ${this.HOURS_PER_MONTH}h = $${monthlyCost.toFixed(2)}/month`,
134
+ '1 KPU = 1 vCPU + 4 GB memory',
135
+ ],
136
+ };
137
+ }
138
+ }
139
+ exports.KinesisCalculator = KinesisCalculator;
140
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"KinesisCalculator.js","sourceRoot":"","sources":["../../../src/pricing/calculators/KinesisCalculator.ts"],"names":[],"mappings":"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","sourcesContent":["import { ResourceWithId } from '../../diff/types';\nimport { ResourceCostCalculator, MonthlyCost, PricingClient } from '../types';\nimport { normalizeRegion } from '../RegionMapper';\n\nexport class KinesisCalculator implements ResourceCostCalculator {\n  private readonly HOURS_PER_MONTH = 730;\n\n  // Data Streams - Provisioned\n  private readonly DEFAULT_SHARD_COUNT = 2;\n  private readonly FALLBACK_SHARD_PRICE = 0.015;\n\n  // Data Streams - On-Demand\n  private readonly DEFAULT_INGESTION_GB = 1000;\n  private readonly DEFAULT_RETRIEVAL_GB = 2000;\n  private readonly FALLBACK_INGESTION_PRICE = 0.040;\n  private readonly FALLBACK_RETRIEVAL_PRICE = 0.015;\n\n  // Firehose\n  private readonly DEFAULT_FIREHOSE_GB = 1000;\n  private readonly FALLBACK_FIREHOSE_PRICE = 0.029;\n\n  // Analytics\n  private readonly DEFAULT_KPUS = 2;\n  private readonly FALLBACK_KPU_PRICE = 0.11;\n\n  constructor(\n    private readonly customShardCount?: number,\n    private readonly customIngestionGB?: number,\n    private readonly customRetrievalGB?: number,\n    private readonly customFirehoseGB?: number,\n    private readonly customKPUs?: number,\n  ) {}\n\n  supports(resourceType: string): boolean {\n    return (\n      resourceType === 'AWS::Kinesis::Stream' ||\n      resourceType === 'AWS::KinesisFirehose::DeliveryStream' ||\n      resourceType === 'AWS::KinesisAnalyticsV2::Application'\n    );\n  }\n\n  async calculateCost(\n    resource: ResourceWithId,\n    region: string,\n    pricingClient: PricingClient,\n  ): Promise<MonthlyCost> {\n    try {\n      switch (resource.type) {\n        case 'AWS::Kinesis::Stream':\n          return await this.calculateDataStreamCost(resource, region, pricingClient);\n        case 'AWS::KinesisFirehose::DeliveryStream':\n          return this.calculateFirehoseCost();\n        case 'AWS::KinesisAnalyticsV2::Application':\n          return this.calculateAnalyticsCost();\n        default:\n          return {\n            amount: 0,\n            currency: 'USD',\n            confidence: 'unknown',\n            assumptions: [`Unsupported Kinesis resource type: ${resource.type}`],\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 async calculateDataStreamCost(\n    resource: ResourceWithId,\n    region: string,\n    pricingClient: PricingClient,\n  ): Promise<MonthlyCost> {\n    const props = resource.properties || {};\n    const streamModeDetails = props.StreamModeDetails as Record<string, unknown> | undefined;\n    const streamMode = (streamModeDetails?.StreamMode as string) || 'PROVISIONED';\n\n    if (streamMode === 'ON_DEMAND') {\n      return this.calculateOnDemandStreamCost();\n    }\n\n    const shardCount = this.customShardCount\n      ?? (props.ShardCount as number | undefined)\n      ?? this.DEFAULT_SHARD_COUNT;\n\n    const shardPrice = await pricingClient.getPrice({\n      serviceCode: 'AmazonKinesis',\n      region: normalizeRegion(region),\n      filters: [\n        { field: 'productFamily', value: 'Kinesis Streams' },\n        { field: 'usagetype', value: 'shardHour-Provisioned' },\n      ],\n    });\n\n    const rate = shardPrice ?? this.FALLBACK_SHARD_PRICE;\n    const monthlyCost = shardCount * rate * this.HOURS_PER_MONTH;\n\n    return {\n      amount: monthlyCost,\n      currency: 'USD',\n      confidence: shardPrice !== null ? 'high' : 'medium',\n      assumptions: [\n        `Provisioned mode: ${shardCount} shards × $${rate.toFixed(4)}/shard-hour × ${this.HOURS_PER_MONTH}h = $${monthlyCost.toFixed(2)}/month`,\n        ...(shardPrice === null ? [`Using fallback pricing (API data not available for region ${region})`] : []),\n      ],\n    };\n  }\n\n  private calculateOnDemandStreamCost(): MonthlyCost {\n    const ingestionGB = this.customIngestionGB ?? this.DEFAULT_INGESTION_GB;\n    const retrievalGB = this.customRetrievalGB ?? this.DEFAULT_RETRIEVAL_GB;\n\n    const ingestionCost = ingestionGB * this.FALLBACK_INGESTION_PRICE;\n    const retrievalCost = retrievalGB * this.FALLBACK_RETRIEVAL_PRICE;\n    const totalCost = ingestionCost + retrievalCost;\n\n    return {\n      amount: totalCost,\n      currency: 'USD',\n      confidence: 'medium',\n      assumptions: [\n        `On-demand mode`,\n        `Ingestion: ${ingestionGB} GB × $${this.FALLBACK_INGESTION_PRICE}/GB = $${ingestionCost.toFixed(2)}/month`,\n        `Retrieval: ${retrievalGB} GB × $${this.FALLBACK_RETRIEVAL_PRICE}/GB = $${retrievalCost.toFixed(2)}/month`,\n      ],\n    };\n  }\n\n  private calculateFirehoseCost(): MonthlyCost {\n    const dataGB = this.customFirehoseGB ?? this.DEFAULT_FIREHOSE_GB;\n    const monthlyCost = dataGB * this.FALLBACK_FIREHOSE_PRICE;\n\n    return {\n      amount: monthlyCost,\n      currency: 'USD',\n      confidence: 'medium',\n      assumptions: [\n        `Firehose ingestion: ${dataGB} GB × $${this.FALLBACK_FIREHOSE_PRICE}/GB = $${monthlyCost.toFixed(2)}/month`,\n        'Does not include format conversion or VPC delivery costs',\n      ],\n    };\n  }\n\n  private calculateAnalyticsCost(): MonthlyCost {\n    const kpus = this.customKPUs ?? this.DEFAULT_KPUS;\n    const monthlyCost = kpus * this.FALLBACK_KPU_PRICE * this.HOURS_PER_MONTH;\n\n    return {\n      amount: monthlyCost,\n      currency: 'USD',\n      confidence: 'medium',\n      assumptions: [\n        `Kinesis Analytics: ${kpus} KPUs × $${this.FALLBACK_KPU_PRICE}/KPU-hour × ${this.HOURS_PER_MONTH}h = $${monthlyCost.toFixed(2)}/month`,\n        '1 KPU = 1 vCPU + 4 GB memory',\n      ],\n    };\n  }\n}\n"]}
@@ -0,0 +1,18 @@
1
+ import { ResourceWithId } from '../../diff/types';
2
+ import { ResourceCostCalculator, MonthlyCost, PricingClient } from '../types';
3
+ export declare class Route53Calculator implements ResourceCostCalculator {
4
+ private readonly customMonthlyQueries?;
5
+ private readonly HOSTED_ZONE_PRICE;
6
+ private readonly BASIC_HEALTH_CHECK_PRICE;
7
+ private readonly HTTPS_HEALTH_CHECK_PRICE;
8
+ private readonly STANDARD_QUERY_PRICE_PER_MILLION;
9
+ private readonly LATENCY_QUERY_PRICE_PER_MILLION;
10
+ private readonly GEO_QUERY_PRICE_PER_MILLION;
11
+ private readonly DEFAULT_MONTHLY_QUERIES;
12
+ constructor(customMonthlyQueries?: number | undefined);
13
+ supports(resourceType: string): boolean;
14
+ calculateCost(resource: ResourceWithId, _region: string, _pricingClient: PricingClient): Promise<MonthlyCost>;
15
+ private calculateHostedZoneCost;
16
+ private calculateHealthCheckCost;
17
+ private calculateRecordSetCost;
18
+ }
@@ -0,0 +1,99 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.Route53Calculator = void 0;
4
+ class Route53Calculator {
5
+ customMonthlyQueries;
6
+ HOSTED_ZONE_PRICE = 0.50;
7
+ BASIC_HEALTH_CHECK_PRICE = 0.50;
8
+ HTTPS_HEALTH_CHECK_PRICE = 1.00;
9
+ STANDARD_QUERY_PRICE_PER_MILLION = 0.40;
10
+ LATENCY_QUERY_PRICE_PER_MILLION = 0.60;
11
+ GEO_QUERY_PRICE_PER_MILLION = 0.70;
12
+ DEFAULT_MONTHLY_QUERIES = 1_000_000;
13
+ constructor(customMonthlyQueries) {
14
+ this.customMonthlyQueries = customMonthlyQueries;
15
+ }
16
+ supports(resourceType) {
17
+ return (resourceType === 'AWS::Route53::HostedZone' ||
18
+ resourceType === 'AWS::Route53::HealthCheck' ||
19
+ resourceType === 'AWS::Route53::RecordSet');
20
+ }
21
+ async calculateCost(resource, _region, _pricingClient) {
22
+ switch (resource.type) {
23
+ case 'AWS::Route53::HostedZone':
24
+ return this.calculateHostedZoneCost();
25
+ case 'AWS::Route53::HealthCheck':
26
+ return this.calculateHealthCheckCost(resource);
27
+ case 'AWS::Route53::RecordSet':
28
+ return this.calculateRecordSetCost(resource);
29
+ default:
30
+ return {
31
+ amount: 0,
32
+ currency: 'USD',
33
+ confidence: 'unknown',
34
+ assumptions: [`Unsupported Route 53 resource type: ${resource.type}`],
35
+ };
36
+ }
37
+ }
38
+ calculateHostedZoneCost() {
39
+ return {
40
+ amount: this.HOSTED_ZONE_PRICE,
41
+ currency: 'USD',
42
+ confidence: 'high',
43
+ assumptions: [
44
+ `Hosted zone: $${this.HOSTED_ZONE_PRICE.toFixed(2)}/month`,
45
+ 'DNS query costs are calculated on RecordSet resources',
46
+ ],
47
+ };
48
+ }
49
+ calculateHealthCheckCost(resource) {
50
+ const props = resource.properties || {};
51
+ const healthCheckConfig = props.HealthCheckConfig;
52
+ const checkType = healthCheckConfig?.Type || 'HTTP';
53
+ const isAdvanced = ['HTTPS', 'HTTP_STR_MATCH', 'HTTPS_STR_MATCH'].includes(checkType);
54
+ const price = isAdvanced ? this.HTTPS_HEALTH_CHECK_PRICE : this.BASIC_HEALTH_CHECK_PRICE;
55
+ return {
56
+ amount: price,
57
+ currency: 'USD',
58
+ confidence: 'high',
59
+ assumptions: [
60
+ `Health check (${checkType}): $${price.toFixed(2)}/month`,
61
+ ],
62
+ };
63
+ }
64
+ calculateRecordSetCost(resource) {
65
+ const props = resource.properties || {};
66
+ // Alias records to AWS resources are free
67
+ if (props.AliasTarget) {
68
+ return {
69
+ amount: 0,
70
+ currency: 'USD',
71
+ confidence: 'high',
72
+ assumptions: ['Alias record to AWS resource: no query charges'],
73
+ };
74
+ }
75
+ const queries = this.customMonthlyQueries ?? this.DEFAULT_MONTHLY_QUERIES;
76
+ const queriesInMillions = queries / 1_000_000;
77
+ let pricePerMillion = this.STANDARD_QUERY_PRICE_PER_MILLION;
78
+ let routingType = 'standard';
79
+ if (props.Region) {
80
+ pricePerMillion = this.LATENCY_QUERY_PRICE_PER_MILLION;
81
+ routingType = 'latency-based';
82
+ }
83
+ else if (props.GeoLocation) {
84
+ pricePerMillion = this.GEO_QUERY_PRICE_PER_MILLION;
85
+ routingType = 'geolocation';
86
+ }
87
+ const monthlyCost = queriesInMillions * pricePerMillion;
88
+ return {
89
+ amount: monthlyCost,
90
+ currency: 'USD',
91
+ confidence: 'medium',
92
+ assumptions: [
93
+ `DNS queries (${routingType}): ${queriesInMillions}M × $${pricePerMillion.toFixed(2)}/M = $${monthlyCost.toFixed(2)}/month`,
94
+ ],
95
+ };
96
+ }
97
+ }
98
+ exports.Route53Calculator = Route53Calculator;
99
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Route53Calculator.js","sourceRoot":"","sources":["../../../src/pricing/calculators/Route53Calculator.ts"],"names":[],"mappings":"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","sourcesContent":["import { ResourceWithId } from '../../diff/types';\nimport { ResourceCostCalculator, MonthlyCost, PricingClient } from '../types';\n\nexport class Route53Calculator implements ResourceCostCalculator {\n  private readonly HOSTED_ZONE_PRICE = 0.50;\n  private readonly BASIC_HEALTH_CHECK_PRICE = 0.50;\n  private readonly HTTPS_HEALTH_CHECK_PRICE = 1.00;\n  private readonly STANDARD_QUERY_PRICE_PER_MILLION = 0.40;\n  private readonly LATENCY_QUERY_PRICE_PER_MILLION = 0.60;\n  private readonly GEO_QUERY_PRICE_PER_MILLION = 0.70;\n  private readonly DEFAULT_MONTHLY_QUERIES = 1_000_000;\n\n  constructor(\n    private readonly customMonthlyQueries?: number,\n  ) {}\n\n  supports(resourceType: string): boolean {\n    return (\n      resourceType === 'AWS::Route53::HostedZone' ||\n      resourceType === 'AWS::Route53::HealthCheck' ||\n      resourceType === 'AWS::Route53::RecordSet'\n    );\n  }\n\n  async calculateCost(\n    resource: ResourceWithId,\n    _region: string,\n    _pricingClient: PricingClient,\n  ): Promise<MonthlyCost> {\n    switch (resource.type) {\n      case 'AWS::Route53::HostedZone':\n        return this.calculateHostedZoneCost();\n      case 'AWS::Route53::HealthCheck':\n        return this.calculateHealthCheckCost(resource);\n      case 'AWS::Route53::RecordSet':\n        return this.calculateRecordSetCost(resource);\n      default:\n        return {\n          amount: 0,\n          currency: 'USD',\n          confidence: 'unknown',\n          assumptions: [`Unsupported Route 53 resource type: ${resource.type}`],\n        };\n    }\n  }\n\n  private calculateHostedZoneCost(): MonthlyCost {\n    return {\n      amount: this.HOSTED_ZONE_PRICE,\n      currency: 'USD',\n      confidence: 'high',\n      assumptions: [\n        `Hosted zone: $${this.HOSTED_ZONE_PRICE.toFixed(2)}/month`,\n        'DNS query costs are calculated on RecordSet resources',\n      ],\n    };\n  }\n\n  private calculateHealthCheckCost(resource: ResourceWithId): MonthlyCost {\n    const props = resource.properties || {};\n    const healthCheckConfig = props.HealthCheckConfig as Record<string, unknown> | undefined;\n    const checkType = (healthCheckConfig?.Type as string) || 'HTTP';\n\n    const isAdvanced = ['HTTPS', 'HTTP_STR_MATCH', 'HTTPS_STR_MATCH'].includes(checkType);\n    const price = isAdvanced ? this.HTTPS_HEALTH_CHECK_PRICE : this.BASIC_HEALTH_CHECK_PRICE;\n\n    return {\n      amount: price,\n      currency: 'USD',\n      confidence: 'high',\n      assumptions: [\n        `Health check (${checkType}): $${price.toFixed(2)}/month`,\n      ],\n    };\n  }\n\n  private calculateRecordSetCost(resource: ResourceWithId): MonthlyCost {\n    const props = resource.properties || {};\n\n    // Alias records to AWS resources are free\n    if (props.AliasTarget) {\n      return {\n        amount: 0,\n        currency: 'USD',\n        confidence: 'high',\n        assumptions: ['Alias record to AWS resource: no query charges'],\n      };\n    }\n\n    const queries = this.customMonthlyQueries ?? this.DEFAULT_MONTHLY_QUERIES;\n    const queriesInMillions = queries / 1_000_000;\n\n    let pricePerMillion = this.STANDARD_QUERY_PRICE_PER_MILLION;\n    let routingType = 'standard';\n\n    if (props.Region) {\n      pricePerMillion = this.LATENCY_QUERY_PRICE_PER_MILLION;\n      routingType = 'latency-based';\n    } else if (props.GeoLocation) {\n      pricePerMillion = this.GEO_QUERY_PRICE_PER_MILLION;\n      routingType = 'geolocation';\n    }\n\n    const monthlyCost = queriesInMillions * pricePerMillion;\n\n    return {\n      amount: monthlyCost,\n      currency: 'USD',\n      confidence: 'medium',\n      assumptions: [\n        `DNS queries (${routingType}): ${queriesInMillions}M × $${pricePerMillion.toFixed(2)}/M = $${monthlyCost.toFixed(2)}/month`,\n      ],\n    };\n  }\n}\n"]}
@@ -0,0 +1,16 @@
1
+ import { ResourceWithId } from '../../diff/types';
2
+ import { ResourceCostCalculator, MonthlyCost, PricingClient } from '../types';
3
+ export declare class TransitGatewayCalculator implements ResourceCostCalculator {
4
+ private readonly customAttachments?;
5
+ private readonly customMonthlyDataGB?;
6
+ private readonly HOURS_PER_MONTH;
7
+ private readonly DEFAULT_ATTACHMENTS;
8
+ private readonly DEFAULT_MONTHLY_DATA_GB;
9
+ private readonly FALLBACK_ATTACHMENT_PRICE;
10
+ private readonly FALLBACK_DATA_PRICE;
11
+ constructor(customAttachments?: number | undefined, customMonthlyDataGB?: number | undefined);
12
+ supports(resourceType: string): boolean;
13
+ calculateCost(resource: ResourceWithId, region: string, pricingClient: PricingClient): Promise<MonthlyCost>;
14
+ private calculateAttachmentCost;
15
+ private calculateTransitGatewayCost;
16
+ }
@@ -0,0 +1,97 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.TransitGatewayCalculator = void 0;
4
+ const RegionMapper_1 = require("../RegionMapper");
5
+ class TransitGatewayCalculator {
6
+ customAttachments;
7
+ customMonthlyDataGB;
8
+ HOURS_PER_MONTH = 730;
9
+ DEFAULT_ATTACHMENTS = 3;
10
+ DEFAULT_MONTHLY_DATA_GB = 1000;
11
+ FALLBACK_ATTACHMENT_PRICE = 0.05;
12
+ FALLBACK_DATA_PRICE = 0.02;
13
+ constructor(customAttachments, customMonthlyDataGB) {
14
+ this.customAttachments = customAttachments;
15
+ this.customMonthlyDataGB = customMonthlyDataGB;
16
+ }
17
+ supports(resourceType) {
18
+ return (resourceType === 'AWS::EC2::TransitGateway' ||
19
+ resourceType === 'AWS::EC2::TransitGatewayAttachment');
20
+ }
21
+ async calculateCost(resource, region, pricingClient) {
22
+ try {
23
+ if (resource.type === 'AWS::EC2::TransitGatewayAttachment') {
24
+ return await this.calculateAttachmentCost(region, pricingClient);
25
+ }
26
+ return await this.calculateTransitGatewayCost(region, pricingClient);
27
+ }
28
+ catch (error) {
29
+ return {
30
+ amount: 0,
31
+ currency: 'USD',
32
+ confidence: 'unknown',
33
+ assumptions: [`Failed to fetch pricing: ${error instanceof Error ? error.message : String(error)}`],
34
+ };
35
+ }
36
+ }
37
+ async calculateAttachmentCost(region, pricingClient) {
38
+ const hourlyRate = await pricingClient.getPrice({
39
+ serviceCode: 'AmazonVPC',
40
+ region: (0, RegionMapper_1.normalizeRegion)(region),
41
+ filters: [
42
+ { field: 'productFamily', value: 'TransitGateway' },
43
+ { field: 'operation', value: 'TransitGatewayVPCAttachment' },
44
+ ],
45
+ });
46
+ const rate = hourlyRate ?? this.FALLBACK_ATTACHMENT_PRICE;
47
+ const monthlyCost = rate * this.HOURS_PER_MONTH;
48
+ return {
49
+ amount: monthlyCost,
50
+ currency: 'USD',
51
+ confidence: hourlyRate !== null ? 'high' : 'medium',
52
+ assumptions: [
53
+ `Transit Gateway attachment: $${rate.toFixed(4)}/hour × ${this.HOURS_PER_MONTH}h = $${monthlyCost.toFixed(2)}/month`,
54
+ ...(hourlyRate === null ? [`Using fallback pricing (API data not available for region ${region})`] : []),
55
+ 'Data processing costs are calculated on the Transit Gateway resource',
56
+ ],
57
+ };
58
+ }
59
+ async calculateTransitGatewayCost(region, pricingClient) {
60
+ const hourlyRate = await pricingClient.getPrice({
61
+ serviceCode: 'AmazonVPC',
62
+ region: (0, RegionMapper_1.normalizeRegion)(region),
63
+ filters: [
64
+ { field: 'productFamily', value: 'TransitGateway' },
65
+ { field: 'operation', value: 'TransitGatewayVPCAttachment' },
66
+ ],
67
+ });
68
+ const dataRate = await pricingClient.getPrice({
69
+ serviceCode: 'AmazonVPC',
70
+ region: (0, RegionMapper_1.normalizeRegion)(region),
71
+ filters: [
72
+ { field: 'productFamily', value: 'TransitGateway' },
73
+ { field: 'operation', value: 'TransitGatewayData' },
74
+ ],
75
+ });
76
+ const attachmentRate = hourlyRate ?? this.FALLBACK_ATTACHMENT_PRICE;
77
+ const dataProcessingRate = dataRate ?? this.FALLBACK_DATA_PRICE;
78
+ const usedFallback = hourlyRate === null || dataRate === null;
79
+ const attachments = this.customAttachments ?? this.DEFAULT_ATTACHMENTS;
80
+ const monthlyDataGB = this.customMonthlyDataGB ?? this.DEFAULT_MONTHLY_DATA_GB;
81
+ const attachmentCost = attachments * attachmentRate * this.HOURS_PER_MONTH;
82
+ const dataProcessingCost = monthlyDataGB * dataProcessingRate;
83
+ const totalCost = attachmentCost + dataProcessingCost;
84
+ return {
85
+ amount: totalCost,
86
+ currency: 'USD',
87
+ confidence: 'medium',
88
+ assumptions: [
89
+ `${attachments} attachments × $${attachmentRate.toFixed(4)}/hour × ${this.HOURS_PER_MONTH}h = $${attachmentCost.toFixed(2)}/month`,
90
+ `Data processing: ${monthlyDataGB} GB × $${dataProcessingRate.toFixed(4)}/GB = $${dataProcessingCost.toFixed(2)}/month`,
91
+ ...(usedFallback ? [`Using fallback pricing (API data not available for region ${region})`] : []),
92
+ ],
93
+ };
94
+ }
95
+ }
96
+ exports.TransitGatewayCalculator = TransitGatewayCalculator;
97
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"TransitGatewayCalculator.js","sourceRoot":"","sources":["../../../src/pricing/calculators/TransitGatewayCalculator.ts"],"names":[],"mappings":"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","sourcesContent":["import { ResourceWithId } from '../../diff/types';\nimport { ResourceCostCalculator, MonthlyCost, PricingClient } from '../types';\nimport { normalizeRegion } from '../RegionMapper';\n\nexport class TransitGatewayCalculator implements ResourceCostCalculator {\n  private readonly HOURS_PER_MONTH = 730;\n  private readonly DEFAULT_ATTACHMENTS = 3;\n  private readonly DEFAULT_MONTHLY_DATA_GB = 1000;\n  private readonly FALLBACK_ATTACHMENT_PRICE = 0.05;\n  private readonly FALLBACK_DATA_PRICE = 0.02;\n\n  constructor(\n    private readonly customAttachments?: number,\n    private readonly customMonthlyDataGB?: number,\n  ) {}\n\n  supports(resourceType: string): boolean {\n    return (\n      resourceType === 'AWS::EC2::TransitGateway' ||\n      resourceType === 'AWS::EC2::TransitGatewayAttachment'\n    );\n  }\n\n  async calculateCost(\n    resource: ResourceWithId,\n    region: string,\n    pricingClient: PricingClient,\n  ): Promise<MonthlyCost> {\n    try {\n      if (resource.type === 'AWS::EC2::TransitGatewayAttachment') {\n        return await this.calculateAttachmentCost(region, pricingClient);\n      }\n      return await this.calculateTransitGatewayCost(region, pricingClient);\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 async calculateAttachmentCost(\n    region: string,\n    pricingClient: PricingClient,\n  ): Promise<MonthlyCost> {\n    const hourlyRate = await pricingClient.getPrice({\n      serviceCode: 'AmazonVPC',\n      region: normalizeRegion(region),\n      filters: [\n        { field: 'productFamily', value: 'TransitGateway' },\n        { field: 'operation', value: 'TransitGatewayVPCAttachment' },\n      ],\n    });\n\n    const rate = hourlyRate ?? this.FALLBACK_ATTACHMENT_PRICE;\n    const monthlyCost = rate * this.HOURS_PER_MONTH;\n\n    return {\n      amount: monthlyCost,\n      currency: 'USD',\n      confidence: hourlyRate !== null ? 'high' : 'medium',\n      assumptions: [\n        `Transit Gateway attachment: $${rate.toFixed(4)}/hour × ${this.HOURS_PER_MONTH}h = $${monthlyCost.toFixed(2)}/month`,\n        ...(hourlyRate === null ? [`Using fallback pricing (API data not available for region ${region})`] : []),\n        'Data processing costs are calculated on the Transit Gateway resource',\n      ],\n    };\n  }\n\n  private async calculateTransitGatewayCost(\n    region: string,\n    pricingClient: PricingClient,\n  ): Promise<MonthlyCost> {\n    const hourlyRate = await pricingClient.getPrice({\n      serviceCode: 'AmazonVPC',\n      region: normalizeRegion(region),\n      filters: [\n        { field: 'productFamily', value: 'TransitGateway' },\n        { field: 'operation', value: 'TransitGatewayVPCAttachment' },\n      ],\n    });\n\n    const dataRate = await pricingClient.getPrice({\n      serviceCode: 'AmazonVPC',\n      region: normalizeRegion(region),\n      filters: [\n        { field: 'productFamily', value: 'TransitGateway' },\n        { field: 'operation', value: 'TransitGatewayData' },\n      ],\n    });\n\n    const attachmentRate = hourlyRate ?? this.FALLBACK_ATTACHMENT_PRICE;\n    const dataProcessingRate = dataRate ?? this.FALLBACK_DATA_PRICE;\n    const usedFallback = hourlyRate === null || dataRate === null;\n\n    const attachments = this.customAttachments ?? this.DEFAULT_ATTACHMENTS;\n    const monthlyDataGB = this.customMonthlyDataGB ?? this.DEFAULT_MONTHLY_DATA_GB;\n\n    const attachmentCost = attachments * attachmentRate * this.HOURS_PER_MONTH;\n    const dataProcessingCost = monthlyDataGB * dataProcessingRate;\n    const totalCost = attachmentCost + dataProcessingCost;\n\n    return {\n      amount: totalCost,\n      currency: 'USD',\n      confidence: 'medium',\n      assumptions: [\n        `${attachments} attachments × $${attachmentRate.toFixed(4)}/hour × ${this.HOURS_PER_MONTH}h = $${attachmentCost.toFixed(2)}/month`,\n        `Data processing: ${monthlyDataGB} GB × $${dataProcessingRate.toFixed(4)}/GB = $${dataProcessingCost.toFixed(2)}/month`,\n        ...(usedFallback ? [`Using fallback pricing (API data not available for region ${region})`] : []),\n      ],\n    };\n  }\n}\n"]}
@@ -1 +1 @@
1
- v0.1.39
1
+ v0.1.41