cdk-cost-analyzer 0.1.39 → 0.1.41
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.cdk-cost-analyzer-cache/metadata.json +8 -8
- package/.husky/pre-commit +6 -0
- package/dist/action/136.index.js +1 -1
- package/dist/action/762.index.js +1 -1
- package/dist/action/998.index.js +1 -1
- package/dist/action/config/types.d.ts +50 -0
- package/dist/action/index.js +3 -3
- package/dist/action/pricing/calculators/AuroraServerlessCalculator.d.ts +20 -0
- package/dist/action/pricing/calculators/EKSCalculator.d.ts +8 -0
- package/dist/action/pricing/calculators/KinesisCalculator.d.ts +27 -0
- package/dist/action/pricing/calculators/Route53Calculator.d.ts +18 -0
- package/dist/action/pricing/calculators/TransitGatewayCalculator.d.ts +16 -0
- package/dist/action/reporter/index.d.ts +1 -0
- package/dist/action/reporter/markdownUtils.d.ts +30 -0
- package/dist/config/types.d.ts +50 -0
- package/dist/config/types.js +1 -1
- package/dist/pricing/PricingService.js +11 -1
- package/dist/pricing/calculators/AuroraServerlessCalculator.d.ts +20 -0
- package/dist/pricing/calculators/AuroraServerlessCalculator.js +92 -0
- package/dist/pricing/calculators/EKSCalculator.d.ts +8 -0
- package/dist/pricing/calculators/EKSCalculator.js +46 -0
- package/dist/pricing/calculators/KinesisCalculator.d.ts +27 -0
- package/dist/pricing/calculators/KinesisCalculator.js +140 -0
- package/dist/pricing/calculators/Route53Calculator.d.ts +18 -0
- package/dist/pricing/calculators/Route53Calculator.js +99 -0
- package/dist/pricing/calculators/TransitGatewayCalculator.d.ts +16 -0
- package/dist/pricing/calculators/TransitGatewayCalculator.js +97 -0
- package/dist/releasetag.txt +1 -1
- package/dist/reporter/Reporter.js +94 -56
- package/dist/reporter/SingleTemplateReporter.js +49 -21
- package/dist/reporter/index.d.ts +1 -0
- package/dist/reporter/index.js +8 -2
- package/dist/reporter/markdownUtils.d.ts +30 -0
- package/dist/reporter/markdownUtils.js +93 -0
- package/package.json +2 -2
|
@@ -1,6 +1,7 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
3
|
exports.SingleTemplateReporter = void 0;
|
|
4
|
+
const markdownUtils_1 = require("./markdownUtils");
|
|
4
5
|
/**
|
|
5
6
|
* Reporter for generating formatted output for single template cost analysis
|
|
6
7
|
*/
|
|
@@ -108,59 +109,86 @@ class SingleTemplateReporter {
|
|
|
108
109
|
generateMarkdownReport(result, options) {
|
|
109
110
|
let report = '';
|
|
110
111
|
// Header
|
|
111
|
-
report += '# Single Template Cost Analysis\n\n';
|
|
112
|
+
report += '# 💰 Single Template Cost Analysis\n\n';
|
|
112
113
|
// Summary
|
|
113
114
|
report += '## Summary\n\n';
|
|
114
115
|
report += `**Total Monthly Cost:** $${result.totalMonthlyCost.toFixed(2)} ${result.currency}\n\n`;
|
|
115
|
-
report +=
|
|
116
|
-
report +=
|
|
117
|
-
report +=
|
|
116
|
+
report += '| Metric | Value |\n';
|
|
117
|
+
report += '|--------|-------|\n';
|
|
118
|
+
report += `| Analysis Date | ${result.metadata.analyzedAt.toISOString()} |\n`;
|
|
119
|
+
report += `| Region | ${result.metadata.region} |\n`;
|
|
120
|
+
report += `| Template Hash | \`${result.metadata.templateHash}\` |\n\n`;
|
|
118
121
|
// Resource counts
|
|
119
|
-
report += '## Resource Overview\n\n';
|
|
120
|
-
report +=
|
|
121
|
-
report +=
|
|
122
|
-
report +=
|
|
122
|
+
report += '## 📊 Resource Overview\n\n';
|
|
123
|
+
report += '| Category | Count |\n';
|
|
124
|
+
report += '|----------|-------|\n';
|
|
125
|
+
report += `| Total Resources | ${result.metadata.resourceCount} |\n`;
|
|
126
|
+
report += `| Supported Resources | ${result.metadata.supportedResourceCount} |\n`;
|
|
127
|
+
report += `| Unsupported Resources | ${result.metadata.unsupportedResourceCount} |\n\n`;
|
|
123
128
|
// Cost breakdown by confidence
|
|
124
129
|
if (result.costBreakdown.byConfidenceLevel.length > 0) {
|
|
125
|
-
report += '## Cost Confidence Breakdown\n\n';
|
|
130
|
+
report += '## 🎯 Cost Confidence Breakdown\n\n';
|
|
126
131
|
report += '| Confidence | Resources | Cost | Percentage |\n';
|
|
127
132
|
report += '|------------|-----------|------|------------|\n';
|
|
128
133
|
for (const conf of result.costBreakdown.byConfidenceLevel) {
|
|
129
134
|
const percentage = ((conf.totalCost / result.totalMonthlyCost) * 100).toFixed(1);
|
|
130
|
-
|
|
135
|
+
const emoji = conf.confidence === 'high' ? '✅' : conf.confidence === 'medium' ? '⚠️' : conf.confidence === 'low' ? '❓' : '❌';
|
|
136
|
+
report += `| ${emoji} ${conf.confidence} | ${conf.count} | $${conf.totalCost.toFixed(2)} | ${percentage}% |\n`;
|
|
131
137
|
}
|
|
132
138
|
report += '\n';
|
|
133
139
|
}
|
|
134
|
-
//
|
|
140
|
+
// Cost breakdown by service
|
|
135
141
|
if (options.showBreakdown && options.groupByType) {
|
|
136
|
-
report += '## Cost Breakdown by
|
|
142
|
+
report += '## 📈 Cost Breakdown by Service\n\n';
|
|
143
|
+
const serviceMap = new Map();
|
|
144
|
+
for (const typeGroup of result.costBreakdown.byResourceType) {
|
|
145
|
+
const service = (0, markdownUtils_1.extractServiceName)(typeGroup.resourceType);
|
|
146
|
+
const existing = serviceMap.get(service) || { totalCost: 0, resources: [] };
|
|
147
|
+
existing.totalCost += typeGroup.totalCost;
|
|
148
|
+
existing.resources.push(...typeGroup.resources);
|
|
149
|
+
serviceMap.set(service, existing);
|
|
150
|
+
}
|
|
151
|
+
const services = Array.from(serviceMap.entries())
|
|
152
|
+
.map(([service, data]) => ({ service, ...data }))
|
|
153
|
+
.sort((a, b) => b.totalCost - a.totalCost);
|
|
154
|
+
report += '| Service | Cost | Percentage | Trend |\n';
|
|
155
|
+
report += '|---------|------|------------|-------|\n';
|
|
156
|
+
for (const svc of services) {
|
|
157
|
+
const percentage = ((svc.totalCost / result.totalMonthlyCost) * 100).toFixed(1);
|
|
158
|
+
const trend = (0, markdownUtils_1.getTrendIndicator)(svc.totalCost);
|
|
159
|
+
report += `| ${svc.service} | $${svc.totalCost.toFixed(2)} | ${percentage}% | ${trend} |\n`;
|
|
160
|
+
}
|
|
161
|
+
report += '\n';
|
|
162
|
+
// Detailed breakdown by resource type
|
|
163
|
+
report += '### Detailed Resource Breakdown\n\n';
|
|
137
164
|
for (const typeGroup of result.costBreakdown.byResourceType) {
|
|
138
165
|
const percentage = ((typeGroup.totalCost / result.totalMonthlyCost) * 100).toFixed(1);
|
|
139
|
-
report +=
|
|
166
|
+
report += `#### \`${typeGroup.resourceType}\`\n\n`;
|
|
140
167
|
report += `- **Count:** ${typeGroup.count}\n`;
|
|
141
168
|
report += `- **Total Cost:** $${typeGroup.totalCost.toFixed(2)} (${percentage}%)\n\n`;
|
|
142
|
-
// Show individual resources in table
|
|
143
169
|
report += '| Resource | Cost | Confidence |\n';
|
|
144
170
|
report += '|----------|------|------------|\n';
|
|
145
171
|
const sortedResources = this.sortResources(typeGroup.resources, options.sortBy);
|
|
146
172
|
for (const resource of sortedResources) {
|
|
147
173
|
const conf = resource.monthlyCost.confidence;
|
|
148
|
-
|
|
174
|
+
const emoji = conf === 'high' ? '✅' : conf === 'medium' ? '⚠️' : conf === 'low' ? '❓' : '❌';
|
|
175
|
+
report += `| ${resource.logicalId} | $${resource.monthlyCost.amount.toFixed(2)} | ${emoji} ${conf} |\n`;
|
|
149
176
|
}
|
|
150
177
|
report += '\n';
|
|
151
178
|
}
|
|
152
179
|
}
|
|
153
|
-
// Assumptions
|
|
180
|
+
// Assumptions (collapsible)
|
|
154
181
|
if (options.showAssumptions && result.costBreakdown.assumptions.length > 0) {
|
|
155
|
-
report += '
|
|
182
|
+
report += '<details>\n';
|
|
183
|
+
report += '<summary><strong>📋 Cost Calculation Assumptions</strong></summary>\n\n';
|
|
156
184
|
for (const assumption of result.costBreakdown.assumptions) {
|
|
157
185
|
report += `- ${assumption}\n`;
|
|
158
186
|
}
|
|
159
|
-
report += '\n';
|
|
187
|
+
report += '\n</details>\n\n';
|
|
160
188
|
}
|
|
161
|
-
//
|
|
189
|
+
// Footer
|
|
162
190
|
report += '---\n\n';
|
|
163
|
-
report += '*Confidence levels: high
|
|
191
|
+
report += '*Powered by [cdk-cost-analyzer](https://github.com/buildinginthecloud/cdk-cost-analyzer) | Confidence levels: ✅ high, ⚠️ medium, ❓ low, ❌ unknown*\n';
|
|
164
192
|
return report;
|
|
165
193
|
}
|
|
166
194
|
/**
|
|
@@ -183,4 +211,4 @@ class SingleTemplateReporter {
|
|
|
183
211
|
}
|
|
184
212
|
}
|
|
185
213
|
exports.SingleTemplateReporter = SingleTemplateReporter;
|
|
186
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"SingleTemplateReporter.js","sourceRoot":"","sources":["../../src/reporter/SingleTemplateReporter.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  SingleTemplateCostResult,\n  SingleTemplateReportOptions,\n} from '../api/single-template-types';\n\n/**\n * Reporter for generating formatted output for single template cost analysis\n */\nexport class SingleTemplateReporter {\n  /**\n   * Generate a formatted report from single template analysis result\n   * \n   * @param result - The analysis result\n   * @param format - Output format (text, json, or markdown)\n   * @param options - Optional formatting preferences\n   * @returns Formatted report string\n   */\n  generateReport(\n    result: SingleTemplateCostResult,\n    format: 'text' | 'json' | 'markdown',\n    options?: SingleTemplateReportOptions,\n  ): string {\n    const opts = {\n      showBreakdown: true,\n      showAssumptions: true,\n      groupByType: true,\n      sortBy: 'cost' as 'cost' | 'name' | 'type',\n      ...options,\n    };\n\n    switch (format) {\n      case 'json':\n        return this.generateJsonReport(result);\n      case 'markdown':\n        return this.generateMarkdownReport(result, opts);\n      case 'text':\n      default:\n        return this.generateTextReport(result, opts);\n    }\n  }\n\n  /**\n   * Generate JSON format report\n   */\n  private generateJsonReport(result: SingleTemplateCostResult): string {\n    return JSON.stringify(result, null, 2);\n  }\n\n  /**\n   * Generate text format report\n   */\n  private generateTextReport(\n    result: SingleTemplateCostResult,\n    options: Required<SingleTemplateReportOptions>,\n  ): string {\n    let report = '';\n\n    // Header\n    report += '='.repeat(80) + '\\n';\n    report += 'Single Template Cost Analysis\\n';\n    report += '='.repeat(80) + '\\n\\n';\n\n    // Summary\n    report += `Total Monthly Cost: $${result.totalMonthlyCost.toFixed(2)} ${result.currency}\\n`;\n    report += `Analysis Date: ${result.metadata.analyzedAt.toISOString()}\\n`;\n    report += `Region: ${result.metadata.region}\\n`;\n    report += `Template Hash: ${result.metadata.templateHash}\\n\\n`;\n\n    // Resource counts\n    report += `Total Resources: ${result.metadata.resourceCount}\\n`;\n    report += `Supported Resources: ${result.metadata.supportedResourceCount}\\n`;\n    report += `Unsupported Resources: ${result.metadata.unsupportedResourceCount}\\n\\n`;\n\n    // Cost breakdown by confidence\n    if (result.costBreakdown.byConfidenceLevel.length > 0) {\n      report += '-'.repeat(80) + '\\n';\n      report += 'Cost Confidence Breakdown\\n';\n      report += '-'.repeat(80) + '\\n\\n';\n\n      for (const conf of result.costBreakdown.byConfidenceLevel) {\n        const percentage = ((conf.totalCost / result.totalMonthlyCost) * 100).toFixed(1);\n        report += `${conf.confidence.toUpperCase().padEnd(10)} ${conf.count} resources  $${conf.totalCost.toFixed(2)} (${percentage}%)\\n`;\n      }\n      report += '\\n';\n    }\n\n    // Resource breakdown\n    if (options.showBreakdown && options.groupByType) {\n      report += '-'.repeat(80) + '\\n';\n      report += 'Cost Breakdown by Resource Type\\n';\n      report += '-'.repeat(80) + '\\n\\n';\n\n      for (const typeGroup of result.costBreakdown.byResourceType) {\n        const percentage = ((typeGroup.totalCost / result.totalMonthlyCost) * 100).toFixed(1);\n        report += `${typeGroup.resourceType}\\n`;\n        report += `  Count: ${typeGroup.count}\\n`;\n        report += `  Total Cost: $${typeGroup.totalCost.toFixed(2)} (${percentage}%)\\n`;\n\n        // Show individual resources\n        const sortedResources = this.sortResources(typeGroup.resources, options.sortBy);\n        for (const resource of sortedResources) {\n          const conf = resource.monthlyCost.confidence;\n          const confIndicator = conf === 'high' ? '✓' : conf === 'medium' ? '~' : conf === 'low' ? '?' : '✗';\n          report += `    ${confIndicator} ${resource.logicalId.padEnd(40)} $${resource.monthlyCost.amount.toFixed(2)}\\n`;\n        }\n        report += '\\n';\n      }\n    }\n\n    // Assumptions\n    if (options.showAssumptions && result.costBreakdown.assumptions.length > 0) {\n      report += '-'.repeat(80) + '\\n';\n      report += 'Cost Calculation Assumptions\\n';\n      report += '-'.repeat(80) + '\\n\\n';\n\n      for (const assumption of result.costBreakdown.assumptions) {\n        report += `  • ${assumption}\\n`;\n      }\n      report += '\\n';\n    }\n\n    // Legend\n    report += '-'.repeat(80) + '\\n';\n    report += 'Legend: ✓ High confidence  ~ Medium confidence  ? Low confidence  ✗ Unknown/Unsupported\\n';\n    report += '='.repeat(80) + '\\n';\n\n    return report;\n  }\n\n  /**\n   * Generate markdown format report\n   */\n  private generateMarkdownReport(\n    result: SingleTemplateCostResult,\n    options: Required<SingleTemplateReportOptions>,\n  ): string {\n    let report = '';\n\n    // Header\n    report += '# Single Template Cost Analysis\\n\\n';\n\n    // Summary\n    report += '## Summary\\n\\n';\n    report += `**Total Monthly Cost:** $${result.totalMonthlyCost.toFixed(2)} ${result.currency}\\n\\n`;\n    report += `- **Analysis Date:** ${result.metadata.analyzedAt.toISOString()}\\n`;\n    report += `- **Region:** ${result.metadata.region}\\n`;\n    report += `- **Template Hash:** ${result.metadata.templateHash}\\n\\n`;\n\n    // Resource counts\n    report += '## Resource Overview\\n\\n';\n    report += `- **Total Resources:** ${result.metadata.resourceCount}\\n`;\n    report += `- **Supported Resources:** ${result.metadata.supportedResourceCount}\\n`;\n    report += `- **Unsupported Resources:** ${result.metadata.unsupportedResourceCount}\\n\\n`;\n\n    // Cost breakdown by confidence\n    if (result.costBreakdown.byConfidenceLevel.length > 0) {\n      report += '## Cost Confidence Breakdown\\n\\n';\n      report += '| Confidence | Resources | Cost | Percentage |\\n';\n      report += '|------------|-----------|------|------------|\\n';\n\n      for (const conf of result.costBreakdown.byConfidenceLevel) {\n        const percentage = ((conf.totalCost / result.totalMonthlyCost) * 100).toFixed(1);\n        report += `| ${conf.confidence} | ${conf.count} | $${conf.totalCost.toFixed(2)} | ${percentage}% |\\n`;\n      }\n      report += '\\n';\n    }\n\n    // Resource breakdown by type\n    if (options.showBreakdown && options.groupByType) {\n      report += '## Cost Breakdown by Resource Type\\n\\n';\n\n      for (const typeGroup of result.costBreakdown.byResourceType) {\n        const percentage = ((typeGroup.totalCost / result.totalMonthlyCost) * 100).toFixed(1);\n        report += `### ${typeGroup.resourceType}\\n\\n`;\n        report += `- **Count:** ${typeGroup.count}\\n`;\n        report += `- **Total Cost:** $${typeGroup.totalCost.toFixed(2)} (${percentage}%)\\n\\n`;\n\n        // Show individual resources in table\n        report += '| Resource | Cost | Confidence |\\n';\n        report += '|----------|------|------------|\\n';\n\n        const sortedResources = this.sortResources(typeGroup.resources, options.sortBy);\n        for (const resource of sortedResources) {\n          const conf = resource.monthlyCost.confidence;\n          report += `| ${resource.logicalId} | $${resource.monthlyCost.amount.toFixed(2)} | ${conf} |\\n`;\n        }\n        report += '\\n';\n      }\n    }\n\n    // Assumptions\n    if (options.showAssumptions && result.costBreakdown.assumptions.length > 0) {\n      report += '## Cost Calculation Assumptions\\n\\n';\n      for (const assumption of result.costBreakdown.assumptions) {\n        report += `- ${assumption}\\n`;\n      }\n      report += '\\n';\n    }\n\n    // Legend\n    report += '---\\n\\n';\n    report += '*Confidence levels: high (✓), medium (~), low (?), unknown/unsupported (✗)*\\n';\n\n    return report;\n  }\n\n  /**\n   * Sort resources based on specified criteria\n   */\n  private sortResources(\n    resources: Array<{ logicalId: string; type: string; monthlyCost: any }>,\n    sortBy: 'cost' | 'name' | 'type',\n  ): Array<{ logicalId: string; type: string; monthlyCost: any }> {\n    const sorted = [...resources];\n\n    switch (sortBy) {\n      case 'cost':\n        sorted.sort((a, b) => b.monthlyCost.amount - a.monthlyCost.amount);\n        break;\n      case 'name':\n        sorted.sort((a, b) => a.logicalId.localeCompare(b.logicalId));\n        break;\n      case 'type':\n        sorted.sort((a, b) => a.type.localeCompare(b.type));\n        break;\n    }\n\n    return sorted;\n  }\n}\n"]}
|
|
214
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"SingleTemplateReporter.js","sourceRoot":"","sources":["../../src/reporter/SingleTemplateReporter.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  SingleTemplateCostResult,\n  SingleTemplateReportOptions,\n} from '../api/single-template-types';\nimport { getTrendIndicator, extractServiceName } from './markdownUtils';\n\n/**\n * Reporter for generating formatted output for single template cost analysis\n */\nexport class SingleTemplateReporter {\n  /**\n   * Generate a formatted report from single template analysis result\n   * \n   * @param result - The analysis result\n   * @param format - Output format (text, json, or markdown)\n   * @param options - Optional formatting preferences\n   * @returns Formatted report string\n   */\n  generateReport(\n    result: SingleTemplateCostResult,\n    format: 'text' | 'json' | 'markdown',\n    options?: SingleTemplateReportOptions,\n  ): string {\n    const opts = {\n      showBreakdown: true,\n      showAssumptions: true,\n      groupByType: true,\n      sortBy: 'cost' as 'cost' | 'name' | 'type',\n      ...options,\n    };\n\n    switch (format) {\n      case 'json':\n        return this.generateJsonReport(result);\n      case 'markdown':\n        return this.generateMarkdownReport(result, opts);\n      case 'text':\n      default:\n        return this.generateTextReport(result, opts);\n    }\n  }\n\n  /**\n   * Generate JSON format report\n   */\n  private generateJsonReport(result: SingleTemplateCostResult): string {\n    return JSON.stringify(result, null, 2);\n  }\n\n  /**\n   * Generate text format report\n   */\n  private generateTextReport(\n    result: SingleTemplateCostResult,\n    options: Required<SingleTemplateReportOptions>,\n  ): string {\n    let report = '';\n\n    // Header\n    report += '='.repeat(80) + '\\n';\n    report += 'Single Template Cost Analysis\\n';\n    report += '='.repeat(80) + '\\n\\n';\n\n    // Summary\n    report += `Total Monthly Cost: $${result.totalMonthlyCost.toFixed(2)} ${result.currency}\\n`;\n    report += `Analysis Date: ${result.metadata.analyzedAt.toISOString()}\\n`;\n    report += `Region: ${result.metadata.region}\\n`;\n    report += `Template Hash: ${result.metadata.templateHash}\\n\\n`;\n\n    // Resource counts\n    report += `Total Resources: ${result.metadata.resourceCount}\\n`;\n    report += `Supported Resources: ${result.metadata.supportedResourceCount}\\n`;\n    report += `Unsupported Resources: ${result.metadata.unsupportedResourceCount}\\n\\n`;\n\n    // Cost breakdown by confidence\n    if (result.costBreakdown.byConfidenceLevel.length > 0) {\n      report += '-'.repeat(80) + '\\n';\n      report += 'Cost Confidence Breakdown\\n';\n      report += '-'.repeat(80) + '\\n\\n';\n\n      for (const conf of result.costBreakdown.byConfidenceLevel) {\n        const percentage = ((conf.totalCost / result.totalMonthlyCost) * 100).toFixed(1);\n        report += `${conf.confidence.toUpperCase().padEnd(10)} ${conf.count} resources  $${conf.totalCost.toFixed(2)} (${percentage}%)\\n`;\n      }\n      report += '\\n';\n    }\n\n    // Resource breakdown\n    if (options.showBreakdown && options.groupByType) {\n      report += '-'.repeat(80) + '\\n';\n      report += 'Cost Breakdown by Resource Type\\n';\n      report += '-'.repeat(80) + '\\n\\n';\n\n      for (const typeGroup of result.costBreakdown.byResourceType) {\n        const percentage = ((typeGroup.totalCost / result.totalMonthlyCost) * 100).toFixed(1);\n        report += `${typeGroup.resourceType}\\n`;\n        report += `  Count: ${typeGroup.count}\\n`;\n        report += `  Total Cost: $${typeGroup.totalCost.toFixed(2)} (${percentage}%)\\n`;\n\n        // Show individual resources\n        const sortedResources = this.sortResources(typeGroup.resources, options.sortBy);\n        for (const resource of sortedResources) {\n          const conf = resource.monthlyCost.confidence;\n          const confIndicator = conf === 'high' ? '✓' : conf === 'medium' ? '~' : conf === 'low' ? '?' : '✗';\n          report += `    ${confIndicator} ${resource.logicalId.padEnd(40)} $${resource.monthlyCost.amount.toFixed(2)}\\n`;\n        }\n        report += '\\n';\n      }\n    }\n\n    // Assumptions\n    if (options.showAssumptions && result.costBreakdown.assumptions.length > 0) {\n      report += '-'.repeat(80) + '\\n';\n      report += 'Cost Calculation Assumptions\\n';\n      report += '-'.repeat(80) + '\\n\\n';\n\n      for (const assumption of result.costBreakdown.assumptions) {\n        report += `  • ${assumption}\\n`;\n      }\n      report += '\\n';\n    }\n\n    // Legend\n    report += '-'.repeat(80) + '\\n';\n    report += 'Legend: ✓ High confidence  ~ Medium confidence  ? Low confidence  ✗ Unknown/Unsupported\\n';\n    report += '='.repeat(80) + '\\n';\n\n    return report;\n  }\n\n  /**\n   * Generate markdown format report\n   */\n  private generateMarkdownReport(\n    result: SingleTemplateCostResult,\n    options: Required<SingleTemplateReportOptions>,\n  ): string {\n    let report = '';\n\n    // Header\n    report += '# 💰 Single Template Cost Analysis\\n\\n';\n\n    // Summary\n    report += '## Summary\\n\\n';\n    report += `**Total Monthly Cost:** $${result.totalMonthlyCost.toFixed(2)} ${result.currency}\\n\\n`;\n    report += '| Metric | Value |\\n';\n    report += '|--------|-------|\\n';\n    report += `| Analysis Date | ${result.metadata.analyzedAt.toISOString()} |\\n`;\n    report += `| Region | ${result.metadata.region} |\\n`;\n    report += `| Template Hash | \\`${result.metadata.templateHash}\\` |\\n\\n`;\n\n    // Resource counts\n    report += '## 📊 Resource Overview\\n\\n';\n    report += '| Category | Count |\\n';\n    report += '|----------|-------|\\n';\n    report += `| Total Resources | ${result.metadata.resourceCount} |\\n`;\n    report += `| Supported Resources | ${result.metadata.supportedResourceCount} |\\n`;\n    report += `| Unsupported Resources | ${result.metadata.unsupportedResourceCount} |\\n\\n`;\n\n    // Cost breakdown by confidence\n    if (result.costBreakdown.byConfidenceLevel.length > 0) {\n      report += '## 🎯 Cost Confidence Breakdown\\n\\n';\n      report += '| Confidence | Resources | Cost | Percentage |\\n';\n      report += '|------------|-----------|------|------------|\\n';\n\n      for (const conf of result.costBreakdown.byConfidenceLevel) {\n        const percentage = ((conf.totalCost / result.totalMonthlyCost) * 100).toFixed(1);\n        const emoji = conf.confidence === 'high' ? '✅' : conf.confidence === 'medium' ? '⚠️' : conf.confidence === 'low' ? '❓' : '❌';\n        report += `| ${emoji} ${conf.confidence} | ${conf.count} | $${conf.totalCost.toFixed(2)} | ${percentage}% |\\n`;\n      }\n      report += '\\n';\n    }\n\n    // Cost breakdown by service\n    if (options.showBreakdown && options.groupByType) {\n      report += '## 📈 Cost Breakdown by Service\\n\\n';\n\n      const serviceMap = new Map<string, { totalCost: number; resources: Array<{ logicalId: string; type: string; monthlyCost: any }> }>();\n\n      for (const typeGroup of result.costBreakdown.byResourceType) {\n        const service = extractServiceName(typeGroup.resourceType);\n        const existing = serviceMap.get(service) || { totalCost: 0, resources: [] };\n        existing.totalCost += typeGroup.totalCost;\n        existing.resources.push(...typeGroup.resources);\n        serviceMap.set(service, existing);\n      }\n\n      const services = Array.from(serviceMap.entries())\n        .map(([service, data]) => ({ service, ...data }))\n        .sort((a, b) => b.totalCost - a.totalCost);\n\n      report += '| Service | Cost | Percentage | Trend |\\n';\n      report += '|---------|------|------------|-------|\\n';\n\n      for (const svc of services) {\n        const percentage = ((svc.totalCost / result.totalMonthlyCost) * 100).toFixed(1);\n        const trend = getTrendIndicator(svc.totalCost);\n        report += `| ${svc.service} | $${svc.totalCost.toFixed(2)} | ${percentage}% | ${trend} |\\n`;\n      }\n      report += '\\n';\n\n      // Detailed breakdown by resource type\n      report += '### Detailed Resource Breakdown\\n\\n';\n\n      for (const typeGroup of result.costBreakdown.byResourceType) {\n        const percentage = ((typeGroup.totalCost / result.totalMonthlyCost) * 100).toFixed(1);\n        report += `#### \\`${typeGroup.resourceType}\\`\\n\\n`;\n        report += `- **Count:** ${typeGroup.count}\\n`;\n        report += `- **Total Cost:** $${typeGroup.totalCost.toFixed(2)} (${percentage}%)\\n\\n`;\n\n        report += '| Resource | Cost | Confidence |\\n';\n        report += '|----------|------|------------|\\n';\n\n        const sortedResources = this.sortResources(typeGroup.resources, options.sortBy);\n        for (const resource of sortedResources) {\n          const conf = resource.monthlyCost.confidence;\n          const emoji = conf === 'high' ? '✅' : conf === 'medium' ? '⚠️' : conf === 'low' ? '❓' : '❌';\n          report += `| ${resource.logicalId} | $${resource.monthlyCost.amount.toFixed(2)} | ${emoji} ${conf} |\\n`;\n        }\n        report += '\\n';\n      }\n    }\n\n    // Assumptions (collapsible)\n    if (options.showAssumptions && result.costBreakdown.assumptions.length > 0) {\n      report += '<details>\\n';\n      report += '<summary><strong>📋 Cost Calculation Assumptions</strong></summary>\\n\\n';\n      for (const assumption of result.costBreakdown.assumptions) {\n        report += `- ${assumption}\\n`;\n      }\n      report += '\\n</details>\\n\\n';\n    }\n\n    // Footer\n    report += '---\\n\\n';\n    report += '*Powered by [cdk-cost-analyzer](https://github.com/buildinginthecloud/cdk-cost-analyzer) | Confidence levels: ✅ high, ⚠️ medium, ❓ low, ❌ unknown*\\n';\n\n    return report;\n  }\n\n  /**\n   * Sort resources based on specified criteria\n   */\n  private sortResources(\n    resources: Array<{ logicalId: string; type: string; monthlyCost: any }>,\n    sortBy: 'cost' | 'name' | 'type',\n  ): Array<{ logicalId: string; type: string; monthlyCost: any }> {\n    const sorted = [...resources];\n\n    switch (sortBy) {\n      case 'cost':\n        sorted.sort((a, b) => b.monthlyCost.amount - a.monthlyCost.amount);\n        break;\n      case 'name':\n        sorted.sort((a, b) => a.logicalId.localeCompare(b.logicalId));\n        break;\n      case 'type':\n        sorted.sort((a, b) => a.type.localeCompare(b.type));\n        break;\n    }\n\n    return sorted;\n  }\n}\n"]}
|
package/dist/reporter/index.d.ts
CHANGED
|
@@ -1,4 +1,5 @@
|
|
|
1
1
|
export { Reporter } from './Reporter';
|
|
2
2
|
export { SingleTemplateReporter } from './SingleTemplateReporter';
|
|
3
3
|
export { GitHubActionReporter } from './GitHubActionReporter';
|
|
4
|
+
export { ServiceBreakdown, getTrendIndicator, getPercentageChange, extractServiceName, groupCostsByService, calculateTotalCosts, } from './markdownUtils';
|
|
4
5
|
export * from './types';
|
package/dist/reporter/index.js
CHANGED
|
@@ -14,12 +14,18 @@ var __exportStar = (this && this.__exportStar) || function(m, exports) {
|
|
|
14
14
|
for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
|
|
15
15
|
};
|
|
16
16
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
17
|
-
exports.GitHubActionReporter = exports.SingleTemplateReporter = exports.Reporter = void 0;
|
|
17
|
+
exports.calculateTotalCosts = exports.groupCostsByService = exports.extractServiceName = exports.getPercentageChange = exports.getTrendIndicator = exports.GitHubActionReporter = exports.SingleTemplateReporter = exports.Reporter = void 0;
|
|
18
18
|
var Reporter_1 = require("./Reporter");
|
|
19
19
|
Object.defineProperty(exports, "Reporter", { enumerable: true, get: function () { return Reporter_1.Reporter; } });
|
|
20
20
|
var SingleTemplateReporter_1 = require("./SingleTemplateReporter");
|
|
21
21
|
Object.defineProperty(exports, "SingleTemplateReporter", { enumerable: true, get: function () { return SingleTemplateReporter_1.SingleTemplateReporter; } });
|
|
22
22
|
var GitHubActionReporter_1 = require("./GitHubActionReporter");
|
|
23
23
|
Object.defineProperty(exports, "GitHubActionReporter", { enumerable: true, get: function () { return GitHubActionReporter_1.GitHubActionReporter; } });
|
|
24
|
+
var markdownUtils_1 = require("./markdownUtils");
|
|
25
|
+
Object.defineProperty(exports, "getTrendIndicator", { enumerable: true, get: function () { return markdownUtils_1.getTrendIndicator; } });
|
|
26
|
+
Object.defineProperty(exports, "getPercentageChange", { enumerable: true, get: function () { return markdownUtils_1.getPercentageChange; } });
|
|
27
|
+
Object.defineProperty(exports, "extractServiceName", { enumerable: true, get: function () { return markdownUtils_1.extractServiceName; } });
|
|
28
|
+
Object.defineProperty(exports, "groupCostsByService", { enumerable: true, get: function () { return markdownUtils_1.groupCostsByService; } });
|
|
29
|
+
Object.defineProperty(exports, "calculateTotalCosts", { enumerable: true, get: function () { return markdownUtils_1.calculateTotalCosts; } });
|
|
24
30
|
__exportStar(require("./types"), exports);
|
|
25
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
31
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvcmVwb3J0ZXIvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFBQSx1Q0FBc0M7QUFBN0Isb0dBQUEsUUFBUSxPQUFBO0FBQ2pCLG1FQUFrRTtBQUF6RCxnSUFBQSxzQkFBc0IsT0FBQTtBQUMvQiwrREFBOEQ7QUFBckQsNEhBQUEsb0JBQW9CLE9BQUE7QUFDN0IsaURBT3lCO0FBTHZCLGtIQUFBLGlCQUFpQixPQUFBO0FBQ2pCLG9IQUFBLG1CQUFtQixPQUFBO0FBQ25CLG1IQUFBLGtCQUFrQixPQUFBO0FBQ2xCLG9IQUFBLG1CQUFtQixPQUFBO0FBQ25CLG9IQUFBLG1CQUFtQixPQUFBO0FBRXJCLDBDQUF3QiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB7IFJlcG9ydGVyIH0gZnJvbSAnLi9SZXBvcnRlcic7XG5leHBvcnQgeyBTaW5nbGVUZW1wbGF0ZVJlcG9ydGVyIH0gZnJvbSAnLi9TaW5nbGVUZW1wbGF0ZVJlcG9ydGVyJztcbmV4cG9ydCB7IEdpdEh1YkFjdGlvblJlcG9ydGVyIH0gZnJvbSAnLi9HaXRIdWJBY3Rpb25SZXBvcnRlcic7XG5leHBvcnQge1xuICBTZXJ2aWNlQnJlYWtkb3duLFxuICBnZXRUcmVuZEluZGljYXRvcixcbiAgZ2V0UGVyY2VudGFnZUNoYW5nZSxcbiAgZXh0cmFjdFNlcnZpY2VOYW1lLFxuICBncm91cENvc3RzQnlTZXJ2aWNlLFxuICBjYWxjdWxhdGVUb3RhbENvc3RzLFxufSBmcm9tICcuL21hcmtkb3duVXRpbHMnO1xuZXhwb3J0ICogZnJvbSAnLi90eXBlcyc7XG4iXX0=
|
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
import { CostDelta } from '../pricing/types';
|
|
2
|
+
/** Service breakdown entry for cost grouping */
|
|
3
|
+
export interface ServiceBreakdown {
|
|
4
|
+
service: string;
|
|
5
|
+
totalCost: number;
|
|
6
|
+
resourceCount: number;
|
|
7
|
+
}
|
|
8
|
+
/**
|
|
9
|
+
* Get trend indicator emoji based on cost change direction
|
|
10
|
+
*/
|
|
11
|
+
export declare function getTrendIndicator(amount: number): string;
|
|
12
|
+
/**
|
|
13
|
+
* Calculate percentage change between old and new amounts
|
|
14
|
+
*/
|
|
15
|
+
export declare function getPercentageChange(oldAmount: number, newAmount: number): string;
|
|
16
|
+
/**
|
|
17
|
+
* Extract AWS service name from resource type (e.g., AWS::EC2::Instance -> EC2)
|
|
18
|
+
*/
|
|
19
|
+
export declare function extractServiceName(resourceType: string): string;
|
|
20
|
+
/**
|
|
21
|
+
* Group costs by AWS service (e.g., EC2, S3, Lambda)
|
|
22
|
+
*/
|
|
23
|
+
export declare function groupCostsByService(costDelta: CostDelta): ServiceBreakdown[];
|
|
24
|
+
/**
|
|
25
|
+
* Calculate total costs before and after changes
|
|
26
|
+
*/
|
|
27
|
+
export declare function calculateTotalCosts(costDelta: CostDelta): {
|
|
28
|
+
before: number;
|
|
29
|
+
after: number;
|
|
30
|
+
};
|
|
@@ -0,0 +1,93 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.getTrendIndicator = getTrendIndicator;
|
|
4
|
+
exports.getPercentageChange = getPercentageChange;
|
|
5
|
+
exports.extractServiceName = extractServiceName;
|
|
6
|
+
exports.groupCostsByService = groupCostsByService;
|
|
7
|
+
exports.calculateTotalCosts = calculateTotalCosts;
|
|
8
|
+
/**
|
|
9
|
+
* Get trend indicator emoji based on cost change direction
|
|
10
|
+
*/
|
|
11
|
+
function getTrendIndicator(amount) {
|
|
12
|
+
if (amount > 0)
|
|
13
|
+
return '↗️';
|
|
14
|
+
if (amount < 0)
|
|
15
|
+
return '↘️';
|
|
16
|
+
return '➡️';
|
|
17
|
+
}
|
|
18
|
+
/**
|
|
19
|
+
* Calculate percentage change between old and new amounts
|
|
20
|
+
*/
|
|
21
|
+
function getPercentageChange(oldAmount, newAmount) {
|
|
22
|
+
if (oldAmount === 0) {
|
|
23
|
+
return newAmount > 0 ? '+∞%' : '0%';
|
|
24
|
+
}
|
|
25
|
+
const percentage = ((newAmount - oldAmount) / oldAmount) * 100;
|
|
26
|
+
if (percentage === 0)
|
|
27
|
+
return '0%';
|
|
28
|
+
const sign = percentage > 0 ? '+' : '';
|
|
29
|
+
return `${sign}${percentage.toFixed(1)}%`;
|
|
30
|
+
}
|
|
31
|
+
/**
|
|
32
|
+
* Extract AWS service name from resource type (e.g., AWS::EC2::Instance -> EC2)
|
|
33
|
+
*/
|
|
34
|
+
function extractServiceName(resourceType) {
|
|
35
|
+
const parts = resourceType.split('::');
|
|
36
|
+
if (parts.length >= 2) {
|
|
37
|
+
return parts[1];
|
|
38
|
+
}
|
|
39
|
+
return resourceType;
|
|
40
|
+
}
|
|
41
|
+
/**
|
|
42
|
+
* Group costs by AWS service (e.g., EC2, S3, Lambda)
|
|
43
|
+
*/
|
|
44
|
+
function groupCostsByService(costDelta) {
|
|
45
|
+
const serviceMap = new Map();
|
|
46
|
+
for (const resource of costDelta.addedCosts) {
|
|
47
|
+
const service = extractServiceName(resource.type);
|
|
48
|
+
const existing = serviceMap.get(service) || { totalCost: 0, resourceCount: 0 };
|
|
49
|
+
existing.totalCost += resource.monthlyCost.amount;
|
|
50
|
+
existing.resourceCount += 1;
|
|
51
|
+
serviceMap.set(service, existing);
|
|
52
|
+
}
|
|
53
|
+
for (const resource of costDelta.removedCosts) {
|
|
54
|
+
const service = extractServiceName(resource.type);
|
|
55
|
+
const existing = serviceMap.get(service) || { totalCost: 0, resourceCount: 0 };
|
|
56
|
+
existing.totalCost -= resource.monthlyCost.amount;
|
|
57
|
+
existing.resourceCount += 1;
|
|
58
|
+
serviceMap.set(service, existing);
|
|
59
|
+
}
|
|
60
|
+
for (const resource of costDelta.modifiedCosts) {
|
|
61
|
+
const service = extractServiceName(resource.type);
|
|
62
|
+
const existing = serviceMap.get(service) || { totalCost: 0, resourceCount: 0 };
|
|
63
|
+
existing.totalCost += resource.costDelta;
|
|
64
|
+
existing.resourceCount += 1;
|
|
65
|
+
serviceMap.set(service, existing);
|
|
66
|
+
}
|
|
67
|
+
return Array.from(serviceMap.entries())
|
|
68
|
+
.map(([service, data]) => ({
|
|
69
|
+
service,
|
|
70
|
+
totalCost: data.totalCost,
|
|
71
|
+
resourceCount: data.resourceCount,
|
|
72
|
+
}))
|
|
73
|
+
.sort((a, b) => Math.abs(b.totalCost) - Math.abs(a.totalCost));
|
|
74
|
+
}
|
|
75
|
+
/**
|
|
76
|
+
* Calculate total costs before and after changes
|
|
77
|
+
*/
|
|
78
|
+
function calculateTotalCosts(costDelta) {
|
|
79
|
+
let before = 0;
|
|
80
|
+
let after = 0;
|
|
81
|
+
for (const resource of costDelta.addedCosts) {
|
|
82
|
+
after += resource.monthlyCost.amount;
|
|
83
|
+
}
|
|
84
|
+
for (const resource of costDelta.removedCosts) {
|
|
85
|
+
before += resource.monthlyCost.amount;
|
|
86
|
+
}
|
|
87
|
+
for (const resource of costDelta.modifiedCosts) {
|
|
88
|
+
before += resource.oldMonthlyCost.amount;
|
|
89
|
+
after += resource.newMonthlyCost.amount;
|
|
90
|
+
}
|
|
91
|
+
return { before, after };
|
|
92
|
+
}
|
|
93
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"markdownUtils.js","sourceRoot":"","sources":["../../src/reporter/markdownUtils.ts"],"names":[],"mappings":"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","sourcesContent":["import { CostDelta } from '../pricing/types';\n\n/** Service breakdown entry for cost grouping */\nexport interface ServiceBreakdown {\n  service: string;\n  totalCost: number;\n  resourceCount: number;\n}\n\n/**\n * Get trend indicator emoji based on cost change direction\n */\nexport function getTrendIndicator(amount: number): string {\n  if (amount > 0) return '↗️';\n  if (amount < 0) return '↘️';\n  return '➡️';\n}\n\n/**\n * Calculate percentage change between old and new amounts\n */\nexport function getPercentageChange(oldAmount: number, newAmount: number): string {\n  if (oldAmount === 0) {\n    return newAmount > 0 ? '+∞%' : '0%';\n  }\n  const percentage = ((newAmount - oldAmount) / oldAmount) * 100;\n  if (percentage === 0) return '0%';\n  const sign = percentage > 0 ? '+' : '';\n  return `${sign}${percentage.toFixed(1)}%`;\n}\n\n/**\n * Extract AWS service name from resource type (e.g., AWS::EC2::Instance -> EC2)\n */\nexport function extractServiceName(resourceType: string): string {\n  const parts = resourceType.split('::');\n  if (parts.length >= 2) {\n    return parts[1];\n  }\n  return resourceType;\n}\n\n/**\n * Group costs by AWS service (e.g., EC2, S3, Lambda)\n */\nexport function groupCostsByService(costDelta: CostDelta): ServiceBreakdown[] {\n  const serviceMap = new Map<string, { totalCost: number; resourceCount: number }>();\n\n  for (const resource of costDelta.addedCosts) {\n    const service = extractServiceName(resource.type);\n    const existing = serviceMap.get(service) || { totalCost: 0, resourceCount: 0 };\n    existing.totalCost += resource.monthlyCost.amount;\n    existing.resourceCount += 1;\n    serviceMap.set(service, existing);\n  }\n\n  for (const resource of costDelta.removedCosts) {\n    const service = extractServiceName(resource.type);\n    const existing = serviceMap.get(service) || { totalCost: 0, resourceCount: 0 };\n    existing.totalCost -= resource.monthlyCost.amount;\n    existing.resourceCount += 1;\n    serviceMap.set(service, existing);\n  }\n\n  for (const resource of costDelta.modifiedCosts) {\n    const service = extractServiceName(resource.type);\n    const existing = serviceMap.get(service) || { totalCost: 0, resourceCount: 0 };\n    existing.totalCost += resource.costDelta;\n    existing.resourceCount += 1;\n    serviceMap.set(service, existing);\n  }\n\n  return Array.from(serviceMap.entries())\n    .map(([service, data]) => ({\n      service,\n      totalCost: data.totalCost,\n      resourceCount: data.resourceCount,\n    }))\n    .sort((a, b) => Math.abs(b.totalCost) - Math.abs(a.totalCost));\n}\n\n/**\n * Calculate total costs before and after changes\n */\nexport function calculateTotalCosts(costDelta: CostDelta): { before: number; after: number } {\n  let before = 0;\n  let after = 0;\n\n  for (const resource of costDelta.addedCosts) {\n    after += resource.monthlyCost.amount;\n  }\n\n  for (const resource of costDelta.removedCosts) {\n    before += resource.monthlyCost.amount;\n  }\n\n  for (const resource of costDelta.modifiedCosts) {\n    before += resource.oldMonthlyCost.amount;\n    after += resource.newMonthlyCost.amount;\n  }\n\n  return { before, after };\n}\n"]}
|
package/package.json
CHANGED
|
@@ -56,7 +56,7 @@
|
|
|
56
56
|
"eslint-plugin-import": "^2.32.0",
|
|
57
57
|
"fast-check": "^3.23.1",
|
|
58
58
|
"husky": "^9.0.0",
|
|
59
|
-
"jest": "^30.
|
|
59
|
+
"jest": "^30.3.0",
|
|
60
60
|
"jest-junit": "^16",
|
|
61
61
|
"lint-staged": "^15.0.0",
|
|
62
62
|
"projen": "^0.99.19",
|
|
@@ -87,7 +87,7 @@
|
|
|
87
87
|
"publishConfig": {
|
|
88
88
|
"access": "public"
|
|
89
89
|
},
|
|
90
|
-
"version": "0.1.
|
|
90
|
+
"version": "0.1.41",
|
|
91
91
|
"bugs": {
|
|
92
92
|
"url": "https://github.com/buildinginthecloud/cdk-cost-analyzer/issues"
|
|
93
93
|
},
|