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,3 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvYXBpL3R5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBSZXNvdXJjZUNvc3QsIE1vZGlmaWVkUmVzb3VyY2VDb3N0IH0gZnJvbSAnLi4vcHJpY2luZy90eXBlcyc7XG5cbmV4cG9ydCBpbnRlcmZhY2UgQW5hbHl6ZU9wdGlvbnMge1xuICBiYXNlVGVtcGxhdGU6IHN0cmluZztcbiAgdGFyZ2V0VGVtcGxhdGU6IHN0cmluZztcbiAgcmVnaW9uPzogc3RyaW5nO1xuICBmb3JtYXQ/OiAndGV4dCcgfCAnanNvbicgfCAnbWFya2Rvd24nO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIENvc3RBbmFseXNpc1Jlc3VsdCB7XG4gIHRvdGFsRGVsdGE6IG51bWJlcjtcbiAgY3VycmVuY3k6IHN0cmluZztcbiAgYWRkZWRSZXNvdXJjZXM6IFJlc291cmNlQ29zdFtdO1xuICByZW1vdmVkUmVzb3VyY2VzOiBSZXNvdXJjZUNvc3RbXTtcbiAgbW9kaWZpZWRSZXNvdXJjZXM6IE1vZGlmaWVkUmVzb3VyY2VDb3N0W107XG4gIHN1bW1hcnk6IHN0cmluZztcbn1cbiJdfQ==
@@ -0,0 +1,2 @@
1
+ #!/usr/bin/env node
2
+ export {};
@@ -0,0 +1,262 @@
1
+ #!/usr/bin/env node
2
+ "use strict";
3
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
4
+ if (k2 === undefined) k2 = k;
5
+ var desc = Object.getOwnPropertyDescriptor(m, k);
6
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
7
+ desc = { enumerable: true, get: function() { return m[k]; } };
8
+ }
9
+ Object.defineProperty(o, k2, desc);
10
+ }) : (function(o, m, k, k2) {
11
+ if (k2 === undefined) k2 = k;
12
+ o[k2] = m[k];
13
+ }));
14
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
15
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
16
+ }) : function(o, v) {
17
+ o["default"] = v;
18
+ });
19
+ var __importStar = (this && this.__importStar) || (function () {
20
+ var ownKeys = function(o) {
21
+ ownKeys = Object.getOwnPropertyNames || function (o) {
22
+ var ar = [];
23
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
24
+ return ar;
25
+ };
26
+ return ownKeys(o);
27
+ };
28
+ return function (mod) {
29
+ if (mod && mod.__esModule) return mod;
30
+ var result = {};
31
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
32
+ __setModuleDefault(result, mod);
33
+ return result;
34
+ };
35
+ })();
36
+ Object.defineProperty(exports, "__esModule", { value: true });
37
+ const fs = __importStar(require("fs"));
38
+ const commander_1 = require("commander");
39
+ const api_1 = require("../api");
40
+ const GitLabIntegration_1 = require("../integrations/GitLabIntegration");
41
+ const PipelineOrchestrator_1 = require("../pipeline/PipelineOrchestrator");
42
+ const program = new commander_1.Command();
43
+ program
44
+ .name('cdk-cost-analyzer')
45
+ .description('Analyze AWS CDK infrastructure changes and provide cost impact summaries')
46
+ .version('1.0.0');
47
+ // Original command for direct template comparison
48
+ program
49
+ .command('compare')
50
+ .description('Compare two CloudFormation templates')
51
+ .argument('<base>', 'Path to base CloudFormation template')
52
+ .argument('<target>', 'Path to target CloudFormation template')
53
+ .option('--region <region>', 'AWS region', 'eu-central-1')
54
+ .option('--format <format>', 'Output format: text|json|markdown', 'text')
55
+ .option('--config <path>', 'Path to configuration file')
56
+ .action(async (basePath, targetPath, options) => {
57
+ try {
58
+ if (!fs.existsSync(basePath)) {
59
+ console.error(`Error: Base template file not found: ${basePath}`);
60
+ process.exit(1);
61
+ }
62
+ if (!fs.existsSync(targetPath)) {
63
+ console.error(`Error: Target template file not found: ${targetPath}`);
64
+ process.exit(1);
65
+ }
66
+ const baseTemplate = fs.readFileSync(basePath, 'utf-8');
67
+ const targetTemplate = fs.readFileSync(targetPath, 'utf-8');
68
+ const result = await (0, api_1.analyzeCosts)({
69
+ baseTemplate,
70
+ targetTemplate,
71
+ region: options.region,
72
+ format: options.format,
73
+ });
74
+ if (options.format === 'json') {
75
+ console.log(JSON.stringify(result, null, 2));
76
+ }
77
+ else {
78
+ console.log(result.summary);
79
+ }
80
+ process.exit(0);
81
+ }
82
+ catch (error) {
83
+ if (error instanceof Error) {
84
+ console.error(`Error: ${error.message}`);
85
+ }
86
+ else {
87
+ console.error(`Error: ${String(error)}`);
88
+ }
89
+ process.exit(1);
90
+ }
91
+ });
92
+ // New pipeline command for CI/CD integration with synthesis
93
+ program
94
+ .command('pipeline')
95
+ .description('Run cost analysis in CI/CD pipeline with automatic synthesis')
96
+ .option('--base <path>', 'Path to base template (if not using synthesis)')
97
+ .option('--target <path>', 'Path to target template (if not using synthesis)')
98
+ .option('--synth', 'Enable automatic CDK synthesis')
99
+ .option('--cdk-app-path <path>', 'Path to CDK application directory')
100
+ .option('--region <region>', 'AWS region', 'eu-central-1')
101
+ .option('--config <path>', 'Path to configuration file')
102
+ .option('--environment <env>', 'Environment name for threshold selection')
103
+ .option('--format <format>', 'Output format: text|json|markdown', 'text')
104
+ .option('--post-to-gitlab', 'Post results to GitLab merge request')
105
+ .action(async (options) => {
106
+ try {
107
+ // Check AWS credentials
108
+ if (!process.env.AWS_ACCESS_KEY_ID && !process.env.AWS_PROFILE) {
109
+ console.error('Error: AWS credentials not configured');
110
+ console.error('');
111
+ console.error('Please set AWS credentials using one of:');
112
+ console.error(' - AWS_ACCESS_KEY_ID and AWS_SECRET_ACCESS_KEY environment variables');
113
+ console.error(' - AWS_PROFILE environment variable');
114
+ console.error(' - AWS credentials file (~/.aws/credentials)');
115
+ console.error('');
116
+ console.error('For GitLab CI, configure AWS credentials in CI/CD variables:');
117
+ console.error(' Settings > CI/CD > Variables');
118
+ console.error(' Add: AWS_ACCESS_KEY_ID, AWS_SECRET_ACCESS_KEY, AWS_REGION');
119
+ process.exit(1);
120
+ }
121
+ const orchestrator = new PipelineOrchestrator_1.PipelineOrchestrator();
122
+ const result = await orchestrator.runPipelineAnalysis({
123
+ baseTemplate: options.base,
124
+ targetTemplate: options.target,
125
+ synthesize: options.synth,
126
+ cdkAppPath: options.cdkAppPath,
127
+ region: options.region,
128
+ configPath: options.config,
129
+ environment: options.environment,
130
+ });
131
+ // Display results
132
+ if (options.format === 'json') {
133
+ console.log(JSON.stringify(result, null, 2));
134
+ }
135
+ else {
136
+ console.log(result.costAnalysis.summary);
137
+ console.log('');
138
+ // Display threshold status
139
+ if (result.thresholdStatus.level !== 'none') {
140
+ console.log('='.repeat(80));
141
+ console.log('THRESHOLD STATUS');
142
+ console.log('='.repeat(80));
143
+ console.log(result.thresholdStatus.message);
144
+ console.log('');
145
+ if (result.thresholdStatus.recommendations.length > 0) {
146
+ console.log('Recommendations:');
147
+ result.thresholdStatus.recommendations.forEach((rec, i) => {
148
+ console.log(`${i + 1}. ${rec}`);
149
+ });
150
+ console.log('');
151
+ }
152
+ }
153
+ // Display config summary
154
+ if (result.configUsed.configPath || result.configUsed.thresholds) {
155
+ console.log('='.repeat(80));
156
+ console.log('CONFIGURATION');
157
+ console.log('='.repeat(80));
158
+ if (result.configUsed.configPath) {
159
+ console.log(`Config file: ${result.configUsed.configPath}`);
160
+ }
161
+ if (result.configUsed.thresholds) {
162
+ console.log(`Thresholds: warning=$${result.configUsed.thresholds.warning || 'none'}, error=$${result.configUsed.thresholds.error || 'none'}`);
163
+ if (result.configUsed.thresholds.environment) {
164
+ console.log(`Environment: ${result.configUsed.thresholds.environment}`);
165
+ }
166
+ }
167
+ if (result.configUsed.excludedResourceTypes?.length) {
168
+ console.log(`Excluded resources: ${result.configUsed.excludedResourceTypes.join(', ')}`);
169
+ }
170
+ console.log('');
171
+ }
172
+ }
173
+ // Post to GitLab if requested
174
+ if (options.postToGitlab) {
175
+ const gitlabToken = process.env.GITLAB_TOKEN || process.env.CI_JOB_TOKEN;
176
+ const projectId = process.env.CI_PROJECT_ID;
177
+ const mergeRequestIid = process.env.CI_MERGE_REQUEST_IID;
178
+ const gitlabUrl = process.env.CI_SERVER_URL;
179
+ if (!gitlabToken || !projectId || !mergeRequestIid || !gitlabUrl) {
180
+ console.error('Warning: GitLab environment variables not found. Skipping GitLab post.');
181
+ console.error('Required: CI_JOB_TOKEN, CI_PROJECT_ID, CI_MERGE_REQUEST_IID, CI_SERVER_URL');
182
+ }
183
+ else {
184
+ try {
185
+ const gitlab = new GitLabIntegration_1.GitLabIntegration({
186
+ token: gitlabToken,
187
+ apiUrl: gitlabUrl,
188
+ });
189
+ await gitlab.postMergeRequestComment(projectId, mergeRequestIid, result.costAnalysis.summary);
190
+ console.log('Results posted to GitLab merge request');
191
+ }
192
+ catch (error) {
193
+ console.error(`Warning: Failed to post to GitLab: ${error instanceof Error ? error.message : String(error)}`);
194
+ }
195
+ }
196
+ }
197
+ // Exit with appropriate code based on threshold
198
+ if (result.thresholdStatus.level === 'error' && !result.thresholdStatus.passed) {
199
+ console.error('Pipeline failed: Cost threshold exceeded');
200
+ process.exit(2);
201
+ }
202
+ process.exit(0);
203
+ }
204
+ catch (error) {
205
+ if (error instanceof Error) {
206
+ console.error(`Error: ${error.message}`);
207
+ }
208
+ else {
209
+ console.error(`Error: ${String(error)}`);
210
+ }
211
+ process.exit(1);
212
+ }
213
+ });
214
+ // Default to compare command for backward compatibility
215
+ program
216
+ .argument('[base]', 'Path to base CloudFormation template')
217
+ .argument('[target]', 'Path to target CloudFormation template')
218
+ .option('--region <region>', 'AWS region', 'eu-central-1')
219
+ .option('--format <format>', 'Output format: text|json|markdown', 'text')
220
+ .option('--config <path>', 'Path to configuration file')
221
+ .action(async (basePath, targetPath, options) => {
222
+ if (!basePath || !targetPath) {
223
+ program.help();
224
+ return;
225
+ }
226
+ try {
227
+ if (!fs.existsSync(basePath)) {
228
+ console.error(`Error: Base template file not found: ${basePath}`);
229
+ process.exit(1);
230
+ }
231
+ if (!fs.existsSync(targetPath)) {
232
+ console.error(`Error: Target template file not found: ${targetPath}`);
233
+ process.exit(1);
234
+ }
235
+ const baseTemplate = fs.readFileSync(basePath, 'utf-8');
236
+ const targetTemplate = fs.readFileSync(targetPath, 'utf-8');
237
+ const result = await (0, api_1.analyzeCosts)({
238
+ baseTemplate,
239
+ targetTemplate,
240
+ region: options?.region || 'eu-central-1',
241
+ format: options?.format || 'text',
242
+ });
243
+ if (options?.format === 'json') {
244
+ console.log(JSON.stringify(result, null, 2));
245
+ }
246
+ else {
247
+ console.log(result.summary);
248
+ }
249
+ process.exit(0);
250
+ }
251
+ catch (error) {
252
+ if (error instanceof Error) {
253
+ console.error(`Error: ${error.message}`);
254
+ }
255
+ else {
256
+ console.error(`Error: ${String(error)}`);
257
+ }
258
+ process.exit(1);
259
+ }
260
+ });
261
+ program.parse();
262
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/cli/index.ts"],"names":[],"mappings":"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","sourcesContent":["#!/usr/bin/env node\n\nimport * as fs from 'fs';\nimport { Command } from 'commander';\nimport { analyzeCosts } from '../api';\nimport { GitLabIntegration } from '../integrations/GitLabIntegration';\nimport { PipelineOrchestrator } from '../pipeline/PipelineOrchestrator';\n\nconst program = new Command();\n\nprogram\n  .name('cdk-cost-analyzer')\n  .description('Analyze AWS CDK infrastructure changes and provide cost impact summaries')\n  .version('1.0.0');\n\n// Original command for direct template comparison\nprogram\n  .command('compare')\n  .description('Compare two CloudFormation templates')\n  .argument('<base>', 'Path to base CloudFormation template')\n  .argument('<target>', 'Path to target CloudFormation template')\n  .option('--region <region>', 'AWS region', 'eu-central-1')\n  .option('--format <format>', 'Output format: text|json|markdown', 'text')\n  .option('--config <path>', 'Path to configuration file')\n  .action(async (basePath: string, targetPath: string, options: { region: string; format: string; config?: string }) => {\n    try {\n      if (!fs.existsSync(basePath)) {\n        console.error(`Error: Base template file not found: ${basePath}`);\n        process.exit(1);\n      }\n\n      if (!fs.existsSync(targetPath)) {\n        console.error(`Error: Target template file not found: ${targetPath}`);\n        process.exit(1);\n      }\n\n      const baseTemplate = fs.readFileSync(basePath, 'utf-8');\n      const targetTemplate = fs.readFileSync(targetPath, 'utf-8');\n\n      const result = await analyzeCosts({\n        baseTemplate,\n        targetTemplate,\n        region: options.region,\n        format: options.format as 'text' | 'json' | 'markdown',\n      });\n\n      if (options.format === 'json') {\n        console.log(JSON.stringify(result, null, 2));\n      } else {\n        console.log(result.summary);\n      }\n\n      process.exit(0);\n    } catch (error) {\n      if (error instanceof Error) {\n        console.error(`Error: ${error.message}`);\n      } else {\n        console.error(`Error: ${String(error)}`);\n      }\n      process.exit(1);\n    }\n  });\n\n// New pipeline command for CI/CD integration with synthesis\nprogram\n  .command('pipeline')\n  .description('Run cost analysis in CI/CD pipeline with automatic synthesis')\n  .option('--base <path>', 'Path to base template (if not using synthesis)')\n  .option('--target <path>', 'Path to target template (if not using synthesis)')\n  .option('--synth', 'Enable automatic CDK synthesis')\n  .option('--cdk-app-path <path>', 'Path to CDK application directory')\n  .option('--region <region>', 'AWS region', 'eu-central-1')\n  .option('--config <path>', 'Path to configuration file')\n  .option('--environment <env>', 'Environment name for threshold selection')\n  .option('--format <format>', 'Output format: text|json|markdown', 'text')\n  .option('--post-to-gitlab', 'Post results to GitLab merge request')\n  .action(async (options: {\n    base?: string;\n    target?: string;\n    synth?: boolean;\n    cdkAppPath?: string;\n    region: string;\n    config?: string;\n    environment?: string;\n    format: string;\n    postToGitlab?: boolean;\n  }) => {\n    try {\n      // Check AWS credentials\n      if (!process.env.AWS_ACCESS_KEY_ID && !process.env.AWS_PROFILE) {\n        console.error('Error: AWS credentials not configured');\n        console.error('');\n        console.error('Please set AWS credentials using one of:');\n        console.error('  - AWS_ACCESS_KEY_ID and AWS_SECRET_ACCESS_KEY environment variables');\n        console.error('  - AWS_PROFILE environment variable');\n        console.error('  - AWS credentials file (~/.aws/credentials)');\n        console.error('');\n        console.error('For GitLab CI, configure AWS credentials in CI/CD variables:');\n        console.error('  Settings > CI/CD > Variables');\n        console.error('  Add: AWS_ACCESS_KEY_ID, AWS_SECRET_ACCESS_KEY, AWS_REGION');\n        process.exit(1);\n      }\n\n      const orchestrator = new PipelineOrchestrator();\n\n      const result = await orchestrator.runPipelineAnalysis({\n        baseTemplate: options.base,\n        targetTemplate: options.target,\n        synthesize: options.synth,\n        cdkAppPath: options.cdkAppPath,\n        region: options.region,\n        configPath: options.config,\n        environment: options.environment,\n      });\n\n      // Display results\n      if (options.format === 'json') {\n        console.log(JSON.stringify(result, null, 2));\n      } else {\n        console.log(result.costAnalysis.summary);\n        console.log('');\n\n        // Display threshold status\n        if (result.thresholdStatus.level !== 'none') {\n          console.log('='.repeat(80));\n          console.log('THRESHOLD STATUS');\n          console.log('='.repeat(80));\n          console.log(result.thresholdStatus.message);\n          console.log('');\n\n          if (result.thresholdStatus.recommendations.length > 0) {\n            console.log('Recommendations:');\n            result.thresholdStatus.recommendations.forEach((rec, i) => {\n              console.log(`${i + 1}. ${rec}`);\n            });\n            console.log('');\n          }\n        }\n\n        // Display config summary\n        if (result.configUsed.configPath || result.configUsed.thresholds) {\n          console.log('='.repeat(80));\n          console.log('CONFIGURATION');\n          console.log('='.repeat(80));\n          if (result.configUsed.configPath) {\n            console.log(`Config file: ${result.configUsed.configPath}`);\n          }\n          if (result.configUsed.thresholds) {\n            console.log(`Thresholds: warning=$${result.configUsed.thresholds.warning || 'none'}, error=$${result.configUsed.thresholds.error || 'none'}`);\n            if (result.configUsed.thresholds.environment) {\n              console.log(`Environment: ${result.configUsed.thresholds.environment}`);\n            }\n          }\n          if (result.configUsed.excludedResourceTypes?.length) {\n            console.log(`Excluded resources: ${result.configUsed.excludedResourceTypes.join(', ')}`);\n          }\n          console.log('');\n        }\n      }\n\n      // Post to GitLab if requested\n      if (options.postToGitlab) {\n        const gitlabToken = process.env.GITLAB_TOKEN || process.env.CI_JOB_TOKEN;\n        const projectId = process.env.CI_PROJECT_ID;\n        const mergeRequestIid = process.env.CI_MERGE_REQUEST_IID;\n        const gitlabUrl = process.env.CI_SERVER_URL;\n\n        if (!gitlabToken || !projectId || !mergeRequestIid || !gitlabUrl) {\n          console.error('Warning: GitLab environment variables not found. Skipping GitLab post.');\n          console.error('Required: CI_JOB_TOKEN, CI_PROJECT_ID, CI_MERGE_REQUEST_IID, CI_SERVER_URL');\n        } else {\n          try {\n            const gitlab = new GitLabIntegration({\n              token: gitlabToken,\n              apiUrl: gitlabUrl,\n            });\n            await gitlab.postMergeRequestComment(\n              projectId,\n              mergeRequestIid,\n              result.costAnalysis.summary,\n            );\n            console.log('Results posted to GitLab merge request');\n          } catch (error) {\n            console.error(`Warning: Failed to post to GitLab: ${error instanceof Error ? error.message : String(error)}`);\n          }\n        }\n      }\n\n      // Exit with appropriate code based on threshold\n      if (result.thresholdStatus.level === 'error' && !result.thresholdStatus.passed) {\n        console.error('Pipeline failed: Cost threshold exceeded');\n        process.exit(2);\n      }\n\n      process.exit(0);\n    } catch (error) {\n      if (error instanceof Error) {\n        console.error(`Error: ${error.message}`);\n      } else {\n        console.error(`Error: ${String(error)}`);\n      }\n      process.exit(1);\n    }\n  });\n\n// Default to compare command for backward compatibility\nprogram\n  .argument('[base]', 'Path to base CloudFormation template')\n  .argument('[target]', 'Path to target CloudFormation template')\n  .option('--region <region>', 'AWS region', 'eu-central-1')\n  .option('--format <format>', 'Output format: text|json|markdown', 'text')\n  .option('--config <path>', 'Path to configuration file')\n  .action(async (basePath?: string, targetPath?: string, options?: { region: string; format: string; config?: string }) => {\n    if (!basePath || !targetPath) {\n      program.help();\n      return;\n    }\n\n    try {\n      if (!fs.existsSync(basePath)) {\n        console.error(`Error: Base template file not found: ${basePath}`);\n        process.exit(1);\n      }\n\n      if (!fs.existsSync(targetPath)) {\n        console.error(`Error: Target template file not found: ${targetPath}`);\n        process.exit(1);\n      }\n\n      const baseTemplate = fs.readFileSync(basePath, 'utf-8');\n      const targetTemplate = fs.readFileSync(targetPath, 'utf-8');\n\n      const result = await analyzeCosts({\n        baseTemplate,\n        targetTemplate,\n        region: options?.region || 'eu-central-1',\n        format: (options?.format as 'text' | 'json' | 'markdown') || 'text',\n      });\n\n      if (options?.format === 'json') {\n        console.log(JSON.stringify(result, null, 2));\n      } else {\n        console.log(result.summary);\n      }\n\n      process.exit(0);\n    } catch (error) {\n      if (error instanceof Error) {\n        console.error(`Error: ${error.message}`);\n      } else {\n        console.error(`Error: ${String(error)}`);\n      }\n      process.exit(1);\n    }\n  });\n\nprogram.parse();\n"]}
@@ -0,0 +1,40 @@
1
+ import { CostAnalyzerConfig, ValidationResult } from './types';
2
+ export declare class ConfigManager {
3
+ private static readonly CONFIG_FILE_NAMES;
4
+ /**
5
+ * Load configuration from file or return default configuration
6
+ */
7
+ loadConfig(configPath?: string): Promise<CostAnalyzerConfig>;
8
+ /**
9
+ * Validate configuration structure and values
10
+ */
11
+ validateConfig(config: CostAnalyzerConfig): ValidationResult;
12
+ /**
13
+ * Resolve configuration file path using search order
14
+ */
15
+ private resolveConfigPath;
16
+ /**
17
+ * Read and parse configuration file (YAML or JSON)
18
+ */
19
+ private readConfigFile;
20
+ /**
21
+ * Check if file exists
22
+ */
23
+ private fileExists;
24
+ /**
25
+ * Validate threshold configuration
26
+ */
27
+ private validateThresholds;
28
+ /**
29
+ * Validate usage assumptions
30
+ */
31
+ private validateUsageAssumptions;
32
+ /**
33
+ * Get default configuration
34
+ */
35
+ private getDefaultConfig;
36
+ /**
37
+ * Merge user configuration with defaults
38
+ */
39
+ private mergeWithDefaults;
40
+ }
@@ -0,0 +1,238 @@
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 __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
35
+ Object.defineProperty(exports, "__esModule", { value: true });
36
+ exports.ConfigManager = void 0;
37
+ const fs = __importStar(require("fs/promises"));
38
+ const path = __importStar(require("path"));
39
+ const yaml = __importStar(require("js-yaml"));
40
+ const types_1 = require("./types");
41
+ class ConfigManager {
42
+ static CONFIG_FILE_NAMES = [
43
+ '.cdk-cost-analyzer.yml',
44
+ '.cdk-cost-analyzer.yaml',
45
+ '.cdk-cost-analyzer.json',
46
+ ];
47
+ /**
48
+ * Load configuration from file or return default configuration
49
+ */
50
+ async loadConfig(configPath) {
51
+ try {
52
+ const resolvedPath = await this.resolveConfigPath(configPath);
53
+ if (!resolvedPath) {
54
+ return this.getDefaultConfig();
55
+ }
56
+ const config = await this.readConfigFile(resolvedPath);
57
+ const validation = this.validateConfig(config);
58
+ if (!validation.valid) {
59
+ throw new types_1.ConfigurationError('Invalid configuration', resolvedPath, validation.errors);
60
+ }
61
+ return this.mergeWithDefaults(config);
62
+ }
63
+ catch (error) {
64
+ if (error instanceof types_1.ConfigurationError) {
65
+ throw error;
66
+ }
67
+ throw new types_1.ConfigurationError(`Failed to load configuration: ${error instanceof Error ? error.message : String(error)}`, configPath || 'unknown', []);
68
+ }
69
+ }
70
+ /**
71
+ * Validate configuration structure and values
72
+ */
73
+ validateConfig(config) {
74
+ const errors = [];
75
+ const warnings = [];
76
+ if (config.thresholds) {
77
+ this.validateThresholds(config.thresholds, errors, warnings);
78
+ }
79
+ if (config.usageAssumptions) {
80
+ this.validateUsageAssumptions(config.usageAssumptions, errors);
81
+ }
82
+ if (config.cache?.durationHours !== undefined) {
83
+ if (config.cache.durationHours <= 0) {
84
+ errors.push('cache.durationHours must be positive');
85
+ }
86
+ }
87
+ return {
88
+ valid: errors.length === 0,
89
+ errors,
90
+ warnings,
91
+ };
92
+ }
93
+ /**
94
+ * Resolve configuration file path using search order
95
+ */
96
+ async resolveConfigPath(configPath) {
97
+ if (configPath) {
98
+ const exists = await this.fileExists(configPath);
99
+ if (!exists) {
100
+ throw new Error(`Configuration file not found: ${configPath}`);
101
+ }
102
+ return configPath;
103
+ }
104
+ // Search in current directory
105
+ for (const fileName of ConfigManager.CONFIG_FILE_NAMES) {
106
+ const filePath = path.join(process.cwd(), fileName);
107
+ if (await this.fileExists(filePath)) {
108
+ return filePath;
109
+ }
110
+ }
111
+ return null;
112
+ }
113
+ /**
114
+ * Read and parse configuration file (YAML or JSON)
115
+ */
116
+ async readConfigFile(filePath) {
117
+ const content = await fs.readFile(filePath, 'utf-8');
118
+ const ext = path.extname(filePath).toLowerCase();
119
+ try {
120
+ if (ext === '.json') {
121
+ return JSON.parse(content);
122
+ }
123
+ else {
124
+ return yaml.load(content);
125
+ }
126
+ }
127
+ catch (error) {
128
+ throw new Error(`Failed to parse configuration file: ${error instanceof Error ? error.message : String(error)}`);
129
+ }
130
+ }
131
+ /**
132
+ * Check if file exists
133
+ */
134
+ async fileExists(filePath) {
135
+ try {
136
+ await fs.access(filePath);
137
+ return true;
138
+ }
139
+ catch {
140
+ return false;
141
+ }
142
+ }
143
+ /**
144
+ * Validate threshold configuration
145
+ */
146
+ validateThresholds(thresholds, errors, warnings) {
147
+ const validateLevels = (levels, prefix) => {
148
+ if (levels.warning !== undefined && levels.warning < 0) {
149
+ errors.push(`${prefix}.warning must be non-negative`);
150
+ }
151
+ if (levels.error !== undefined && levels.error < 0) {
152
+ errors.push(`${prefix}.error must be non-negative`);
153
+ }
154
+ if (levels.warning !== undefined &&
155
+ levels.error !== undefined &&
156
+ levels.warning > levels.error) {
157
+ warnings.push(`${prefix}.warning (${levels.warning}) is greater than ${prefix}.error (${levels.error})`);
158
+ }
159
+ };
160
+ if (thresholds.default) {
161
+ validateLevels(thresholds.default, 'thresholds.default');
162
+ }
163
+ if (thresholds.environments) {
164
+ for (const [env, levels] of Object.entries(thresholds.environments)) {
165
+ validateLevels(levels, `thresholds.environments.${env}`);
166
+ }
167
+ }
168
+ }
169
+ /**
170
+ * Validate usage assumptions
171
+ */
172
+ validateUsageAssumptions(assumptions, errors) {
173
+ const validatePositive = (value, configPath) => {
174
+ if (value !== undefined && value < 0) {
175
+ errors.push(`${configPath} must be non-negative`);
176
+ }
177
+ };
178
+ if (assumptions.s3) {
179
+ validatePositive(assumptions.s3.storageGB, 'usageAssumptions.s3.storageGB');
180
+ validatePositive(assumptions.s3.getRequests, 'usageAssumptions.s3.getRequests');
181
+ validatePositive(assumptions.s3.putRequests, 'usageAssumptions.s3.putRequests');
182
+ }
183
+ if (assumptions.lambda) {
184
+ validatePositive(assumptions.lambda.invocationsPerMonth, 'usageAssumptions.lambda.invocationsPerMonth');
185
+ validatePositive(assumptions.lambda.averageDurationMs, 'usageAssumptions.lambda.averageDurationMs');
186
+ }
187
+ if (assumptions.natGateway) {
188
+ validatePositive(assumptions.natGateway.dataProcessedGB, 'usageAssumptions.natGateway.dataProcessedGB');
189
+ }
190
+ if (assumptions.alb) {
191
+ validatePositive(assumptions.alb.newConnectionsPerSecond, 'usageAssumptions.alb.newConnectionsPerSecond');
192
+ validatePositive(assumptions.alb.activeConnectionsPerMinute, 'usageAssumptions.alb.activeConnectionsPerMinute');
193
+ validatePositive(assumptions.alb.processedBytesGB, 'usageAssumptions.alb.processedBytesGB');
194
+ }
195
+ if (assumptions.nlb) {
196
+ validatePositive(assumptions.nlb.newConnectionsPerSecond, 'usageAssumptions.nlb.newConnectionsPerSecond');
197
+ validatePositive(assumptions.nlb.activeConnectionsPerMinute, 'usageAssumptions.nlb.activeConnectionsPerMinute');
198
+ validatePositive(assumptions.nlb.processedBytesGB, 'usageAssumptions.nlb.processedBytesGB');
199
+ }
200
+ if (assumptions.cloudfront) {
201
+ validatePositive(assumptions.cloudfront.dataTransferGB, 'usageAssumptions.cloudfront.dataTransferGB');
202
+ validatePositive(assumptions.cloudfront.requests, 'usageAssumptions.cloudfront.requests');
203
+ }
204
+ if (assumptions.apiGateway) {
205
+ validatePositive(assumptions.apiGateway.requestsPerMonth, 'usageAssumptions.apiGateway.requestsPerMonth');
206
+ }
207
+ if (assumptions.vpcEndpoint) {
208
+ validatePositive(assumptions.vpcEndpoint.dataProcessedGB, 'usageAssumptions.vpcEndpoint.dataProcessedGB');
209
+ }
210
+ }
211
+ /**
212
+ * Get default configuration
213
+ */
214
+ getDefaultConfig() {
215
+ return {
216
+ cache: {
217
+ enabled: true,
218
+ durationHours: 24,
219
+ },
220
+ };
221
+ }
222
+ /**
223
+ * Merge user configuration with defaults
224
+ */
225
+ mergeWithDefaults(config) {
226
+ const defaults = this.getDefaultConfig();
227
+ return {
228
+ ...defaults,
229
+ ...config,
230
+ cache: {
231
+ ...defaults.cache,
232
+ ...config.cache,
233
+ },
234
+ };
235
+ }
236
+ }
237
+ exports.ConfigManager = ConfigManager;
238
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ConfigManager.js","sourceRoot":"","sources":["../../src/config/ConfigManager.ts"],"names":[],"mappings":"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","sourcesContent":["import * as fs from 'fs/promises';\nimport * as path from 'path';\nimport * as yaml from 'js-yaml';\nimport {\n  CostAnalyzerConfig,\n  ValidationResult,\n  ConfigurationError,\n  ThresholdLevels,\n} from './types';\n\nexport class ConfigManager {\n  private static readonly CONFIG_FILE_NAMES = [\n    '.cdk-cost-analyzer.yml',\n    '.cdk-cost-analyzer.yaml',\n    '.cdk-cost-analyzer.json',\n  ];\n\n  /**\n   * Load configuration from file or return default configuration\n   */\n  async loadConfig(configPath?: string): Promise<CostAnalyzerConfig> {\n    try {\n      const resolvedPath = await this.resolveConfigPath(configPath);\n\n      if (!resolvedPath) {\n        return this.getDefaultConfig();\n      }\n\n      const config = await this.readConfigFile(resolvedPath);\n      const validation = this.validateConfig(config);\n\n      if (!validation.valid) {\n        throw new ConfigurationError(\n          'Invalid configuration',\n          resolvedPath,\n          validation.errors,\n        );\n      }\n\n      return this.mergeWithDefaults(config);\n    } catch (error) {\n      if (error instanceof ConfigurationError) {\n        throw error;\n      }\n      throw new ConfigurationError(\n        `Failed to load configuration: ${error instanceof Error ? error.message : String(error)}`,\n        configPath || 'unknown',\n        [],\n      );\n    }\n  }\n\n  /**\n   * Validate configuration structure and values\n   */\n  validateConfig(config: CostAnalyzerConfig): ValidationResult {\n    const errors: string[] = [];\n    const warnings: string[] = [];\n\n    if (config.thresholds) {\n      this.validateThresholds(config.thresholds, errors, warnings);\n    }\n\n    if (config.usageAssumptions) {\n      this.validateUsageAssumptions(config.usageAssumptions, errors);\n    }\n\n    if (config.cache?.durationHours !== undefined) {\n      if (config.cache.durationHours <= 0) {\n        errors.push('cache.durationHours must be positive');\n      }\n    }\n\n    return {\n      valid: errors.length === 0,\n      errors,\n      warnings,\n    };\n  }\n\n  /**\n   * Resolve configuration file path using search order\n   */\n  private async resolveConfigPath(configPath?: string): Promise<string | null> {\n    if (configPath) {\n      const exists = await this.fileExists(configPath);\n      if (!exists) {\n        throw new Error(`Configuration file not found: ${configPath}`);\n      }\n      return configPath;\n    }\n\n    // Search in current directory\n    for (const fileName of ConfigManager.CONFIG_FILE_NAMES) {\n      const filePath = path.join(process.cwd(), fileName);\n      if (await this.fileExists(filePath)) {\n        return filePath;\n      }\n    }\n\n    return null;\n  }\n\n  /**\n   * Read and parse configuration file (YAML or JSON)\n   */\n  private async readConfigFile(filePath: string): Promise<CostAnalyzerConfig> {\n    const content = await fs.readFile(filePath, 'utf-8');\n    const ext = path.extname(filePath).toLowerCase();\n\n    try {\n      if (ext === '.json') {\n        return JSON.parse(content);\n      } else {\n        return yaml.load(content) as CostAnalyzerConfig;\n      }\n    } catch (error) {\n      throw new Error(\n        `Failed to parse configuration file: ${error instanceof Error ? error.message : String(error)}`,\n      );\n    }\n  }\n\n  /**\n   * Check if file exists\n   */\n  private async fileExists(filePath: string): Promise<boolean> {\n    try {\n      await fs.access(filePath);\n      return true;\n    } catch {\n      return false;\n    }\n  }\n\n  /**\n   * Validate threshold configuration\n   */\n  private validateThresholds(\n    thresholds: NonNullable<CostAnalyzerConfig['thresholds']>,\n    errors: string[],\n    warnings: string[],\n  ): void {\n    const validateLevels = (levels: ThresholdLevels, prefix: string) => {\n      if (levels.warning !== undefined && levels.warning < 0) {\n        errors.push(`${prefix}.warning must be non-negative`);\n      }\n      if (levels.error !== undefined && levels.error < 0) {\n        errors.push(`${prefix}.error must be non-negative`);\n      }\n      if (\n        levels.warning !== undefined &&\n        levels.error !== undefined &&\n        levels.warning > levels.error\n      ) {\n        warnings.push(\n          `${prefix}.warning (${levels.warning}) is greater than ${prefix}.error (${levels.error})`,\n        );\n      }\n    };\n\n    if (thresholds.default) {\n      validateLevels(thresholds.default, 'thresholds.default');\n    }\n\n    if (thresholds.environments) {\n      for (const [env, levels] of Object.entries(thresholds.environments)) {\n        validateLevels(levels, `thresholds.environments.${env}`);\n      }\n    }\n  }\n\n  /**\n   * Validate usage assumptions\n   */\n  private validateUsageAssumptions(\n    assumptions: NonNullable<CostAnalyzerConfig['usageAssumptions']>,\n    errors: string[],\n  ): void {\n    const validatePositive = (value: number | undefined, configPath: string) => {\n      if (value !== undefined && value < 0) {\n        errors.push(`${configPath} must be non-negative`);\n      }\n    };\n\n    if (assumptions.s3) {\n      validatePositive(assumptions.s3.storageGB, 'usageAssumptions.s3.storageGB');\n      validatePositive(assumptions.s3.getRequests, 'usageAssumptions.s3.getRequests');\n      validatePositive(assumptions.s3.putRequests, 'usageAssumptions.s3.putRequests');\n    }\n\n    if (assumptions.lambda) {\n      validatePositive(\n        assumptions.lambda.invocationsPerMonth,\n        'usageAssumptions.lambda.invocationsPerMonth',\n      );\n      validatePositive(\n        assumptions.lambda.averageDurationMs,\n        'usageAssumptions.lambda.averageDurationMs',\n      );\n    }\n\n    if (assumptions.natGateway) {\n      validatePositive(\n        assumptions.natGateway.dataProcessedGB,\n        'usageAssumptions.natGateway.dataProcessedGB',\n      );\n    }\n\n    if (assumptions.alb) {\n      validatePositive(\n        assumptions.alb.newConnectionsPerSecond,\n        'usageAssumptions.alb.newConnectionsPerSecond',\n      );\n      validatePositive(\n        assumptions.alb.activeConnectionsPerMinute,\n        'usageAssumptions.alb.activeConnectionsPerMinute',\n      );\n      validatePositive(\n        assumptions.alb.processedBytesGB,\n        'usageAssumptions.alb.processedBytesGB',\n      );\n    }\n\n    if (assumptions.nlb) {\n      validatePositive(\n        assumptions.nlb.newConnectionsPerSecond,\n        'usageAssumptions.nlb.newConnectionsPerSecond',\n      );\n      validatePositive(\n        assumptions.nlb.activeConnectionsPerMinute,\n        'usageAssumptions.nlb.activeConnectionsPerMinute',\n      );\n      validatePositive(\n        assumptions.nlb.processedBytesGB,\n        'usageAssumptions.nlb.processedBytesGB',\n      );\n    }\n\n    if (assumptions.cloudfront) {\n      validatePositive(\n        assumptions.cloudfront.dataTransferGB,\n        'usageAssumptions.cloudfront.dataTransferGB',\n      );\n      validatePositive(\n        assumptions.cloudfront.requests,\n        'usageAssumptions.cloudfront.requests',\n      );\n    }\n\n    if (assumptions.apiGateway) {\n      validatePositive(\n        assumptions.apiGateway.requestsPerMonth,\n        'usageAssumptions.apiGateway.requestsPerMonth',\n      );\n    }\n\n    if (assumptions.vpcEndpoint) {\n      validatePositive(\n        assumptions.vpcEndpoint.dataProcessedGB,\n        'usageAssumptions.vpcEndpoint.dataProcessedGB',\n      );\n    }\n  }\n\n  /**\n   * Get default configuration\n   */\n  private getDefaultConfig(): CostAnalyzerConfig {\n    return {\n      cache: {\n        enabled: true,\n        durationHours: 24,\n      },\n    };\n  }\n\n  /**\n   * Merge user configuration with defaults\n   */\n  private mergeWithDefaults(config: CostAnalyzerConfig): CostAnalyzerConfig {\n    const defaults = this.getDefaultConfig();\n    return {\n      ...defaults,\n      ...config,\n      cache: {\n        ...defaults.cache,\n        ...config.cache,\n      },\n    };\n  }\n}\n"]}
@@ -0,0 +1,2 @@
1
+ export * from './ConfigManager';
2
+ export * from './types';
@@ -0,0 +1,19 @@
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
+ __exportStar(require("./ConfigManager"), exports);
18
+ __exportStar(require("./types"), exports);
19
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvY29uZmlnL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7QUFBQSxrREFBZ0M7QUFDaEMsMENBQXdCIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9Db25maWdNYW5hZ2VyJztcbmV4cG9ydCAqIGZyb20gJy4vdHlwZXMnO1xuIl19