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.
- package/.cdk-cost-analyzer-cache/metadata.json +12 -0
- package/.gitlab-ci.yml +214 -0
- package/.husky/pre-commit +12 -0
- package/.kiro/hooks/accessibility-audit.kiro.hook +18 -0
- package/.kiro/hooks/api-schema-validation.kiro.hook +21 -0
- package/.kiro/hooks/auto-test-on-save.kiro.hook +19 -0
- package/.kiro/hooks/cdk-synth-on-change.kiro.hook +20 -0
- package/.kiro/hooks/code-coverage-check.kiro.hook +14 -0
- package/.kiro/hooks/commit-message-helper.kiro.hook +14 -0
- package/.kiro/hooks/dependency-update-check.kiro.hook +14 -0
- package/.kiro/hooks/env-file-validation.kiro.hook +18 -0
- package/.kiro/hooks/lint-and-format-on-save.kiro.hook +21 -0
- package/.kiro/hooks/mcp-config-validation.kiro.hook +17 -0
- package/.kiro/hooks/mcp-server-test.kiro.hook +14 -0
- package/.kiro/hooks/performance-analysis.kiro.hook +14 -0
- package/.kiro/hooks/readme-spell-check.kiro.hook +14 -0
- package/.kiro/hooks/security-scan-on-dependency-change.kiro.hook +21 -0
- package/.kiro/hooks/translation-update.kiro.hook +18 -0
- package/.kiro/hooks/update-documentation.kiro.hook +18 -0
- package/.kiro/settings/mcp.json +20 -0
- package/.kiro/specs/cdk-cost-analyzer/design.md +620 -0
- package/.kiro/specs/cdk-cost-analyzer/requirements.md +183 -0
- package/.kiro/specs/cdk-cost-analyzer/tasks.md +357 -0
- package/.kiro/specs/github-actions-ci/design.md +281 -0
- package/.kiro/specs/github-actions-ci/requirements.md +86 -0
- package/.kiro/specs/github-actions-ci/tasks.md +115 -0
- package/.kiro/specs/nlb-calculator-test-coverage/design.md +190 -0
- package/.kiro/specs/nlb-calculator-test-coverage/requirements.md +84 -0
- package/.kiro/specs/nlb-calculator-test-coverage/tasks.md +150 -0
- package/.kiro/specs/production-readiness/design.md +1213 -0
- package/.kiro/specs/production-readiness/requirements.md +312 -0
- package/.kiro/specs/production-readiness/tasks.md +269 -0
- package/.kiro/specs/repository-cleanup/design.md +283 -0
- package/.kiro/specs/repository-cleanup/requirements.md +74 -0
- package/.kiro/specs/repository-cleanup/tasks.md +64 -0
- package/.kiro/steering/aws-cli-best-practices.md +41 -0
- package/.kiro/steering/cdk-best-practices.md +49 -0
- package/.kiro/steering/development-standards.md +54 -0
- package/.kiro/steering/docker-best-practices.md +34 -0
- package/.kiro/steering/documentation-style.md +151 -0
- package/.kiro/steering/git-best-practices.md +37 -0
- package/.kiro/steering/mcp-best-practices.md +95 -0
- package/.kiro/steering/python-best-practices.md +48 -0
- package/.kiro/steering/react-best-practices.md +44 -0
- package/.kiro/steering/security-best-practices.md +41 -0
- package/.kiro/steering/testing-best-practices.md +59 -0
- package/.kiro/steering/typescript-best-practices.md +40 -0
- package/CHANGELOG.md +49 -0
- package/CONTRIBUTING.md +258 -0
- package/LICENSE +19 -0
- package/README.md +480 -0
- package/SECURITY.md +117 -0
- package/dist/api/index.d.ts +11 -0
- package/dist/api/index.js +65 -0
- package/dist/api/types.d.ts +15 -0
- package/dist/api/types.js +3 -0
- package/dist/cli/index.d.ts +2 -0
- package/dist/cli/index.js +262 -0
- package/dist/config/ConfigManager.d.ts +40 -0
- package/dist/config/ConfigManager.js +238 -0
- package/dist/config/index.d.ts +2 -0
- package/dist/config/index.js +19 -0
- package/dist/config/types.d.ts +72 -0
- package/dist/config/types.js +15 -0
- package/dist/diff/DiffEngine.d.ts +7 -0
- package/dist/diff/DiffEngine.js +73 -0
- package/dist/diff/index.d.ts +2 -0
- package/dist/diff/index.js +21 -0
- package/dist/diff/types.d.ts +20 -0
- package/dist/diff/types.js +3 -0
- package/dist/integrations/GitLabIntegration.d.ts +7 -0
- package/dist/integrations/GitLabIntegration.js +45 -0
- package/dist/integrations/index.d.ts +2 -0
- package/dist/integrations/index.js +21 -0
- package/dist/integrations/types.d.ts +11 -0
- package/dist/integrations/types.js +13 -0
- package/dist/parser/TemplateParser.d.ts +8 -0
- package/dist/parser/TemplateParser.js +75 -0
- package/dist/parser/index.d.ts +2 -0
- package/dist/parser/index.js +22 -0
- package/dist/parser/types.d.ts +30 -0
- package/dist/parser/types.js +3 -0
- package/dist/pipeline/PipelineOrchestrator.d.ts +23 -0
- package/dist/pipeline/PipelineOrchestrator.js +191 -0
- package/dist/pipeline/index.d.ts +2 -0
- package/dist/pipeline/index.js +19 -0
- package/dist/pipeline/types.d.ts +41 -0
- package/dist/pipeline/types.js +13 -0
- package/dist/pricing/CacheManager.d.ts +75 -0
- package/dist/pricing/CacheManager.js +195 -0
- package/dist/pricing/PricingClient.d.ts +17 -0
- package/dist/pricing/PricingClient.js +122 -0
- package/dist/pricing/PricingService.d.ts +16 -0
- package/dist/pricing/PricingService.js +149 -0
- package/dist/pricing/calculators/ALBCalculator.d.ts +16 -0
- package/dist/pricing/calculators/ALBCalculator.js +163 -0
- package/dist/pricing/calculators/APIGatewayCalculator.d.ts +10 -0
- package/dist/pricing/calculators/APIGatewayCalculator.js +177 -0
- package/dist/pricing/calculators/CloudFrontCalculator.d.ts +59 -0
- package/dist/pricing/calculators/CloudFrontCalculator.js +151 -0
- package/dist/pricing/calculators/DynamoDBCalculator.d.ts +9 -0
- package/dist/pricing/calculators/DynamoDBCalculator.js +146 -0
- package/dist/pricing/calculators/EC2Calculator.d.ts +7 -0
- package/dist/pricing/calculators/EC2Calculator.js +80 -0
- package/dist/pricing/calculators/ECSCalculator.d.ts +9 -0
- package/dist/pricing/calculators/ECSCalculator.js +116 -0
- package/dist/pricing/calculators/ElastiCacheCalculator.d.ts +8 -0
- package/dist/pricing/calculators/ElastiCacheCalculator.js +106 -0
- package/dist/pricing/calculators/LambdaCalculator.d.ts +13 -0
- package/dist/pricing/calculators/LambdaCalculator.js +111 -0
- package/dist/pricing/calculators/NLBCalculator.d.ts +16 -0
- package/dist/pricing/calculators/NLBCalculator.js +138 -0
- package/dist/pricing/calculators/NatGatewayCalculator.d.ts +12 -0
- package/dist/pricing/calculators/NatGatewayCalculator.js +116 -0
- package/dist/pricing/calculators/RDSCalculator.d.ts +9 -0
- package/dist/pricing/calculators/RDSCalculator.js +103 -0
- package/dist/pricing/calculators/S3Calculator.d.ts +8 -0
- package/dist/pricing/calculators/S3Calculator.js +68 -0
- package/dist/pricing/calculators/VPCEndpointCalculator.d.ts +12 -0
- package/dist/pricing/calculators/VPCEndpointCalculator.js +129 -0
- package/dist/pricing/index.d.ts +10 -0
- package/dist/pricing/index.js +37 -0
- package/dist/pricing/types.d.ts +53 -0
- package/dist/pricing/types.js +22 -0
- package/dist/releasetag.txt +1 -0
- package/dist/reporter/Reporter.d.ts +18 -0
- package/dist/reporter/Reporter.js +412 -0
- package/dist/reporter/index.d.ts +2 -0
- package/dist/reporter/index.js +21 -0
- package/dist/reporter/types.d.ts +72 -0
- package/dist/reporter/types.js +3 -0
- package/dist/synthesis/SynthesisOrchestrator.d.ts +26 -0
- package/dist/synthesis/SynthesisOrchestrator.js +243 -0
- package/dist/synthesis/index.d.ts +2 -0
- package/dist/synthesis/index.js +19 -0
- package/dist/synthesis/types.d.ts +17 -0
- package/dist/synthesis/types.js +13 -0
- package/dist/threshold/ThresholdEnforcer.d.ts +29 -0
- package/dist/threshold/ThresholdEnforcer.js +143 -0
- package/dist/threshold/index.d.ts +2 -0
- package/dist/threshold/index.js +19 -0
- package/dist/threshold/types.d.ts +15 -0
- package/dist/threshold/types.js +17 -0
- package/docs/CALCULATORS.md +820 -0
- package/docs/CI_CD.md +608 -0
- package/docs/CONFIGURATION.md +407 -0
- package/docs/DEVELOPMENT.md +387 -0
- package/docs/RELEASE.md +223 -0
- package/docs/TROUBLESHOOTING.md +847 -0
- package/examples/.cdk-cost-analyzer.yml +85 -0
- package/examples/.gitlab-ci.yml +125 -0
- package/examples/api-usage.js +26 -0
- package/examples/complex/base.json +16 -0
- package/examples/complex/target.json +29 -0
- package/examples/monorepo/.gitlab-ci.yml +251 -0
- package/examples/monorepo/README.md +341 -0
- package/examples/monorepo/package.json +27 -0
- package/examples/monorepo/packages/backend-infra/.cdk-cost-analyzer.yml +34 -0
- package/examples/monorepo/packages/backend-infra/bin/app.ts +16 -0
- package/examples/monorepo/packages/backend-infra/cdk.json +7 -0
- package/examples/monorepo/packages/backend-infra/lib/backend-stack.ts +128 -0
- package/examples/monorepo/packages/backend-infra/package.json +30 -0
- package/examples/monorepo/packages/backend-infra/tsconfig.json +11 -0
- package/examples/monorepo/packages/data-infra/.cdk-cost-analyzer.yml +38 -0
- package/examples/monorepo/packages/data-infra/bin/app.ts +16 -0
- package/examples/monorepo/packages/data-infra/cdk.json +7 -0
- package/examples/monorepo/packages/data-infra/lib/data-stack.ts +121 -0
- package/examples/monorepo/packages/data-infra/package.json +30 -0
- package/examples/monorepo/packages/data-infra/tsconfig.json +11 -0
- package/examples/monorepo/packages/frontend-infra/.cdk-cost-analyzer.yml +31 -0
- package/examples/monorepo/packages/frontend-infra/bin/app.ts +16 -0
- package/examples/monorepo/packages/frontend-infra/cdk.json +7 -0
- package/examples/monorepo/packages/frontend-infra/lib/frontend-stack.ts +60 -0
- package/examples/monorepo/packages/frontend-infra/package.json +30 -0
- package/examples/monorepo/packages/frontend-infra/tsconfig.json +11 -0
- package/examples/monorepo/tsconfig.json +35 -0
- package/examples/multi-stack/.cdk-cost-analyzer.yml +72 -0
- package/examples/multi-stack/.gitlab-ci.yml +184 -0
- package/examples/multi-stack/README.md +279 -0
- package/examples/multi-stack/bin/app.ts +36 -0
- package/examples/multi-stack/cdk.json +72 -0
- package/examples/multi-stack/lib/compute-stack.ts +128 -0
- package/examples/multi-stack/lib/networking-stack.ts +69 -0
- package/examples/multi-stack/lib/storage-stack.ts +141 -0
- package/examples/multi-stack/package-lock.json +4437 -0
- package/examples/multi-stack/package.json +42 -0
- package/examples/multi-stack/tsconfig.json +34 -0
- package/examples/simple/base.json +8 -0
- package/examples/simple/target.json +14 -0
- package/examples/single-stack/.NVP +0 -0
- package/examples/single-stack/.cdk-cost-analyzer.yml +52 -0
- package/examples/single-stack/.gitlab-ci.yml +126 -0
- package/examples/single-stack/README.md +184 -0
- package/examples/single-stack/UeK +0 -0
- package/examples/single-stack/bin/app.ts +16 -0
- package/examples/single-stack/cdk.json +72 -0
- package/examples/single-stack/lib/infrastructure-stack.ts +119 -0
- package/examples/single-stack/package-lock.json +4443 -0
- package/examples/single-stack/package.json +38 -0
- package/examples/single-stack/tsconfig.json +34 -0
- package/package.json +139 -0
- package/test-cdk-project/README-COMPUTE.md +141 -0
- package/test-cdk-project/README.md +95 -0
- package/test-cdk-project/app-with-compute.js +102 -0
- package/test-cdk-project/app.js +81 -0
- package/test-cdk-project/cdk-compute.json +3 -0
- package/test-cdk-project/cdk.context.json +7 -0
- package/test-cdk-project/cdk.json +3 -0
- package/test-cdk-project/cdk.out/TestStack.assets.json +21 -0
- package/test-cdk-project/cdk.out/TestStack.template.json +115 -0
- package/test-cdk-project/cdk.out/cdk.out +1 -0
- package/test-cdk-project/cdk.out/manifest.json +503 -0
- package/test-cdk-project/cdk.out/tree.json +1 -0
- package/test-cdk-project/cdk.out.base/TestStack.assets.json +21 -0
- package/test-cdk-project/cdk.out.base/TestStack.template.json +115 -0
- package/test-cdk-project/cdk.out.base/cdk.out +1 -0
- package/test-cdk-project/cdk.out.base/manifest.json +503 -0
- package/test-cdk-project/cdk.out.base/tree.json +1 -0
- package/test-cdk-project/cdk.out.target/TestStack.assets.json +21 -0
- package/test-cdk-project/cdk.out.target/TestStack.template.json +183 -0
- package/test-cdk-project/cdk.out.target/cdk.out +1 -0
- package/test-cdk-project/cdk.out.target/manifest.json +521 -0
- package/test-cdk-project/cdk.out.target/tree.json +1 -0
- package/test-cdk-project/package-lock.json +422 -0
- package/test-cdk-project/package.json +17 -0
- package/tools/workflows/README.md +102 -0
- package/tools/workflows/validate-workflows.js +109 -0
- package/tools/workflows/workflow-utils.ts +181 -0
|
@@ -0,0 +1,177 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.APIGatewayCalculator = void 0;
|
|
4
|
+
class APIGatewayCalculator {
|
|
5
|
+
supports(resourceType) {
|
|
6
|
+
return resourceType === 'AWS::ApiGateway::RestApi' ||
|
|
7
|
+
resourceType === 'AWS::ApiGatewayV2::Api';
|
|
8
|
+
}
|
|
9
|
+
async calculateCost(resource, region, pricingClient) {
|
|
10
|
+
const isV2 = resource.type === 'AWS::ApiGatewayV2::Api';
|
|
11
|
+
const protocolType = isV2 ? resource.properties.ProtocolType : 'REST';
|
|
12
|
+
if (protocolType === 'WEBSOCKET') {
|
|
13
|
+
return this.calculateWebSocketCost(region, pricingClient);
|
|
14
|
+
}
|
|
15
|
+
else if (protocolType === 'HTTP') {
|
|
16
|
+
return this.calculateHttpApiCost(region, pricingClient);
|
|
17
|
+
}
|
|
18
|
+
else {
|
|
19
|
+
return this.calculateRestApiCost(region, pricingClient);
|
|
20
|
+
}
|
|
21
|
+
}
|
|
22
|
+
async calculateRestApiCost(region, pricingClient) {
|
|
23
|
+
try {
|
|
24
|
+
const assumedRequests = 1_000_000; // 1M requests per month
|
|
25
|
+
const costPerMillion = await pricingClient.getPrice({
|
|
26
|
+
serviceCode: 'AmazonApiGateway',
|
|
27
|
+
region: this.normalizeRegion(region),
|
|
28
|
+
filters: [
|
|
29
|
+
{ field: 'productFamily', value: 'API Calls' },
|
|
30
|
+
{ field: 'groupDescription', value: 'ApiGatewayRequest' },
|
|
31
|
+
],
|
|
32
|
+
});
|
|
33
|
+
if (costPerMillion === null) {
|
|
34
|
+
return {
|
|
35
|
+
amount: 0,
|
|
36
|
+
currency: 'USD',
|
|
37
|
+
confidence: 'unknown',
|
|
38
|
+
assumptions: ['Pricing data not available for API Gateway REST API'],
|
|
39
|
+
};
|
|
40
|
+
}
|
|
41
|
+
const monthlyCost = (assumedRequests / 1_000_000) * costPerMillion;
|
|
42
|
+
return {
|
|
43
|
+
amount: monthlyCost,
|
|
44
|
+
currency: 'USD',
|
|
45
|
+
confidence: 'medium',
|
|
46
|
+
assumptions: [
|
|
47
|
+
`Assumes ${assumedRequests.toLocaleString()} REST API requests per month`,
|
|
48
|
+
'REST API type',
|
|
49
|
+
'Does not include data transfer, caching, or other features',
|
|
50
|
+
'First 333M requests may have tiered pricing (not calculated)',
|
|
51
|
+
],
|
|
52
|
+
};
|
|
53
|
+
}
|
|
54
|
+
catch (error) {
|
|
55
|
+
return {
|
|
56
|
+
amount: 0,
|
|
57
|
+
currency: 'USD',
|
|
58
|
+
confidence: 'unknown',
|
|
59
|
+
assumptions: [`Failed to fetch pricing: ${error instanceof Error ? error.message : String(error)}`],
|
|
60
|
+
};
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
async calculateHttpApiCost(region, pricingClient) {
|
|
64
|
+
try {
|
|
65
|
+
const assumedRequests = 1_000_000; // 1M requests per month
|
|
66
|
+
const costPerMillion = await pricingClient.getPrice({
|
|
67
|
+
serviceCode: 'AmazonApiGateway',
|
|
68
|
+
region: this.normalizeRegion(region),
|
|
69
|
+
filters: [
|
|
70
|
+
{ field: 'productFamily', value: 'API Calls' },
|
|
71
|
+
{ field: 'groupDescription', value: 'ApiGatewayHttpRequest' },
|
|
72
|
+
],
|
|
73
|
+
});
|
|
74
|
+
if (costPerMillion === null) {
|
|
75
|
+
return {
|
|
76
|
+
amount: 0,
|
|
77
|
+
currency: 'USD',
|
|
78
|
+
confidence: 'unknown',
|
|
79
|
+
assumptions: ['Pricing data not available for API Gateway HTTP API'],
|
|
80
|
+
};
|
|
81
|
+
}
|
|
82
|
+
const monthlyCost = (assumedRequests / 1_000_000) * costPerMillion;
|
|
83
|
+
return {
|
|
84
|
+
amount: monthlyCost,
|
|
85
|
+
currency: 'USD',
|
|
86
|
+
confidence: 'medium',
|
|
87
|
+
assumptions: [
|
|
88
|
+
`Assumes ${assumedRequests.toLocaleString()} HTTP API requests per month`,
|
|
89
|
+
'HTTP API type',
|
|
90
|
+
'Does not include data transfer costs',
|
|
91
|
+
'First 300M requests may have tiered pricing (not calculated)',
|
|
92
|
+
],
|
|
93
|
+
};
|
|
94
|
+
}
|
|
95
|
+
catch (error) {
|
|
96
|
+
return {
|
|
97
|
+
amount: 0,
|
|
98
|
+
currency: 'USD',
|
|
99
|
+
confidence: 'unknown',
|
|
100
|
+
assumptions: [`Failed to fetch pricing: ${error instanceof Error ? error.message : String(error)}`],
|
|
101
|
+
};
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
async calculateWebSocketCost(region, pricingClient) {
|
|
105
|
+
try {
|
|
106
|
+
const assumedMessages = 1_000_000; // 1M messages per month
|
|
107
|
+
const assumedConnectionMinutes = 100_000; // 100K connection minutes
|
|
108
|
+
const messageCostPerMillion = await pricingClient.getPrice({
|
|
109
|
+
serviceCode: 'AmazonApiGateway',
|
|
110
|
+
region: this.normalizeRegion(region),
|
|
111
|
+
filters: [
|
|
112
|
+
{ field: 'productFamily', value: 'WebSocket' },
|
|
113
|
+
{ field: 'groupDescription', value: 'ApiGatewayMessage' },
|
|
114
|
+
],
|
|
115
|
+
});
|
|
116
|
+
const connectionCostPerMinute = await pricingClient.getPrice({
|
|
117
|
+
serviceCode: 'AmazonApiGateway',
|
|
118
|
+
region: this.normalizeRegion(region),
|
|
119
|
+
filters: [
|
|
120
|
+
{ field: 'productFamily', value: 'WebSocket' },
|
|
121
|
+
{ field: 'groupDescription', value: 'ApiGatewayConnectionMinute' },
|
|
122
|
+
],
|
|
123
|
+
});
|
|
124
|
+
if (messageCostPerMillion === null || connectionCostPerMinute === null) {
|
|
125
|
+
return {
|
|
126
|
+
amount: 0,
|
|
127
|
+
currency: 'USD',
|
|
128
|
+
confidence: 'unknown',
|
|
129
|
+
assumptions: ['Pricing data not available for API Gateway WebSocket API'],
|
|
130
|
+
};
|
|
131
|
+
}
|
|
132
|
+
const messageCost = (assumedMessages / 1_000_000) * messageCostPerMillion;
|
|
133
|
+
const connectionCost = assumedConnectionMinutes * connectionCostPerMinute;
|
|
134
|
+
const monthlyCost = messageCost + connectionCost;
|
|
135
|
+
return {
|
|
136
|
+
amount: monthlyCost,
|
|
137
|
+
currency: 'USD',
|
|
138
|
+
confidence: 'medium',
|
|
139
|
+
assumptions: [
|
|
140
|
+
`Assumes ${assumedMessages.toLocaleString()} WebSocket messages per month`,
|
|
141
|
+
`Assumes ${assumedConnectionMinutes.toLocaleString()} connection minutes per month`,
|
|
142
|
+
'WebSocket API type',
|
|
143
|
+
'Does not include data transfer costs',
|
|
144
|
+
],
|
|
145
|
+
};
|
|
146
|
+
}
|
|
147
|
+
catch (error) {
|
|
148
|
+
return {
|
|
149
|
+
amount: 0,
|
|
150
|
+
currency: 'USD',
|
|
151
|
+
confidence: 'unknown',
|
|
152
|
+
assumptions: [`Failed to fetch pricing: ${error instanceof Error ? error.message : String(error)}`],
|
|
153
|
+
};
|
|
154
|
+
}
|
|
155
|
+
}
|
|
156
|
+
normalizeRegion(region) {
|
|
157
|
+
const regionMap = {
|
|
158
|
+
'us-east-1': 'US East (N. Virginia)',
|
|
159
|
+
'us-east-2': 'US East (Ohio)',
|
|
160
|
+
'us-west-1': 'US West (N. California)',
|
|
161
|
+
'us-west-2': 'US West (Oregon)',
|
|
162
|
+
'eu-west-1': 'EU (Ireland)',
|
|
163
|
+
'eu-west-2': 'EU (London)',
|
|
164
|
+
'eu-west-3': 'EU (Paris)',
|
|
165
|
+
'eu-central-1': 'EU (Frankfurt)',
|
|
166
|
+
'eu-north-1': 'EU (Stockholm)',
|
|
167
|
+
'ap-south-1': 'Asia Pacific (Mumbai)',
|
|
168
|
+
'ap-southeast-1': 'Asia Pacific (Singapore)',
|
|
169
|
+
'ap-southeast-2': 'Asia Pacific (Sydney)',
|
|
170
|
+
'ap-northeast-1': 'Asia Pacific (Tokyo)',
|
|
171
|
+
'ap-northeast-2': 'Asia Pacific (Seoul)',
|
|
172
|
+
};
|
|
173
|
+
return regionMap[region] || region;
|
|
174
|
+
}
|
|
175
|
+
}
|
|
176
|
+
exports.APIGatewayCalculator = APIGatewayCalculator;
|
|
177
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"APIGatewayCalculator.js","sourceRoot":"","sources":["../../../src/pricing/calculators/APIGatewayCalculator.ts"],"names":[],"mappings":"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","sourcesContent":["import { ResourceWithId } from '../../diff/types';\nimport { ResourceCostCalculator, MonthlyCost, PricingClient } from '../types';\n\nexport class APIGatewayCalculator implements ResourceCostCalculator {\n  supports(resourceType: string): boolean {\n    return resourceType === 'AWS::ApiGateway::RestApi' ||\n           resourceType === 'AWS::ApiGatewayV2::Api';\n  }\n\n  async calculateCost(\n    resource: ResourceWithId,\n    region: string,\n    pricingClient: PricingClient,\n  ): Promise<MonthlyCost> {\n    const isV2 = resource.type === 'AWS::ApiGatewayV2::Api';\n    const protocolType = isV2 ? (resource.properties.ProtocolType as string) : 'REST';\n\n    if (protocolType === 'WEBSOCKET') {\n      return this.calculateWebSocketCost(region, pricingClient);\n    } else if (protocolType === 'HTTP') {\n      return this.calculateHttpApiCost(region, pricingClient);\n    } else {\n      return this.calculateRestApiCost(region, pricingClient);\n    }\n  }\n\n  private async calculateRestApiCost(\n    region: string,\n    pricingClient: PricingClient,\n  ): Promise<MonthlyCost> {\n    try {\n      const assumedRequests = 1_000_000; // 1M requests per month\n\n      const costPerMillion = await pricingClient.getPrice({\n        serviceCode: 'AmazonApiGateway',\n        region: this.normalizeRegion(region),\n        filters: [\n          { field: 'productFamily', value: 'API Calls' },\n          { field: 'groupDescription', value: 'ApiGatewayRequest' },\n        ],\n      });\n\n      if (costPerMillion === null) {\n        return {\n          amount: 0,\n          currency: 'USD',\n          confidence: 'unknown',\n          assumptions: ['Pricing data not available for API Gateway REST API'],\n        };\n      }\n\n      const monthlyCost = (assumedRequests / 1_000_000) * costPerMillion;\n\n      return {\n        amount: monthlyCost,\n        currency: 'USD',\n        confidence: 'medium',\n        assumptions: [\n          `Assumes ${assumedRequests.toLocaleString()} REST API requests per month`,\n          'REST API type',\n          'Does not include data transfer, caching, or other features',\n          'First 333M requests may have tiered pricing (not calculated)',\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 calculateHttpApiCost(\n    region: string,\n    pricingClient: PricingClient,\n  ): Promise<MonthlyCost> {\n    try {\n      const assumedRequests = 1_000_000; // 1M requests per month\n\n      const costPerMillion = await pricingClient.getPrice({\n        serviceCode: 'AmazonApiGateway',\n        region: this.normalizeRegion(region),\n        filters: [\n          { field: 'productFamily', value: 'API Calls' },\n          { field: 'groupDescription', value: 'ApiGatewayHttpRequest' },\n        ],\n      });\n\n      if (costPerMillion === null) {\n        return {\n          amount: 0,\n          currency: 'USD',\n          confidence: 'unknown',\n          assumptions: ['Pricing data not available for API Gateway HTTP API'],\n        };\n      }\n\n      const monthlyCost = (assumedRequests / 1_000_000) * costPerMillion;\n\n      return {\n        amount: monthlyCost,\n        currency: 'USD',\n        confidence: 'medium',\n        assumptions: [\n          `Assumes ${assumedRequests.toLocaleString()} HTTP API requests per month`,\n          'HTTP API type',\n          'Does not include data transfer costs',\n          'First 300M requests may have tiered pricing (not calculated)',\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 calculateWebSocketCost(\n    region: string,\n    pricingClient: PricingClient,\n  ): Promise<MonthlyCost> {\n    try {\n      const assumedMessages = 1_000_000; // 1M messages per month\n      const assumedConnectionMinutes = 100_000; // 100K connection minutes\n\n      const messageCostPerMillion = await pricingClient.getPrice({\n        serviceCode: 'AmazonApiGateway',\n        region: this.normalizeRegion(region),\n        filters: [\n          { field: 'productFamily', value: 'WebSocket' },\n          { field: 'groupDescription', value: 'ApiGatewayMessage' },\n        ],\n      });\n\n      const connectionCostPerMinute = await pricingClient.getPrice({\n        serviceCode: 'AmazonApiGateway',\n        region: this.normalizeRegion(region),\n        filters: [\n          { field: 'productFamily', value: 'WebSocket' },\n          { field: 'groupDescription', value: 'ApiGatewayConnectionMinute' },\n        ],\n      });\n\n      if (messageCostPerMillion === null || connectionCostPerMinute === null) {\n        return {\n          amount: 0,\n          currency: 'USD',\n          confidence: 'unknown',\n          assumptions: ['Pricing data not available for API Gateway WebSocket API'],\n        };\n      }\n\n      const messageCost = (assumedMessages / 1_000_000) * messageCostPerMillion;\n      const connectionCost = assumedConnectionMinutes * connectionCostPerMinute;\n      const monthlyCost = messageCost + connectionCost;\n\n      return {\n        amount: monthlyCost,\n        currency: 'USD',\n        confidence: 'medium',\n        assumptions: [\n          `Assumes ${assumedMessages.toLocaleString()} WebSocket messages per month`,\n          `Assumes ${assumedConnectionMinutes.toLocaleString()} connection minutes per month`,\n          'WebSocket API type',\n          'Does not include data transfer costs',\n        ],\n      };\n    } catch (error) {\n      return {\n        amount: 0,\n        currency: 'USD',\n        confidence: 'unknown',\n        assumptions: [`Failed to fetch pricing: ${error instanceof Error ? error.message : String(error)}`],\n      };\n    }\n  }\n\n  private normalizeRegion(region: string): string {\n    const regionMap: Record<string, string> = {\n      'us-east-1': 'US East (N. Virginia)',\n      'us-east-2': 'US East (Ohio)',\n      'us-west-1': 'US West (N. California)',\n      'us-west-2': 'US West (Oregon)',\n      'eu-west-1': 'EU (Ireland)',\n      'eu-west-2': 'EU (London)',\n      'eu-west-3': 'EU (Paris)',\n      'eu-central-1': 'EU (Frankfurt)',\n      'eu-north-1': 'EU (Stockholm)',\n      'ap-south-1': 'Asia Pacific (Mumbai)',\n      'ap-southeast-1': 'Asia Pacific (Singapore)',\n      'ap-southeast-2': 'Asia Pacific (Sydney)',\n      'ap-northeast-1': 'Asia Pacific (Tokyo)',\n      'ap-northeast-2': 'Asia Pacific (Seoul)',\n    };\n\n    return regionMap[region] || region;\n  }\n}\n"]}
|
|
@@ -0,0 +1,59 @@
|
|
|
1
|
+
import { ResourceWithId } from '../../diff/types';
|
|
2
|
+
import { ResourceCostCalculator, MonthlyCost, PricingClient } from '../types';
|
|
3
|
+
/**
|
|
4
|
+
* Calculator for AWS CloudFront distribution costs.
|
|
5
|
+
*
|
|
6
|
+
* Estimates monthly costs based on data transfer and request volumes.
|
|
7
|
+
* CloudFront pricing includes:
|
|
8
|
+
* - Data transfer out to internet (per GB)
|
|
9
|
+
* - HTTP/HTTPS requests (per 10,000 requests)
|
|
10
|
+
*
|
|
11
|
+
* @see https://aws.amazon.com/cloudfront/pricing/
|
|
12
|
+
*/
|
|
13
|
+
export declare class CloudFrontCalculator implements ResourceCostCalculator {
|
|
14
|
+
private readonly dataTransferGB?;
|
|
15
|
+
private readonly requests?;
|
|
16
|
+
private readonly DEFAULT_DATA_TRANSFER_GB;
|
|
17
|
+
private readonly DEFAULT_REQUESTS;
|
|
18
|
+
/**
|
|
19
|
+
* Creates a CloudFront cost calculator.
|
|
20
|
+
*
|
|
21
|
+
* @param dataTransferGB - Optional custom data transfer volume in GB per month
|
|
22
|
+
* @param requests - Optional custom request count per month
|
|
23
|
+
*/
|
|
24
|
+
constructor(dataTransferGB?: number | undefined, requests?: number | undefined);
|
|
25
|
+
/**
|
|
26
|
+
* Checks if this calculator supports the given resource type.
|
|
27
|
+
*
|
|
28
|
+
* @param resourceType - CloudFormation resource type
|
|
29
|
+
* @returns true if resource type is AWS::CloudFront::Distribution
|
|
30
|
+
*/
|
|
31
|
+
supports(resourceType: string): boolean;
|
|
32
|
+
/**
|
|
33
|
+
* Calculates monthly cost for a CloudFront distribution.
|
|
34
|
+
*
|
|
35
|
+
* Cost components:
|
|
36
|
+
* - Data transfer out to internet (based on volume in GB)
|
|
37
|
+
* - HTTP/HTTPS requests (per 10,000 requests)
|
|
38
|
+
*
|
|
39
|
+
* Default assumptions:
|
|
40
|
+
* - 100 GB data transfer per month
|
|
41
|
+
* - 1,000,000 requests per month
|
|
42
|
+
*
|
|
43
|
+
* @param _resource - CloudFormation resource (properties not used for CloudFront)
|
|
44
|
+
* @param region - AWS region for pricing (CloudFront is global but pricing varies by region)
|
|
45
|
+
* @param pricingClient - Client for querying AWS Pricing API
|
|
46
|
+
* @returns Monthly cost estimate with assumptions and confidence level
|
|
47
|
+
*/
|
|
48
|
+
calculateCost(_resource: ResourceWithId, region: string, pricingClient: PricingClient): Promise<MonthlyCost>;
|
|
49
|
+
/**
|
|
50
|
+
* Normalizes AWS region code to AWS Pricing API region name.
|
|
51
|
+
*
|
|
52
|
+
* CloudFront is a global service, but pricing varies by region and the
|
|
53
|
+
* AWS Pricing API requires region names in specific format.
|
|
54
|
+
*
|
|
55
|
+
* @param region - AWS region code (e.g., 'us-east-1')
|
|
56
|
+
* @returns AWS Pricing API region name (e.g., 'US East (N. Virginia)')
|
|
57
|
+
*/
|
|
58
|
+
private normalizeRegion;
|
|
59
|
+
}
|
|
@@ -0,0 +1,151 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.CloudFrontCalculator = void 0;
|
|
4
|
+
/**
|
|
5
|
+
* Calculator for AWS CloudFront distribution costs.
|
|
6
|
+
*
|
|
7
|
+
* Estimates monthly costs based on data transfer and request volumes.
|
|
8
|
+
* CloudFront pricing includes:
|
|
9
|
+
* - Data transfer out to internet (per GB)
|
|
10
|
+
* - HTTP/HTTPS requests (per 10,000 requests)
|
|
11
|
+
*
|
|
12
|
+
* @see https://aws.amazon.com/cloudfront/pricing/
|
|
13
|
+
*/
|
|
14
|
+
class CloudFrontCalculator {
|
|
15
|
+
dataTransferGB;
|
|
16
|
+
requests;
|
|
17
|
+
DEFAULT_DATA_TRANSFER_GB = 100;
|
|
18
|
+
DEFAULT_REQUESTS = 1000000;
|
|
19
|
+
/**
|
|
20
|
+
* Creates a CloudFront cost calculator.
|
|
21
|
+
*
|
|
22
|
+
* @param dataTransferGB - Optional custom data transfer volume in GB per month
|
|
23
|
+
* @param requests - Optional custom request count per month
|
|
24
|
+
*/
|
|
25
|
+
constructor(dataTransferGB, requests) {
|
|
26
|
+
this.dataTransferGB = dataTransferGB;
|
|
27
|
+
this.requests = requests;
|
|
28
|
+
}
|
|
29
|
+
/**
|
|
30
|
+
* Checks if this calculator supports the given resource type.
|
|
31
|
+
*
|
|
32
|
+
* @param resourceType - CloudFormation resource type
|
|
33
|
+
* @returns true if resource type is AWS::CloudFront::Distribution
|
|
34
|
+
*/
|
|
35
|
+
supports(resourceType) {
|
|
36
|
+
return resourceType === 'AWS::CloudFront::Distribution';
|
|
37
|
+
}
|
|
38
|
+
/**
|
|
39
|
+
* Calculates monthly cost for a CloudFront distribution.
|
|
40
|
+
*
|
|
41
|
+
* Cost components:
|
|
42
|
+
* - Data transfer out to internet (based on volume in GB)
|
|
43
|
+
* - HTTP/HTTPS requests (per 10,000 requests)
|
|
44
|
+
*
|
|
45
|
+
* Default assumptions:
|
|
46
|
+
* - 100 GB data transfer per month
|
|
47
|
+
* - 1,000,000 requests per month
|
|
48
|
+
*
|
|
49
|
+
* @param _resource - CloudFormation resource (properties not used for CloudFront)
|
|
50
|
+
* @param region - AWS region for pricing (CloudFront is global but pricing varies by region)
|
|
51
|
+
* @param pricingClient - Client for querying AWS Pricing API
|
|
52
|
+
* @returns Monthly cost estimate with assumptions and confidence level
|
|
53
|
+
*/
|
|
54
|
+
async calculateCost(_resource, region, pricingClient) {
|
|
55
|
+
const dataTransfer = this.dataTransferGB ?? this.DEFAULT_DATA_TRANSFER_GB;
|
|
56
|
+
const requestCount = this.requests ?? this.DEFAULT_REQUESTS;
|
|
57
|
+
try {
|
|
58
|
+
// Query pricing for data transfer out to internet
|
|
59
|
+
const dataTransferPrice = await pricingClient.getPrice({
|
|
60
|
+
serviceCode: 'AmazonCloudFront',
|
|
61
|
+
region: this.normalizeRegion(region),
|
|
62
|
+
filters: [{ field: 'transferType', value: 'CloudFront to Internet' }],
|
|
63
|
+
});
|
|
64
|
+
// Query pricing for HTTP/HTTPS requests
|
|
65
|
+
const requestPrice = await pricingClient.getPrice({
|
|
66
|
+
serviceCode: 'AmazonCloudFront',
|
|
67
|
+
region: this.normalizeRegion(region),
|
|
68
|
+
filters: [{ field: 'requestType', value: 'HTTP-Requests' }],
|
|
69
|
+
});
|
|
70
|
+
if (dataTransferPrice === null || requestPrice === null) {
|
|
71
|
+
const assumptions = [
|
|
72
|
+
`Pricing data not available for CloudFront in region ${region}`,
|
|
73
|
+
`Would assume ${dataTransfer} GB of data transfer out to internet`,
|
|
74
|
+
`Would assume ${requestCount.toLocaleString()} HTTP/HTTPS requests per month`,
|
|
75
|
+
];
|
|
76
|
+
if (this.dataTransferGB !== undefined) {
|
|
77
|
+
assumptions.push(`Using custom data transfer assumption: ${dataTransfer} GB from configuration`);
|
|
78
|
+
}
|
|
79
|
+
if (this.requests !== undefined) {
|
|
80
|
+
assumptions.push(`Using custom request count assumption: ${requestCount.toLocaleString()} requests from configuration`);
|
|
81
|
+
}
|
|
82
|
+
return {
|
|
83
|
+
amount: 0,
|
|
84
|
+
currency: 'USD',
|
|
85
|
+
confidence: 'unknown',
|
|
86
|
+
assumptions,
|
|
87
|
+
};
|
|
88
|
+
}
|
|
89
|
+
// Calculate costs
|
|
90
|
+
const dataTransferCost = dataTransfer * dataTransferPrice;
|
|
91
|
+
const requestCost = (requestCount / 10000) * requestPrice; // Pricing is per 10,000 requests
|
|
92
|
+
const totalCost = dataTransferCost + requestCost;
|
|
93
|
+
const assumptions = [
|
|
94
|
+
`Assumes ${dataTransfer} GB of data transfer out to internet`,
|
|
95
|
+
`Assumes ${requestCount.toLocaleString()} HTTP/HTTPS requests per month`,
|
|
96
|
+
];
|
|
97
|
+
if (this.dataTransferGB !== undefined) {
|
|
98
|
+
assumptions.push(`Using custom data transfer assumption: ${dataTransfer} GB from configuration`);
|
|
99
|
+
}
|
|
100
|
+
if (this.requests !== undefined) {
|
|
101
|
+
assumptions.push(`Using custom request count assumption: ${requestCount.toLocaleString()} requests from configuration`);
|
|
102
|
+
}
|
|
103
|
+
return {
|
|
104
|
+
amount: totalCost,
|
|
105
|
+
currency: 'USD',
|
|
106
|
+
confidence: 'medium',
|
|
107
|
+
assumptions,
|
|
108
|
+
};
|
|
109
|
+
}
|
|
110
|
+
catch (error) {
|
|
111
|
+
return {
|
|
112
|
+
amount: 0,
|
|
113
|
+
currency: 'USD',
|
|
114
|
+
confidence: 'unknown',
|
|
115
|
+
assumptions: [
|
|
116
|
+
`Failed to fetch pricing: ${error instanceof Error ? error.message : String(error)}`,
|
|
117
|
+
],
|
|
118
|
+
};
|
|
119
|
+
}
|
|
120
|
+
}
|
|
121
|
+
/**
|
|
122
|
+
* Normalizes AWS region code to AWS Pricing API region name.
|
|
123
|
+
*
|
|
124
|
+
* CloudFront is a global service, but pricing varies by region and the
|
|
125
|
+
* AWS Pricing API requires region names in specific format.
|
|
126
|
+
*
|
|
127
|
+
* @param region - AWS region code (e.g., 'us-east-1')
|
|
128
|
+
* @returns AWS Pricing API region name (e.g., 'US East (N. Virginia)')
|
|
129
|
+
*/
|
|
130
|
+
normalizeRegion(region) {
|
|
131
|
+
const regionMap = {
|
|
132
|
+
'us-east-1': 'US East (N. Virginia)',
|
|
133
|
+
'us-east-2': 'US East (Ohio)',
|
|
134
|
+
'us-west-1': 'US West (N. California)',
|
|
135
|
+
'us-west-2': 'US West (Oregon)',
|
|
136
|
+
'eu-west-1': 'EU (Ireland)',
|
|
137
|
+
'eu-west-2': 'EU (London)',
|
|
138
|
+
'eu-west-3': 'EU (Paris)',
|
|
139
|
+
'eu-central-1': 'EU (Frankfurt)',
|
|
140
|
+
'eu-north-1': 'EU (Stockholm)',
|
|
141
|
+
'ap-south-1': 'Asia Pacific (Mumbai)',
|
|
142
|
+
'ap-southeast-1': 'Asia Pacific (Singapore)',
|
|
143
|
+
'ap-southeast-2': 'Asia Pacific (Sydney)',
|
|
144
|
+
'ap-northeast-1': 'Asia Pacific (Tokyo)',
|
|
145
|
+
'ap-northeast-2': 'Asia Pacific (Seoul)',
|
|
146
|
+
};
|
|
147
|
+
return regionMap[region] || region;
|
|
148
|
+
}
|
|
149
|
+
}
|
|
150
|
+
exports.CloudFrontCalculator = CloudFrontCalculator;
|
|
151
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"CloudFrontCalculator.js","sourceRoot":"","sources":["../../../src/pricing/calculators/CloudFrontCalculator.ts"],"names":[],"mappings":"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","sourcesContent":["import { ResourceWithId } from '../../diff/types';\nimport { ResourceCostCalculator, MonthlyCost, PricingClient } from '../types';\n\n/**\n * Calculator for AWS CloudFront distribution costs.\n *\n * Estimates monthly costs based on data transfer and request volumes.\n * CloudFront pricing includes:\n * - Data transfer out to internet (per GB)\n * - HTTP/HTTPS requests (per 10,000 requests)\n *\n * @see https://aws.amazon.com/cloudfront/pricing/\n */\nexport class CloudFrontCalculator implements ResourceCostCalculator {\n  private readonly DEFAULT_DATA_TRANSFER_GB = 100;\n  private readonly DEFAULT_REQUESTS = 1000000;\n\n  /**\n   * Creates a CloudFront cost calculator.\n   *\n   * @param dataTransferGB - Optional custom data transfer volume in GB per month\n   * @param requests - Optional custom request count per month\n   */\n  constructor(\n    private readonly dataTransferGB?: number,\n    private readonly requests?: number,\n  ) {}\n\n  /**\n   * Checks if this calculator supports the given resource type.\n   *\n   * @param resourceType - CloudFormation resource type\n   * @returns true if resource type is AWS::CloudFront::Distribution\n   */\n  supports(resourceType: string): boolean {\n    return resourceType === 'AWS::CloudFront::Distribution';\n  }\n\n  /**\n   * Calculates monthly cost for a CloudFront distribution.\n   *\n   * Cost components:\n   * - Data transfer out to internet (based on volume in GB)\n   * - HTTP/HTTPS requests (per 10,000 requests)\n   *\n   * Default assumptions:\n   * - 100 GB data transfer per month\n   * - 1,000,000 requests per month\n   *\n   * @param _resource - CloudFormation resource (properties not used for CloudFront)\n   * @param region - AWS region for pricing (CloudFront is global but pricing varies by region)\n   * @param pricingClient - Client for querying AWS Pricing API\n   * @returns Monthly cost estimate with assumptions and confidence level\n   */\n  async calculateCost(\n    _resource: ResourceWithId,\n    region: string,\n    pricingClient: PricingClient,\n  ): Promise<MonthlyCost> {\n    const dataTransfer = this.dataTransferGB ?? this.DEFAULT_DATA_TRANSFER_GB;\n    const requestCount = this.requests ?? this.DEFAULT_REQUESTS;\n\n    try {\n      // Query pricing for data transfer out to internet\n      const dataTransferPrice = await pricingClient.getPrice({\n        serviceCode: 'AmazonCloudFront',\n        region: this.normalizeRegion(region),\n        filters: [{ field: 'transferType', value: 'CloudFront to Internet' }],\n      });\n\n      // Query pricing for HTTP/HTTPS requests\n      const requestPrice = await pricingClient.getPrice({\n        serviceCode: 'AmazonCloudFront',\n        region: this.normalizeRegion(region),\n        filters: [{ field: 'requestType', value: 'HTTP-Requests' }],\n      });\n\n      if (dataTransferPrice === null || requestPrice === null) {\n        const assumptions = [\n          `Pricing data not available for CloudFront in region ${region}`,\n          `Would assume ${dataTransfer} GB of data transfer out to internet`,\n          `Would assume ${requestCount.toLocaleString()} HTTP/HTTPS requests per month`,\n        ];\n\n        if (this.dataTransferGB !== undefined) {\n          assumptions.push(\n            `Using custom data transfer assumption: ${dataTransfer} GB from configuration`,\n          );\n        }\n        if (this.requests !== undefined) {\n          assumptions.push(\n            `Using custom request count assumption: ${requestCount.toLocaleString()} requests from configuration`,\n          );\n        }\n\n        return {\n          amount: 0,\n          currency: 'USD',\n          confidence: 'unknown',\n          assumptions,\n        };\n      }\n\n      // Calculate costs\n      const dataTransferCost = dataTransfer * dataTransferPrice;\n      const requestCost = (requestCount / 10000) * requestPrice; // Pricing is per 10,000 requests\n\n      const totalCost = dataTransferCost + requestCost;\n\n      const assumptions = [\n        `Assumes ${dataTransfer} GB of data transfer out to internet`,\n        `Assumes ${requestCount.toLocaleString()} HTTP/HTTPS requests per month`,\n      ];\n\n      if (this.dataTransferGB !== undefined) {\n        assumptions.push(\n          `Using custom data transfer assumption: ${dataTransfer} GB from configuration`,\n        );\n      }\n      if (this.requests !== undefined) {\n        assumptions.push(\n          `Using custom request count assumption: ${requestCount.toLocaleString()} requests from configuration`,\n        );\n      }\n\n      return {\n        amount: totalCost,\n        currency: 'USD',\n        confidence: 'medium',\n        assumptions,\n      };\n    } catch (error) {\n      return {\n        amount: 0,\n        currency: 'USD',\n        confidence: 'unknown',\n        assumptions: [\n          `Failed to fetch pricing: ${error instanceof Error ? error.message : String(error)}`,\n        ],\n      };\n    }\n  }\n\n  /**\n   * Normalizes AWS region code to AWS Pricing API region name.\n   *\n   * CloudFront is a global service, but pricing varies by region and the\n   * AWS Pricing API requires region names in specific format.\n   *\n   * @param region - AWS region code (e.g., 'us-east-1')\n   * @returns AWS Pricing API region name (e.g., 'US East (N. Virginia)')\n   */\n  private normalizeRegion(region: string): string {\n    const regionMap: Record<string, string> = {\n      'us-east-1': 'US East (N. Virginia)',\n      'us-east-2': 'US East (Ohio)',\n      'us-west-1': 'US West (N. California)',\n      'us-west-2': 'US West (Oregon)',\n      'eu-west-1': 'EU (Ireland)',\n      'eu-west-2': 'EU (London)',\n      'eu-west-3': 'EU (Paris)',\n      'eu-central-1': 'EU (Frankfurt)',\n      'eu-north-1': 'EU (Stockholm)',\n      'ap-south-1': 'Asia Pacific (Mumbai)',\n      'ap-southeast-1': 'Asia Pacific (Singapore)',\n      'ap-southeast-2': 'Asia Pacific (Sydney)',\n      'ap-northeast-1': 'Asia Pacific (Tokyo)',\n      'ap-northeast-2': 'Asia Pacific (Seoul)',\n    };\n\n    return regionMap[region] || region;\n  }\n}\n"]}
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
import { ResourceWithId } from '../../diff/types';
|
|
2
|
+
import { ResourceCostCalculator, MonthlyCost, PricingClient } from '../types';
|
|
3
|
+
export declare class DynamoDBCalculator implements ResourceCostCalculator {
|
|
4
|
+
supports(resourceType: string): boolean;
|
|
5
|
+
calculateCost(resource: ResourceWithId, region: string, pricingClient: PricingClient): Promise<MonthlyCost>;
|
|
6
|
+
private calculateOnDemandCost;
|
|
7
|
+
private calculateProvisionedCost;
|
|
8
|
+
private normalizeRegion;
|
|
9
|
+
}
|
|
@@ -0,0 +1,146 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.DynamoDBCalculator = void 0;
|
|
4
|
+
class DynamoDBCalculator {
|
|
5
|
+
supports(resourceType) {
|
|
6
|
+
return resourceType === 'AWS::DynamoDB::Table';
|
|
7
|
+
}
|
|
8
|
+
async calculateCost(resource, region, pricingClient) {
|
|
9
|
+
const billingMode = resource.properties.BillingMode || 'PROVISIONED';
|
|
10
|
+
if (billingMode === 'PAY_PER_REQUEST') {
|
|
11
|
+
return this.calculateOnDemandCost(resource, region, pricingClient);
|
|
12
|
+
}
|
|
13
|
+
else {
|
|
14
|
+
return this.calculateProvisionedCost(resource, region, pricingClient);
|
|
15
|
+
}
|
|
16
|
+
}
|
|
17
|
+
async calculateOnDemandCost(_resource, region, pricingClient) {
|
|
18
|
+
try {
|
|
19
|
+
// Default assumptions for on-demand mode
|
|
20
|
+
const assumedReadRequests = 10_000_000; // 10M read requests per month
|
|
21
|
+
const assumedWriteRequests = 1_000_000; // 1M write requests per month
|
|
22
|
+
const readCostPerMillion = await pricingClient.getPrice({
|
|
23
|
+
serviceCode: 'AmazonDynamoDB',
|
|
24
|
+
region: this.normalizeRegion(region),
|
|
25
|
+
filters: [
|
|
26
|
+
{ field: 'group', value: 'DDB-ReadUnits' },
|
|
27
|
+
{ field: 'groupDescription', value: 'OnDemand ReadRequestUnits' },
|
|
28
|
+
],
|
|
29
|
+
});
|
|
30
|
+
const writeCostPerMillion = await pricingClient.getPrice({
|
|
31
|
+
serviceCode: 'AmazonDynamoDB',
|
|
32
|
+
region: this.normalizeRegion(region),
|
|
33
|
+
filters: [
|
|
34
|
+
{ field: 'group', value: 'DDB-WriteUnits' },
|
|
35
|
+
{ field: 'groupDescription', value: 'OnDemand WriteRequestUnits' },
|
|
36
|
+
],
|
|
37
|
+
});
|
|
38
|
+
if (readCostPerMillion === null || writeCostPerMillion === null) {
|
|
39
|
+
return {
|
|
40
|
+
amount: 0,
|
|
41
|
+
currency: 'USD',
|
|
42
|
+
confidence: 'unknown',
|
|
43
|
+
assumptions: ['Pricing data not available for DynamoDB on-demand mode'],
|
|
44
|
+
};
|
|
45
|
+
}
|
|
46
|
+
const readCost = (assumedReadRequests / 1_000_000) * readCostPerMillion;
|
|
47
|
+
const writeCost = (assumedWriteRequests / 1_000_000) * writeCostPerMillion;
|
|
48
|
+
const monthlyCost = readCost + writeCost;
|
|
49
|
+
return {
|
|
50
|
+
amount: monthlyCost,
|
|
51
|
+
currency: 'USD',
|
|
52
|
+
confidence: 'medium',
|
|
53
|
+
assumptions: [
|
|
54
|
+
`Assumes ${assumedReadRequests.toLocaleString()} read requests per month`,
|
|
55
|
+
`Assumes ${assumedWriteRequests.toLocaleString()} write requests per month`,
|
|
56
|
+
'On-demand billing mode',
|
|
57
|
+
'Does not include storage costs or other features (streams, backups, etc.)',
|
|
58
|
+
],
|
|
59
|
+
};
|
|
60
|
+
}
|
|
61
|
+
catch (error) {
|
|
62
|
+
return {
|
|
63
|
+
amount: 0,
|
|
64
|
+
currency: 'USD',
|
|
65
|
+
confidence: 'unknown',
|
|
66
|
+
assumptions: [`Failed to fetch pricing: ${error instanceof Error ? error.message : String(error)}`],
|
|
67
|
+
};
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
async calculateProvisionedCost(resource, region, pricingClient) {
|
|
71
|
+
try {
|
|
72
|
+
const provisionedThroughput = resource.properties.ProvisionedThroughput;
|
|
73
|
+
const readCapacity = provisionedThroughput?.ReadCapacityUnits || 5;
|
|
74
|
+
const writeCapacity = provisionedThroughput?.WriteCapacityUnits || 5;
|
|
75
|
+
const readCostPerHour = await pricingClient.getPrice({
|
|
76
|
+
serviceCode: 'AmazonDynamoDB',
|
|
77
|
+
region: this.normalizeRegion(region),
|
|
78
|
+
filters: [
|
|
79
|
+
{ field: 'group', value: 'DDB-ReadUnits' },
|
|
80
|
+
{ field: 'groupDescription', value: 'Provisioned ReadCapacityUnit-Hrs' },
|
|
81
|
+
],
|
|
82
|
+
});
|
|
83
|
+
const writeCostPerHour = await pricingClient.getPrice({
|
|
84
|
+
serviceCode: 'AmazonDynamoDB',
|
|
85
|
+
region: this.normalizeRegion(region),
|
|
86
|
+
filters: [
|
|
87
|
+
{ field: 'group', value: 'DDB-WriteUnits' },
|
|
88
|
+
{ field: 'groupDescription', value: 'Provisioned WriteCapacityUnit-Hrs' },
|
|
89
|
+
],
|
|
90
|
+
});
|
|
91
|
+
if (readCostPerHour === null || writeCostPerHour === null) {
|
|
92
|
+
return {
|
|
93
|
+
amount: 0,
|
|
94
|
+
currency: 'USD',
|
|
95
|
+
confidence: 'unknown',
|
|
96
|
+
assumptions: ['Pricing data not available for DynamoDB provisioned mode'],
|
|
97
|
+
};
|
|
98
|
+
}
|
|
99
|
+
const hoursPerMonth = 730;
|
|
100
|
+
const readCost = readCapacity * hoursPerMonth * readCostPerHour;
|
|
101
|
+
const writeCost = writeCapacity * hoursPerMonth * writeCostPerHour;
|
|
102
|
+
const monthlyCost = readCost + writeCost;
|
|
103
|
+
return {
|
|
104
|
+
amount: monthlyCost,
|
|
105
|
+
currency: 'USD',
|
|
106
|
+
confidence: 'high',
|
|
107
|
+
assumptions: [
|
|
108
|
+
`${readCapacity} provisioned read capacity units`,
|
|
109
|
+
`${writeCapacity} provisioned write capacity units`,
|
|
110
|
+
`Assumes ${hoursPerMonth} hours per month (24/7 operation)`,
|
|
111
|
+
'Provisioned billing mode',
|
|
112
|
+
'Does not include storage costs or other features (streams, backups, etc.)',
|
|
113
|
+
],
|
|
114
|
+
};
|
|
115
|
+
}
|
|
116
|
+
catch (error) {
|
|
117
|
+
return {
|
|
118
|
+
amount: 0,
|
|
119
|
+
currency: 'USD',
|
|
120
|
+
confidence: 'unknown',
|
|
121
|
+
assumptions: [`Failed to fetch pricing: ${error instanceof Error ? error.message : String(error)}`],
|
|
122
|
+
};
|
|
123
|
+
}
|
|
124
|
+
}
|
|
125
|
+
normalizeRegion(region) {
|
|
126
|
+
const regionMap = {
|
|
127
|
+
'us-east-1': 'US East (N. Virginia)',
|
|
128
|
+
'us-east-2': 'US East (Ohio)',
|
|
129
|
+
'us-west-1': 'US West (N. California)',
|
|
130
|
+
'us-west-2': 'US West (Oregon)',
|
|
131
|
+
'eu-west-1': 'EU (Ireland)',
|
|
132
|
+
'eu-west-2': 'EU (London)',
|
|
133
|
+
'eu-west-3': 'EU (Paris)',
|
|
134
|
+
'eu-central-1': 'EU (Frankfurt)',
|
|
135
|
+
'eu-north-1': 'EU (Stockholm)',
|
|
136
|
+
'ap-south-1': 'Asia Pacific (Mumbai)',
|
|
137
|
+
'ap-southeast-1': 'Asia Pacific (Singapore)',
|
|
138
|
+
'ap-southeast-2': 'Asia Pacific (Sydney)',
|
|
139
|
+
'ap-northeast-1': 'Asia Pacific (Tokyo)',
|
|
140
|
+
'ap-northeast-2': 'Asia Pacific (Seoul)',
|
|
141
|
+
};
|
|
142
|
+
return regionMap[region] || region;
|
|
143
|
+
}
|
|
144
|
+
}
|
|
145
|
+
exports.DynamoDBCalculator = DynamoDBCalculator;
|
|
146
|
+
//# 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';\n\nexport class DynamoDBCalculator implements ResourceCostCalculator {\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\n    if (billingMode === 'PAY_PER_REQUEST') {\n      return this.calculateOnDemandCost(resource, region, pricingClient);\n    } else {\n      return this.calculateProvisionedCost(resource, region, pricingClient);\n    }\n  }\n\n  private async calculateOnDemandCost(\n    _resource: ResourceWithId,\n    region: string,\n    pricingClient: PricingClient,\n  ): Promise<MonthlyCost> {\n    try {\n      // Default assumptions for on-demand mode\n      const assumedReadRequests = 10_000_000; // 10M read requests per month\n      const assumedWriteRequests = 1_000_000; // 1M write requests per month\n\n      const readCostPerMillion = await pricingClient.getPrice({\n        serviceCode: 'AmazonDynamoDB',\n        region: this.normalizeRegion(region),\n        filters: [\n          { field: 'group', value: 'DDB-ReadUnits' },\n          { field: 'groupDescription', value: 'OnDemand ReadRequestUnits' },\n        ],\n      });\n\n      const writeCostPerMillion = await pricingClient.getPrice({\n        serviceCode: 'AmazonDynamoDB',\n        region: this.normalizeRegion(region),\n        filters: [\n          { field: 'group', value: 'DDB-WriteUnits' },\n          { field: 'groupDescription', value: 'OnDemand WriteRequestUnits' },\n        ],\n      });\n\n      if (readCostPerMillion === null || writeCostPerMillion === null) {\n        return {\n          amount: 0,\n          currency: 'USD',\n          confidence: 'unknown',\n          assumptions: ['Pricing data not available for DynamoDB on-demand mode'],\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: [`Failed to fetch pricing: ${error instanceof Error ? error.message : String(error)}`],\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      const readCostPerHour = await pricingClient.getPrice({\n        serviceCode: 'AmazonDynamoDB',\n        region: this.normalizeRegion(region),\n        filters: [\n          { field: 'group', value: 'DDB-ReadUnits' },\n          { field: 'groupDescription', value: 'Provisioned ReadCapacityUnit-Hrs' },\n        ],\n      });\n\n      const writeCostPerHour = await pricingClient.getPrice({\n        serviceCode: 'AmazonDynamoDB',\n        region: this.normalizeRegion(region),\n        filters: [\n          { field: 'group', value: 'DDB-WriteUnits' },\n          { field: 'groupDescription', value: 'Provisioned WriteCapacityUnit-Hrs' },\n        ],\n      });\n\n      if (readCostPerHour === null || writeCostPerHour === null) {\n        return {\n          amount: 0,\n          currency: 'USD',\n          confidence: 'unknown',\n          assumptions: ['Pricing data not available for DynamoDB provisioned mode'],\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: [`Failed to fetch pricing: ${error instanceof Error ? error.message : String(error)}`],\n      };\n    }\n  }\n\n  private normalizeRegion(region: string): string {\n    const regionMap: Record<string, string> = {\n      'us-east-1': 'US East (N. Virginia)',\n      'us-east-2': 'US East (Ohio)',\n      'us-west-1': 'US West (N. California)',\n      'us-west-2': 'US West (Oregon)',\n      'eu-west-1': 'EU (Ireland)',\n      'eu-west-2': 'EU (London)',\n      'eu-west-3': 'EU (Paris)',\n      'eu-central-1': 'EU (Frankfurt)',\n      'eu-north-1': 'EU (Stockholm)',\n      'ap-south-1': 'Asia Pacific (Mumbai)',\n      'ap-southeast-1': 'Asia Pacific (Singapore)',\n      'ap-southeast-2': 'Asia Pacific (Sydney)',\n      'ap-northeast-1': 'Asia Pacific (Tokyo)',\n      'ap-northeast-2': 'Asia Pacific (Seoul)',\n    };\n\n    return regionMap[region] || region;\n  }\n}\n"]}
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
import { ResourceWithId } from '../../diff/types';
|
|
2
|
+
import { ResourceCostCalculator, MonthlyCost, PricingClient } from '../types';
|
|
3
|
+
export declare class EC2Calculator implements ResourceCostCalculator {
|
|
4
|
+
supports(resourceType: string): boolean;
|
|
5
|
+
calculateCost(resource: ResourceWithId, region: string, pricingClient: PricingClient): Promise<MonthlyCost>;
|
|
6
|
+
private normalizeRegion;
|
|
7
|
+
}
|