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.
Files changed (228) hide show
  1. package/.cdk-cost-analyzer-cache/metadata.json +12 -0
  2. package/.gitlab-ci.yml +214 -0
  3. package/.husky/pre-commit +12 -0
  4. package/.kiro/hooks/accessibility-audit.kiro.hook +18 -0
  5. package/.kiro/hooks/api-schema-validation.kiro.hook +21 -0
  6. package/.kiro/hooks/auto-test-on-save.kiro.hook +19 -0
  7. package/.kiro/hooks/cdk-synth-on-change.kiro.hook +20 -0
  8. package/.kiro/hooks/code-coverage-check.kiro.hook +14 -0
  9. package/.kiro/hooks/commit-message-helper.kiro.hook +14 -0
  10. package/.kiro/hooks/dependency-update-check.kiro.hook +14 -0
  11. package/.kiro/hooks/env-file-validation.kiro.hook +18 -0
  12. package/.kiro/hooks/lint-and-format-on-save.kiro.hook +21 -0
  13. package/.kiro/hooks/mcp-config-validation.kiro.hook +17 -0
  14. package/.kiro/hooks/mcp-server-test.kiro.hook +14 -0
  15. package/.kiro/hooks/performance-analysis.kiro.hook +14 -0
  16. package/.kiro/hooks/readme-spell-check.kiro.hook +14 -0
  17. package/.kiro/hooks/security-scan-on-dependency-change.kiro.hook +21 -0
  18. package/.kiro/hooks/translation-update.kiro.hook +18 -0
  19. package/.kiro/hooks/update-documentation.kiro.hook +18 -0
  20. package/.kiro/settings/mcp.json +20 -0
  21. package/.kiro/specs/cdk-cost-analyzer/design.md +620 -0
  22. package/.kiro/specs/cdk-cost-analyzer/requirements.md +183 -0
  23. package/.kiro/specs/cdk-cost-analyzer/tasks.md +357 -0
  24. package/.kiro/specs/github-actions-ci/design.md +281 -0
  25. package/.kiro/specs/github-actions-ci/requirements.md +86 -0
  26. package/.kiro/specs/github-actions-ci/tasks.md +115 -0
  27. package/.kiro/specs/nlb-calculator-test-coverage/design.md +190 -0
  28. package/.kiro/specs/nlb-calculator-test-coverage/requirements.md +84 -0
  29. package/.kiro/specs/nlb-calculator-test-coverage/tasks.md +150 -0
  30. package/.kiro/specs/production-readiness/design.md +1213 -0
  31. package/.kiro/specs/production-readiness/requirements.md +312 -0
  32. package/.kiro/specs/production-readiness/tasks.md +269 -0
  33. package/.kiro/specs/repository-cleanup/design.md +283 -0
  34. package/.kiro/specs/repository-cleanup/requirements.md +74 -0
  35. package/.kiro/specs/repository-cleanup/tasks.md +64 -0
  36. package/.kiro/steering/aws-cli-best-practices.md +41 -0
  37. package/.kiro/steering/cdk-best-practices.md +49 -0
  38. package/.kiro/steering/development-standards.md +54 -0
  39. package/.kiro/steering/docker-best-practices.md +34 -0
  40. package/.kiro/steering/documentation-style.md +151 -0
  41. package/.kiro/steering/git-best-practices.md +37 -0
  42. package/.kiro/steering/mcp-best-practices.md +95 -0
  43. package/.kiro/steering/python-best-practices.md +48 -0
  44. package/.kiro/steering/react-best-practices.md +44 -0
  45. package/.kiro/steering/security-best-practices.md +41 -0
  46. package/.kiro/steering/testing-best-practices.md +59 -0
  47. package/.kiro/steering/typescript-best-practices.md +40 -0
  48. package/CHANGELOG.md +49 -0
  49. package/CONTRIBUTING.md +258 -0
  50. package/LICENSE +19 -0
  51. package/README.md +480 -0
  52. package/SECURITY.md +117 -0
  53. package/dist/api/index.d.ts +11 -0
  54. package/dist/api/index.js +65 -0
  55. package/dist/api/types.d.ts +15 -0
  56. package/dist/api/types.js +3 -0
  57. package/dist/cli/index.d.ts +2 -0
  58. package/dist/cli/index.js +262 -0
  59. package/dist/config/ConfigManager.d.ts +40 -0
  60. package/dist/config/ConfigManager.js +238 -0
  61. package/dist/config/index.d.ts +2 -0
  62. package/dist/config/index.js +19 -0
  63. package/dist/config/types.d.ts +72 -0
  64. package/dist/config/types.js +15 -0
  65. package/dist/diff/DiffEngine.d.ts +7 -0
  66. package/dist/diff/DiffEngine.js +73 -0
  67. package/dist/diff/index.d.ts +2 -0
  68. package/dist/diff/index.js +21 -0
  69. package/dist/diff/types.d.ts +20 -0
  70. package/dist/diff/types.js +3 -0
  71. package/dist/integrations/GitLabIntegration.d.ts +7 -0
  72. package/dist/integrations/GitLabIntegration.js +45 -0
  73. package/dist/integrations/index.d.ts +2 -0
  74. package/dist/integrations/index.js +21 -0
  75. package/dist/integrations/types.d.ts +11 -0
  76. package/dist/integrations/types.js +13 -0
  77. package/dist/parser/TemplateParser.d.ts +8 -0
  78. package/dist/parser/TemplateParser.js +75 -0
  79. package/dist/parser/index.d.ts +2 -0
  80. package/dist/parser/index.js +22 -0
  81. package/dist/parser/types.d.ts +30 -0
  82. package/dist/parser/types.js +3 -0
  83. package/dist/pipeline/PipelineOrchestrator.d.ts +23 -0
  84. package/dist/pipeline/PipelineOrchestrator.js +191 -0
  85. package/dist/pipeline/index.d.ts +2 -0
  86. package/dist/pipeline/index.js +19 -0
  87. package/dist/pipeline/types.d.ts +41 -0
  88. package/dist/pipeline/types.js +13 -0
  89. package/dist/pricing/CacheManager.d.ts +75 -0
  90. package/dist/pricing/CacheManager.js +195 -0
  91. package/dist/pricing/PricingClient.d.ts +17 -0
  92. package/dist/pricing/PricingClient.js +122 -0
  93. package/dist/pricing/PricingService.d.ts +16 -0
  94. package/dist/pricing/PricingService.js +149 -0
  95. package/dist/pricing/calculators/ALBCalculator.d.ts +16 -0
  96. package/dist/pricing/calculators/ALBCalculator.js +163 -0
  97. package/dist/pricing/calculators/APIGatewayCalculator.d.ts +10 -0
  98. package/dist/pricing/calculators/APIGatewayCalculator.js +177 -0
  99. package/dist/pricing/calculators/CloudFrontCalculator.d.ts +59 -0
  100. package/dist/pricing/calculators/CloudFrontCalculator.js +151 -0
  101. package/dist/pricing/calculators/DynamoDBCalculator.d.ts +9 -0
  102. package/dist/pricing/calculators/DynamoDBCalculator.js +146 -0
  103. package/dist/pricing/calculators/EC2Calculator.d.ts +7 -0
  104. package/dist/pricing/calculators/EC2Calculator.js +80 -0
  105. package/dist/pricing/calculators/ECSCalculator.d.ts +9 -0
  106. package/dist/pricing/calculators/ECSCalculator.js +116 -0
  107. package/dist/pricing/calculators/ElastiCacheCalculator.d.ts +8 -0
  108. package/dist/pricing/calculators/ElastiCacheCalculator.js +106 -0
  109. package/dist/pricing/calculators/LambdaCalculator.d.ts +13 -0
  110. package/dist/pricing/calculators/LambdaCalculator.js +111 -0
  111. package/dist/pricing/calculators/NLBCalculator.d.ts +16 -0
  112. package/dist/pricing/calculators/NLBCalculator.js +138 -0
  113. package/dist/pricing/calculators/NatGatewayCalculator.d.ts +12 -0
  114. package/dist/pricing/calculators/NatGatewayCalculator.js +116 -0
  115. package/dist/pricing/calculators/RDSCalculator.d.ts +9 -0
  116. package/dist/pricing/calculators/RDSCalculator.js +103 -0
  117. package/dist/pricing/calculators/S3Calculator.d.ts +8 -0
  118. package/dist/pricing/calculators/S3Calculator.js +68 -0
  119. package/dist/pricing/calculators/VPCEndpointCalculator.d.ts +12 -0
  120. package/dist/pricing/calculators/VPCEndpointCalculator.js +129 -0
  121. package/dist/pricing/index.d.ts +10 -0
  122. package/dist/pricing/index.js +37 -0
  123. package/dist/pricing/types.d.ts +53 -0
  124. package/dist/pricing/types.js +22 -0
  125. package/dist/releasetag.txt +1 -0
  126. package/dist/reporter/Reporter.d.ts +18 -0
  127. package/dist/reporter/Reporter.js +412 -0
  128. package/dist/reporter/index.d.ts +2 -0
  129. package/dist/reporter/index.js +21 -0
  130. package/dist/reporter/types.d.ts +72 -0
  131. package/dist/reporter/types.js +3 -0
  132. package/dist/synthesis/SynthesisOrchestrator.d.ts +26 -0
  133. package/dist/synthesis/SynthesisOrchestrator.js +243 -0
  134. package/dist/synthesis/index.d.ts +2 -0
  135. package/dist/synthesis/index.js +19 -0
  136. package/dist/synthesis/types.d.ts +17 -0
  137. package/dist/synthesis/types.js +13 -0
  138. package/dist/threshold/ThresholdEnforcer.d.ts +29 -0
  139. package/dist/threshold/ThresholdEnforcer.js +143 -0
  140. package/dist/threshold/index.d.ts +2 -0
  141. package/dist/threshold/index.js +19 -0
  142. package/dist/threshold/types.d.ts +15 -0
  143. package/dist/threshold/types.js +17 -0
  144. package/docs/CALCULATORS.md +820 -0
  145. package/docs/CI_CD.md +608 -0
  146. package/docs/CONFIGURATION.md +407 -0
  147. package/docs/DEVELOPMENT.md +387 -0
  148. package/docs/RELEASE.md +223 -0
  149. package/docs/TROUBLESHOOTING.md +847 -0
  150. package/examples/.cdk-cost-analyzer.yml +85 -0
  151. package/examples/.gitlab-ci.yml +125 -0
  152. package/examples/api-usage.js +26 -0
  153. package/examples/complex/base.json +16 -0
  154. package/examples/complex/target.json +29 -0
  155. package/examples/monorepo/.gitlab-ci.yml +251 -0
  156. package/examples/monorepo/README.md +341 -0
  157. package/examples/monorepo/package.json +27 -0
  158. package/examples/monorepo/packages/backend-infra/.cdk-cost-analyzer.yml +34 -0
  159. package/examples/monorepo/packages/backend-infra/bin/app.ts +16 -0
  160. package/examples/monorepo/packages/backend-infra/cdk.json +7 -0
  161. package/examples/monorepo/packages/backend-infra/lib/backend-stack.ts +128 -0
  162. package/examples/monorepo/packages/backend-infra/package.json +30 -0
  163. package/examples/monorepo/packages/backend-infra/tsconfig.json +11 -0
  164. package/examples/monorepo/packages/data-infra/.cdk-cost-analyzer.yml +38 -0
  165. package/examples/monorepo/packages/data-infra/bin/app.ts +16 -0
  166. package/examples/monorepo/packages/data-infra/cdk.json +7 -0
  167. package/examples/monorepo/packages/data-infra/lib/data-stack.ts +121 -0
  168. package/examples/monorepo/packages/data-infra/package.json +30 -0
  169. package/examples/monorepo/packages/data-infra/tsconfig.json +11 -0
  170. package/examples/monorepo/packages/frontend-infra/.cdk-cost-analyzer.yml +31 -0
  171. package/examples/monorepo/packages/frontend-infra/bin/app.ts +16 -0
  172. package/examples/monorepo/packages/frontend-infra/cdk.json +7 -0
  173. package/examples/monorepo/packages/frontend-infra/lib/frontend-stack.ts +60 -0
  174. package/examples/monorepo/packages/frontend-infra/package.json +30 -0
  175. package/examples/monorepo/packages/frontend-infra/tsconfig.json +11 -0
  176. package/examples/monorepo/tsconfig.json +35 -0
  177. package/examples/multi-stack/.cdk-cost-analyzer.yml +72 -0
  178. package/examples/multi-stack/.gitlab-ci.yml +184 -0
  179. package/examples/multi-stack/README.md +279 -0
  180. package/examples/multi-stack/bin/app.ts +36 -0
  181. package/examples/multi-stack/cdk.json +72 -0
  182. package/examples/multi-stack/lib/compute-stack.ts +128 -0
  183. package/examples/multi-stack/lib/networking-stack.ts +69 -0
  184. package/examples/multi-stack/lib/storage-stack.ts +141 -0
  185. package/examples/multi-stack/package-lock.json +4437 -0
  186. package/examples/multi-stack/package.json +42 -0
  187. package/examples/multi-stack/tsconfig.json +34 -0
  188. package/examples/simple/base.json +8 -0
  189. package/examples/simple/target.json +14 -0
  190. package/examples/single-stack/.NVP +0 -0
  191. package/examples/single-stack/.cdk-cost-analyzer.yml +52 -0
  192. package/examples/single-stack/.gitlab-ci.yml +126 -0
  193. package/examples/single-stack/README.md +184 -0
  194. package/examples/single-stack/UeK +0 -0
  195. package/examples/single-stack/bin/app.ts +16 -0
  196. package/examples/single-stack/cdk.json +72 -0
  197. package/examples/single-stack/lib/infrastructure-stack.ts +119 -0
  198. package/examples/single-stack/package-lock.json +4443 -0
  199. package/examples/single-stack/package.json +38 -0
  200. package/examples/single-stack/tsconfig.json +34 -0
  201. package/package.json +139 -0
  202. package/test-cdk-project/README-COMPUTE.md +141 -0
  203. package/test-cdk-project/README.md +95 -0
  204. package/test-cdk-project/app-with-compute.js +102 -0
  205. package/test-cdk-project/app.js +81 -0
  206. package/test-cdk-project/cdk-compute.json +3 -0
  207. package/test-cdk-project/cdk.context.json +7 -0
  208. package/test-cdk-project/cdk.json +3 -0
  209. package/test-cdk-project/cdk.out/TestStack.assets.json +21 -0
  210. package/test-cdk-project/cdk.out/TestStack.template.json +115 -0
  211. package/test-cdk-project/cdk.out/cdk.out +1 -0
  212. package/test-cdk-project/cdk.out/manifest.json +503 -0
  213. package/test-cdk-project/cdk.out/tree.json +1 -0
  214. package/test-cdk-project/cdk.out.base/TestStack.assets.json +21 -0
  215. package/test-cdk-project/cdk.out.base/TestStack.template.json +115 -0
  216. package/test-cdk-project/cdk.out.base/cdk.out +1 -0
  217. package/test-cdk-project/cdk.out.base/manifest.json +503 -0
  218. package/test-cdk-project/cdk.out.base/tree.json +1 -0
  219. package/test-cdk-project/cdk.out.target/TestStack.assets.json +21 -0
  220. package/test-cdk-project/cdk.out.target/TestStack.template.json +183 -0
  221. package/test-cdk-project/cdk.out.target/cdk.out +1 -0
  222. package/test-cdk-project/cdk.out.target/manifest.json +521 -0
  223. package/test-cdk-project/cdk.out.target/tree.json +1 -0
  224. package/test-cdk-project/package-lock.json +422 -0
  225. package/test-cdk-project/package.json +17 -0
  226. package/tools/workflows/README.md +102 -0
  227. package/tools/workflows/validate-workflows.js +109 -0
  228. 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":"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","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,2 @@
1
+ export { Reporter } from './Reporter';
2
+ export * from './types';
@@ -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
+ }