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,412 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.Reporter = void 0;
|
|
4
|
+
class Reporter {
|
|
5
|
+
generateReport(costDelta, format, options) {
|
|
6
|
+
switch (format) {
|
|
7
|
+
case 'text':
|
|
8
|
+
return this.generateTextReport(costDelta, options);
|
|
9
|
+
case 'json':
|
|
10
|
+
return this.generateJsonReport(costDelta, options);
|
|
11
|
+
case 'markdown':
|
|
12
|
+
return this.generateMarkdownReport(costDelta, options);
|
|
13
|
+
default:
|
|
14
|
+
throw new Error(`Unsupported report format: ${format}`);
|
|
15
|
+
}
|
|
16
|
+
}
|
|
17
|
+
generateTextReport(costDelta, options) {
|
|
18
|
+
const lines = [];
|
|
19
|
+
lines.push('='.repeat(60));
|
|
20
|
+
lines.push('CDK Cost Analysis Report');
|
|
21
|
+
lines.push('='.repeat(60));
|
|
22
|
+
lines.push('');
|
|
23
|
+
// Add configuration summary if provided
|
|
24
|
+
if (options?.configSummary) {
|
|
25
|
+
lines.push(...this.formatConfigSummaryText(options.configSummary));
|
|
26
|
+
lines.push('');
|
|
27
|
+
}
|
|
28
|
+
// Add threshold status if provided
|
|
29
|
+
if (options?.thresholdStatus) {
|
|
30
|
+
lines.push(...this.formatThresholdStatusText(options.thresholdStatus));
|
|
31
|
+
lines.push('');
|
|
32
|
+
}
|
|
33
|
+
lines.push(`Total Cost Delta: ${this.formatDelta(costDelta.totalDelta, costDelta.currency)}`);
|
|
34
|
+
lines.push('');
|
|
35
|
+
if (costDelta.addedCosts.length > 0) {
|
|
36
|
+
lines.push('ADDED RESOURCES:');
|
|
37
|
+
lines.push('-'.repeat(60));
|
|
38
|
+
const sortedAdded = [...costDelta.addedCosts].sort((a, b) => b.monthlyCost.amount - a.monthlyCost.amount);
|
|
39
|
+
for (const resource of sortedAdded) {
|
|
40
|
+
lines.push(this.formatResourceLine(resource));
|
|
41
|
+
}
|
|
42
|
+
lines.push('');
|
|
43
|
+
}
|
|
44
|
+
if (costDelta.removedCosts.length > 0) {
|
|
45
|
+
lines.push('REMOVED RESOURCES:');
|
|
46
|
+
lines.push('-'.repeat(60));
|
|
47
|
+
const sortedRemoved = [...costDelta.removedCosts].sort((a, b) => b.monthlyCost.amount - a.monthlyCost.amount);
|
|
48
|
+
for (const resource of sortedRemoved) {
|
|
49
|
+
lines.push(this.formatResourceLine(resource));
|
|
50
|
+
}
|
|
51
|
+
lines.push('');
|
|
52
|
+
}
|
|
53
|
+
if (costDelta.modifiedCosts.length > 0) {
|
|
54
|
+
lines.push('MODIFIED RESOURCES:');
|
|
55
|
+
lines.push('-'.repeat(60));
|
|
56
|
+
const sortedModified = [...costDelta.modifiedCosts].sort((a, b) => Math.abs(b.costDelta) - Math.abs(a.costDelta));
|
|
57
|
+
for (const resource of sortedModified) {
|
|
58
|
+
lines.push(this.formatModifiedResourceLine(resource));
|
|
59
|
+
}
|
|
60
|
+
lines.push('');
|
|
61
|
+
}
|
|
62
|
+
if (costDelta.addedCosts.length === 0 &&
|
|
63
|
+
costDelta.removedCosts.length === 0 &&
|
|
64
|
+
costDelta.modifiedCosts.length === 0) {
|
|
65
|
+
lines.push('No resource changes detected.');
|
|
66
|
+
lines.push('');
|
|
67
|
+
}
|
|
68
|
+
lines.push('='.repeat(60));
|
|
69
|
+
return lines.join('\n');
|
|
70
|
+
}
|
|
71
|
+
generateJsonReport(costDelta, options) {
|
|
72
|
+
const report = {
|
|
73
|
+
...costDelta,
|
|
74
|
+
};
|
|
75
|
+
if (options?.configSummary) {
|
|
76
|
+
report.configSummary = options.configSummary;
|
|
77
|
+
}
|
|
78
|
+
if (options?.thresholdStatus) {
|
|
79
|
+
report.thresholdStatus = options.thresholdStatus;
|
|
80
|
+
}
|
|
81
|
+
if (options?.stackName) {
|
|
82
|
+
report.stackName = options.stackName;
|
|
83
|
+
}
|
|
84
|
+
return JSON.stringify(report, null, 2);
|
|
85
|
+
}
|
|
86
|
+
generateMarkdownReport(costDelta, options) {
|
|
87
|
+
const lines = [];
|
|
88
|
+
lines.push('# CDK Cost Analysis Report');
|
|
89
|
+
lines.push('');
|
|
90
|
+
// Add configuration summary if provided
|
|
91
|
+
if (options?.configSummary) {
|
|
92
|
+
lines.push(...this.formatConfigSummaryMarkdown(options.configSummary));
|
|
93
|
+
lines.push('');
|
|
94
|
+
}
|
|
95
|
+
// Add threshold status if provided
|
|
96
|
+
if (options?.thresholdStatus) {
|
|
97
|
+
lines.push(...this.formatThresholdStatusMarkdown(options.thresholdStatus, costDelta));
|
|
98
|
+
lines.push('');
|
|
99
|
+
}
|
|
100
|
+
// Show total cost delta
|
|
101
|
+
lines.push(`**Total Cost Delta:** ${this.formatDelta(costDelta.totalDelta, costDelta.currency)}`);
|
|
102
|
+
lines.push('');
|
|
103
|
+
// If multi-stack, show per-stack breakdowns
|
|
104
|
+
if (options?.multiStack && options?.stacks && options.stacks.length > 1) {
|
|
105
|
+
lines.push('## Per-Stack Cost Breakdown');
|
|
106
|
+
lines.push('');
|
|
107
|
+
lines.push('| Stack | Cost Delta |');
|
|
108
|
+
lines.push('|-------|------------|');
|
|
109
|
+
for (const stack of options.stacks) {
|
|
110
|
+
lines.push(`| ${stack.stackName} | ${this.formatDelta(stack.costDelta.totalDelta, stack.costDelta.currency)} |`);
|
|
111
|
+
}
|
|
112
|
+
lines.push('');
|
|
113
|
+
lines.push('<details>');
|
|
114
|
+
lines.push('<summary><strong>View Detailed Stack Breakdowns</strong></summary>');
|
|
115
|
+
lines.push('');
|
|
116
|
+
for (const stack of options.stacks) {
|
|
117
|
+
lines.push(`### ${stack.stackName}`);
|
|
118
|
+
lines.push('');
|
|
119
|
+
lines.push(...this.formatStackDetailsMarkdown(stack.costDelta));
|
|
120
|
+
lines.push('');
|
|
121
|
+
}
|
|
122
|
+
lines.push('</details>');
|
|
123
|
+
lines.push('');
|
|
124
|
+
}
|
|
125
|
+
if (costDelta.addedCosts.length > 0) {
|
|
126
|
+
lines.push('## Added Resources');
|
|
127
|
+
lines.push('');
|
|
128
|
+
lines.push('| Logical ID | Type | Monthly Cost |');
|
|
129
|
+
lines.push('|------------|------|--------------|');
|
|
130
|
+
const sortedAdded = [...costDelta.addedCosts].sort((a, b) => b.monthlyCost.amount - a.monthlyCost.amount);
|
|
131
|
+
for (const resource of sortedAdded) {
|
|
132
|
+
lines.push(`| ${resource.logicalId} | ${resource.type} | ${this.formatCurrency(resource.monthlyCost.amount, costDelta.currency)} |`);
|
|
133
|
+
}
|
|
134
|
+
lines.push('');
|
|
135
|
+
}
|
|
136
|
+
if (costDelta.removedCosts.length > 0) {
|
|
137
|
+
lines.push('## Removed Resources');
|
|
138
|
+
lines.push('');
|
|
139
|
+
lines.push('| Logical ID | Type | Monthly Cost |');
|
|
140
|
+
lines.push('|------------|------|--------------|');
|
|
141
|
+
const sortedRemoved = [...costDelta.removedCosts].sort((a, b) => b.monthlyCost.amount - a.monthlyCost.amount);
|
|
142
|
+
for (const resource of sortedRemoved) {
|
|
143
|
+
lines.push(`| ${resource.logicalId} | ${resource.type} | ${this.formatCurrency(resource.monthlyCost.amount, costDelta.currency)} |`);
|
|
144
|
+
}
|
|
145
|
+
lines.push('');
|
|
146
|
+
}
|
|
147
|
+
if (costDelta.modifiedCosts.length > 0) {
|
|
148
|
+
lines.push('## Modified Resources');
|
|
149
|
+
lines.push('');
|
|
150
|
+
lines.push('| Logical ID | Type | Old Cost | New Cost | Delta |');
|
|
151
|
+
lines.push('|------------|------|----------|----------|-------|');
|
|
152
|
+
const sortedModified = [...costDelta.modifiedCosts].sort((a, b) => Math.abs(b.costDelta) - Math.abs(a.costDelta));
|
|
153
|
+
for (const resource of sortedModified) {
|
|
154
|
+
lines.push(`| ${resource.logicalId} | ${resource.type} | ` +
|
|
155
|
+
`${this.formatCurrency(resource.oldMonthlyCost.amount, costDelta.currency)} | ` +
|
|
156
|
+
`${this.formatCurrency(resource.newMonthlyCost.amount, costDelta.currency)} | ` +
|
|
157
|
+
`${this.formatDelta(resource.costDelta, costDelta.currency)} |`);
|
|
158
|
+
}
|
|
159
|
+
lines.push('');
|
|
160
|
+
}
|
|
161
|
+
return lines.join('\n');
|
|
162
|
+
}
|
|
163
|
+
formatResourceLine(resource) {
|
|
164
|
+
const cost = this.formatCurrency(resource.monthlyCost.amount, 'USD');
|
|
165
|
+
const confidence = resource.monthlyCost.confidence;
|
|
166
|
+
return ` • ${resource.logicalId} (${resource.type}): ${cost} [${confidence}]`;
|
|
167
|
+
}
|
|
168
|
+
formatModifiedResourceLine(resource) {
|
|
169
|
+
const oldCost = this.formatCurrency(resource.oldMonthlyCost.amount, 'USD');
|
|
170
|
+
const newCost = this.formatCurrency(resource.newMonthlyCost.amount, 'USD');
|
|
171
|
+
const delta = this.formatDelta(resource.costDelta, 'USD');
|
|
172
|
+
return ` • ${resource.logicalId} (${resource.type}): ${oldCost} → ${newCost} (${delta})`;
|
|
173
|
+
}
|
|
174
|
+
formatCurrency(amount, currency) {
|
|
175
|
+
const symbol = currency === 'USD' ? '$' : currency;
|
|
176
|
+
return `${symbol}${amount.toFixed(2)}`;
|
|
177
|
+
}
|
|
178
|
+
formatDelta(amount, currency) {
|
|
179
|
+
const sign = amount > 0 ? '+' : amount < 0 ? '-' : '';
|
|
180
|
+
const absAmount = Math.abs(amount);
|
|
181
|
+
const formatted = this.formatCurrency(absAmount, currency);
|
|
182
|
+
return amount === 0 ? formatted : `${sign}${formatted}`;
|
|
183
|
+
}
|
|
184
|
+
formatConfigSummaryText(config) {
|
|
185
|
+
const lines = [];
|
|
186
|
+
lines.push('CONFIGURATION:');
|
|
187
|
+
lines.push('-'.repeat(60));
|
|
188
|
+
if (config.configPath) {
|
|
189
|
+
lines.push(` Configuration File: ${config.configPath}`);
|
|
190
|
+
}
|
|
191
|
+
else {
|
|
192
|
+
lines.push(' Configuration File: Using defaults');
|
|
193
|
+
}
|
|
194
|
+
if (config.thresholds) {
|
|
195
|
+
if (config.thresholds.environment) {
|
|
196
|
+
lines.push(` Environment: ${config.thresholds.environment}`);
|
|
197
|
+
}
|
|
198
|
+
if (config.thresholds.warning !== undefined) {
|
|
199
|
+
lines.push(` Warning Threshold: $${config.thresholds.warning.toFixed(2)}/month`);
|
|
200
|
+
}
|
|
201
|
+
if (config.thresholds.error !== undefined) {
|
|
202
|
+
lines.push(` Error Threshold: $${config.thresholds.error.toFixed(2)}/month`);
|
|
203
|
+
}
|
|
204
|
+
}
|
|
205
|
+
if (config.excludedResourceTypes &&
|
|
206
|
+
config.excludedResourceTypes.length > 0) {
|
|
207
|
+
lines.push(` Excluded Resource Types: ${config.excludedResourceTypes.join(', ')}`);
|
|
208
|
+
}
|
|
209
|
+
if (config.usageAssumptions &&
|
|
210
|
+
Object.keys(config.usageAssumptions).length > 0) {
|
|
211
|
+
lines.push(' Custom Usage Assumptions:');
|
|
212
|
+
for (const [resourceType, assumptions] of Object.entries(config.usageAssumptions)) {
|
|
213
|
+
lines.push(` - ${resourceType}: ${JSON.stringify(assumptions)}`);
|
|
214
|
+
}
|
|
215
|
+
}
|
|
216
|
+
return lines;
|
|
217
|
+
}
|
|
218
|
+
formatThresholdStatusText(threshold) {
|
|
219
|
+
const lines = [];
|
|
220
|
+
lines.push('THRESHOLD STATUS:');
|
|
221
|
+
lines.push('-'.repeat(60));
|
|
222
|
+
if (threshold.level === 'none') {
|
|
223
|
+
lines.push(' No thresholds configured');
|
|
224
|
+
}
|
|
225
|
+
else {
|
|
226
|
+
const status = threshold.passed ? 'PASSED' : 'EXCEEDED';
|
|
227
|
+
lines.push(` Status: ${status}`);
|
|
228
|
+
if (threshold.threshold !== undefined) {
|
|
229
|
+
lines.push(` Threshold: $${threshold.threshold.toFixed(2)}/month (${threshold.level})`);
|
|
230
|
+
}
|
|
231
|
+
lines.push(` Actual Delta: $${Math.abs(threshold.delta).toFixed(2)}/month`);
|
|
232
|
+
if (!threshold.passed &&
|
|
233
|
+
threshold.recommendations &&
|
|
234
|
+
threshold.recommendations.length > 0) {
|
|
235
|
+
lines.push(' Recommendations:');
|
|
236
|
+
for (const rec of threshold.recommendations) {
|
|
237
|
+
lines.push(` - ${rec}`);
|
|
238
|
+
}
|
|
239
|
+
}
|
|
240
|
+
}
|
|
241
|
+
return lines;
|
|
242
|
+
}
|
|
243
|
+
formatConfigSummaryMarkdown(config) {
|
|
244
|
+
const lines = [];
|
|
245
|
+
lines.push('<details>');
|
|
246
|
+
lines.push('<summary><strong>Configuration Summary</strong></summary>');
|
|
247
|
+
lines.push('');
|
|
248
|
+
if (config.configPath) {
|
|
249
|
+
lines.push(`**Configuration File:** \`${config.configPath}\``);
|
|
250
|
+
}
|
|
251
|
+
else {
|
|
252
|
+
lines.push('**Configuration File:** Using defaults');
|
|
253
|
+
}
|
|
254
|
+
lines.push('');
|
|
255
|
+
if (config.thresholds) {
|
|
256
|
+
lines.push('**Thresholds:**');
|
|
257
|
+
if (config.thresholds.environment) {
|
|
258
|
+
lines.push(`- Environment: ${config.thresholds.environment}`);
|
|
259
|
+
}
|
|
260
|
+
if (config.thresholds.warning !== undefined) {
|
|
261
|
+
lines.push(`- Warning: $${config.thresholds.warning.toFixed(2)}/month`);
|
|
262
|
+
}
|
|
263
|
+
if (config.thresholds.error !== undefined) {
|
|
264
|
+
lines.push(`- Error: $${config.thresholds.error.toFixed(2)}/month`);
|
|
265
|
+
}
|
|
266
|
+
lines.push('');
|
|
267
|
+
}
|
|
268
|
+
if (config.excludedResourceTypes &&
|
|
269
|
+
config.excludedResourceTypes.length > 0) {
|
|
270
|
+
lines.push('**Excluded Resource Types:**');
|
|
271
|
+
for (const type of config.excludedResourceTypes) {
|
|
272
|
+
lines.push(`- \`${type}\``);
|
|
273
|
+
}
|
|
274
|
+
lines.push('');
|
|
275
|
+
}
|
|
276
|
+
if (config.usageAssumptions &&
|
|
277
|
+
Object.keys(config.usageAssumptions).length > 0) {
|
|
278
|
+
lines.push('**Custom Usage Assumptions:**');
|
|
279
|
+
for (const [resourceType, assumptions] of Object.entries(config.usageAssumptions)) {
|
|
280
|
+
lines.push(`- **${resourceType}:**`);
|
|
281
|
+
const assumptionObj = assumptions;
|
|
282
|
+
for (const [key, value] of Object.entries(assumptionObj)) {
|
|
283
|
+
lines.push(` - ${key}: ${value}`);
|
|
284
|
+
}
|
|
285
|
+
}
|
|
286
|
+
lines.push('');
|
|
287
|
+
}
|
|
288
|
+
lines.push('</details>');
|
|
289
|
+
lines.push('');
|
|
290
|
+
return lines;
|
|
291
|
+
}
|
|
292
|
+
formatThresholdStatusMarkdown(threshold, costDelta) {
|
|
293
|
+
const lines = [];
|
|
294
|
+
if (threshold.level === 'none') {
|
|
295
|
+
return lines;
|
|
296
|
+
}
|
|
297
|
+
const passed = threshold.passed;
|
|
298
|
+
const status = passed ? 'PASSED' : 'EXCEEDED';
|
|
299
|
+
lines.push(`## Threshold Status: ${status}`);
|
|
300
|
+
lines.push('');
|
|
301
|
+
if (threshold.threshold !== undefined) {
|
|
302
|
+
lines.push(`**Threshold:** $${threshold.threshold.toFixed(2)}/month (${threshold.level})`);
|
|
303
|
+
}
|
|
304
|
+
lines.push(`**Actual Delta:** ${this.formatDelta(threshold.delta, costDelta.currency)}/month`);
|
|
305
|
+
lines.push('');
|
|
306
|
+
if (!passed) {
|
|
307
|
+
lines.push('### Action Required');
|
|
308
|
+
lines.push('');
|
|
309
|
+
lines.push(threshold.message);
|
|
310
|
+
lines.push('');
|
|
311
|
+
if (threshold.recommendations && threshold.recommendations.length > 0) {
|
|
312
|
+
lines.push('### Recommendations');
|
|
313
|
+
lines.push('');
|
|
314
|
+
for (const rec of threshold.recommendations) {
|
|
315
|
+
lines.push(`- ${rec}`);
|
|
316
|
+
}
|
|
317
|
+
lines.push('');
|
|
318
|
+
}
|
|
319
|
+
// Show top cost contributors
|
|
320
|
+
const topContributors = this.getTopCostContributors(costDelta, 5);
|
|
321
|
+
if (topContributors.length > 0) {
|
|
322
|
+
lines.push('### Top Cost Contributors');
|
|
323
|
+
lines.push('');
|
|
324
|
+
lines.push('| Resource | Type | Impact |');
|
|
325
|
+
lines.push('|----------|------|--------|');
|
|
326
|
+
for (const contributor of topContributors) {
|
|
327
|
+
lines.push(`| ${contributor.logicalId} | ${contributor.type} | ${this.formatDelta(contributor.impact, costDelta.currency)} |`);
|
|
328
|
+
}
|
|
329
|
+
lines.push('');
|
|
330
|
+
}
|
|
331
|
+
}
|
|
332
|
+
return lines;
|
|
333
|
+
}
|
|
334
|
+
getTopCostContributors(costDelta, limit) {
|
|
335
|
+
const contributors = [];
|
|
336
|
+
// Add added resources
|
|
337
|
+
for (const resource of costDelta.addedCosts) {
|
|
338
|
+
contributors.push({
|
|
339
|
+
logicalId: resource.logicalId,
|
|
340
|
+
type: resource.type,
|
|
341
|
+
impact: resource.monthlyCost.amount,
|
|
342
|
+
});
|
|
343
|
+
}
|
|
344
|
+
// Add removed resources (negative impact)
|
|
345
|
+
for (const resource of costDelta.removedCosts) {
|
|
346
|
+
contributors.push({
|
|
347
|
+
logicalId: resource.logicalId,
|
|
348
|
+
type: resource.type,
|
|
349
|
+
impact: -resource.monthlyCost.amount,
|
|
350
|
+
});
|
|
351
|
+
}
|
|
352
|
+
// Add modified resources
|
|
353
|
+
for (const resource of costDelta.modifiedCosts) {
|
|
354
|
+
contributors.push({
|
|
355
|
+
logicalId: resource.logicalId,
|
|
356
|
+
type: resource.type,
|
|
357
|
+
impact: resource.costDelta,
|
|
358
|
+
});
|
|
359
|
+
}
|
|
360
|
+
// Sort by absolute impact (descending)
|
|
361
|
+
contributors.sort((a, b) => Math.abs(b.impact) - Math.abs(a.impact));
|
|
362
|
+
return contributors.slice(0, limit);
|
|
363
|
+
}
|
|
364
|
+
formatStackDetailsMarkdown(costDelta) {
|
|
365
|
+
const lines = [];
|
|
366
|
+
if (costDelta.addedCosts.length > 0) {
|
|
367
|
+
lines.push('**Added Resources:**');
|
|
368
|
+
lines.push('');
|
|
369
|
+
lines.push('| Logical ID | Type | Monthly Cost |');
|
|
370
|
+
lines.push('|------------|------|--------------|');
|
|
371
|
+
const sortedAdded = [...costDelta.addedCosts].sort((a, b) => b.monthlyCost.amount - a.monthlyCost.amount);
|
|
372
|
+
for (const resource of sortedAdded) {
|
|
373
|
+
lines.push(`| ${resource.logicalId} | ${resource.type} | ${this.formatCurrency(resource.monthlyCost.amount, costDelta.currency)} |`);
|
|
374
|
+
}
|
|
375
|
+
lines.push('');
|
|
376
|
+
}
|
|
377
|
+
if (costDelta.removedCosts.length > 0) {
|
|
378
|
+
lines.push('**Removed Resources:**');
|
|
379
|
+
lines.push('');
|
|
380
|
+
lines.push('| Logical ID | Type | Monthly Cost |');
|
|
381
|
+
lines.push('|------------|------|--------------|');
|
|
382
|
+
const sortedRemoved = [...costDelta.removedCosts].sort((a, b) => b.monthlyCost.amount - a.monthlyCost.amount);
|
|
383
|
+
for (const resource of sortedRemoved) {
|
|
384
|
+
lines.push(`| ${resource.logicalId} | ${resource.type} | ${this.formatCurrency(resource.monthlyCost.amount, costDelta.currency)} |`);
|
|
385
|
+
}
|
|
386
|
+
lines.push('');
|
|
387
|
+
}
|
|
388
|
+
if (costDelta.modifiedCosts.length > 0) {
|
|
389
|
+
lines.push('**Modified Resources:**');
|
|
390
|
+
lines.push('');
|
|
391
|
+
lines.push('| Logical ID | Type | Old Cost | New Cost | Delta |');
|
|
392
|
+
lines.push('|------------|------|----------|----------|-------|');
|
|
393
|
+
const sortedModified = [...costDelta.modifiedCosts].sort((a, b) => Math.abs(b.costDelta) - Math.abs(a.costDelta));
|
|
394
|
+
for (const resource of sortedModified) {
|
|
395
|
+
lines.push(`| ${resource.logicalId} | ${resource.type} | ` +
|
|
396
|
+
`${this.formatCurrency(resource.oldMonthlyCost.amount, costDelta.currency)} | ` +
|
|
397
|
+
`${this.formatCurrency(resource.newMonthlyCost.amount, costDelta.currency)} | ` +
|
|
398
|
+
`${this.formatDelta(resource.costDelta, costDelta.currency)} |`);
|
|
399
|
+
}
|
|
400
|
+
lines.push('');
|
|
401
|
+
}
|
|
402
|
+
if (costDelta.addedCosts.length === 0 &&
|
|
403
|
+
costDelta.removedCosts.length === 0 &&
|
|
404
|
+
costDelta.modifiedCosts.length === 0) {
|
|
405
|
+
lines.push('No resource changes detected.');
|
|
406
|
+
lines.push('');
|
|
407
|
+
}
|
|
408
|
+
return lines;
|
|
409
|
+
}
|
|
410
|
+
}
|
|
411
|
+
exports.Reporter = Reporter;
|
|
412
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Reporter.js","sourceRoot":"","sources":["../../src/reporter/Reporter.ts"],"names":[],"mappings":";;;AAOA,MAAa,QAAQ;IACnB,cAAc,CACZ,SAAoB,EACpB,MAAoB,EACpB,OAAuB;QAEvB,QAAQ,MAAM,EAAE,CAAC;YACf,KAAK,MAAM;gBACT,OAAO,IAAI,CAAC,kBAAkB,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC;YACrD,KAAK,MAAM;gBACT,OAAO,IAAI,CAAC,kBAAkB,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC;YACrD,KAAK,UAAU;gBACb,OAAO,IAAI,CAAC,sBAAsB,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC;YACzD;gBACE,MAAM,IAAI,KAAK,CAAC,8BAA8B,MAAM,EAAE,CAAC,CAAC;QAC5D,CAAC;IACH,CAAC;IAEO,kBAAkB,CACxB,SAAoB,EACpB,OAAuB;QAEvB,MAAM,KAAK,GAAa,EAAE,CAAC;QAE3B,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC;QAC3B,KAAK,CAAC,IAAI,CAAC,0BAA0B,CAAC,CAAC;QACvC,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC;QAC3B,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QAEf,wCAAwC;QACxC,IAAI,OAAO,EAAE,aAAa,EAAE,CAAC;YAC3B,KAAK,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,uBAAuB,CAAC,OAAO,CAAC,aAAa,CAAC,CAAC,CAAC;YACnE,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACjB,CAAC;QAED,mCAAmC;QACnC,IAAI,OAAO,EAAE,eAAe,EAAE,CAAC;YAC7B,KAAK,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,yBAAyB,CAAC,OAAO,CAAC,eAAe,CAAC,CAAC,CAAC;YACvE,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACjB,CAAC;QAED,KAAK,CAAC,IAAI,CACR,qBAAqB,IAAI,CAAC,WAAW,CAAC,SAAS,CAAC,UAAU,EAAE,SAAS,CAAC,QAAQ,CAAC,EAAE,CAClF,CAAC;QACF,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QAEf,IAAI,SAAS,CAAC,UAAU,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACpC,KAAK,CAAC,IAAI,CAAC,kBAAkB,CAAC,CAAC;YAC/B,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC;YAE3B,MAAM,WAAW,GAAG,CAAC,GAAG,SAAS,CAAC,UAAU,CAAC,CAAC,IAAI,CAChD,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,WAAW,CAAC,MAAM,GAAG,CAAC,CAAC,WAAW,CAAC,MAAM,CACtD,CAAC;YAEF,KAAK,MAAM,QAAQ,IAAI,WAAW,EAAE,CAAC;gBACnC,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,kBAAkB,CAAC,QAAQ,CAAC,CAAC,CAAC;YAChD,CAAC;YACD,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACjB,CAAC;QAED,IAAI,SAAS,CAAC,YAAY,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACtC,KAAK,CAAC,IAAI,CAAC,oBAAoB,CAAC,CAAC;YACjC,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC;YAE3B,MAAM,aAAa,GAAG,CAAC,GAAG,SAAS,CAAC,YAAY,CAAC,CAAC,IAAI,CACpD,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,WAAW,CAAC,MAAM,GAAG,CAAC,CAAC,WAAW,CAAC,MAAM,CACtD,CAAC;YAEF,KAAK,MAAM,QAAQ,IAAI,aAAa,EAAE,CAAC;gBACrC,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,kBAAkB,CAAC,QAAQ,CAAC,CAAC,CAAC;YAChD,CAAC;YACD,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACjB,CAAC;QAED,IAAI,SAAS,CAAC,aAAa,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACvC,KAAK,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC;YAClC,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC;YAE3B,MAAM,cAAc,GAAG,CAAC,GAAG,SAAS,CAAC,aAAa,CAAC,CAAC,IAAI,CACtD,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,SAAS,CAAC,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,SAAS,CAAC,CACxD,CAAC;YAEF,KAAK,MAAM,QAAQ,IAAI,cAAc,EAAE,CAAC;gBACtC,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,0BAA0B,CAAC,QAAQ,CAAC,CAAC,CAAC;YACxD,CAAC;YACD,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACjB,CAAC;QAED,IACE,SAAS,CAAC,UAAU,CAAC,MAAM,KAAK,CAAC;YACjC,SAAS,CAAC,YAAY,CAAC,MAAM,KAAK,CAAC;YACnC,SAAS,CAAC,aAAa,CAAC,MAAM,KAAK,CAAC,EACpC,CAAC;YACD,KAAK,CAAC,IAAI,CAAC,+BAA+B,CAAC,CAAC;YAC5C,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACjB,CAAC;QAED,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC;QAE3B,OAAO,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IAC1B,CAAC;IAEO,kBAAkB,CACxB,SAAoB,EACpB,OAAuB;QAEvB,MAAM,MAAM,GAAQ;YAClB,GAAG,SAAS;SACb,CAAC;QAEF,IAAI,OAAO,EAAE,aAAa,EAAE,CAAC;YAC3B,MAAM,CAAC,aAAa,GAAG,OAAO,CAAC,aAAa,CAAC;QAC/C,CAAC;QAED,IAAI,OAAO,EAAE,eAAe,EAAE,CAAC;YAC7B,MAAM,CAAC,eAAe,GAAG,OAAO,CAAC,eAAe,CAAC;QACnD,CAAC;QAED,IAAI,OAAO,EAAE,SAAS,EAAE,CAAC;YACvB,MAAM,CAAC,SAAS,GAAG,OAAO,CAAC,SAAS,CAAC;QACvC,CAAC;QAED,OAAO,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC;IACzC,CAAC;IAEO,sBAAsB,CAC5B,SAAoB,EACpB,OAAuB;QAEvB,MAAM,KAAK,GAAa,EAAE,CAAC;QAE3B,KAAK,CAAC,IAAI,CAAC,4BAA4B,CAAC,CAAC;QACzC,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QAEf,wCAAwC;QACxC,IAAI,OAAO,EAAE,aAAa,EAAE,CAAC;YAC3B,KAAK,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,2BAA2B,CAAC,OAAO,CAAC,aAAa,CAAC,CAAC,CAAC;YACvE,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACjB,CAAC;QAED,mCAAmC;QACnC,IAAI,OAAO,EAAE,eAAe,EAAE,CAAC;YAC7B,KAAK,CAAC,IAAI,CACR,GAAG,IAAI,CAAC,6BAA6B,CACnC,OAAO,CAAC,eAAe,EACvB,SAAS,CACV,CACF,CAAC;YACF,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACjB,CAAC;QAED,wBAAwB;QACxB,KAAK,CAAC,IAAI,CACR,yBAAyB,IAAI,CAAC,WAAW,CAAC,SAAS,CAAC,UAAU,EAAE,SAAS,CAAC,QAAQ,CAAC,EAAE,CACtF,CAAC;QACF,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QAEf,4CAA4C;QAC5C,IAAI,OAAO,EAAE,UAAU,IAAI,OAAO,EAAE,MAAM,IAAI,OAAO,CAAC,MAAM,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACxE,KAAK,CAAC,IAAI,CAAC,6BAA6B,CAAC,CAAC;YAC1C,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YACf,KAAK,CAAC,IAAI,CAAC,wBAAwB,CAAC,CAAC;YACrC,KAAK,CAAC,IAAI,CAAC,wBAAwB,CAAC,CAAC;YACrC,KAAK,MAAM,KAAK,IAAI,OAAO,CAAC,MAAM,EAAE,CAAC;gBACnC,KAAK,CAAC,IAAI,CACR,KAAK,KAAK,CAAC,SAAS,MAAM,IAAI,CAAC,WAAW,CAAC,KAAK,CAAC,SAAS,CAAC,UAAU,EAAE,KAAK,CAAC,SAAS,CAAC,QAAQ,CAAC,IAAI,CACrG,CAAC;YACJ,CAAC;YACD,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YACf,KAAK,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;YACxB,KAAK,CAAC,IAAI,CACR,oEAAoE,CACrE,CAAC;YACF,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YACf,KAAK,MAAM,KAAK,IAAI,OAAO,CAAC,MAAM,EAAE,CAAC;gBACnC,KAAK,CAAC,IAAI,CAAC,OAAO,KAAK,CAAC,SAAS,EAAE,CAAC,CAAC;gBACrC,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;gBACf,KAAK,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,0BAA0B,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC,CAAC;gBAChE,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YACjB,CAAC;YACD,KAAK,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;YACzB,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACjB,CAAC;QAED,IAAI,SAAS,CAAC,UAAU,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACpC,KAAK,CAAC,IAAI,CAAC,oBAAoB,CAAC,CAAC;YACjC,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YACf,KAAK,CAAC,IAAI,CAAC,sCAAsC,CAAC,CAAC;YACnD,KAAK,CAAC,IAAI,CAAC,sCAAsC,CAAC,CAAC;YAEnD,MAAM,WAAW,GAAG,CAAC,GAAG,SAAS,CAAC,UAAU,CAAC,CAAC,IAAI,CAChD,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,WAAW,CAAC,MAAM,GAAG,CAAC,CAAC,WAAW,CAAC,MAAM,CACtD,CAAC;YAEF,KAAK,MAAM,QAAQ,IAAI,WAAW,EAAE,CAAC;gBACnC,KAAK,CAAC,IAAI,CACR,KAAK,QAAQ,CAAC,SAAS,MAAM,QAAQ,CAAC,IAAI,MAAM,IAAI,CAAC,cAAc,CAAC,QAAQ,CAAC,WAAW,CAAC,MAAM,EAAE,SAAS,CAAC,QAAQ,CAAC,IAAI,CACzH,CAAC;YACJ,CAAC;YACD,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACjB,CAAC;QAED,IAAI,SAAS,CAAC,YAAY,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACtC,KAAK,CAAC,IAAI,CAAC,sBAAsB,CAAC,CAAC;YACnC,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YACf,KAAK,CAAC,IAAI,CAAC,sCAAsC,CAAC,CAAC;YACnD,KAAK,CAAC,IAAI,CAAC,sCAAsC,CAAC,CAAC;YAEnD,MAAM,aAAa,GAAG,CAAC,GAAG,SAAS,CAAC,YAAY,CAAC,CAAC,IAAI,CACpD,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,WAAW,CAAC,MAAM,GAAG,CAAC,CAAC,WAAW,CAAC,MAAM,CACtD,CAAC;YAEF,KAAK,MAAM,QAAQ,IAAI,aAAa,EAAE,CAAC;gBACrC,KAAK,CAAC,IAAI,CACR,KAAK,QAAQ,CAAC,SAAS,MAAM,QAAQ,CAAC,IAAI,MAAM,IAAI,CAAC,cAAc,CAAC,QAAQ,CAAC,WAAW,CAAC,MAAM,EAAE,SAAS,CAAC,QAAQ,CAAC,IAAI,CACzH,CAAC;YACJ,CAAC;YACD,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACjB,CAAC;QAED,IAAI,SAAS,CAAC,aAAa,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACvC,KAAK,CAAC,IAAI,CAAC,uBAAuB,CAAC,CAAC;YACpC,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YACf,KAAK,CAAC,IAAI,CAAC,qDAAqD,CAAC,CAAC;YAClE,KAAK,CAAC,IAAI,CAAC,qDAAqD,CAAC,CAAC;YAElE,MAAM,cAAc,GAAG,CAAC,GAAG,SAAS,CAAC,aAAa,CAAC,CAAC,IAAI,CACtD,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,SAAS,CAAC,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,SAAS,CAAC,CACxD,CAAC;YAEF,KAAK,MAAM,QAAQ,IAAI,cAAc,EAAE,CAAC;gBACtC,KAAK,CAAC,IAAI,CACR,KAAK,QAAQ,CAAC,SAAS,MAAM,QAAQ,CAAC,IAAI,KAAK;oBAC7C,GAAG,IAAI,CAAC,cAAc,CAAC,QAAQ,CAAC,cAAc,CAAC,MAAM,EAAE,SAAS,CAAC,QAAQ,CAAC,KAAK;oBAC/E,GAAG,IAAI,CAAC,cAAc,CAAC,QAAQ,CAAC,cAAc,CAAC,MAAM,EAAE,SAAS,CAAC,QAAQ,CAAC,KAAK;oBAC/E,GAAG,IAAI,CAAC,WAAW,CAAC,QAAQ,CAAC,SAAS,EAAE,SAAS,CAAC,QAAQ,CAAC,IAAI,CAClE,CAAC;YACJ,CAAC;YACD,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACjB,CAAC;QAED,OAAO,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IAC1B,CAAC;IAEO,kBAAkB,CAAC,QAAsB;QAC/C,MAAM,IAAI,GAAG,IAAI,CAAC,cAAc,CAAC,QAAQ,CAAC,WAAW,CAAC,MAAM,EAAE,KAAK,CAAC,CAAC;QACrE,MAAM,UAAU,GAAG,QAAQ,CAAC,WAAW,CAAC,UAAU,CAAC;QACnD,OAAO,OAAO,QAAQ,CAAC,SAAS,KAAK,QAAQ,CAAC,IAAI,MAAM,IAAI,KAAK,UAAU,GAAG,CAAC;IACjF,CAAC;IAEO,0BAA0B,CAAC,QAA8B;QAC/D,MAAM,OAAO,GAAG,IAAI,CAAC,cAAc,CAAC,QAAQ,CAAC,cAAc,CAAC,MAAM,EAAE,KAAK,CAAC,CAAC;QAC3E,MAAM,OAAO,GAAG,IAAI,CAAC,cAAc,CAAC,QAAQ,CAAC,cAAc,CAAC,MAAM,EAAE,KAAK,CAAC,CAAC;QAC3E,MAAM,KAAK,GAAG,IAAI,CAAC,WAAW,CAAC,QAAQ,CAAC,SAAS,EAAE,KAAK,CAAC,CAAC;QAC1D,OAAO,OAAO,QAAQ,CAAC,SAAS,KAAK,QAAQ,CAAC,IAAI,MAAM,OAAO,MAAM,OAAO,KAAK,KAAK,GAAG,CAAC;IAC5F,CAAC;IAEO,cAAc,CAAC,MAAc,EAAE,QAAgB;QACrD,MAAM,MAAM,GAAG,QAAQ,KAAK,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,QAAQ,CAAC;QACnD,OAAO,GAAG,MAAM,GAAG,MAAM,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC;IACzC,CAAC;IAEO,WAAW,CAAC,MAAc,EAAE,QAAgB;QAClD,MAAM,IAAI,GAAG,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC;QACtD,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;QACnC,MAAM,SAAS,GAAG,IAAI,CAAC,cAAc,CAAC,SAAS,EAAE,QAAQ,CAAC,CAAC;QAC3D,OAAO,MAAM,KAAK,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,GAAG,IAAI,GAAG,SAAS,EAAE,CAAC;IAC1D,CAAC;IAEO,uBAAuB,CAAC,MAAW;QACzC,MAAM,KAAK,GAAa,EAAE,CAAC;QAC3B,KAAK,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAC;QAC7B,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC;QAE3B,IAAI,MAAM,CAAC,UAAU,EAAE,CAAC;YACtB,KAAK,CAAC,IAAI,CAAC,yBAAyB,MAAM,CAAC,UAAU,EAAE,CAAC,CAAC;QAC3D,CAAC;aAAM,CAAC;YACN,KAAK,CAAC,IAAI,CAAC,sCAAsC,CAAC,CAAC;QACrD,CAAC;QAED,IAAI,MAAM,CAAC,UAAU,EAAE,CAAC;YACtB,IAAI,MAAM,CAAC,UAAU,CAAC,WAAW,EAAE,CAAC;gBAClC,KAAK,CAAC,IAAI,CAAC,kBAAkB,MAAM,CAAC,UAAU,CAAC,WAAW,EAAE,CAAC,CAAC;YAChE,CAAC;YACD,IAAI,MAAM,CAAC,UAAU,CAAC,OAAO,KAAK,SAAS,EAAE,CAAC;gBAC5C,KAAK,CAAC,IAAI,CACR,yBAAyB,MAAM,CAAC,UAAU,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC,CAAC,QAAQ,CACtE,CAAC;YACJ,CAAC;YACD,IAAI,MAAM,CAAC,UAAU,CAAC,KAAK,KAAK,SAAS,EAAE,CAAC;gBAC1C,KAAK,CAAC,IAAI,CACR,uBAAuB,MAAM,CAAC,UAAU,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,QAAQ,CAClE,CAAC;YACJ,CAAC;QACH,CAAC;QAED,IACE,MAAM,CAAC,qBAAqB;YAC5B,MAAM,CAAC,qBAAqB,CAAC,MAAM,GAAG,CAAC,EACvC,CAAC;YACD,KAAK,CAAC,IAAI,CACR,8BAA8B,MAAM,CAAC,qBAAqB,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CACxE,CAAC;QACJ,CAAC;QAED,IACE,MAAM,CAAC,gBAAgB;YACvB,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,gBAAgB,CAAC,CAAC,MAAM,GAAG,CAAC,EAC/C,CAAC;YACD,KAAK,CAAC,IAAI,CAAC,6BAA6B,CAAC,CAAC;YAC1C,KAAK,MAAM,CAAC,YAAY,EAAE,WAAW,CAAC,IAAI,MAAM,CAAC,OAAO,CACtD,MAAM,CAAC,gBAAgB,CACxB,EAAE,CAAC;gBACF,KAAK,CAAC,IAAI,CAAC,SAAS,YAAY,KAAK,IAAI,CAAC,SAAS,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC;YACtE,CAAC;QACH,CAAC;QAED,OAAO,KAAK,CAAC;IACf,CAAC;IAEO,yBAAyB,CAAC,SAAc;QAC9C,MAAM,KAAK,GAAa,EAAE,CAAC;QAC3B,KAAK,CAAC,IAAI,CAAC,mBAAmB,CAAC,CAAC;QAChC,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC;QAE3B,IAAI,SAAS,CAAC,KAAK,KAAK,MAAM,EAAE,CAAC;YAC/B,KAAK,CAAC,IAAI,CAAC,4BAA4B,CAAC,CAAC;QAC3C,CAAC;aAAM,CAAC;YACN,MAAM,MAAM,GAAG,SAAS,CAAC,MAAM,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,UAAU,CAAC;YACxD,KAAK,CAAC,IAAI,CAAC,aAAa,MAAM,EAAE,CAAC,CAAC;YAElC,IAAI,SAAS,CAAC,SAAS,KAAK,SAAS,EAAE,CAAC;gBACtC,KAAK,CAAC,IAAI,CACR,iBAAiB,SAAS,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC,CAAC,WAAW,SAAS,CAAC,KAAK,GAAG,CAC7E,CAAC;YACJ,CAAC;YACD,KAAK,CAAC,IAAI,CACR,oBAAoB,IAAI,CAAC,GAAG,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,QAAQ,CACjE,CAAC;YAEF,IACE,CAAC,SAAS,CAAC,MAAM;gBACjB,SAAS,CAAC,eAAe;gBACzB,SAAS,CAAC,eAAe,CAAC,MAAM,GAAG,CAAC,EACpC,CAAC;gBACD,KAAK,CAAC,IAAI,CAAC,oBAAoB,CAAC,CAAC;gBACjC,KAAK,MAAM,GAAG,IAAI,SAAS,CAAC,eAAe,EAAE,CAAC;oBAC5C,KAAK,CAAC,IAAI,CAAC,SAAS,GAAG,EAAE,CAAC,CAAC;gBAC7B,CAAC;YACH,CAAC;QACH,CAAC;QAED,OAAO,KAAK,CAAC;IACf,CAAC;IAEO,2BAA2B,CAAC,MAAW;QAC7C,MAAM,KAAK,GAAa,EAAE,CAAC;QAC3B,KAAK,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;QACxB,KAAK,CAAC,IAAI,CAAC,2DAA2D,CAAC,CAAC;QACxE,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QAEf,IAAI,MAAM,CAAC,UAAU,EAAE,CAAC;YACtB,KAAK,CAAC,IAAI,CAAC,6BAA6B,MAAM,CAAC,UAAU,IAAI,CAAC,CAAC;QACjE,CAAC;aAAM,CAAC;YACN,KAAK,CAAC,IAAI,CAAC,wCAAwC,CAAC,CAAC;QACvD,CAAC;QACD,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QAEf,IAAI,MAAM,CAAC,UAAU,EAAE,CAAC;YACtB,KAAK,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC;YAC9B,IAAI,MAAM,CAAC,UAAU,CAAC,WAAW,EAAE,CAAC;gBAClC,KAAK,CAAC,IAAI,CAAC,kBAAkB,MAAM,CAAC,UAAU,CAAC,WAAW,EAAE,CAAC,CAAC;YAChE,CAAC;YACD,IAAI,MAAM,CAAC,UAAU,CAAC,OAAO,KAAK,SAAS,EAAE,CAAC;gBAC5C,KAAK,CAAC,IAAI,CAAC,eAAe,MAAM,CAAC,UAAU,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC;YAC1E,CAAC;YACD,IAAI,MAAM,CAAC,UAAU,CAAC,KAAK,KAAK,SAAS,EAAE,CAAC;gBAC1C,KAAK,CAAC,IAAI,CAAC,aAAa,MAAM,CAAC,UAAU,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC;YACtE,CAAC;YACD,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACjB,CAAC;QAED,IACE,MAAM,CAAC,qBAAqB;YAC5B,MAAM,CAAC,qBAAqB,CAAC,MAAM,GAAG,CAAC,EACvC,CAAC;YACD,KAAK,CAAC,IAAI,CAAC,8BAA8B,CAAC,CAAC;YAC3C,KAAK,MAAM,IAAI,IAAI,MAAM,CAAC,qBAAqB,EAAE,CAAC;gBAChD,KAAK,CAAC,IAAI,CAAC,OAAO,IAAI,IAAI,CAAC,CAAC;YAC9B,CAAC;YACD,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACjB,CAAC;QAED,IACE,MAAM,CAAC,gBAAgB;YACvB,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,gBAAgB,CAAC,CAAC,MAAM,GAAG,CAAC,EAC/C,CAAC;YACD,KAAK,CAAC,IAAI,CAAC,+BAA+B,CAAC,CAAC;YAC5C,KAAK,MAAM,CAAC,YAAY,EAAE,WAAW,CAAC,IAAI,MAAM,CAAC,OAAO,CACtD,MAAM,CAAC,gBAAgB,CACxB,EAAE,CAAC;gBACF,KAAK,CAAC,IAAI,CAAC,OAAO,YAAY,KAAK,CAAC,CAAC;gBACrC,MAAM,aAAa,GAAG,WAAkC,CAAC;gBACzD,KAAK,MAAM,CAAC,GAAG,EAAE,KAAK,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,aAAa,CAAC,EAAE,CAAC;oBACzD,KAAK,CAAC,IAAI,CAAC,OAAO,GAAG,KAAK,KAAK,EAAE,CAAC,CAAC;gBACrC,CAAC;YACH,CAAC;YACD,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACjB,CAAC;QAED,KAAK,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;QACzB,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QAEf,OAAO,KAAK,CAAC;IACf,CAAC;IAEO,6BAA6B,CACnC,SAAc,EACd,SAAoB;QAEpB,MAAM,KAAK,GAAa,EAAE,CAAC;QAE3B,IAAI,SAAS,CAAC,KAAK,KAAK,MAAM,EAAE,CAAC;YAC/B,OAAO,KAAK,CAAC;QACf,CAAC;QAED,MAAM,MAAM,GAAG,SAAS,CAAC,MAAM,CAAC;QAChC,MAAM,MAAM,GAAG,MAAM,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,UAAU,CAAC;QAE9C,KAAK,CAAC,IAAI,CAAC,wBAAwB,MAAM,EAAE,CAAC,CAAC;QAC7C,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QAEf,IAAI,SAAS,CAAC,SAAS,KAAK,SAAS,EAAE,CAAC;YACtC,KAAK,CAAC,IAAI,CACR,mBAAmB,SAAS,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC,CAAC,WAAW,SAAS,CAAC,KAAK,GAAG,CAC/E,CAAC;QACJ,CAAC;QACD,KAAK,CAAC,IAAI,CACR,qBAAqB,IAAI,CAAC,WAAW,CAAC,SAAS,CAAC,KAAK,EAAE,SAAS,CAAC,QAAQ,CAAC,QAAQ,CACnF,CAAC;QACF,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QAEf,IAAI,CAAC,MAAM,EAAE,CAAC;YACZ,KAAK,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC;YAClC,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YACf,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC;YAC9B,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YAEf,IAAI,SAAS,CAAC,eAAe,IAAI,SAAS,CAAC,eAAe,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBACtE,KAAK,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC;gBAClC,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;gBACf,KAAK,MAAM,GAAG,IAAI,SAAS,CAAC,eAAe,EAAE,CAAC;oBAC5C,KAAK,CAAC,IAAI,CAAC,KAAK,GAAG,EAAE,CAAC,CAAC;gBACzB,CAAC;gBACD,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YACjB,CAAC;YAED,6BAA6B;YAC7B,MAAM,eAAe,GAAG,IAAI,CAAC,sBAAsB,CAAC,SAAS,EAAE,CAAC,CAAC,CAAC;YAClE,IAAI,eAAe,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBAC/B,KAAK,CAAC,IAAI,CAAC,2BAA2B,CAAC,CAAC;gBACxC,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;gBACf,KAAK,CAAC,IAAI,CAAC,8BAA8B,CAAC,CAAC;gBAC3C,KAAK,CAAC,IAAI,CAAC,8BAA8B,CAAC,CAAC;gBAC3C,KAAK,MAAM,WAAW,IAAI,eAAe,EAAE,CAAC;oBAC1C,KAAK,CAAC,IAAI,CACR,KAAK,WAAW,CAAC,SAAS,MAAM,WAAW,CAAC,IAAI,MAAM,IAAI,CAAC,WAAW,CAAC,WAAW,CAAC,MAAM,EAAE,SAAS,CAAC,QAAQ,CAAC,IAAI,CACnH,CAAC;gBACJ,CAAC;gBACD,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YACjB,CAAC;QACH,CAAC;QAED,OAAO,KAAK,CAAC;IACf,CAAC;IAEO,sBAAsB,CAC5B,SAAoB,EACpB,KAAa;QAEb,MAAM,YAAY,GAIb,EAAE,CAAC;QAER,sBAAsB;QACtB,KAAK,MAAM,QAAQ,IAAI,SAAS,CAAC,UAAU,EAAE,CAAC;YAC5C,YAAY,CAAC,IAAI,CAAC;gBAChB,SAAS,EAAE,QAAQ,CAAC,SAAS;gBAC7B,IAAI,EAAE,QAAQ,CAAC,IAAI;gBACnB,MAAM,EAAE,QAAQ,CAAC,WAAW,CAAC,MAAM;aACpC,CAAC,CAAC;QACL,CAAC;QAED,0CAA0C;QAC1C,KAAK,MAAM,QAAQ,IAAI,SAAS,CAAC,YAAY,EAAE,CAAC;YAC9C,YAAY,CAAC,IAAI,CAAC;gBAChB,SAAS,EAAE,QAAQ,CAAC,SAAS;gBAC7B,IAAI,EAAE,QAAQ,CAAC,IAAI;gBACnB,MAAM,EAAE,CAAC,QAAQ,CAAC,WAAW,CAAC,MAAM;aACrC,CAAC,CAAC;QACL,CAAC;QAED,yBAAyB;QACzB,KAAK,MAAM,QAAQ,IAAI,SAAS,CAAC,aAAa,EAAE,CAAC;YAC/C,YAAY,CAAC,IAAI,CAAC;gBAChB,SAAS,EAAE,QAAQ,CAAC,SAAS;gBAC7B,IAAI,EAAE,QAAQ,CAAC,IAAI;gBACnB,MAAM,EAAE,QAAQ,CAAC,SAAS;aAC3B,CAAC,CAAC;QACL,CAAC;QAED,uCAAuC;QACvC,YAAY,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC;QAErE,OAAO,YAAY,CAAC,KAAK,CAAC,CAAC,EAAE,KAAK,CAAC,CAAC;IACtC,CAAC;IAEO,0BAA0B,CAAC,SAAoB;QACrD,MAAM,KAAK,GAAa,EAAE,CAAC;QAE3B,IAAI,SAAS,CAAC,UAAU,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACpC,KAAK,CAAC,IAAI,CAAC,sBAAsB,CAAC,CAAC;YACnC,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YACf,KAAK,CAAC,IAAI,CAAC,sCAAsC,CAAC,CAAC;YACnD,KAAK,CAAC,IAAI,CAAC,sCAAsC,CAAC,CAAC;YAEnD,MAAM,WAAW,GAAG,CAAC,GAAG,SAAS,CAAC,UAAU,CAAC,CAAC,IAAI,CAChD,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,WAAW,CAAC,MAAM,GAAG,CAAC,CAAC,WAAW,CAAC,MAAM,CACtD,CAAC;YAEF,KAAK,MAAM,QAAQ,IAAI,WAAW,EAAE,CAAC;gBACnC,KAAK,CAAC,IAAI,CACR,KAAK,QAAQ,CAAC,SAAS,MAAM,QAAQ,CAAC,IAAI,MAAM,IAAI,CAAC,cAAc,CAAC,QAAQ,CAAC,WAAW,CAAC,MAAM,EAAE,SAAS,CAAC,QAAQ,CAAC,IAAI,CACzH,CAAC;YACJ,CAAC;YACD,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACjB,CAAC;QAED,IAAI,SAAS,CAAC,YAAY,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACtC,KAAK,CAAC,IAAI,CAAC,wBAAwB,CAAC,CAAC;YACrC,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YACf,KAAK,CAAC,IAAI,CAAC,sCAAsC,CAAC,CAAC;YACnD,KAAK,CAAC,IAAI,CAAC,sCAAsC,CAAC,CAAC;YAEnD,MAAM,aAAa,GAAG,CAAC,GAAG,SAAS,CAAC,YAAY,CAAC,CAAC,IAAI,CACpD,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,WAAW,CAAC,MAAM,GAAG,CAAC,CAAC,WAAW,CAAC,MAAM,CACtD,CAAC;YAEF,KAAK,MAAM,QAAQ,IAAI,aAAa,EAAE,CAAC;gBACrC,KAAK,CAAC,IAAI,CACR,KAAK,QAAQ,CAAC,SAAS,MAAM,QAAQ,CAAC,IAAI,MAAM,IAAI,CAAC,cAAc,CAAC,QAAQ,CAAC,WAAW,CAAC,MAAM,EAAE,SAAS,CAAC,QAAQ,CAAC,IAAI,CACzH,CAAC;YACJ,CAAC;YACD,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACjB,CAAC;QAED,IAAI,SAAS,CAAC,aAAa,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACvC,KAAK,CAAC,IAAI,CAAC,yBAAyB,CAAC,CAAC;YACtC,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YACf,KAAK,CAAC,IAAI,CAAC,qDAAqD,CAAC,CAAC;YAClE,KAAK,CAAC,IAAI,CAAC,qDAAqD,CAAC,CAAC;YAElE,MAAM,cAAc,GAAG,CAAC,GAAG,SAAS,CAAC,aAAa,CAAC,CAAC,IAAI,CACtD,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,SAAS,CAAC,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,SAAS,CAAC,CACxD,CAAC;YAEF,KAAK,MAAM,QAAQ,IAAI,cAAc,EAAE,CAAC;gBACtC,KAAK,CAAC,IAAI,CACR,KAAK,QAAQ,CAAC,SAAS,MAAM,QAAQ,CAAC,IAAI,KAAK;oBAC7C,GAAG,IAAI,CAAC,cAAc,CAAC,QAAQ,CAAC,cAAc,CAAC,MAAM,EAAE,SAAS,CAAC,QAAQ,CAAC,KAAK;oBAC/E,GAAG,IAAI,CAAC,cAAc,CAAC,QAAQ,CAAC,cAAc,CAAC,MAAM,EAAE,SAAS,CAAC,QAAQ,CAAC,KAAK;oBAC/E,GAAG,IAAI,CAAC,WAAW,CAAC,QAAQ,CAAC,SAAS,EAAE,SAAS,CAAC,QAAQ,CAAC,IAAI,CAClE,CAAC;YACJ,CAAC;YACD,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACjB,CAAC;QAED,IACE,SAAS,CAAC,UAAU,CAAC,MAAM,KAAK,CAAC;YACjC,SAAS,CAAC,YAAY,CAAC,MAAM,KAAK,CAAC;YACnC,SAAS,CAAC,aAAa,CAAC,MAAM,KAAK,CAAC,EACpC,CAAC;YACD,KAAK,CAAC,IAAI,CAAC,+BAA+B,CAAC,CAAC;YAC5C,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACjB,CAAC;QAED,OAAO,KAAK,CAAC;IACf,CAAC;CACF;AA9kBD,4BA8kBC","sourcesContent":["import { Reporter as IReporter, ReportFormat, ReportOptions } from './types';\nimport {\n  CostDelta,\n  ResourceCost,\n  ModifiedResourceCost,\n} from '../pricing/types';\n\nexport class Reporter implements IReporter {\n  generateReport(\n    costDelta: CostDelta,\n    format: ReportFormat,\n    options?: ReportOptions,\n  ): string {\n    switch (format) {\n      case 'text':\n        return this.generateTextReport(costDelta, options);\n      case 'json':\n        return this.generateJsonReport(costDelta, options);\n      case 'markdown':\n        return this.generateMarkdownReport(costDelta, options);\n      default:\n        throw new Error(`Unsupported report format: ${format}`);\n    }\n  }\n\n  private generateTextReport(\n    costDelta: CostDelta,\n    options?: ReportOptions,\n  ): string {\n    const lines: string[] = [];\n\n    lines.push('='.repeat(60));\n    lines.push('CDK Cost Analysis Report');\n    lines.push('='.repeat(60));\n    lines.push('');\n\n    // Add configuration summary if provided\n    if (options?.configSummary) {\n      lines.push(...this.formatConfigSummaryText(options.configSummary));\n      lines.push('');\n    }\n\n    // Add threshold status if provided\n    if (options?.thresholdStatus) {\n      lines.push(...this.formatThresholdStatusText(options.thresholdStatus));\n      lines.push('');\n    }\n\n    lines.push(\n      `Total Cost Delta: ${this.formatDelta(costDelta.totalDelta, costDelta.currency)}`,\n    );\n    lines.push('');\n\n    if (costDelta.addedCosts.length > 0) {\n      lines.push('ADDED RESOURCES:');\n      lines.push('-'.repeat(60));\n\n      const sortedAdded = [...costDelta.addedCosts].sort(\n        (a, b) => b.monthlyCost.amount - a.monthlyCost.amount,\n      );\n\n      for (const resource of sortedAdded) {\n        lines.push(this.formatResourceLine(resource));\n      }\n      lines.push('');\n    }\n\n    if (costDelta.removedCosts.length > 0) {\n      lines.push('REMOVED RESOURCES:');\n      lines.push('-'.repeat(60));\n\n      const sortedRemoved = [...costDelta.removedCosts].sort(\n        (a, b) => b.monthlyCost.amount - a.monthlyCost.amount,\n      );\n\n      for (const resource of sortedRemoved) {\n        lines.push(this.formatResourceLine(resource));\n      }\n      lines.push('');\n    }\n\n    if (costDelta.modifiedCosts.length > 0) {\n      lines.push('MODIFIED RESOURCES:');\n      lines.push('-'.repeat(60));\n\n      const sortedModified = [...costDelta.modifiedCosts].sort(\n        (a, b) => Math.abs(b.costDelta) - Math.abs(a.costDelta),\n      );\n\n      for (const resource of sortedModified) {\n        lines.push(this.formatModifiedResourceLine(resource));\n      }\n      lines.push('');\n    }\n\n    if (\n      costDelta.addedCosts.length === 0 &&\n      costDelta.removedCosts.length === 0 &&\n      costDelta.modifiedCosts.length === 0\n    ) {\n      lines.push('No resource changes detected.');\n      lines.push('');\n    }\n\n    lines.push('='.repeat(60));\n\n    return lines.join('\\n');\n  }\n\n  private generateJsonReport(\n    costDelta: CostDelta,\n    options?: ReportOptions,\n  ): string {\n    const report: any = {\n      ...costDelta,\n    };\n\n    if (options?.configSummary) {\n      report.configSummary = options.configSummary;\n    }\n\n    if (options?.thresholdStatus) {\n      report.thresholdStatus = options.thresholdStatus;\n    }\n\n    if (options?.stackName) {\n      report.stackName = options.stackName;\n    }\n\n    return JSON.stringify(report, null, 2);\n  }\n\n  private generateMarkdownReport(\n    costDelta: CostDelta,\n    options?: ReportOptions,\n  ): string {\n    const lines: string[] = [];\n\n    lines.push('# CDK Cost Analysis Report');\n    lines.push('');\n\n    // Add configuration summary if provided\n    if (options?.configSummary) {\n      lines.push(...this.formatConfigSummaryMarkdown(options.configSummary));\n      lines.push('');\n    }\n\n    // Add threshold status if provided\n    if (options?.thresholdStatus) {\n      lines.push(\n        ...this.formatThresholdStatusMarkdown(\n          options.thresholdStatus,\n          costDelta,\n        ),\n      );\n      lines.push('');\n    }\n\n    // Show total cost delta\n    lines.push(\n      `**Total Cost Delta:** ${this.formatDelta(costDelta.totalDelta, costDelta.currency)}`,\n    );\n    lines.push('');\n\n    // If multi-stack, show per-stack breakdowns\n    if (options?.multiStack && options?.stacks && options.stacks.length > 1) {\n      lines.push('## Per-Stack Cost Breakdown');\n      lines.push('');\n      lines.push('| Stack | Cost Delta |');\n      lines.push('|-------|------------|');\n      for (const stack of options.stacks) {\n        lines.push(\n          `| ${stack.stackName} | ${this.formatDelta(stack.costDelta.totalDelta, stack.costDelta.currency)} |`,\n        );\n      }\n      lines.push('');\n      lines.push('<details>');\n      lines.push(\n        '<summary><strong>View Detailed Stack Breakdowns</strong></summary>',\n      );\n      lines.push('');\n      for (const stack of options.stacks) {\n        lines.push(`### ${stack.stackName}`);\n        lines.push('');\n        lines.push(...this.formatStackDetailsMarkdown(stack.costDelta));\n        lines.push('');\n      }\n      lines.push('</details>');\n      lines.push('');\n    }\n\n    if (costDelta.addedCosts.length > 0) {\n      lines.push('## Added Resources');\n      lines.push('');\n      lines.push('| Logical ID | Type | Monthly Cost |');\n      lines.push('|------------|------|--------------|');\n\n      const sortedAdded = [...costDelta.addedCosts].sort(\n        (a, b) => b.monthlyCost.amount - a.monthlyCost.amount,\n      );\n\n      for (const resource of sortedAdded) {\n        lines.push(\n          `| ${resource.logicalId} | ${resource.type} | ${this.formatCurrency(resource.monthlyCost.amount, costDelta.currency)} |`,\n        );\n      }\n      lines.push('');\n    }\n\n    if (costDelta.removedCosts.length > 0) {\n      lines.push('## Removed Resources');\n      lines.push('');\n      lines.push('| Logical ID | Type | Monthly Cost |');\n      lines.push('|------------|------|--------------|');\n\n      const sortedRemoved = [...costDelta.removedCosts].sort(\n        (a, b) => b.monthlyCost.amount - a.monthlyCost.amount,\n      );\n\n      for (const resource of sortedRemoved) {\n        lines.push(\n          `| ${resource.logicalId} | ${resource.type} | ${this.formatCurrency(resource.monthlyCost.amount, costDelta.currency)} |`,\n        );\n      }\n      lines.push('');\n    }\n\n    if (costDelta.modifiedCosts.length > 0) {\n      lines.push('## Modified Resources');\n      lines.push('');\n      lines.push('| Logical ID | Type | Old Cost | New Cost | Delta |');\n      lines.push('|------------|------|----------|----------|-------|');\n\n      const sortedModified = [...costDelta.modifiedCosts].sort(\n        (a, b) => Math.abs(b.costDelta) - Math.abs(a.costDelta),\n      );\n\n      for (const resource of sortedModified) {\n        lines.push(\n          `| ${resource.logicalId} | ${resource.type} | ` +\n            `${this.formatCurrency(resource.oldMonthlyCost.amount, costDelta.currency)} | ` +\n            `${this.formatCurrency(resource.newMonthlyCost.amount, costDelta.currency)} | ` +\n            `${this.formatDelta(resource.costDelta, costDelta.currency)} |`,\n        );\n      }\n      lines.push('');\n    }\n\n    return lines.join('\\n');\n  }\n\n  private formatResourceLine(resource: ResourceCost): string {\n    const cost = this.formatCurrency(resource.monthlyCost.amount, 'USD');\n    const confidence = resource.monthlyCost.confidence;\n    return `  • ${resource.logicalId} (${resource.type}): ${cost} [${confidence}]`;\n  }\n\n  private formatModifiedResourceLine(resource: ModifiedResourceCost): string {\n    const oldCost = this.formatCurrency(resource.oldMonthlyCost.amount, 'USD');\n    const newCost = this.formatCurrency(resource.newMonthlyCost.amount, 'USD');\n    const delta = this.formatDelta(resource.costDelta, 'USD');\n    return `  • ${resource.logicalId} (${resource.type}): ${oldCost} → ${newCost} (${delta})`;\n  }\n\n  private formatCurrency(amount: number, currency: string): string {\n    const symbol = currency === 'USD' ? '$' : currency;\n    return `${symbol}${amount.toFixed(2)}`;\n  }\n\n  private formatDelta(amount: number, currency: string): string {\n    const sign = amount > 0 ? '+' : amount < 0 ? '-' : '';\n    const absAmount = Math.abs(amount);\n    const formatted = this.formatCurrency(absAmount, currency);\n    return amount === 0 ? formatted : `${sign}${formatted}`;\n  }\n\n  private formatConfigSummaryText(config: any): string[] {\n    const lines: string[] = [];\n    lines.push('CONFIGURATION:');\n    lines.push('-'.repeat(60));\n\n    if (config.configPath) {\n      lines.push(`  Configuration File: ${config.configPath}`);\n    } else {\n      lines.push('  Configuration File: Using defaults');\n    }\n\n    if (config.thresholds) {\n      if (config.thresholds.environment) {\n        lines.push(`  Environment: ${config.thresholds.environment}`);\n      }\n      if (config.thresholds.warning !== undefined) {\n        lines.push(\n          `  Warning Threshold: $${config.thresholds.warning.toFixed(2)}/month`,\n        );\n      }\n      if (config.thresholds.error !== undefined) {\n        lines.push(\n          `  Error Threshold: $${config.thresholds.error.toFixed(2)}/month`,\n        );\n      }\n    }\n\n    if (\n      config.excludedResourceTypes &&\n      config.excludedResourceTypes.length > 0\n    ) {\n      lines.push(\n        `  Excluded Resource Types: ${config.excludedResourceTypes.join(', ')}`,\n      );\n    }\n\n    if (\n      config.usageAssumptions &&\n      Object.keys(config.usageAssumptions).length > 0\n    ) {\n      lines.push('  Custom Usage Assumptions:');\n      for (const [resourceType, assumptions] of Object.entries(\n        config.usageAssumptions,\n      )) {\n        lines.push(`    - ${resourceType}: ${JSON.stringify(assumptions)}`);\n      }\n    }\n\n    return lines;\n  }\n\n  private formatThresholdStatusText(threshold: any): string[] {\n    const lines: string[] = [];\n    lines.push('THRESHOLD STATUS:');\n    lines.push('-'.repeat(60));\n\n    if (threshold.level === 'none') {\n      lines.push('  No thresholds configured');\n    } else {\n      const status = threshold.passed ? 'PASSED' : 'EXCEEDED';\n      lines.push(`  Status: ${status}`);\n\n      if (threshold.threshold !== undefined) {\n        lines.push(\n          `  Threshold: $${threshold.threshold.toFixed(2)}/month (${threshold.level})`,\n        );\n      }\n      lines.push(\n        `  Actual Delta: $${Math.abs(threshold.delta).toFixed(2)}/month`,\n      );\n\n      if (\n        !threshold.passed &&\n        threshold.recommendations &&\n        threshold.recommendations.length > 0\n      ) {\n        lines.push('  Recommendations:');\n        for (const rec of threshold.recommendations) {\n          lines.push(`    - ${rec}`);\n        }\n      }\n    }\n\n    return lines;\n  }\n\n  private formatConfigSummaryMarkdown(config: any): string[] {\n    const lines: string[] = [];\n    lines.push('<details>');\n    lines.push('<summary><strong>Configuration Summary</strong></summary>');\n    lines.push('');\n\n    if (config.configPath) {\n      lines.push(`**Configuration File:** \\`${config.configPath}\\``);\n    } else {\n      lines.push('**Configuration File:** Using defaults');\n    }\n    lines.push('');\n\n    if (config.thresholds) {\n      lines.push('**Thresholds:**');\n      if (config.thresholds.environment) {\n        lines.push(`- Environment: ${config.thresholds.environment}`);\n      }\n      if (config.thresholds.warning !== undefined) {\n        lines.push(`- Warning: $${config.thresholds.warning.toFixed(2)}/month`);\n      }\n      if (config.thresholds.error !== undefined) {\n        lines.push(`- Error: $${config.thresholds.error.toFixed(2)}/month`);\n      }\n      lines.push('');\n    }\n\n    if (\n      config.excludedResourceTypes &&\n      config.excludedResourceTypes.length > 0\n    ) {\n      lines.push('**Excluded Resource Types:**');\n      for (const type of config.excludedResourceTypes) {\n        lines.push(`- \\`${type}\\``);\n      }\n      lines.push('');\n    }\n\n    if (\n      config.usageAssumptions &&\n      Object.keys(config.usageAssumptions).length > 0\n    ) {\n      lines.push('**Custom Usage Assumptions:**');\n      for (const [resourceType, assumptions] of Object.entries(\n        config.usageAssumptions,\n      )) {\n        lines.push(`- **${resourceType}:**`);\n        const assumptionObj = assumptions as Record<string, any>;\n        for (const [key, value] of Object.entries(assumptionObj)) {\n          lines.push(`  - ${key}: ${value}`);\n        }\n      }\n      lines.push('');\n    }\n\n    lines.push('</details>');\n    lines.push('');\n\n    return lines;\n  }\n\n  private formatThresholdStatusMarkdown(\n    threshold: any,\n    costDelta: CostDelta,\n  ): string[] {\n    const lines: string[] = [];\n\n    if (threshold.level === 'none') {\n      return lines;\n    }\n\n    const passed = threshold.passed;\n    const status = passed ? 'PASSED' : 'EXCEEDED';\n\n    lines.push(`## Threshold Status: ${status}`);\n    lines.push('');\n\n    if (threshold.threshold !== undefined) {\n      lines.push(\n        `**Threshold:** $${threshold.threshold.toFixed(2)}/month (${threshold.level})`,\n      );\n    }\n    lines.push(\n      `**Actual Delta:** ${this.formatDelta(threshold.delta, costDelta.currency)}/month`,\n    );\n    lines.push('');\n\n    if (!passed) {\n      lines.push('### Action Required');\n      lines.push('');\n      lines.push(threshold.message);\n      lines.push('');\n\n      if (threshold.recommendations && threshold.recommendations.length > 0) {\n        lines.push('### Recommendations');\n        lines.push('');\n        for (const rec of threshold.recommendations) {\n          lines.push(`- ${rec}`);\n        }\n        lines.push('');\n      }\n\n      // Show top cost contributors\n      const topContributors = this.getTopCostContributors(costDelta, 5);\n      if (topContributors.length > 0) {\n        lines.push('### Top Cost Contributors');\n        lines.push('');\n        lines.push('| Resource | Type | Impact |');\n        lines.push('|----------|------|--------|');\n        for (const contributor of topContributors) {\n          lines.push(\n            `| ${contributor.logicalId} | ${contributor.type} | ${this.formatDelta(contributor.impact, costDelta.currency)} |`,\n          );\n        }\n        lines.push('');\n      }\n    }\n\n    return lines;\n  }\n\n  private getTopCostContributors(\n    costDelta: CostDelta,\n    limit: number,\n  ): Array<{ logicalId: string; type: string; impact: number }> {\n    const contributors: Array<{\n      logicalId: string;\n      type: string;\n      impact: number;\n    }> = [];\n\n    // Add added resources\n    for (const resource of costDelta.addedCosts) {\n      contributors.push({\n        logicalId: resource.logicalId,\n        type: resource.type,\n        impact: resource.monthlyCost.amount,\n      });\n    }\n\n    // Add removed resources (negative impact)\n    for (const resource of costDelta.removedCosts) {\n      contributors.push({\n        logicalId: resource.logicalId,\n        type: resource.type,\n        impact: -resource.monthlyCost.amount,\n      });\n    }\n\n    // Add modified resources\n    for (const resource of costDelta.modifiedCosts) {\n      contributors.push({\n        logicalId: resource.logicalId,\n        type: resource.type,\n        impact: resource.costDelta,\n      });\n    }\n\n    // Sort by absolute impact (descending)\n    contributors.sort((a, b) => Math.abs(b.impact) - Math.abs(a.impact));\n\n    return contributors.slice(0, limit);\n  }\n\n  private formatStackDetailsMarkdown(costDelta: CostDelta): string[] {\n    const lines: string[] = [];\n\n    if (costDelta.addedCosts.length > 0) {\n      lines.push('**Added Resources:**');\n      lines.push('');\n      lines.push('| Logical ID | Type | Monthly Cost |');\n      lines.push('|------------|------|--------------|');\n\n      const sortedAdded = [...costDelta.addedCosts].sort(\n        (a, b) => b.monthlyCost.amount - a.monthlyCost.amount,\n      );\n\n      for (const resource of sortedAdded) {\n        lines.push(\n          `| ${resource.logicalId} | ${resource.type} | ${this.formatCurrency(resource.monthlyCost.amount, costDelta.currency)} |`,\n        );\n      }\n      lines.push('');\n    }\n\n    if (costDelta.removedCosts.length > 0) {\n      lines.push('**Removed Resources:**');\n      lines.push('');\n      lines.push('| Logical ID | Type | Monthly Cost |');\n      lines.push('|------------|------|--------------|');\n\n      const sortedRemoved = [...costDelta.removedCosts].sort(\n        (a, b) => b.monthlyCost.amount - a.monthlyCost.amount,\n      );\n\n      for (const resource of sortedRemoved) {\n        lines.push(\n          `| ${resource.logicalId} | ${resource.type} | ${this.formatCurrency(resource.monthlyCost.amount, costDelta.currency)} |`,\n        );\n      }\n      lines.push('');\n    }\n\n    if (costDelta.modifiedCosts.length > 0) {\n      lines.push('**Modified Resources:**');\n      lines.push('');\n      lines.push('| Logical ID | Type | Old Cost | New Cost | Delta |');\n      lines.push('|------------|------|----------|----------|-------|');\n\n      const sortedModified = [...costDelta.modifiedCosts].sort(\n        (a, b) => Math.abs(b.costDelta) - Math.abs(a.costDelta),\n      );\n\n      for (const resource of sortedModified) {\n        lines.push(\n          `| ${resource.logicalId} | ${resource.type} | ` +\n            `${this.formatCurrency(resource.oldMonthlyCost.amount, costDelta.currency)} | ` +\n            `${this.formatCurrency(resource.newMonthlyCost.amount, costDelta.currency)} | ` +\n            `${this.formatDelta(resource.costDelta, costDelta.currency)} |`,\n        );\n      }\n      lines.push('');\n    }\n\n    if (\n      costDelta.addedCosts.length === 0 &&\n      costDelta.removedCosts.length === 0 &&\n      costDelta.modifiedCosts.length === 0\n    ) {\n      lines.push('No resource changes detected.');\n      lines.push('');\n    }\n\n    return lines;\n  }\n}\n"]}
|
|
@@ -0,0 +1,21 @@
|
|
|
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.Reporter = void 0;
|
|
18
|
+
var Reporter_1 = require("./Reporter");
|
|
19
|
+
Object.defineProperty(exports, "Reporter", { enumerable: true, get: function () { return Reporter_1.Reporter; } });
|
|
20
|
+
__exportStar(require("./types"), exports);
|
|
21
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvcmVwb3J0ZXIvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFBQSx1Q0FBc0M7QUFBN0Isb0dBQUEsUUFBUSxPQUFBO0FBQ2pCLDBDQUF3QiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB7IFJlcG9ydGVyIH0gZnJvbSAnLi9SZXBvcnRlcic7XG5leHBvcnQgKiBmcm9tICcuL3R5cGVzJztcbiJdfQ==
|
|
@@ -0,0 +1,72 @@
|
|
|
1
|
+
import { ConfigSummary } from '../pipeline/types';
|
|
2
|
+
import { CostDelta } from '../pricing/types';
|
|
3
|
+
import { ThresholdEvaluation } from '../threshold/types';
|
|
4
|
+
/**
|
|
5
|
+
* Reporter interface for generating cost analysis reports in various formats.
|
|
6
|
+
*/
|
|
7
|
+
export interface Reporter {
|
|
8
|
+
/**
|
|
9
|
+
* Generate a cost analysis report.
|
|
10
|
+
*
|
|
11
|
+
* @param costDelta - The cost delta containing added, removed, and modified resources
|
|
12
|
+
* @param format - The output format (text, json, or markdown)
|
|
13
|
+
* @param options - Optional reporting options for enhanced reports
|
|
14
|
+
* @returns Formatted report string
|
|
15
|
+
*/
|
|
16
|
+
generateReport(costDelta: CostDelta, format: ReportFormat, options?: ReportOptions): string;
|
|
17
|
+
}
|
|
18
|
+
/**
|
|
19
|
+
* Options for customizing report generation.
|
|
20
|
+
*
|
|
21
|
+
* These options enable enhanced reporting features including:
|
|
22
|
+
* - Configuration summaries showing applied settings
|
|
23
|
+
* - Threshold status with actionable guidance
|
|
24
|
+
* - Multi-stack breakdowns for complex applications
|
|
25
|
+
*/
|
|
26
|
+
export interface ReportOptions {
|
|
27
|
+
/**
|
|
28
|
+
* Configuration summary to include in the report.
|
|
29
|
+
* Shows thresholds, usage assumptions, and exclusions applied.
|
|
30
|
+
*/
|
|
31
|
+
configSummary?: ConfigSummary;
|
|
32
|
+
/**
|
|
33
|
+
* Threshold evaluation status to include in the report.
|
|
34
|
+
* Shows whether cost thresholds were exceeded and provides recommendations.
|
|
35
|
+
*/
|
|
36
|
+
thresholdStatus?: ThresholdEvaluation;
|
|
37
|
+
/**
|
|
38
|
+
* Name of the stack being analyzed (for multi-stack reports).
|
|
39
|
+
*/
|
|
40
|
+
stackName?: string;
|
|
41
|
+
/**
|
|
42
|
+
* Whether this is a multi-stack analysis.
|
|
43
|
+
* When true, enables per-stack cost breakdowns in the report.
|
|
44
|
+
*/
|
|
45
|
+
multiStack?: boolean;
|
|
46
|
+
/**
|
|
47
|
+
* Stack-level cost details for multi-stack reports.
|
|
48
|
+
* Each entry contains the stack name and its cost delta.
|
|
49
|
+
*/
|
|
50
|
+
stacks?: StackCostDelta[];
|
|
51
|
+
}
|
|
52
|
+
/**
|
|
53
|
+
* Stack-level cost delta for multi-stack reporting.
|
|
54
|
+
*/
|
|
55
|
+
export interface StackCostDelta {
|
|
56
|
+
/**
|
|
57
|
+
* Name of the CloudFormation stack.
|
|
58
|
+
*/
|
|
59
|
+
stackName: string;
|
|
60
|
+
/**
|
|
61
|
+
* Cost delta for this specific stack.
|
|
62
|
+
*/
|
|
63
|
+
costDelta: CostDelta;
|
|
64
|
+
}
|
|
65
|
+
/**
|
|
66
|
+
* Supported report output formats.
|
|
67
|
+
*
|
|
68
|
+
* - text: Human-readable console output with ASCII formatting
|
|
69
|
+
* - json: Structured JSON for programmatic processing
|
|
70
|
+
* - markdown: Formatted markdown suitable for GitLab merge request comments
|
|
71
|
+
*/
|
|
72
|
+
export type ReportFormat = 'text' | 'json' | 'markdown';
|
|
@@ -0,0 +1,3 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvcmVwb3J0ZXIvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbmZpZ1N1bW1hcnkgfSBmcm9tICcuLi9waXBlbGluZS90eXBlcyc7XG5pbXBvcnQgeyBDb3N0RGVsdGEgfSBmcm9tICcuLi9wcmljaW5nL3R5cGVzJztcbmltcG9ydCB7IFRocmVzaG9sZEV2YWx1YXRpb24gfSBmcm9tICcuLi90aHJlc2hvbGQvdHlwZXMnO1xuXG4vKipcbiAqIFJlcG9ydGVyIGludGVyZmFjZSBmb3IgZ2VuZXJhdGluZyBjb3N0IGFuYWx5c2lzIHJlcG9ydHMgaW4gdmFyaW91cyBmb3JtYXRzLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIFJlcG9ydGVyIHtcbiAgLyoqXG4gICAqIEdlbmVyYXRlIGEgY29zdCBhbmFseXNpcyByZXBvcnQuXG4gICAqXG4gICAqIEBwYXJhbSBjb3N0RGVsdGEgLSBUaGUgY29zdCBkZWx0YSBjb250YWluaW5nIGFkZGVkLCByZW1vdmVkLCBhbmQgbW9kaWZpZWQgcmVzb3VyY2VzXG4gICAqIEBwYXJhbSBmb3JtYXQgLSBUaGUgb3V0cHV0IGZvcm1hdCAodGV4dCwganNvbiwgb3IgbWFya2Rvd24pXG4gICAqIEBwYXJhbSBvcHRpb25zIC0gT3B0aW9uYWwgcmVwb3J0aW5nIG9wdGlvbnMgZm9yIGVuaGFuY2VkIHJlcG9ydHNcbiAgICogQHJldHVybnMgRm9ybWF0dGVkIHJlcG9ydCBzdHJpbmdcbiAgICovXG4gIGdlbmVyYXRlUmVwb3J0KFxuICAgIGNvc3REZWx0YTogQ29zdERlbHRhLFxuICAgIGZvcm1hdDogUmVwb3J0Rm9ybWF0LFxuICAgIG9wdGlvbnM/OiBSZXBvcnRPcHRpb25zLFxuICApOiBzdHJpbmc7XG59XG5cbi8qKlxuICogT3B0aW9ucyBmb3IgY3VzdG9taXppbmcgcmVwb3J0IGdlbmVyYXRpb24uXG4gKlxuICogVGhlc2Ugb3B0aW9ucyBlbmFibGUgZW5oYW5jZWQgcmVwb3J0aW5nIGZlYXR1cmVzIGluY2x1ZGluZzpcbiAqIC0gQ29uZmlndXJhdGlvbiBzdW1tYXJpZXMgc2hvd2luZyBhcHBsaWVkIHNldHRpbmdzXG4gKiAtIFRocmVzaG9sZCBzdGF0dXMgd2l0aCBhY3Rpb25hYmxlIGd1aWRhbmNlXG4gKiAtIE11bHRpLXN0YWNrIGJyZWFrZG93bnMgZm9yIGNvbXBsZXggYXBwbGljYXRpb25zXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgUmVwb3J0T3B0aW9ucyB7XG4gIC8qKlxuICAgKiBDb25maWd1cmF0aW9uIHN1bW1hcnkgdG8gaW5jbHVkZSBpbiB0aGUgcmVwb3J0LlxuICAgKiBTaG93cyB0aHJlc2hvbGRzLCB1c2FnZSBhc3N1bXB0aW9ucywgYW5kIGV4Y2x1c2lvbnMgYXBwbGllZC5cbiAgICovXG4gIGNvbmZpZ1N1bW1hcnk/OiBDb25maWdTdW1tYXJ5O1xuXG4gIC8qKlxuICAgKiBUaHJlc2hvbGQgZXZhbHVhdGlvbiBzdGF0dXMgdG8gaW5jbHVkZSBpbiB0aGUgcmVwb3J0LlxuICAgKiBTaG93cyB3aGV0aGVyIGNvc3QgdGhyZXNob2xkcyB3ZXJlIGV4Y2VlZGVkIGFuZCBwcm92aWRlcyByZWNvbW1lbmRhdGlvbnMuXG4gICAqL1xuICB0aHJlc2hvbGRTdGF0dXM/OiBUaHJlc2hvbGRFdmFsdWF0aW9uO1xuXG4gIC8qKlxuICAgKiBOYW1lIG9mIHRoZSBzdGFjayBiZWluZyBhbmFseXplZCAoZm9yIG11bHRpLXN0YWNrIHJlcG9ydHMpLlxuICAgKi9cbiAgc3RhY2tOYW1lPzogc3RyaW5nO1xuXG4gIC8qKlxuICAgKiBXaGV0aGVyIHRoaXMgaXMgYSBtdWx0aS1zdGFjayBhbmFseXNpcy5cbiAgICogV2hlbiB0cnVlLCBlbmFibGVzIHBlci1zdGFjayBjb3N0IGJyZWFrZG93bnMgaW4gdGhlIHJlcG9ydC5cbiAgICovXG4gIG11bHRpU3RhY2s/OiBib29sZWFuO1xuXG4gIC8qKlxuICAgKiBTdGFjay1sZXZlbCBjb3N0IGRldGFpbHMgZm9yIG11bHRpLXN0YWNrIHJlcG9ydHMuXG4gICAqIEVhY2ggZW50cnkgY29udGFpbnMgdGhlIHN0YWNrIG5hbWUgYW5kIGl0cyBjb3N0IGRlbHRhLlxuICAgKi9cbiAgc3RhY2tzPzogU3RhY2tDb3N0RGVsdGFbXTtcbn1cblxuLyoqXG4gKiBTdGFjay1sZXZlbCBjb3N0IGRlbHRhIGZvciBtdWx0aS1zdGFjayByZXBvcnRpbmcuXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgU3RhY2tDb3N0RGVsdGEge1xuICAvKipcbiAgICogTmFtZSBvZiB0aGUgQ2xvdWRGb3JtYXRpb24gc3RhY2suXG4gICAqL1xuICBzdGFja05hbWU6IHN0cmluZztcblxuICAvKipcbiAgICogQ29zdCBkZWx0YSBmb3IgdGhpcyBzcGVjaWZpYyBzdGFjay5cbiAgICovXG4gIGNvc3REZWx0YTogQ29zdERlbHRhO1xufVxuXG4vKipcbiAqIFN1cHBvcnRlZCByZXBvcnQgb3V0cHV0IGZvcm1hdHMuXG4gKlxuICogLSB0ZXh0OiBIdW1hbi1yZWFkYWJsZSBjb25zb2xlIG91dHB1dCB3aXRoIEFTQ0lJIGZvcm1hdHRpbmdcbiAqIC0ganNvbjogU3RydWN0dXJlZCBKU09OIGZvciBwcm9ncmFtbWF0aWMgcHJvY2Vzc2luZ1xuICogLSBtYXJrZG93bjogRm9ybWF0dGVkIG1hcmtkb3duIHN1aXRhYmxlIGZvciBHaXRMYWIgbWVyZ2UgcmVxdWVzdCBjb21tZW50c1xuICovXG5leHBvcnQgdHlwZSBSZXBvcnRGb3JtYXQgPSAndGV4dCcgfCAnanNvbicgfCAnbWFya2Rvd24nO1xuIl19
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
import { SynthesisOptions, SynthesisResult } from './types';
|
|
2
|
+
export declare class SynthesisOrchestrator {
|
|
3
|
+
private readonly DEFAULT_OUTPUT_PATH;
|
|
4
|
+
/**
|
|
5
|
+
* Execute CDK synthesis
|
|
6
|
+
*/
|
|
7
|
+
synthesize(options: SynthesisOptions): Promise<SynthesisResult>;
|
|
8
|
+
/**
|
|
9
|
+
* Execute synthesis command
|
|
10
|
+
*
|
|
11
|
+
* Uses shell: false for security to prevent command injection attacks.
|
|
12
|
+
* Arguments are passed as an array to avoid shell interpretation.
|
|
13
|
+
*
|
|
14
|
+
* Implements a 15-second timeout to prevent hanging processes in CI:
|
|
15
|
+
* - Sends SIGTERM for graceful termination
|
|
16
|
+
* - Follows up with SIGKILL after 1 second if process doesn't exit
|
|
17
|
+
* - Prevents duplicate resolution using isResolved flag
|
|
18
|
+
* - Ensures all event listeners are cleaned up
|
|
19
|
+
* - Uses process.kill as fallback for stubborn processes
|
|
20
|
+
*/
|
|
21
|
+
private executeSynthesis;
|
|
22
|
+
/**
|
|
23
|
+
* Find all CloudFormation templates in output directory
|
|
24
|
+
*/
|
|
25
|
+
private findTemplates;
|
|
26
|
+
}
|