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,129 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.VPCEndpointCalculator = void 0;
|
|
4
|
+
class VPCEndpointCalculator {
|
|
5
|
+
customDataProcessedGB;
|
|
6
|
+
DEFAULT_DATA_PROCESSED_GB = 100;
|
|
7
|
+
HOURS_PER_MONTH = 730;
|
|
8
|
+
constructor(customDataProcessedGB) {
|
|
9
|
+
this.customDataProcessedGB = customDataProcessedGB;
|
|
10
|
+
}
|
|
11
|
+
supports(resourceType) {
|
|
12
|
+
return resourceType === 'AWS::EC2::VPCEndpoint';
|
|
13
|
+
}
|
|
14
|
+
async calculateCost(resource, region, pricingClient) {
|
|
15
|
+
const explicitEndpointType = resource.properties?.VpcEndpointType;
|
|
16
|
+
const serviceName = resource.properties?.ServiceName || '';
|
|
17
|
+
// Determine if this is a Gateway endpoint
|
|
18
|
+
// Priority: explicit type > inferred from service name
|
|
19
|
+
const isGatewayEndpoint = explicitEndpointType === 'Gateway' ||
|
|
20
|
+
(explicitEndpointType === undefined &&
|
|
21
|
+
(serviceName.includes('s3') || serviceName.includes('dynamodb')));
|
|
22
|
+
// Gateway endpoints (S3 and DynamoDB) are free
|
|
23
|
+
if (isGatewayEndpoint) {
|
|
24
|
+
return {
|
|
25
|
+
amount: 0,
|
|
26
|
+
currency: 'USD',
|
|
27
|
+
confidence: 'high',
|
|
28
|
+
assumptions: [
|
|
29
|
+
'Gateway VPC endpoints for S3 and DynamoDB are free',
|
|
30
|
+
'No data processing charges for gateway endpoints',
|
|
31
|
+
],
|
|
32
|
+
};
|
|
33
|
+
}
|
|
34
|
+
// Interface endpoints have hourly and data processing costs
|
|
35
|
+
try {
|
|
36
|
+
// Get hourly rate per endpoint
|
|
37
|
+
const hourlyRate = await pricingClient.getPrice({
|
|
38
|
+
serviceCode: 'AmazonVPC',
|
|
39
|
+
region: this.normalizeRegion(region),
|
|
40
|
+
filters: [
|
|
41
|
+
{ field: 'productFamily', value: 'VpcEndpoint' },
|
|
42
|
+
{ field: 'usagetype', value: `${this.getRegionPrefix(region)}VpcEndpoint-Hours` },
|
|
43
|
+
],
|
|
44
|
+
});
|
|
45
|
+
// Get data processing rate
|
|
46
|
+
const dataProcessingRate = await pricingClient.getPrice({
|
|
47
|
+
serviceCode: 'AmazonVPC',
|
|
48
|
+
region: this.normalizeRegion(region),
|
|
49
|
+
filters: [
|
|
50
|
+
{ field: 'productFamily', value: 'VpcEndpoint' },
|
|
51
|
+
{ field: 'usagetype', value: `${this.getRegionPrefix(region)}VpcEndpoint-Bytes` },
|
|
52
|
+
],
|
|
53
|
+
});
|
|
54
|
+
if (hourlyRate === null || dataProcessingRate === null) {
|
|
55
|
+
return {
|
|
56
|
+
amount: 0,
|
|
57
|
+
currency: 'USD',
|
|
58
|
+
confidence: 'unknown',
|
|
59
|
+
assumptions: [
|
|
60
|
+
`Pricing data not available for VPC Endpoint in region ${region}`,
|
|
61
|
+
],
|
|
62
|
+
};
|
|
63
|
+
}
|
|
64
|
+
const dataProcessedGB = this.customDataProcessedGB || this.DEFAULT_DATA_PROCESSED_GB;
|
|
65
|
+
const hourlyCost = hourlyRate * this.HOURS_PER_MONTH;
|
|
66
|
+
const dataProcessingCost = dataProcessingRate * dataProcessedGB;
|
|
67
|
+
const totalCost = hourlyCost + dataProcessingCost;
|
|
68
|
+
return {
|
|
69
|
+
amount: totalCost,
|
|
70
|
+
currency: 'USD',
|
|
71
|
+
confidence: 'medium',
|
|
72
|
+
assumptions: [
|
|
73
|
+
'Interface VPC Endpoint type',
|
|
74
|
+
`Hourly rate: $${hourlyRate.toFixed(4)}/hour × ${this.HOURS_PER_MONTH} hours = $${hourlyCost.toFixed(2)}/month`,
|
|
75
|
+
`Data processing: $${dataProcessingRate.toFixed(4)}/GB × ${dataProcessedGB} GB = $${dataProcessingCost.toFixed(2)}/month`,
|
|
76
|
+
`Total: $${totalCost.toFixed(2)}/month`,
|
|
77
|
+
],
|
|
78
|
+
};
|
|
79
|
+
}
|
|
80
|
+
catch (error) {
|
|
81
|
+
return {
|
|
82
|
+
amount: 0,
|
|
83
|
+
currency: 'USD',
|
|
84
|
+
confidence: 'unknown',
|
|
85
|
+
assumptions: [`Failed to fetch pricing: ${error instanceof Error ? error.message : String(error)}`],
|
|
86
|
+
};
|
|
87
|
+
}
|
|
88
|
+
}
|
|
89
|
+
normalizeRegion(region) {
|
|
90
|
+
const regionMap = {
|
|
91
|
+
'us-east-1': 'US East (N. Virginia)',
|
|
92
|
+
'us-east-2': 'US East (Ohio)',
|
|
93
|
+
'us-west-1': 'US West (N. California)',
|
|
94
|
+
'us-west-2': 'US West (Oregon)',
|
|
95
|
+
'eu-west-1': 'EU (Ireland)',
|
|
96
|
+
'eu-west-2': 'EU (London)',
|
|
97
|
+
'eu-west-3': 'EU (Paris)',
|
|
98
|
+
'eu-central-1': 'EU (Frankfurt)',
|
|
99
|
+
'eu-north-1': 'EU (Stockholm)',
|
|
100
|
+
'ap-south-1': 'Asia Pacific (Mumbai)',
|
|
101
|
+
'ap-southeast-1': 'Asia Pacific (Singapore)',
|
|
102
|
+
'ap-southeast-2': 'Asia Pacific (Sydney)',
|
|
103
|
+
'ap-northeast-1': 'Asia Pacific (Tokyo)',
|
|
104
|
+
'ap-northeast-2': 'Asia Pacific (Seoul)',
|
|
105
|
+
};
|
|
106
|
+
return regionMap[region] || region;
|
|
107
|
+
}
|
|
108
|
+
getRegionPrefix(region) {
|
|
109
|
+
const prefixMap = {
|
|
110
|
+
'us-east-1': 'USE1',
|
|
111
|
+
'us-east-2': 'USE2',
|
|
112
|
+
'us-west-1': 'USW1',
|
|
113
|
+
'us-west-2': 'USW2',
|
|
114
|
+
'eu-west-1': 'EUW1',
|
|
115
|
+
'eu-west-2': 'EUW2',
|
|
116
|
+
'eu-west-3': 'EUW3',
|
|
117
|
+
'eu-central-1': 'EUC1',
|
|
118
|
+
'eu-north-1': 'EUN1',
|
|
119
|
+
'ap-south-1': 'APS1',
|
|
120
|
+
'ap-southeast-1': 'APS2',
|
|
121
|
+
'ap-southeast-2': 'APS3',
|
|
122
|
+
'ap-northeast-1': 'APN1',
|
|
123
|
+
'ap-northeast-2': 'APN2',
|
|
124
|
+
};
|
|
125
|
+
return prefixMap[region] || '';
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
exports.VPCEndpointCalculator = VPCEndpointCalculator;
|
|
129
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"VPCEndpointCalculator.js","sourceRoot":"","sources":["../../../src/pricing/calculators/VPCEndpointCalculator.ts"],"names":[],"mappings":"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","sourcesContent":["import { ResourceWithId } from '../../diff/types';\nimport { ResourceCostCalculator, MonthlyCost, PricingClient } from '../types';\n\nexport class VPCEndpointCalculator implements ResourceCostCalculator {\n  private readonly DEFAULT_DATA_PROCESSED_GB = 100;\n  private readonly HOURS_PER_MONTH = 730;\n\n  constructor(private customDataProcessedGB?: number) {}\n\n  supports(resourceType: string): boolean {\n    return resourceType === 'AWS::EC2::VPCEndpoint';\n  }\n\n  async calculateCost(\n    resource: ResourceWithId,\n    region: string,\n    pricingClient: PricingClient,\n  ): Promise<MonthlyCost> {\n    const explicitEndpointType = resource.properties?.VpcEndpointType as string | undefined;\n    const serviceName = (resource.properties?.ServiceName as string | undefined) || '';\n\n    // Determine if this is a Gateway endpoint\n    // Priority: explicit type > inferred from service name\n    const isGatewayEndpoint =\n      explicitEndpointType === 'Gateway' ||\n      (explicitEndpointType === undefined &&\n        (serviceName.includes('s3') || serviceName.includes('dynamodb')));\n\n    // Gateway endpoints (S3 and DynamoDB) are free\n    if (isGatewayEndpoint) {\n      return {\n        amount: 0,\n        currency: 'USD',\n        confidence: 'high',\n        assumptions: [\n          'Gateway VPC endpoints for S3 and DynamoDB are free',\n          'No data processing charges for gateway endpoints',\n        ],\n      };\n    }\n\n    // Interface endpoints have hourly and data processing costs\n    try {\n      // Get hourly rate per endpoint\n      const hourlyRate = await pricingClient.getPrice({\n        serviceCode: 'AmazonVPC',\n        region: this.normalizeRegion(region),\n        filters: [\n          { field: 'productFamily', value: 'VpcEndpoint' },\n          { field: 'usagetype', value: `${this.getRegionPrefix(region)}VpcEndpoint-Hours` },\n        ],\n      });\n\n      // Get data processing rate\n      const dataProcessingRate = await pricingClient.getPrice({\n        serviceCode: 'AmazonVPC',\n        region: this.normalizeRegion(region),\n        filters: [\n          { field: 'productFamily', value: 'VpcEndpoint' },\n          { field: 'usagetype', value: `${this.getRegionPrefix(region)}VpcEndpoint-Bytes` },\n        ],\n      });\n\n      if (hourlyRate === null || dataProcessingRate === null) {\n        return {\n          amount: 0,\n          currency: 'USD',\n          confidence: 'unknown',\n          assumptions: [\n            `Pricing data not available for VPC Endpoint in region ${region}`,\n          ],\n        };\n      }\n\n      const dataProcessedGB = this.customDataProcessedGB || this.DEFAULT_DATA_PROCESSED_GB;\n      const hourlyCost = hourlyRate * this.HOURS_PER_MONTH;\n      const dataProcessingCost = dataProcessingRate * dataProcessedGB;\n      const totalCost = hourlyCost + dataProcessingCost;\n\n      return {\n        amount: totalCost,\n        currency: 'USD',\n        confidence: 'medium',\n        assumptions: [\n          'Interface VPC Endpoint type',\n          `Hourly rate: $${hourlyRate.toFixed(4)}/hour × ${this.HOURS_PER_MONTH} hours = $${hourlyCost.toFixed(2)}/month`,\n          `Data processing: $${dataProcessingRate.toFixed(4)}/GB × ${dataProcessedGB} GB = $${dataProcessingCost.toFixed(2)}/month`,\n          `Total: $${totalCost.toFixed(2)}/month`,\n        ],\n      };\n    } catch (error) {\n      return {\n        amount: 0,\n        currency: 'USD',\n        confidence: 'unknown',\n        assumptions: [`Failed to fetch pricing: ${error instanceof Error ? error.message : String(error)}`],\n      };\n    }\n  }\n\n  private normalizeRegion(region: string): string {\n    const regionMap: Record<string, string> = {\n      'us-east-1': 'US East (N. Virginia)',\n      'us-east-2': 'US East (Ohio)',\n      'us-west-1': 'US West (N. California)',\n      'us-west-2': 'US West (Oregon)',\n      'eu-west-1': 'EU (Ireland)',\n      'eu-west-2': 'EU (London)',\n      'eu-west-3': 'EU (Paris)',\n      'eu-central-1': 'EU (Frankfurt)',\n      'eu-north-1': 'EU (Stockholm)',\n      'ap-south-1': 'Asia Pacific (Mumbai)',\n      'ap-southeast-1': 'Asia Pacific (Singapore)',\n      'ap-southeast-2': 'Asia Pacific (Sydney)',\n      'ap-northeast-1': 'Asia Pacific (Tokyo)',\n      'ap-northeast-2': 'Asia Pacific (Seoul)',\n    };\n\n    return regionMap[region] || region;\n  }\n\n  private getRegionPrefix(region: string): string {\n    const prefixMap: Record<string, string> = {\n      'us-east-1': 'USE1',\n      'us-east-2': 'USE2',\n      'us-west-1': 'USW1',\n      'us-west-2': 'USW2',\n      'eu-west-1': 'EUW1',\n      'eu-west-2': 'EUW2',\n      'eu-west-3': 'EUW3',\n      'eu-central-1': 'EUC1',\n      'eu-north-1': 'EUN1',\n      'ap-south-1': 'APS1',\n      'ap-southeast-1': 'APS2',\n      'ap-southeast-2': 'APS3',\n      'ap-northeast-1': 'APN1',\n      'ap-northeast-2': 'APN2',\n    };\n\n    return prefixMap[region] || '';\n  }\n}\n"]}
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
export { PricingService } from './PricingService';
|
|
2
|
+
export { PricingClient } from './PricingClient';
|
|
3
|
+
export { CacheManager } from './CacheManager';
|
|
4
|
+
export { EC2Calculator } from './calculators/EC2Calculator';
|
|
5
|
+
export { S3Calculator } from './calculators/S3Calculator';
|
|
6
|
+
export { LambdaCalculator } from './calculators/LambdaCalculator';
|
|
7
|
+
export { RDSCalculator } from './calculators/RDSCalculator';
|
|
8
|
+
export { CloudFrontCalculator } from './calculators/CloudFrontCalculator';
|
|
9
|
+
export { ElastiCacheCalculator } from './calculators/ElastiCacheCalculator';
|
|
10
|
+
export * from './types';
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __exportStar = (this && this.__exportStar) || function(m, exports) {
|
|
14
|
+
for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
|
|
15
|
+
};
|
|
16
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
17
|
+
exports.ElastiCacheCalculator = exports.CloudFrontCalculator = exports.RDSCalculator = exports.LambdaCalculator = exports.S3Calculator = exports.EC2Calculator = exports.CacheManager = exports.PricingClient = exports.PricingService = void 0;
|
|
18
|
+
var PricingService_1 = require("./PricingService");
|
|
19
|
+
Object.defineProperty(exports, "PricingService", { enumerable: true, get: function () { return PricingService_1.PricingService; } });
|
|
20
|
+
var PricingClient_1 = require("./PricingClient");
|
|
21
|
+
Object.defineProperty(exports, "PricingClient", { enumerable: true, get: function () { return PricingClient_1.PricingClient; } });
|
|
22
|
+
var CacheManager_1 = require("./CacheManager");
|
|
23
|
+
Object.defineProperty(exports, "CacheManager", { enumerable: true, get: function () { return CacheManager_1.CacheManager; } });
|
|
24
|
+
var EC2Calculator_1 = require("./calculators/EC2Calculator");
|
|
25
|
+
Object.defineProperty(exports, "EC2Calculator", { enumerable: true, get: function () { return EC2Calculator_1.EC2Calculator; } });
|
|
26
|
+
var S3Calculator_1 = require("./calculators/S3Calculator");
|
|
27
|
+
Object.defineProperty(exports, "S3Calculator", { enumerable: true, get: function () { return S3Calculator_1.S3Calculator; } });
|
|
28
|
+
var LambdaCalculator_1 = require("./calculators/LambdaCalculator");
|
|
29
|
+
Object.defineProperty(exports, "LambdaCalculator", { enumerable: true, get: function () { return LambdaCalculator_1.LambdaCalculator; } });
|
|
30
|
+
var RDSCalculator_1 = require("./calculators/RDSCalculator");
|
|
31
|
+
Object.defineProperty(exports, "RDSCalculator", { enumerable: true, get: function () { return RDSCalculator_1.RDSCalculator; } });
|
|
32
|
+
var CloudFrontCalculator_1 = require("./calculators/CloudFrontCalculator");
|
|
33
|
+
Object.defineProperty(exports, "CloudFrontCalculator", { enumerable: true, get: function () { return CloudFrontCalculator_1.CloudFrontCalculator; } });
|
|
34
|
+
var ElastiCacheCalculator_1 = require("./calculators/ElastiCacheCalculator");
|
|
35
|
+
Object.defineProperty(exports, "ElastiCacheCalculator", { enumerable: true, get: function () { return ElastiCacheCalculator_1.ElastiCacheCalculator; } });
|
|
36
|
+
__exportStar(require("./types"), exports);
|
|
37
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvcHJpY2luZy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7OztBQUFBLG1EQUFrRDtBQUF6QyxnSEFBQSxjQUFjLE9BQUE7QUFDdkIsaURBQWdEO0FBQXZDLDhHQUFBLGFBQWEsT0FBQTtBQUN0QiwrQ0FBOEM7QUFBckMsNEdBQUEsWUFBWSxPQUFBO0FBQ3JCLDZEQUE0RDtBQUFuRCw4R0FBQSxhQUFhLE9BQUE7QUFDdEIsMkRBQTBEO0FBQWpELDRHQUFBLFlBQVksT0FBQTtBQUNyQixtRUFBa0U7QUFBekQsb0hBQUEsZ0JBQWdCLE9BQUE7QUFDekIsNkRBQTREO0FBQW5ELDhHQUFBLGFBQWEsT0FBQTtBQUN0QiwyRUFBMEU7QUFBakUsNEhBQUEsb0JBQW9CLE9BQUE7QUFDN0IsNkVBQTRFO0FBQW5FLDhIQUFBLHFCQUFxQixPQUFBO0FBQzlCLDBDQUF3QiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB7IFByaWNpbmdTZXJ2aWNlIH0gZnJvbSAnLi9QcmljaW5nU2VydmljZSc7XG5leHBvcnQgeyBQcmljaW5nQ2xpZW50IH0gZnJvbSAnLi9QcmljaW5nQ2xpZW50JztcbmV4cG9ydCB7IENhY2hlTWFuYWdlciB9IGZyb20gJy4vQ2FjaGVNYW5hZ2VyJztcbmV4cG9ydCB7IEVDMkNhbGN1bGF0b3IgfSBmcm9tICcuL2NhbGN1bGF0b3JzL0VDMkNhbGN1bGF0b3InO1xuZXhwb3J0IHsgUzNDYWxjdWxhdG9yIH0gZnJvbSAnLi9jYWxjdWxhdG9ycy9TM0NhbGN1bGF0b3InO1xuZXhwb3J0IHsgTGFtYmRhQ2FsY3VsYXRvciB9IGZyb20gJy4vY2FsY3VsYXRvcnMvTGFtYmRhQ2FsY3VsYXRvcic7XG5leHBvcnQgeyBSRFNDYWxjdWxhdG9yIH0gZnJvbSAnLi9jYWxjdWxhdG9ycy9SRFNDYWxjdWxhdG9yJztcbmV4cG9ydCB7IENsb3VkRnJvbnRDYWxjdWxhdG9yIH0gZnJvbSAnLi9jYWxjdWxhdG9ycy9DbG91ZEZyb250Q2FsY3VsYXRvcic7XG5leHBvcnQgeyBFbGFzdGlDYWNoZUNhbGN1bGF0b3IgfSBmcm9tICcuL2NhbGN1bGF0b3JzL0VsYXN0aUNhY2hlQ2FsY3VsYXRvcic7XG5leHBvcnQgKiBmcm9tICcuL3R5cGVzJztcbiJdfQ==
|
|
@@ -0,0 +1,53 @@
|
|
|
1
|
+
import { ResourceWithId, ResourceDiff } from '../diff/types';
|
|
2
|
+
export interface PricingService {
|
|
3
|
+
getResourceCost(resource: ResourceWithId, region: string): Promise<MonthlyCost>;
|
|
4
|
+
getCostDelta(diff: ResourceDiff, region: string): Promise<CostDelta>;
|
|
5
|
+
}
|
|
6
|
+
export interface MonthlyCost {
|
|
7
|
+
amount: number;
|
|
8
|
+
currency: string;
|
|
9
|
+
confidence: 'high' | 'medium' | 'low' | 'unknown';
|
|
10
|
+
assumptions: string[];
|
|
11
|
+
}
|
|
12
|
+
export interface CostDelta {
|
|
13
|
+
totalDelta: number;
|
|
14
|
+
currency: string;
|
|
15
|
+
addedCosts: ResourceCost[];
|
|
16
|
+
removedCosts: ResourceCost[];
|
|
17
|
+
modifiedCosts: ModifiedResourceCost[];
|
|
18
|
+
}
|
|
19
|
+
export interface ResourceCost {
|
|
20
|
+
logicalId: string;
|
|
21
|
+
type: string;
|
|
22
|
+
monthlyCost: MonthlyCost;
|
|
23
|
+
}
|
|
24
|
+
export interface ModifiedResourceCost extends ResourceCost {
|
|
25
|
+
oldMonthlyCost: MonthlyCost;
|
|
26
|
+
newMonthlyCost: MonthlyCost;
|
|
27
|
+
costDelta: number;
|
|
28
|
+
}
|
|
29
|
+
export interface ResourceCostCalculator {
|
|
30
|
+
supports(resourceType: string): boolean;
|
|
31
|
+
calculateCost(resource: ResourceWithId, region: string, pricingClient: PricingClient): Promise<MonthlyCost>;
|
|
32
|
+
}
|
|
33
|
+
export interface PricingClient {
|
|
34
|
+
getPrice(params: PriceQueryParams): Promise<number | null>;
|
|
35
|
+
}
|
|
36
|
+
export interface PriceQueryParams {
|
|
37
|
+
serviceCode: string;
|
|
38
|
+
region: string;
|
|
39
|
+
filters: PriceFilter[];
|
|
40
|
+
}
|
|
41
|
+
export interface PriceFilter {
|
|
42
|
+
field: string;
|
|
43
|
+
value: string;
|
|
44
|
+
type?: 'TERM_MATCH';
|
|
45
|
+
}
|
|
46
|
+
export declare class PricingAPIError extends Error {
|
|
47
|
+
retryable: boolean;
|
|
48
|
+
constructor(message: string, retryable?: boolean);
|
|
49
|
+
}
|
|
50
|
+
export declare class UnsupportedResourceError extends Error {
|
|
51
|
+
resourceType: string;
|
|
52
|
+
constructor(resourceType: string);
|
|
53
|
+
}
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.UnsupportedResourceError = exports.PricingAPIError = void 0;
|
|
4
|
+
class PricingAPIError extends Error {
|
|
5
|
+
retryable;
|
|
6
|
+
constructor(message, retryable = true) {
|
|
7
|
+
super(message);
|
|
8
|
+
this.retryable = retryable;
|
|
9
|
+
this.name = 'PricingAPIError';
|
|
10
|
+
}
|
|
11
|
+
}
|
|
12
|
+
exports.PricingAPIError = PricingAPIError;
|
|
13
|
+
class UnsupportedResourceError extends Error {
|
|
14
|
+
resourceType;
|
|
15
|
+
constructor(resourceType) {
|
|
16
|
+
super(`Resource type ${resourceType} is not supported`);
|
|
17
|
+
this.resourceType = resourceType;
|
|
18
|
+
this.name = 'UnsupportedResourceError';
|
|
19
|
+
}
|
|
20
|
+
}
|
|
21
|
+
exports.UnsupportedResourceError = UnsupportedResourceError;
|
|
22
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvcHJpY2luZy90eXBlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUEyREEsTUFBYSxlQUFnQixTQUFRLEtBQUs7SUFDSjtJQUFwQyxZQUFZLE9BQWUsRUFBUyxZQUFxQixJQUFJO1FBQzNELEtBQUssQ0FBQyxPQUFPLENBQUMsQ0FBQztRQURtQixjQUFTLEdBQVQsU0FBUyxDQUFnQjtRQUUzRCxJQUFJLENBQUMsSUFBSSxHQUFHLGlCQUFpQixDQUFDO0lBQ2hDLENBQUM7Q0FDRjtBQUxELDBDQUtDO0FBRUQsTUFBYSx3QkFBeUIsU0FBUSxLQUFLO0lBQzlCO0lBQW5CLFlBQW1CLFlBQW9CO1FBQ3JDLEtBQUssQ0FBQyxpQkFBaUIsWUFBWSxtQkFBbUIsQ0FBQyxDQUFDO1FBRHZDLGlCQUFZLEdBQVosWUFBWSxDQUFRO1FBRXJDLElBQUksQ0FBQyxJQUFJLEdBQUcsMEJBQTBCLENBQUM7SUFDekMsQ0FBQztDQUNGO0FBTEQsNERBS0MiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBSZXNvdXJjZVdpdGhJZCwgUmVzb3VyY2VEaWZmIH0gZnJvbSAnLi4vZGlmZi90eXBlcyc7XG5cbmV4cG9ydCBpbnRlcmZhY2UgUHJpY2luZ1NlcnZpY2Uge1xuICBnZXRSZXNvdXJjZUNvc3QocmVzb3VyY2U6IFJlc291cmNlV2l0aElkLCByZWdpb246IHN0cmluZyk6IFByb21pc2U8TW9udGhseUNvc3Q+O1xuICBnZXRDb3N0RGVsdGEoZGlmZjogUmVzb3VyY2VEaWZmLCByZWdpb246IHN0cmluZyk6IFByb21pc2U8Q29zdERlbHRhPjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBNb250aGx5Q29zdCB7XG4gIGFtb3VudDogbnVtYmVyO1xuICBjdXJyZW5jeTogc3RyaW5nO1xuICBjb25maWRlbmNlOiAnaGlnaCcgfCAnbWVkaXVtJyB8ICdsb3cnIHwgJ3Vua25vd24nO1xuICBhc3N1bXB0aW9uczogc3RyaW5nW107XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQ29zdERlbHRhIHtcbiAgdG90YWxEZWx0YTogbnVtYmVyO1xuICBjdXJyZW5jeTogc3RyaW5nO1xuICBhZGRlZENvc3RzOiBSZXNvdXJjZUNvc3RbXTtcbiAgcmVtb3ZlZENvc3RzOiBSZXNvdXJjZUNvc3RbXTtcbiAgbW9kaWZpZWRDb3N0czogTW9kaWZpZWRSZXNvdXJjZUNvc3RbXTtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBSZXNvdXJjZUNvc3Qge1xuICBsb2dpY2FsSWQ6IHN0cmluZztcbiAgdHlwZTogc3RyaW5nO1xuICBtb250aGx5Q29zdDogTW9udGhseUNvc3Q7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgTW9kaWZpZWRSZXNvdXJjZUNvc3QgZXh0ZW5kcyBSZXNvdXJjZUNvc3Qge1xuICBvbGRNb250aGx5Q29zdDogTW9udGhseUNvc3Q7XG4gIG5ld01vbnRobHlDb3N0OiBNb250aGx5Q29zdDtcbiAgY29zdERlbHRhOiBudW1iZXI7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgUmVzb3VyY2VDb3N0Q2FsY3VsYXRvciB7XG4gIHN1cHBvcnRzKHJlc291cmNlVHlwZTogc3RyaW5nKTogYm9vbGVhbjtcbiAgY2FsY3VsYXRlQ29zdChcbiAgICByZXNvdXJjZTogUmVzb3VyY2VXaXRoSWQsXG4gICAgcmVnaW9uOiBzdHJpbmcsXG4gICAgcHJpY2luZ0NsaWVudDogUHJpY2luZ0NsaWVudFxuICApOiBQcm9taXNlPE1vbnRobHlDb3N0Pjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBQcmljaW5nQ2xpZW50IHtcbiAgZ2V0UHJpY2UocGFyYW1zOiBQcmljZVF1ZXJ5UGFyYW1zKTogUHJvbWlzZTxudW1iZXIgfCBudWxsPjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBQcmljZVF1ZXJ5UGFyYW1zIHtcbiAgc2VydmljZUNvZGU6IHN0cmluZztcbiAgcmVnaW9uOiBzdHJpbmc7XG4gIGZpbHRlcnM6IFByaWNlRmlsdGVyW107XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgUHJpY2VGaWx0ZXIge1xuICBmaWVsZDogc3RyaW5nO1xuICB2YWx1ZTogc3RyaW5nO1xuICB0eXBlPzogJ1RFUk1fTUFUQ0gnO1xufVxuXG5leHBvcnQgY2xhc3MgUHJpY2luZ0FQSUVycm9yIGV4dGVuZHMgRXJyb3Ige1xuICBjb25zdHJ1Y3RvcihtZXNzYWdlOiBzdHJpbmcsIHB1YmxpYyByZXRyeWFibGU6IGJvb2xlYW4gPSB0cnVlKSB7XG4gICAgc3VwZXIobWVzc2FnZSk7XG4gICAgdGhpcy5uYW1lID0gJ1ByaWNpbmdBUElFcnJvcic7XG4gIH1cbn1cblxuZXhwb3J0IGNsYXNzIFVuc3VwcG9ydGVkUmVzb3VyY2VFcnJvciBleHRlbmRzIEVycm9yIHtcbiAgY29uc3RydWN0b3IocHVibGljIHJlc291cmNlVHlwZTogc3RyaW5nKSB7XG4gICAgc3VwZXIoYFJlc291cmNlIHR5cGUgJHtyZXNvdXJjZVR5cGV9IGlzIG5vdCBzdXBwb3J0ZWRgKTtcbiAgICB0aGlzLm5hbWUgPSAnVW5zdXBwb3J0ZWRSZXNvdXJjZUVycm9yJztcbiAgfVxufVxuIl19
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
v0.1.1
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
import { Reporter as IReporter, ReportFormat, ReportOptions } from './types';
|
|
2
|
+
import { CostDelta } from '../pricing/types';
|
|
3
|
+
export declare class Reporter implements IReporter {
|
|
4
|
+
generateReport(costDelta: CostDelta, format: ReportFormat, options?: ReportOptions): string;
|
|
5
|
+
private generateTextReport;
|
|
6
|
+
private generateJsonReport;
|
|
7
|
+
private generateMarkdownReport;
|
|
8
|
+
private formatResourceLine;
|
|
9
|
+
private formatModifiedResourceLine;
|
|
10
|
+
private formatCurrency;
|
|
11
|
+
private formatDelta;
|
|
12
|
+
private formatConfigSummaryText;
|
|
13
|
+
private formatThresholdStatusText;
|
|
14
|
+
private formatConfigSummaryMarkdown;
|
|
15
|
+
private formatThresholdStatusMarkdown;
|
|
16
|
+
private getTopCostContributors;
|
|
17
|
+
private formatStackDetailsMarkdown;
|
|
18
|
+
}
|