cdk-cost-analyzer 0.1.18 → 0.1.19

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.
@@ -2,35 +2,35 @@
2
2
  "entries": {
3
3
  "AmazonS3:US East (N. Virginia):storageClass:General Purpose|volumeType:Standard": {
4
4
  "price": 0.023,
5
- "timestamp": 1770040481029
5
+ "timestamp": 1770240270476
6
6
  },
7
- "AmazonDynamoDB:US East (N. Virginia):group:DDB-ReadUnits|groupDescription:OnDemand ReadRequestUnits": {
7
+ "AmazonDynamoDB:US East (N. Virginia):group:DDB-ReadUnits|productFamily:Amazon DynamoDB PayPerRequest Throughput": {
8
8
  "price": 0.023,
9
- "timestamp": 1770040481036
9
+ "timestamp": 1770240270484
10
10
  },
11
- "AmazonDynamoDB:US East (N. Virginia):group:DDB-WriteUnits|groupDescription:OnDemand WriteRequestUnits": {
11
+ "AmazonDynamoDB:US East (N. Virginia):group:DDB-WriteUnits|productFamily:Amazon DynamoDB PayPerRequest Throughput": {
12
12
  "price": 0.023,
13
- "timestamp": 1770040481037
13
+ "timestamp": 1770240270484
14
14
  },
15
15
  "AmazonEC2:US East (N. Virginia):capacitystatus:Used|instanceType:t3.micro|operatingSystem:Linux|preInstalledSw:NA|tenancy:Shared": {
16
16
  "price": 0.023,
17
- "timestamp": 1770040481046
17
+ "timestamp": 1770240270494
18
18
  },
19
19
  "AWSLambda:US East (N. Virginia):group:AWS-Lambda-Requests": {
20
20
  "price": 0.023,
21
- "timestamp": 1770040481055
21
+ "timestamp": 1770240270499
22
22
  },
23
23
  "AWSLambda:US East (N. Virginia):group:AWS-Lambda-Duration": {
24
24
  "price": 0.023,
25
- "timestamp": 1770040481055
25
+ "timestamp": 1770240270499
26
26
  },
27
27
  "AmazonS3:EU (Frankfurt):storageClass:General Purpose|volumeType:Standard": {
28
28
  "price": 0.023,
29
- "timestamp": 1770040490319
29
+ "timestamp": 1770240279952
30
30
  },
31
31
  "AmazonS3:invalid-region-123:storageClass:General Purpose|volumeType:Standard": {
32
32
  "price": 0.023,
33
- "timestamp": 1770040490367
33
+ "timestamp": 1770240279996
34
34
  }
35
35
  }
36
36
  }
@@ -33,23 +33,26 @@ class DynamoDBCalculator {
33
33
  const { readRequests: assumedReadRequests, writeRequests: assumedWriteRequests } = this.getUsageAssumptions();
34
34
  // Normalize region for pricing queries
35
35
  const normalizedRegion = (0, RegionMapper_1.normalizeRegion)(region);
36
- const readCostPerMillion = await pricingClient.getPrice({
36
+ // Query pricing using productFamily filter for more accurate on-demand pricing
37
+ // Note: AWS Pricing API returns price per request unit (not per million)
38
+ // See: https://docs.aws.amazon.com/amazondynamodb/latest/developerguide/HowItWorks.ReadWriteCapacityMode.html#HowItWorks.OnDemand
39
+ const readCostPerUnit = await pricingClient.getPrice({
37
40
  serviceCode: 'AmazonDynamoDB',
38
41
  region: normalizedRegion,
39
42
  filters: [
43
+ { field: 'productFamily', value: 'Amazon DynamoDB PayPerRequest Throughput', type: 'TERM_MATCH' },
40
44
  { field: 'group', value: 'DDB-ReadUnits', type: 'TERM_MATCH' },
41
- { field: 'groupDescription', value: 'OnDemand ReadRequestUnits', type: 'TERM_MATCH' },
42
45
  ],
43
46
  });
44
- const writeCostPerMillion = await pricingClient.getPrice({
47
+ const writeCostPerUnit = await pricingClient.getPrice({
45
48
  serviceCode: 'AmazonDynamoDB',
46
49
  region: normalizedRegion,
47
50
  filters: [
51
+ { field: 'productFamily', value: 'Amazon DynamoDB PayPerRequest Throughput', type: 'TERM_MATCH' },
48
52
  { field: 'group', value: 'DDB-WriteUnits', type: 'TERM_MATCH' },
49
- { field: 'groupDescription', value: 'OnDemand WriteRequestUnits', type: 'TERM_MATCH' },
50
53
  ],
51
54
  });
52
- if (readCostPerMillion === null || writeCostPerMillion === null) {
55
+ if (readCostPerUnit === null || writeCostPerUnit === null) {
53
56
  return {
54
57
  amount: 0,
55
58
  currency: 'USD',
@@ -60,8 +63,9 @@ class DynamoDBCalculator {
60
63
  ],
61
64
  };
62
65
  }
63
- const readCost = (assumedReadRequests / 1_000_000) * readCostPerMillion;
64
- const writeCost = (assumedWriteRequests / 1_000_000) * writeCostPerMillion;
66
+ // Calculate total cost: requests × price per request unit
67
+ const readCost = assumedReadRequests * readCostPerUnit;
68
+ const writeCost = assumedWriteRequests * writeCostPerUnit;
65
69
  const monthlyCost = readCost + writeCost;
66
70
  return {
67
71
  amount: monthlyCost,
@@ -150,4 +154,4 @@ class DynamoDBCalculator {
150
154
  }
151
155
  }
152
156
  exports.DynamoDBCalculator = DynamoDBCalculator;
153
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"DynamoDBCalculator.js","sourceRoot":"","sources":["../../../src/pricing/calculators/DynamoDBCalculator.ts"],"names":[],"mappings":"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","sourcesContent":["import { ResourceWithId } from '../../diff/types';\nimport { ResourceCostCalculator, MonthlyCost, PricingClient } from '../types';\nimport { normalizeRegion } from '../RegionMapper';\nimport { CostAnalyzerConfig } from '../../config/types';\n\nexport class DynamoDBCalculator implements ResourceCostCalculator {\n  private config?: CostAnalyzerConfig;\n\n  constructor(config?: CostAnalyzerConfig) {\n    this.config = config;\n  }\n\n  supports(resourceType: string): boolean {\n    return resourceType === 'AWS::DynamoDB::Table';\n  }\n\n  async calculateCost(\n    resource: ResourceWithId,\n    region: string,\n    pricingClient: PricingClient,\n  ): Promise<MonthlyCost> {\n    const billingMode = (resource.properties.BillingMode as string) || 'PROVISIONED';\n    const hasProvisionedThroughput = resource.properties.ProvisionedThroughput !== undefined;\n\n    // Per requirement 1.4: When ProvisionedThroughput is defined, treat as provisioned mode\n    if (hasProvisionedThroughput || billingMode === 'PROVISIONED') {\n      return this.calculateProvisionedCost(resource, region, pricingClient);\n    } else {\n      return this.calculateOnDemandCost(resource, region, pricingClient);\n    }\n  }\n\n  private getUsageAssumptions(): { readRequests: number; writeRequests: number } {\n    return {\n      readRequests: this.config?.usageAssumptions?.dynamodb?.readRequestsPerMonth ?? 10_000_000,\n      writeRequests: this.config?.usageAssumptions?.dynamodb?.writeRequestsPerMonth ?? 1_000_000,\n    };\n  }\n\n  private async calculateOnDemandCost(\n    _resource: ResourceWithId,\n    region: string,\n    pricingClient: PricingClient,\n  ): Promise<MonthlyCost> {\n    try {\n      // Get usage assumptions from config or use defaults\n      const { readRequests: assumedReadRequests, writeRequests: assumedWriteRequests } = this.getUsageAssumptions();\n\n      // Normalize region for pricing queries\n      const normalizedRegion = normalizeRegion(region);\n\n      const readCostPerMillion = await pricingClient.getPrice({\n        serviceCode: 'AmazonDynamoDB',\n        region: normalizedRegion,\n        filters: [\n          { field: 'group', value: 'DDB-ReadUnits', type: 'TERM_MATCH' },\n          { field: 'groupDescription', value: 'OnDemand ReadRequestUnits', type: 'TERM_MATCH' },\n        ],\n      });\n\n      const writeCostPerMillion = await pricingClient.getPrice({\n        serviceCode: 'AmazonDynamoDB',\n        region: normalizedRegion,\n        filters: [\n          { field: 'group', value: 'DDB-WriteUnits', type: 'TERM_MATCH' },\n          { field: 'groupDescription', value: 'OnDemand WriteRequestUnits', type: 'TERM_MATCH' },\n        ],\n      });\n\n      if (readCostPerMillion === null || writeCostPerMillion === null) {\n        return {\n          amount: 0,\n          currency: 'USD',\n          confidence: 'unknown',\n          assumptions: [\n            'Pricing data not available for DynamoDB on-demand mode',\n            'On-demand billing mode',\n          ],\n        };\n      }\n\n      const readCost = (assumedReadRequests / 1_000_000) * readCostPerMillion;\n      const writeCost = (assumedWriteRequests / 1_000_000) * writeCostPerMillion;\n      const monthlyCost = readCost + writeCost;\n\n      return {\n        amount: monthlyCost,\n        currency: 'USD',\n        confidence: 'medium',\n        assumptions: [\n          `Assumes ${assumedReadRequests.toLocaleString()} read requests per month`,\n          `Assumes ${assumedWriteRequests.toLocaleString()} write requests per month`,\n          'On-demand billing mode',\n          'Does not include storage costs or other features (streams, backups, etc.)',\n        ],\n      };\n    } catch (error) {\n      return {\n        amount: 0,\n        currency: 'USD',\n        confidence: 'unknown',\n        assumptions: [\n          `Failed to fetch pricing: ${error instanceof Error ? error.message : String(error)}`,\n          'On-demand billing mode',\n        ],\n      };\n    }\n  }\n\n  private async calculateProvisionedCost(\n    resource: ResourceWithId,\n    region: string,\n    pricingClient: PricingClient,\n  ): Promise<MonthlyCost> {\n    try {\n      const provisionedThroughput = resource.properties.ProvisionedThroughput as any;\n      const readCapacity = provisionedThroughput?.ReadCapacityUnits || 5;\n      const writeCapacity = provisionedThroughput?.WriteCapacityUnits || 5;\n\n      // Normalize region for pricing queries\n      const normalizedRegion = normalizeRegion(region);\n\n      const readCostPerHour = await pricingClient.getPrice({\n        serviceCode: 'AmazonDynamoDB',\n        region: normalizedRegion,\n        filters: [\n          { field: 'usagetype', value: `${region}-ReadCapacityUnit-Hrs`, type: 'TERM_MATCH' },\n        ],\n      });\n\n      const writeCostPerHour = await pricingClient.getPrice({\n        serviceCode: 'AmazonDynamoDB',\n        region: normalizedRegion,\n        filters: [\n          { field: 'usagetype', value: `${region}-WriteCapacityUnit-Hrs`, type: 'TERM_MATCH' },\n        ],\n      });\n\n      if (readCostPerHour === null || writeCostPerHour === null) {\n        return {\n          amount: 0,\n          currency: 'USD',\n          confidence: 'unknown',\n          assumptions: [\n            'Pricing data not available for DynamoDB provisioned mode',\n            'Provisioned billing mode',\n          ],\n        };\n      }\n\n      const hoursPerMonth = 730;\n      const readCost = readCapacity * hoursPerMonth * readCostPerHour;\n      const writeCost = writeCapacity * hoursPerMonth * writeCostPerHour;\n      const monthlyCost = readCost + writeCost;\n\n      return {\n        amount: monthlyCost,\n        currency: 'USD',\n        confidence: 'high',\n        assumptions: [\n          `${readCapacity} provisioned read capacity units`,\n          `${writeCapacity} provisioned write capacity units`,\n          `Assumes ${hoursPerMonth} hours per month (24/7 operation)`,\n          'Provisioned billing mode',\n          'Does not include storage costs or other features (streams, backups, etc.)',\n        ],\n      };\n    } catch (error) {\n      return {\n        amount: 0,\n        currency: 'USD',\n        confidence: 'unknown',\n        assumptions: [\n          `Failed to fetch pricing: ${error instanceof Error ? error.message : String(error)}`,\n          'Provisioned billing mode',\n        ],\n      };\n    }\n  }\n\n}\n"]}
157
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"DynamoDBCalculator.js","sourceRoot":"","sources":["../../../src/pricing/calculators/DynamoDBCalculator.ts"],"names":[],"mappings":"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","sourcesContent":["import { ResourceWithId } from '../../diff/types';\nimport { ResourceCostCalculator, MonthlyCost, PricingClient } from '../types';\nimport { normalizeRegion } from '../RegionMapper';\nimport { CostAnalyzerConfig } from '../../config/types';\n\nexport class DynamoDBCalculator implements ResourceCostCalculator {\n  private config?: CostAnalyzerConfig;\n\n  constructor(config?: CostAnalyzerConfig) {\n    this.config = config;\n  }\n\n  supports(resourceType: string): boolean {\n    return resourceType === 'AWS::DynamoDB::Table';\n  }\n\n  async calculateCost(\n    resource: ResourceWithId,\n    region: string,\n    pricingClient: PricingClient,\n  ): Promise<MonthlyCost> {\n    const billingMode = (resource.properties.BillingMode as string) || 'PROVISIONED';\n    const hasProvisionedThroughput = resource.properties.ProvisionedThroughput !== undefined;\n\n    // Per requirement 1.4: When ProvisionedThroughput is defined, treat as provisioned mode\n    if (hasProvisionedThroughput || billingMode === 'PROVISIONED') {\n      return this.calculateProvisionedCost(resource, region, pricingClient);\n    } else {\n      return this.calculateOnDemandCost(resource, region, pricingClient);\n    }\n  }\n\n  private getUsageAssumptions(): { readRequests: number; writeRequests: number } {\n    return {\n      readRequests: this.config?.usageAssumptions?.dynamodb?.readRequestsPerMonth ?? 10_000_000,\n      writeRequests: this.config?.usageAssumptions?.dynamodb?.writeRequestsPerMonth ?? 1_000_000,\n    };\n  }\n\n  private async calculateOnDemandCost(\n    _resource: ResourceWithId,\n    region: string,\n    pricingClient: PricingClient,\n  ): Promise<MonthlyCost> {\n    try {\n      // Get usage assumptions from config or use defaults\n      const { readRequests: assumedReadRequests, writeRequests: assumedWriteRequests } = this.getUsageAssumptions();\n\n      // Normalize region for pricing queries\n      const normalizedRegion = normalizeRegion(region);\n\n      // Query pricing using productFamily filter for more accurate on-demand pricing\n      // Note: AWS Pricing API returns price per request unit (not per million)\n      // See: https://docs.aws.amazon.com/amazondynamodb/latest/developerguide/HowItWorks.ReadWriteCapacityMode.html#HowItWorks.OnDemand\n      const readCostPerUnit = await pricingClient.getPrice({\n        serviceCode: 'AmazonDynamoDB',\n        region: normalizedRegion,\n        filters: [\n          { field: 'productFamily', value: 'Amazon DynamoDB PayPerRequest Throughput', type: 'TERM_MATCH' },\n          { field: 'group', value: 'DDB-ReadUnits', type: 'TERM_MATCH' },\n        ],\n      });\n\n      const writeCostPerUnit = await pricingClient.getPrice({\n        serviceCode: 'AmazonDynamoDB',\n        region: normalizedRegion,\n        filters: [\n          { field: 'productFamily', value: 'Amazon DynamoDB PayPerRequest Throughput', type: 'TERM_MATCH' },\n          { field: 'group', value: 'DDB-WriteUnits', type: 'TERM_MATCH' },\n        ],\n      });\n\n      if (readCostPerUnit === null || writeCostPerUnit === null) {\n        return {\n          amount: 0,\n          currency: 'USD',\n          confidence: 'unknown',\n          assumptions: [\n            'Pricing data not available for DynamoDB on-demand mode',\n            'On-demand billing mode',\n          ],\n        };\n      }\n\n      // Calculate total cost: requests × price per request unit\n      const readCost = assumedReadRequests * readCostPerUnit;\n      const writeCost = assumedWriteRequests * writeCostPerUnit;\n      const monthlyCost = readCost + writeCost;\n\n      return {\n        amount: monthlyCost,\n        currency: 'USD',\n        confidence: 'medium',\n        assumptions: [\n          `Assumes ${assumedReadRequests.toLocaleString()} read requests per month`,\n          `Assumes ${assumedWriteRequests.toLocaleString()} write requests per month`,\n          'On-demand billing mode',\n          'Does not include storage costs or other features (streams, backups, etc.)',\n        ],\n      };\n    } catch (error) {\n      return {\n        amount: 0,\n        currency: 'USD',\n        confidence: 'unknown',\n        assumptions: [\n          `Failed to fetch pricing: ${error instanceof Error ? error.message : String(error)}`,\n          'On-demand billing mode',\n        ],\n      };\n    }\n  }\n\n  private async calculateProvisionedCost(\n    resource: ResourceWithId,\n    region: string,\n    pricingClient: PricingClient,\n  ): Promise<MonthlyCost> {\n    try {\n      const provisionedThroughput = resource.properties.ProvisionedThroughput as any;\n      const readCapacity = provisionedThroughput?.ReadCapacityUnits || 5;\n      const writeCapacity = provisionedThroughput?.WriteCapacityUnits || 5;\n\n      // Normalize region for pricing queries\n      const normalizedRegion = normalizeRegion(region);\n\n      const readCostPerHour = await pricingClient.getPrice({\n        serviceCode: 'AmazonDynamoDB',\n        region: normalizedRegion,\n        filters: [\n          { field: 'usagetype', value: `${region}-ReadCapacityUnit-Hrs`, type: 'TERM_MATCH' },\n        ],\n      });\n\n      const writeCostPerHour = await pricingClient.getPrice({\n        serviceCode: 'AmazonDynamoDB',\n        region: normalizedRegion,\n        filters: [\n          { field: 'usagetype', value: `${region}-WriteCapacityUnit-Hrs`, type: 'TERM_MATCH' },\n        ],\n      });\n\n      if (readCostPerHour === null || writeCostPerHour === null) {\n        return {\n          amount: 0,\n          currency: 'USD',\n          confidence: 'unknown',\n          assumptions: [\n            'Pricing data not available for DynamoDB provisioned mode',\n            'Provisioned billing mode',\n          ],\n        };\n      }\n\n      const hoursPerMonth = 730;\n      const readCost = readCapacity * hoursPerMonth * readCostPerHour;\n      const writeCost = writeCapacity * hoursPerMonth * writeCostPerHour;\n      const monthlyCost = readCost + writeCost;\n\n      return {\n        amount: monthlyCost,\n        currency: 'USD',\n        confidence: 'high',\n        assumptions: [\n          `${readCapacity} provisioned read capacity units`,\n          `${writeCapacity} provisioned write capacity units`,\n          `Assumes ${hoursPerMonth} hours per month (24/7 operation)`,\n          'Provisioned billing mode',\n          'Does not include storage costs or other features (streams, backups, etc.)',\n        ],\n      };\n    } catch (error) {\n      return {\n        amount: 0,\n        currency: 'USD',\n        confidence: 'unknown',\n        assumptions: [\n          `Failed to fetch pricing: ${error instanceof Error ? error.message : String(error)}`,\n          'Provisioned billing mode',\n        ],\n      };\n    }\n  }\n\n}\n"]}
@@ -1 +1 @@
1
- v0.1.18
1
+ v0.1.19
package/package.json CHANGED
@@ -82,7 +82,7 @@
82
82
  "publishConfig": {
83
83
  "access": "public"
84
84
  },
85
- "version": "0.1.18",
85
+ "version": "0.1.19",
86
86
  "bugs": {
87
87
  "url": "https://github.com/buildinginthecloud/cdk-cost-analyzer/issues"
88
88
  },