cdk-cost-analyzer 0.1.41 → 0.1.43

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 (60) hide show
  1. package/.cdk-cost-analyzer-cache/metadata.json +8 -8
  2. package/dist/action/api/index.d.ts +1 -0
  3. package/dist/action/api/single-template-types.d.ts +4 -0
  4. package/dist/action/api/types.d.ts +3 -0
  5. package/dist/action/config/types.d.ts +2 -0
  6. package/dist/action/optimization/OptimizationEngine.d.ts +11 -0
  7. package/dist/action/optimization/analyzers/GravitonMigrationAnalyzer.d.ts +17 -0
  8. package/dist/action/optimization/analyzers/NATGatewayOptimizationAnalyzer.d.ts +13 -0
  9. package/dist/action/optimization/analyzers/ReservedInstanceAnalyzer.d.ts +13 -0
  10. package/dist/action/optimization/analyzers/RightSizingAnalyzer.d.ts +13 -0
  11. package/dist/action/optimization/analyzers/SavingsPlansAnalyzer.d.ts +11 -0
  12. package/dist/action/optimization/analyzers/SpotInstanceAnalyzer.d.ts +12 -0
  13. package/dist/action/optimization/analyzers/StorageOptimizationAnalyzer.d.ts +14 -0
  14. package/dist/action/optimization/defaults.d.ts +2 -0
  15. package/dist/action/optimization/index.d.ts +10 -0
  16. package/dist/action/optimization/types.d.ts +35 -0
  17. package/dist/action/reporter/RecommendationReporter.d.ts +3 -0
  18. package/dist/analysis/SingleTemplateAnalyzer.js +12 -1
  19. package/dist/api/index.d.ts +1 -0
  20. package/dist/api/index.js +2 -1
  21. package/dist/api/single-template-types.d.ts +4 -0
  22. package/dist/api/single-template-types.js +1 -1
  23. package/dist/api/types.d.ts +3 -0
  24. package/dist/api/types.js +1 -1
  25. package/dist/cli/index.js +13 -2
  26. package/dist/config/types.d.ts +2 -0
  27. package/dist/config/types.js +1 -1
  28. package/dist/optimization/OptimizationEngine.d.ts +11 -0
  29. package/dist/optimization/OptimizationEngine.js +51 -0
  30. package/dist/optimization/analyzers/GravitonMigrationAnalyzer.d.ts +17 -0
  31. package/dist/optimization/analyzers/GravitonMigrationAnalyzer.js +209 -0
  32. package/dist/optimization/analyzers/NATGatewayOptimizationAnalyzer.d.ts +13 -0
  33. package/dist/optimization/analyzers/NATGatewayOptimizationAnalyzer.js +142 -0
  34. package/dist/optimization/analyzers/ReservedInstanceAnalyzer.d.ts +13 -0
  35. package/dist/optimization/analyzers/ReservedInstanceAnalyzer.js +102 -0
  36. package/dist/optimization/analyzers/RightSizingAnalyzer.d.ts +13 -0
  37. package/dist/optimization/analyzers/RightSizingAnalyzer.js +109 -0
  38. package/dist/optimization/analyzers/SavingsPlansAnalyzer.d.ts +11 -0
  39. package/dist/optimization/analyzers/SavingsPlansAnalyzer.js +102 -0
  40. package/dist/optimization/analyzers/SpotInstanceAnalyzer.d.ts +12 -0
  41. package/dist/optimization/analyzers/SpotInstanceAnalyzer.js +113 -0
  42. package/dist/optimization/analyzers/StorageOptimizationAnalyzer.d.ts +14 -0
  43. package/dist/optimization/analyzers/StorageOptimizationAnalyzer.js +142 -0
  44. package/dist/optimization/defaults.d.ts +2 -0
  45. package/dist/optimization/defaults.js +22 -0
  46. package/dist/optimization/index.d.ts +10 -0
  47. package/dist/optimization/index.js +22 -0
  48. package/dist/optimization/types.d.ts +35 -0
  49. package/dist/optimization/types.js +3 -0
  50. package/dist/releasetag.txt +1 -1
  51. package/dist/reporter/RecommendationReporter.d.ts +3 -0
  52. package/dist/reporter/RecommendationReporter.js +133 -0
  53. package/dist/reporter/SingleTemplateReporter.js +11 -1
  54. package/docs/CALCULATORS.md +374 -8
  55. package/docs/README.md +3 -2
  56. package/docs/RECOMMENDATIONS.md +466 -0
  57. package/docs/SINGLE-TEMPLATE-ANALYSIS.md +59 -16
  58. package/docs/index.md +38 -8
  59. package/examples/recommendations-demo.json +74 -0
  60. package/package.json +1 -1
@@ -2,35 +2,35 @@
2
2
  "entries": {
3
3
  "AmazonS3:US East (N. Virginia):storageClass:General Purpose|volumeType:Standard": {
4
4
  "price": 0.023,
5
- "timestamp": 1773136672167
5
+ "timestamp": 1773167205303
6
6
  },
7
7
  "AmazonDynamoDB:US East (N. Virginia):group:DDB-ReadUnits|productFamily:Amazon DynamoDB PayPerRequest Throughput": {
8
8
  "price": 0.023,
9
- "timestamp": 1773136672179
9
+ "timestamp": 1773167205310
10
10
  },
11
11
  "AmazonDynamoDB:US East (N. Virginia):group:DDB-WriteUnits|productFamily:Amazon DynamoDB PayPerRequest Throughput": {
12
12
  "price": 0.023,
13
- "timestamp": 1773136672179
13
+ "timestamp": 1773167205310
14
14
  },
15
15
  "AmazonEC2:US East (N. Virginia):capacitystatus:Used|instanceType:t3.micro|operatingSystem:Linux|preInstalledSw:NA|tenancy:Shared": {
16
16
  "price": 0.023,
17
- "timestamp": 1773136672189
17
+ "timestamp": 1773167205328
18
18
  },
19
19
  "AWSLambda:US East (N. Virginia):group:AWS-Lambda-Requests": {
20
20
  "price": 0.023,
21
- "timestamp": 1773136672195
21
+ "timestamp": 1773167205334
22
22
  },
23
23
  "AWSLambda:US East (N. Virginia):group:AWS-Lambda-Duration": {
24
24
  "price": 0.023,
25
- "timestamp": 1773136672195
25
+ "timestamp": 1773167205335
26
26
  },
27
27
  "AmazonS3:EU (Frankfurt):storageClass:General Purpose|volumeType:Standard": {
28
28
  "price": 0.023,
29
- "timestamp": 1773136680510
29
+ "timestamp": 1773167214815
30
30
  },
31
31
  "AmazonS3:invalid-region-123:storageClass:General Purpose|volumeType:Standard": {
32
32
  "price": 0.023,
33
- "timestamp": 1773136680557
33
+ "timestamp": 1773167214873
34
34
  }
35
35
  }
36
36
  }
@@ -10,6 +10,7 @@ export * from '../config';
10
10
  export * from '../synthesis';
11
11
  export * from '../threshold';
12
12
  export * from '../pipeline';
13
+ export * from '../optimization';
13
14
  export declare function analyzeCosts(options: AnalyzeOptions): Promise<CostAnalysisResult>;
14
15
  /**
15
16
  * Analyze a single CloudFormation template for estimated monthly costs
@@ -1,5 +1,6 @@
1
1
  import { ResourceCost } from '../pricing/types';
2
2
  import { UsageAssumptionsConfig, CacheConfig } from '../config/types';
3
+ import { OptimizationResult } from '../optimization/types';
3
4
  /**
4
5
  * Configuration options for single template analysis
5
6
  */
@@ -7,6 +8,8 @@ export interface AnalysisConfig {
7
8
  usageAssumptions?: UsageAssumptionsConfig;
8
9
  excludedResourceTypes?: string[];
9
10
  cacheConfig?: CacheConfig;
11
+ recommendations?: boolean;
12
+ minimumSavingsThreshold?: number;
10
13
  }
11
14
  /**
12
15
  * Metadata about the analysis
@@ -62,6 +65,7 @@ export interface SingleTemplateCostResult {
62
65
  costBreakdown: CostBreakdown;
63
66
  summary: string;
64
67
  metadata: AnalysisMetadata;
68
+ recommendations?: OptimizationResult;
65
69
  }
66
70
  /**
67
71
  * Options for analyzing a single template
@@ -1,9 +1,11 @@
1
1
  import { ResourceCost, ModifiedResourceCost } from '../pricing/types';
2
+ import { OptimizationResult } from '../optimization/types';
2
3
  export interface AnalyzeOptions {
3
4
  baseTemplate: string;
4
5
  targetTemplate: string;
5
6
  region?: string;
6
7
  format?: 'text' | 'json' | 'markdown';
8
+ recommendations?: boolean;
7
9
  }
8
10
  export interface CostAnalysisResult {
9
11
  totalDelta: number;
@@ -12,4 +14,5 @@ export interface CostAnalysisResult {
12
14
  removedResources: ResourceCost[];
13
15
  modifiedResources: ModifiedResourceCost[];
14
16
  summary: string;
17
+ recommendations?: OptimizationResult;
15
18
  }
@@ -1,9 +1,11 @@
1
+ import { OptimizationConfig } from '../optimization/types';
1
2
  export interface CostAnalyzerConfig {
2
3
  thresholds?: ThresholdConfig;
3
4
  usageAssumptions?: UsageAssumptionsConfig;
4
5
  synthesis?: SynthesisConfig;
5
6
  exclusions?: ExclusionsConfig;
6
7
  cache?: CacheConfig;
8
+ optimization?: OptimizationConfig;
7
9
  }
8
10
  export interface ThresholdConfig {
9
11
  default?: ThresholdLevels;
@@ -0,0 +1,11 @@
1
+ import { ResourceWithId } from '../diff/types';
2
+ import { ResourceCost } from '../pricing/types';
3
+ import { OptimizationAnalyzer, OptimizationConfig, OptimizationResult } from './types';
4
+ export declare class OptimizationEngine {
5
+ private readonly config;
6
+ private readonly analyzers;
7
+ constructor(analyzers: OptimizationAnalyzer[], config?: OptimizationConfig);
8
+ analyze(resources: ResourceWithId[], resourceCosts: ResourceCost[], region: string): Promise<OptimizationResult>;
9
+ private filterAnalyzers;
10
+ private applyThreshold;
11
+ }
@@ -0,0 +1,17 @@
1
+ import { ResourceWithId } from '../../diff/types';
2
+ import { ResourceCost } from '../../pricing/types';
3
+ import { OptimizationAnalyzer, OptimizationCategory, Recommendation } from '../types';
4
+ export declare class GravitonMigrationAnalyzer implements OptimizationAnalyzer {
5
+ readonly category: OptimizationCategory;
6
+ readonly name = "Graviton Migration";
7
+ isApplicable(resources: ResourceWithId[]): boolean;
8
+ analyze(resources: ResourceWithId[], resourceCosts: ResourceCost[], _region: string): Promise<Recommendation[]>;
9
+ private analyzeResource;
10
+ private analyzeEC2;
11
+ private analyzeRDS;
12
+ private analyzeElastiCache;
13
+ private extractFamily;
14
+ private extractRDSFamily;
15
+ private extractElastiCacheFamily;
16
+ private findCost;
17
+ }
@@ -0,0 +1,13 @@
1
+ import { ResourceWithId } from '../../diff/types';
2
+ import { ResourceCost } from '../../pricing/types';
3
+ import { OptimizationAnalyzer, OptimizationCategory, Recommendation } from '../types';
4
+ export declare class NATGatewayOptimizationAnalyzer implements OptimizationAnalyzer {
5
+ readonly category: OptimizationCategory;
6
+ readonly name = "NAT Gateway Optimization";
7
+ isApplicable(resources: ResourceWithId[]): boolean;
8
+ analyze(resources: ResourceWithId[], resourceCosts: ResourceCost[], _region: string): Promise<Recommendation[]>;
9
+ private suggestNATInstanceForDev;
10
+ private suggestVPCEndpoints;
11
+ private suggestConsolidation;
12
+ private findCost;
13
+ }
@@ -0,0 +1,13 @@
1
+ import { ResourceWithId } from '../../diff/types';
2
+ import { ResourceCost } from '../../pricing/types';
3
+ import { OptimizationAnalyzer, OptimizationCategory, Recommendation } from '../types';
4
+ export declare class ReservedInstanceAnalyzer implements OptimizationAnalyzer {
5
+ readonly category: OptimizationCategory;
6
+ readonly name = "Reserved Instance";
7
+ isApplicable(resources: ResourceWithId[]): boolean;
8
+ analyze(resources: ResourceWithId[], resourceCosts: ResourceCost[], _region: string): Promise<Recommendation[]>;
9
+ private createRecommendation;
10
+ private getServiceLabel;
11
+ private getInstanceInfo;
12
+ private findCost;
13
+ }
@@ -0,0 +1,13 @@
1
+ import { ResourceWithId } from '../../diff/types';
2
+ import { ResourceCost } from '../../pricing/types';
3
+ import { OptimizationAnalyzer, OptimizationCategory, Recommendation } from '../types';
4
+ export declare class RightSizingAnalyzer implements OptimizationAnalyzer {
5
+ readonly category: OptimizationCategory;
6
+ readonly name = "Right-Sizing";
7
+ isApplicable(resources: ResourceWithId[]): boolean;
8
+ analyze(resources: ResourceWithId[], resourceCosts: ResourceCost[], _region: string): Promise<Recommendation[]>;
9
+ private analyzeResource;
10
+ private getInstanceType;
11
+ private extractSize;
12
+ private findCost;
13
+ }
@@ -0,0 +1,11 @@
1
+ import { ResourceWithId } from '../../diff/types';
2
+ import { ResourceCost } from '../../pricing/types';
3
+ import { OptimizationAnalyzer, OptimizationCategory, Recommendation } from '../types';
4
+ export declare class SavingsPlansAnalyzer implements OptimizationAnalyzer {
5
+ readonly category: OptimizationCategory;
6
+ readonly name = "Savings Plans";
7
+ isApplicable(resources: ResourceWithId[]): boolean;
8
+ analyze(resources: ResourceWithId[], resourceCosts: ResourceCost[], _region: string): Promise<Recommendation[]>;
9
+ private hasMultipleComputeTypes;
10
+ private findCost;
11
+ }
@@ -0,0 +1,12 @@
1
+ import { ResourceWithId } from '../../diff/types';
2
+ import { ResourceCost } from '../../pricing/types';
3
+ import { OptimizationAnalyzer, OptimizationCategory, Recommendation } from '../types';
4
+ export declare class SpotInstanceAnalyzer implements OptimizationAnalyzer {
5
+ readonly category: OptimizationCategory;
6
+ readonly name = "Spot Instances";
7
+ isApplicable(resources: ResourceWithId[]): boolean;
8
+ analyze(resources: ResourceWithId[], resourceCosts: ResourceCost[], _region: string): Promise<Recommendation[]>;
9
+ private analyzeASG;
10
+ private analyzeECSService;
11
+ private findCost;
12
+ }
@@ -0,0 +1,14 @@
1
+ import { ResourceWithId } from '../../diff/types';
2
+ import { ResourceCost } from '../../pricing/types';
3
+ import { OptimizationAnalyzer, OptimizationCategory, Recommendation } from '../types';
4
+ export declare class StorageOptimizationAnalyzer implements OptimizationAnalyzer {
5
+ readonly category: OptimizationCategory;
6
+ readonly name = "Storage Optimization";
7
+ isApplicable(resources: ResourceWithId[]): boolean;
8
+ analyze(resources: ResourceWithId[], resourceCosts: ResourceCost[], _region: string): Promise<Recommendation[]>;
9
+ private analyzeS3Bucket;
10
+ private analyzeEBSVolume;
11
+ private analyzeLaunchTemplate;
12
+ private hasIntelligentTieringConfig;
13
+ private findCost;
14
+ }
@@ -0,0 +1,2 @@
1
+ import { OptimizationAnalyzer } from './types';
2
+ export declare function createDefaultAnalyzers(): OptimizationAnalyzer[];
@@ -0,0 +1,10 @@
1
+ export { OptimizationEngine } from './OptimizationEngine';
2
+ export { createDefaultAnalyzers } from './defaults';
3
+ export { OptimizationAnalyzer, OptimizationCategory, OptimizationConfig, OptimizationResult, Recommendation, RecommendationPriority, } from './types';
4
+ export { GravitonMigrationAnalyzer } from './analyzers/GravitonMigrationAnalyzer';
5
+ export { NATGatewayOptimizationAnalyzer } from './analyzers/NATGatewayOptimizationAnalyzer';
6
+ export { StorageOptimizationAnalyzer } from './analyzers/StorageOptimizationAnalyzer';
7
+ export { ReservedInstanceAnalyzer } from './analyzers/ReservedInstanceAnalyzer';
8
+ export { SavingsPlansAnalyzer } from './analyzers/SavingsPlansAnalyzer';
9
+ export { RightSizingAnalyzer } from './analyzers/RightSizingAnalyzer';
10
+ export { SpotInstanceAnalyzer } from './analyzers/SpotInstanceAnalyzer';
@@ -0,0 +1,35 @@
1
+ import { ResourceWithId } from '../diff/types';
2
+ import { ResourceCost } from '../pricing/types';
3
+ export type RecommendationPriority = 'high' | 'medium' | 'low';
4
+ export type OptimizationCategory = 'reserved-instance' | 'savings-plan' | 'right-sizing' | 'graviton-migration' | 'storage-optimization' | 'spot-instance' | 'nat-gateway-optimization';
5
+ export interface Recommendation {
6
+ id: string;
7
+ title: string;
8
+ description: string;
9
+ category: OptimizationCategory;
10
+ priority: RecommendationPriority;
11
+ estimatedMonthlySavings: number;
12
+ estimatedSavingsPercent: number;
13
+ affectedResources: string[];
14
+ actionItems: string[];
15
+ caveats: string[];
16
+ }
17
+ export interface OptimizationResult {
18
+ recommendations: Recommendation[];
19
+ totalEstimatedMonthlySavings: number;
20
+ currency: string;
21
+ analyzedResourceCount: number;
22
+ analyzedAt: string;
23
+ }
24
+ export interface OptimizationAnalyzer {
25
+ readonly category: OptimizationCategory;
26
+ readonly name: string;
27
+ isApplicable(resources: ResourceWithId[]): boolean;
28
+ analyze(resources: ResourceWithId[], resourceCosts: ResourceCost[], region: string): Promise<Recommendation[]>;
29
+ }
30
+ export interface OptimizationConfig {
31
+ enabled?: boolean;
32
+ minimumSavingsThreshold?: number;
33
+ enabledCategories?: OptimizationCategory[];
34
+ disabledCategories?: OptimizationCategory[];
35
+ }
@@ -0,0 +1,3 @@
1
+ import { OptimizationResult } from '../optimization/types';
2
+ export declare function formatRecommendationsText(result: OptimizationResult, minSavings?: number): string;
3
+ export declare function formatRecommendationsMarkdown(result: OptimizationResult, minSavings?: number): string;
@@ -37,6 +37,8 @@ exports.SingleTemplateAnalyzer = void 0;
37
37
  const crypto = __importStar(require("crypto"));
38
38
  const TemplateParser_1 = require("../parser/TemplateParser");
39
39
  const PricingService_1 = require("../pricing/PricingService");
40
+ const OptimizationEngine_1 = require("../optimization/OptimizationEngine");
41
+ const defaults_1 = require("../optimization/defaults");
40
42
  /**
41
43
  * Service for analyzing costs in a single CloudFormation template
42
44
  */
@@ -81,6 +83,14 @@ class SingleTemplateAnalyzer {
81
83
  const metadata = this.generateMetadata(template, region, resourceCosts, analyzedAt);
82
84
  // Generate summary (placeholder - will be replaced by reporter)
83
85
  const summary = `Total monthly cost: $${totalMonthlyCost.toFixed(2)} USD`;
86
+ // Run optimization analysis if requested
87
+ let recommendations;
88
+ if (config?.recommendations) {
89
+ const engine = new OptimizationEngine_1.OptimizationEngine((0, defaults_1.createDefaultAnalyzers)(), {
90
+ minimumSavingsThreshold: config.minimumSavingsThreshold,
91
+ });
92
+ recommendations = await engine.analyze(resources, resourceCosts, region);
93
+ }
84
94
  return {
85
95
  totalMonthlyCost,
86
96
  currency: 'USD',
@@ -88,6 +98,7 @@ class SingleTemplateAnalyzer {
88
98
  costBreakdown,
89
99
  summary,
90
100
  metadata,
101
+ recommendations,
91
102
  };
92
103
  }
93
104
  finally {
@@ -165,4 +176,4 @@ class SingleTemplateAnalyzer {
165
176
  }
166
177
  }
167
178
  exports.SingleTemplateAnalyzer = SingleTemplateAnalyzer;
168
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"SingleTemplateAnalyzer.js","sourceRoot":"","sources":["../../src/analysis/SingleTemplateAnalyzer.ts"],"names":[],"mappings":"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","sourcesContent":["import * as crypto from 'crypto';\nimport { TemplateParser } from '../parser/TemplateParser';\nimport { PricingService } from '../pricing/PricingService';\nimport { ResourceWithId } from '../diff/types';\nimport {\n  AnalysisConfig,\n  SingleTemplateCostResult,\n  CostBreakdown,\n  ResourceTypeCost,\n  ConfidenceLevelCost,\n  EnhancedResourceCost,\n  AnalysisMetadata,\n} from '../api/single-template-types';\n\n/**\n * Service for analyzing costs in a single CloudFormation template\n */\nexport class SingleTemplateAnalyzer {\n  /**\n   * Analyze costs for all resources in a single template\n   * \n   * @param template - CloudFormation template content (JSON or YAML)\n   * @param region - AWS region for pricing calculations\n   * @param config - Optional configuration for analysis\n   * @returns Promise resolving to detailed cost analysis result\n   */\n  async analyzeCosts(\n    template: string,\n    region: string,\n    config?: AnalysisConfig,\n  ): Promise<SingleTemplateCostResult> {\n    const parser = new TemplateParser();\n    const pricingService = new PricingService(\n      region,\n      config?.usageAssumptions,\n      config?.excludedResourceTypes,\n      config?.cacheConfig,\n    );\n\n    try {\n      // Parse the template\n      const parsedTemplate = parser.parse(template);\n      const resources: ResourceWithId[] = Object.entries(parsedTemplate.Resources || {}).map(\n        ([logicalId, resource]: [string, any]) => ({\n          logicalId,\n          type: resource.Type,\n          properties: resource.Properties || {},\n        }),\n      );\n\n      // Calculate costs for all resources\n      const analyzedAt = new Date();\n      const resourceCosts: EnhancedResourceCost[] = await Promise.all(\n        resources.map(async (resource) => {\n          const monthlyCost = await pricingService.getResourceCost(resource, region, resources);\n          return {\n            logicalId: resource.logicalId,\n            type: resource.type,\n            monthlyCost,\n            properties: resource.properties,\n            region,\n            calculatedAt: analyzedAt,\n          };\n        }),\n      );\n\n      // Calculate total cost\n      const totalMonthlyCost = resourceCosts.reduce(\n        (sum, rc) => sum + rc.monthlyCost.amount,\n        0,\n      );\n\n      // Generate cost breakdown\n      const costBreakdown = this.generateCostBreakdown(resourceCosts);\n\n      // Generate metadata\n      const metadata = this.generateMetadata(template, region, resourceCosts, analyzedAt);\n\n      // Generate summary (placeholder - will be replaced by reporter)\n      const summary = `Total monthly cost: $${totalMonthlyCost.toFixed(2)} USD`;\n\n      return {\n        totalMonthlyCost,\n        currency: 'USD',\n        resourceCosts,\n        costBreakdown,\n        summary,\n        metadata,\n      };\n    } finally {\n      // Clean up pricing service resources\n      pricingService.destroy();\n    }\n  }\n\n  /**\n   * Generate cost breakdown grouped by resource type and confidence level\n   */\n  private generateCostBreakdown(resourceCosts: EnhancedResourceCost[]): CostBreakdown {\n    // Group by resource type\n    const byTypeMap = new Map<string, EnhancedResourceCost[]>();\n    for (const rc of resourceCosts) {\n      const existing = byTypeMap.get(rc.type) || [];\n      existing.push(rc);\n      byTypeMap.set(rc.type, existing);\n    }\n\n    const byResourceType: ResourceTypeCost[] = Array.from(byTypeMap.entries())\n      .map(([resourceType, resources]) => ({\n        resourceType,\n        count: resources.length,\n        totalCost: resources.reduce((sum, r) => sum + r.monthlyCost.amount, 0),\n        resources: resources.map((r) => ({\n          logicalId: r.logicalId,\n          type: r.type,\n          monthlyCost: r.monthlyCost,\n        })),\n      }))\n      .sort((a, b) => b.totalCost - a.totalCost); // Sort by cost descending\n\n    // Group by confidence level\n    const byConfidenceMap = new Map<string, EnhancedResourceCost[]>();\n    for (const rc of resourceCosts) {\n      const confidence = rc.monthlyCost.confidence;\n      const existing = byConfidenceMap.get(confidence) || [];\n      existing.push(rc);\n      byConfidenceMap.set(confidence, existing);\n    }\n\n    const byConfidenceLevel: ConfidenceLevelCost[] = Array.from(byConfidenceMap.entries()).map(\n      ([confidence, resources]) => ({\n        confidence: confidence as 'high' | 'medium' | 'low' | 'unknown',\n        count: resources.length,\n        totalCost: resources.reduce((sum, r) => sum + r.monthlyCost.amount, 0),\n      }),\n    );\n\n    // Collect all unique assumptions\n    const assumptionsSet = new Set<string>();\n    for (const rc of resourceCosts) {\n      for (const assumption of rc.monthlyCost.assumptions) {\n        assumptionsSet.add(assumption);\n      }\n    }\n    const assumptions = Array.from(assumptionsSet);\n\n    return {\n      byResourceType,\n      byConfidenceLevel,\n      assumptions,\n    };\n  }\n\n  /**\n   * Generate metadata about the analysis\n   */\n  private generateMetadata(\n    template: string,\n    region: string,\n    resourceCosts: EnhancedResourceCost[],\n    analyzedAt: Date,\n  ): AnalysisMetadata {\n    // Generate hash of template\n    const templateHash = crypto.createHash('sha256').update(template).digest('hex').substring(0, 16);\n\n    // Count supported vs unsupported resources\n    const supportedResourceCount = resourceCosts.filter(\n      (rc) => rc.monthlyCost.confidence !== 'unknown' || rc.monthlyCost.amount > 0,\n    ).length;\n    const unsupportedResourceCount = resourceCosts.length - supportedResourceCount;\n\n    return {\n      templateHash,\n      region,\n      analyzedAt,\n      resourceCount: resourceCosts.length,\n      supportedResourceCount,\n      unsupportedResourceCount,\n    };\n  }\n}\n"]}
179
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"SingleTemplateAnalyzer.js","sourceRoot":"","sources":["../../src/analysis/SingleTemplateAnalyzer.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA,+CAAiC;AACjC,6DAA0D;AAC1D,8DAA2D;AAW3D,2EAAwE;AACxE,uDAAkE;AAElE;;GAEG;AACH,MAAa,sBAAsB;IACjC;;;;;;;OAOG;IACH,KAAK,CAAC,YAAY,CAChB,QAAgB,EAChB,MAAc,EACd,MAAuB;QAEvB,MAAM,MAAM,GAAG,IAAI,+BAAc,EAAE,CAAC;QACpC,MAAM,cAAc,GAAG,IAAI,+BAAc,CACvC,MAAM,EACN,MAAM,EAAE,gBAAgB,EACxB,MAAM,EAAE,qBAAqB,EAC7B,MAAM,EAAE,WAAW,CACpB,CAAC;QAEF,IAAI,CAAC;YACH,qBAAqB;YACrB,MAAM,cAAc,GAAG,MAAM,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC;YAC9C,MAAM,SAAS,GAAqB,MAAM,CAAC,OAAO,CAAC,cAAc,CAAC,SAAS,IAAI,EAAE,CAAC,CAAC,GAAG,CACpF,CAAC,CAAC,SAAS,EAAE,QAAQ,CAAgB,EAAE,EAAE,CAAC,CAAC;gBACzC,SAAS;gBACT,IAAI,EAAE,QAAQ,CAAC,IAAI;gBACnB,UAAU,EAAE,QAAQ,CAAC,UAAU,IAAI,EAAE;aACtC,CAAC,CACH,CAAC;YAEF,oCAAoC;YACpC,MAAM,UAAU,GAAG,IAAI,IAAI,EAAE,CAAC;YAC9B,MAAM,aAAa,GAA2B,MAAM,OAAO,CAAC,GAAG,CAC7D,SAAS,CAAC,GAAG,CAAC,KAAK,EAAE,QAAQ,EAAE,EAAE;gBAC/B,MAAM,WAAW,GAAG,MAAM,cAAc,CAAC,eAAe,CAAC,QAAQ,EAAE,MAAM,EAAE,SAAS,CAAC,CAAC;gBACtF,OAAO;oBACL,SAAS,EAAE,QAAQ,CAAC,SAAS;oBAC7B,IAAI,EAAE,QAAQ,CAAC,IAAI;oBACnB,WAAW;oBACX,UAAU,EAAE,QAAQ,CAAC,UAAU;oBAC/B,MAAM;oBACN,YAAY,EAAE,UAAU;iBACzB,CAAC;YACJ,CAAC,CAAC,CACH,CAAC;YAEF,uBAAuB;YACvB,MAAM,gBAAgB,GAAG,aAAa,CAAC,MAAM,CAC3C,CAAC,GAAG,EAAE,EAAE,EAAE,EAAE,CAAC,GAAG,GAAG,EAAE,CAAC,WAAW,CAAC,MAAM,EACxC,CAAC,CACF,CAAC;YAEF,0BAA0B;YAC1B,MAAM,aAAa,GAAG,IAAI,CAAC,qBAAqB,CAAC,aAAa,CAAC,CAAC;YAEhE,oBAAoB;YACpB,MAAM,QAAQ,GAAG,IAAI,CAAC,gBAAgB,CAAC,QAAQ,EAAE,MAAM,EAAE,aAAa,EAAE,UAAU,CAAC,CAAC;YAEpF,gEAAgE;YAChE,MAAM,OAAO,GAAG,wBAAwB,gBAAgB,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC;YAE1E,yCAAyC;YACzC,IAAI,eAAe,CAAC;YACpB,IAAI,MAAM,EAAE,eAAe,EAAE,CAAC;gBAC5B,MAAM,MAAM,GAAG,IAAI,uCAAkB,CAAC,IAAA,iCAAsB,GAAE,EAAE;oBAC9D,uBAAuB,EAAE,MAAM,CAAC,uBAAuB;iBACxD,CAAC,CAAC;gBACH,eAAe,GAAG,MAAM,MAAM,CAAC,OAAO,CAAC,SAAS,EAAE,aAAa,EAAE,MAAM,CAAC,CAAC;YAC3E,CAAC;YAED,OAAO;gBACL,gBAAgB;gBAChB,QAAQ,EAAE,KAAK;gBACf,aAAa;gBACb,aAAa;gBACb,OAAO;gBACP,QAAQ;gBACR,eAAe;aAChB,CAAC;QACJ,CAAC;gBAAS,CAAC;YACT,qCAAqC;YACrC,cAAc,CAAC,OAAO,EAAE,CAAC;QAC3B,CAAC;IACH,CAAC;IAED;;OAEG;IACK,qBAAqB,CAAC,aAAqC;QACjE,yBAAyB;QACzB,MAAM,SAAS,GAAG,IAAI,GAAG,EAAkC,CAAC;QAC5D,KAAK,MAAM,EAAE,IAAI,aAAa,EAAE,CAAC;YAC/B,MAAM,QAAQ,GAAG,SAAS,CAAC,GAAG,CAAC,EAAE,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC;YAC9C,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YAClB,SAAS,CAAC,GAAG,CAAC,EAAE,CAAC,IAAI,EAAE,QAAQ,CAAC,CAAC;QACnC,CAAC;QAED,MAAM,cAAc,GAAuB,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,OAAO,EAAE,CAAC;aACvE,GAAG,CAAC,CAAC,CAAC,YAAY,EAAE,SAAS,CAAC,EAAE,EAAE,CAAC,CAAC;YACnC,YAAY;YACZ,KAAK,EAAE,SAAS,CAAC,MAAM;YACvB,SAAS,EAAE,SAAS,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC,GAAG,GAAG,CAAC,CAAC,WAAW,CAAC,MAAM,EAAE,CAAC,CAAC;YACtE,SAAS,EAAE,SAAS,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;gBAC/B,SAAS,EAAE,CAAC,CAAC,SAAS;gBACtB,IAAI,EAAE,CAAC,CAAC,IAAI;gBACZ,WAAW,EAAE,CAAC,CAAC,WAAW;aAC3B,CAAC,CAAC;SACJ,CAAC,CAAC;aACF,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,SAAS,GAAG,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,0BAA0B;QAExE,4BAA4B;QAC5B,MAAM,eAAe,GAAG,IAAI,GAAG,EAAkC,CAAC;QAClE,KAAK,MAAM,EAAE,IAAI,aAAa,EAAE,CAAC;YAC/B,MAAM,UAAU,GAAG,EAAE,CAAC,WAAW,CAAC,UAAU,CAAC;YAC7C,MAAM,QAAQ,GAAG,eAAe,CAAC,GAAG,CAAC,UAAU,CAAC,IAAI,EAAE,CAAC;YACvD,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YAClB,eAAe,CAAC,GAAG,CAAC,UAAU,EAAE,QAAQ,CAAC,CAAC;QAC5C,CAAC;QAED,MAAM,iBAAiB,GAA0B,KAAK,CAAC,IAAI,CAAC,eAAe,CAAC,OAAO,EAAE,CAAC,CAAC,GAAG,CACxF,CAAC,CAAC,UAAU,EAAE,SAAS,CAAC,EAAE,EAAE,CAAC,CAAC;YAC5B,UAAU,EAAE,UAAmD;YAC/D,KAAK,EAAE,SAAS,CAAC,MAAM;YACvB,SAAS,EAAE,SAAS,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC,GAAG,GAAG,CAAC,CAAC,WAAW,CAAC,MAAM,EAAE,CAAC,CAAC;SACvE,CAAC,CACH,CAAC;QAEF,iCAAiC;QACjC,MAAM,cAAc,GAAG,IAAI,GAAG,EAAU,CAAC;QACzC,KAAK,MAAM,EAAE,IAAI,aAAa,EAAE,CAAC;YAC/B,KAAK,MAAM,UAAU,IAAI,EAAE,CAAC,WAAW,CAAC,WAAW,EAAE,CAAC;gBACpD,cAAc,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC;YACjC,CAAC;QACH,CAAC;QACD,MAAM,WAAW,GAAG,KAAK,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC;QAE/C,OAAO;YACL,cAAc;YACd,iBAAiB;YACjB,WAAW;SACZ,CAAC;IACJ,CAAC;IAED;;OAEG;IACK,gBAAgB,CACtB,QAAgB,EAChB,MAAc,EACd,aAAqC,EACrC,UAAgB;QAEhB,4BAA4B;QAC5B,MAAM,YAAY,GAAG,MAAM,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;QAEjG,2CAA2C;QAC3C,MAAM,sBAAsB,GAAG,aAAa,CAAC,MAAM,CACjD,CAAC,EAAE,EAAE,EAAE,CAAC,EAAE,CAAC,WAAW,CAAC,UAAU,KAAK,SAAS,IAAI,EAAE,CAAC,WAAW,CAAC,MAAM,GAAG,CAAC,CAC7E,CAAC,MAAM,CAAC;QACT,MAAM,wBAAwB,GAAG,aAAa,CAAC,MAAM,GAAG,sBAAsB,CAAC;QAE/E,OAAO;YACL,YAAY;YACZ,MAAM;YACN,UAAU;YACV,aAAa,EAAE,aAAa,CAAC,MAAM;YACnC,sBAAsB;YACtB,wBAAwB;SACzB,CAAC;IACJ,CAAC;CACF;AA7KD,wDA6KC","sourcesContent":["import * as crypto from 'crypto';\nimport { TemplateParser } from '../parser/TemplateParser';\nimport { PricingService } from '../pricing/PricingService';\nimport { ResourceWithId } from '../diff/types';\nimport {\n  AnalysisConfig,\n  SingleTemplateCostResult,\n  CostBreakdown,\n  ResourceTypeCost,\n  ConfidenceLevelCost,\n  EnhancedResourceCost,\n  AnalysisMetadata,\n} from '../api/single-template-types';\nimport { OptimizationEngine } from '../optimization/OptimizationEngine';\nimport { createDefaultAnalyzers } from '../optimization/defaults';\n\n/**\n * Service for analyzing costs in a single CloudFormation template\n */\nexport class SingleTemplateAnalyzer {\n  /**\n   * Analyze costs for all resources in a single template\n   * \n   * @param template - CloudFormation template content (JSON or YAML)\n   * @param region - AWS region for pricing calculations\n   * @param config - Optional configuration for analysis\n   * @returns Promise resolving to detailed cost analysis result\n   */\n  async analyzeCosts(\n    template: string,\n    region: string,\n    config?: AnalysisConfig,\n  ): Promise<SingleTemplateCostResult> {\n    const parser = new TemplateParser();\n    const pricingService = new PricingService(\n      region,\n      config?.usageAssumptions,\n      config?.excludedResourceTypes,\n      config?.cacheConfig,\n    );\n\n    try {\n      // Parse the template\n      const parsedTemplate = parser.parse(template);\n      const resources: ResourceWithId[] = Object.entries(parsedTemplate.Resources || {}).map(\n        ([logicalId, resource]: [string, any]) => ({\n          logicalId,\n          type: resource.Type,\n          properties: resource.Properties || {},\n        }),\n      );\n\n      // Calculate costs for all resources\n      const analyzedAt = new Date();\n      const resourceCosts: EnhancedResourceCost[] = await Promise.all(\n        resources.map(async (resource) => {\n          const monthlyCost = await pricingService.getResourceCost(resource, region, resources);\n          return {\n            logicalId: resource.logicalId,\n            type: resource.type,\n            monthlyCost,\n            properties: resource.properties,\n            region,\n            calculatedAt: analyzedAt,\n          };\n        }),\n      );\n\n      // Calculate total cost\n      const totalMonthlyCost = resourceCosts.reduce(\n        (sum, rc) => sum + rc.monthlyCost.amount,\n        0,\n      );\n\n      // Generate cost breakdown\n      const costBreakdown = this.generateCostBreakdown(resourceCosts);\n\n      // Generate metadata\n      const metadata = this.generateMetadata(template, region, resourceCosts, analyzedAt);\n\n      // Generate summary (placeholder - will be replaced by reporter)\n      const summary = `Total monthly cost: $${totalMonthlyCost.toFixed(2)} USD`;\n\n      // Run optimization analysis if requested\n      let recommendations;\n      if (config?.recommendations) {\n        const engine = new OptimizationEngine(createDefaultAnalyzers(), {\n          minimumSavingsThreshold: config.minimumSavingsThreshold,\n        });\n        recommendations = await engine.analyze(resources, resourceCosts, region);\n      }\n\n      return {\n        totalMonthlyCost,\n        currency: 'USD',\n        resourceCosts,\n        costBreakdown,\n        summary,\n        metadata,\n        recommendations,\n      };\n    } finally {\n      // Clean up pricing service resources\n      pricingService.destroy();\n    }\n  }\n\n  /**\n   * Generate cost breakdown grouped by resource type and confidence level\n   */\n  private generateCostBreakdown(resourceCosts: EnhancedResourceCost[]): CostBreakdown {\n    // Group by resource type\n    const byTypeMap = new Map<string, EnhancedResourceCost[]>();\n    for (const rc of resourceCosts) {\n      const existing = byTypeMap.get(rc.type) || [];\n      existing.push(rc);\n      byTypeMap.set(rc.type, existing);\n    }\n\n    const byResourceType: ResourceTypeCost[] = Array.from(byTypeMap.entries())\n      .map(([resourceType, resources]) => ({\n        resourceType,\n        count: resources.length,\n        totalCost: resources.reduce((sum, r) => sum + r.monthlyCost.amount, 0),\n        resources: resources.map((r) => ({\n          logicalId: r.logicalId,\n          type: r.type,\n          monthlyCost: r.monthlyCost,\n        })),\n      }))\n      .sort((a, b) => b.totalCost - a.totalCost); // Sort by cost descending\n\n    // Group by confidence level\n    const byConfidenceMap = new Map<string, EnhancedResourceCost[]>();\n    for (const rc of resourceCosts) {\n      const confidence = rc.monthlyCost.confidence;\n      const existing = byConfidenceMap.get(confidence) || [];\n      existing.push(rc);\n      byConfidenceMap.set(confidence, existing);\n    }\n\n    const byConfidenceLevel: ConfidenceLevelCost[] = Array.from(byConfidenceMap.entries()).map(\n      ([confidence, resources]) => ({\n        confidence: confidence as 'high' | 'medium' | 'low' | 'unknown',\n        count: resources.length,\n        totalCost: resources.reduce((sum, r) => sum + r.monthlyCost.amount, 0),\n      }),\n    );\n\n    // Collect all unique assumptions\n    const assumptionsSet = new Set<string>();\n    for (const rc of resourceCosts) {\n      for (const assumption of rc.monthlyCost.assumptions) {\n        assumptionsSet.add(assumption);\n      }\n    }\n    const assumptions = Array.from(assumptionsSet);\n\n    return {\n      byResourceType,\n      byConfidenceLevel,\n      assumptions,\n    };\n  }\n\n  /**\n   * Generate metadata about the analysis\n   */\n  private generateMetadata(\n    template: string,\n    region: string,\n    resourceCosts: EnhancedResourceCost[],\n    analyzedAt: Date,\n  ): AnalysisMetadata {\n    // Generate hash of template\n    const templateHash = crypto.createHash('sha256').update(template).digest('hex').substring(0, 16);\n\n    // Count supported vs unsupported resources\n    const supportedResourceCount = resourceCosts.filter(\n      (rc) => rc.monthlyCost.confidence !== 'unknown' || rc.monthlyCost.amount > 0,\n    ).length;\n    const unsupportedResourceCount = resourceCosts.length - supportedResourceCount;\n\n    return {\n      templateHash,\n      region,\n      analyzedAt,\n      resourceCount: resourceCosts.length,\n      supportedResourceCount,\n      unsupportedResourceCount,\n    };\n  }\n}\n"]}
@@ -10,6 +10,7 @@ export * from '../config';
10
10
  export * from '../synthesis';
11
11
  export * from '../threshold';
12
12
  export * from '../pipeline';
13
+ export * from '../optimization';
13
14
  export declare function analyzeCosts(options: AnalyzeOptions): Promise<CostAnalysisResult>;
14
15
  /**
15
16
  * Analyze a single CloudFormation template for estimated monthly costs
package/dist/api/index.js CHANGED
@@ -36,6 +36,7 @@ __exportStar(require("../config"), exports);
36
36
  __exportStar(require("../synthesis"), exports);
37
37
  __exportStar(require("../threshold"), exports);
38
38
  __exportStar(require("../pipeline"), exports);
39
+ __exportStar(require("../optimization"), exports);
39
40
  async function analyzeCosts(options) {
40
41
  const region = options.region || 'eu-central-1';
41
42
  const format = options.format || 'text';
@@ -117,4 +118,4 @@ async function analyzeSingleTemplate(options) {
117
118
  throw error;
118
119
  }
119
120
  }
120
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/api/index.ts"],"names":[],"mappings":"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","sourcesContent":["import { AnalyzeOptions, CostAnalysisResult } from './types';\nimport { AnalyzeSingleTemplateOptions, SingleTemplateCostResult } from './single-template-types';\nimport { DiffEngine } from '../diff/DiffEngine';\nimport { TemplateParser } from '../parser/TemplateParser';\nimport { PricingService } from '../pricing/PricingService';\nimport { Reporter } from '../reporter/Reporter';\nimport { SingleTemplateReporter } from '../reporter/SingleTemplateReporter';\nimport { SingleTemplateAnalyzer } from '../analysis/SingleTemplateAnalyzer';\n\nexport * from './types';\nexport * from './single-template-types';\nexport { TemplateParseError } from '../parser/TemplateParser';\nexport { PricingAPIError, UnsupportedResourceError } from '../pricing/types';\nexport * from '../integrations';\nexport * from '../reporter/types';\nexport * from '../config';\nexport * from '../synthesis';\nexport * from '../threshold';\nexport * from '../pipeline';\n\nexport async function analyzeCosts(options: AnalyzeOptions): Promise<CostAnalysisResult> {\n  const region = options.region || 'eu-central-1';\n  const format = options.format || 'text';\n\n  if (!options.baseTemplate || !options.targetTemplate) {\n    throw new Error('Both baseTemplate and targetTemplate are required');\n  }\n\n  const parser = new TemplateParser();\n  const diffEngine = new DiffEngine();\n  const pricingService = new PricingService();\n  const reporter = new Reporter();\n\n  try {\n    const baseTemplateObj = parser.parse(options.baseTemplate);\n    const targetTemplateObj = parser.parse(options.targetTemplate);\n\n    const diff = diffEngine.diff(baseTemplateObj, targetTemplateObj);\n\n    const costDelta = await pricingService.getCostDelta(diff, region);\n\n    const summary = reporter.generateReport(costDelta, format);\n\n    return {\n      totalDelta: costDelta.totalDelta,\n      currency: costDelta.currency,\n      addedResources: costDelta.addedCosts,\n      removedResources: costDelta.removedCosts,\n      modifiedResources: costDelta.modifiedCosts,\n      summary,\n    };\n  } finally {\n    // Clean up resources to prevent hanging connections\n    pricingService.destroy();\n  }\n}\n\n/**\n * Analyze a single CloudFormation template for estimated monthly costs\n * \n * @param options - Configuration options including template content, region, format, and analysis config\n * @returns Promise resolving to detailed cost analysis result\n * \n * @example\n * ```typescript\n * const result = await analyzeSingleTemplate({\n *   template: fs.readFileSync('template.json', 'utf-8'),\n *   region: 'us-east-1',\n *   format: 'text',\n *   config: {\n *     usageAssumptions: {\n *       lambda: { invocationsPerMonth: 1000000 }\n *     }\n *   }\n * });\n * console.log(result.summary);\n * ```\n */\nexport async function analyzeSingleTemplate(\n  options: AnalyzeSingleTemplateOptions,\n): Promise<SingleTemplateCostResult> {\n  const region = options.region || 'eu-central-1';\n  const format = options.format || 'text';\n\n  if (!options.template) {\n    throw new Error('Template content is required');\n  }\n\n  const analyzer = new SingleTemplateAnalyzer();\n  const reporter = new SingleTemplateReporter();\n\n  try {\n    // Analyze the template\n    const result = await analyzer.analyzeCosts(options.template, region, options.config);\n\n    // Generate formatted summary using the reporter\n    const summary = reporter.generateReport(result, format);\n\n    return {\n      ...result,\n      summary,\n    };\n  } catch (error) {\n    // Handle AWS credentials errors\n    if (error instanceof Error && error.message.includes('Could not load credentials')) {\n      throw new Error(\n        'AWS credentials not configured. Please set AWS credentials using one of:\\n' +\n        '  - AWS_ACCESS_KEY_ID and AWS_SECRET_ACCESS_KEY environment variables\\n' +\n        '  - AWS_PROFILE environment variable\\n' +\n        '  - AWS credentials file (~/.aws/credentials)\\n' +\n        '  - For CI/CD, configure AWS credentials in your pipeline',\n      );\n    }\n    throw error;\n  }\n}\n\n"]}
121
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/api/index.ts"],"names":[],"mappings":"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","sourcesContent":["import { AnalyzeOptions, CostAnalysisResult } from './types';\nimport { AnalyzeSingleTemplateOptions, SingleTemplateCostResult } from './single-template-types';\nimport { DiffEngine } from '../diff/DiffEngine';\nimport { TemplateParser } from '../parser/TemplateParser';\nimport { PricingService } from '../pricing/PricingService';\nimport { Reporter } from '../reporter/Reporter';\nimport { SingleTemplateReporter } from '../reporter/SingleTemplateReporter';\nimport { SingleTemplateAnalyzer } from '../analysis/SingleTemplateAnalyzer';\n\nexport * from './types';\nexport * from './single-template-types';\nexport { TemplateParseError } from '../parser/TemplateParser';\nexport { PricingAPIError, UnsupportedResourceError } from '../pricing/types';\nexport * from '../integrations';\nexport * from '../reporter/types';\nexport * from '../config';\nexport * from '../synthesis';\nexport * from '../threshold';\nexport * from '../pipeline';\nexport * from '../optimization';\n\nexport async function analyzeCosts(options: AnalyzeOptions): Promise<CostAnalysisResult> {\n  const region = options.region || 'eu-central-1';\n  const format = options.format || 'text';\n\n  if (!options.baseTemplate || !options.targetTemplate) {\n    throw new Error('Both baseTemplate and targetTemplate are required');\n  }\n\n  const parser = new TemplateParser();\n  const diffEngine = new DiffEngine();\n  const pricingService = new PricingService();\n  const reporter = new Reporter();\n\n  try {\n    const baseTemplateObj = parser.parse(options.baseTemplate);\n    const targetTemplateObj = parser.parse(options.targetTemplate);\n\n    const diff = diffEngine.diff(baseTemplateObj, targetTemplateObj);\n\n    const costDelta = await pricingService.getCostDelta(diff, region);\n\n    const summary = reporter.generateReport(costDelta, format);\n\n    return {\n      totalDelta: costDelta.totalDelta,\n      currency: costDelta.currency,\n      addedResources: costDelta.addedCosts,\n      removedResources: costDelta.removedCosts,\n      modifiedResources: costDelta.modifiedCosts,\n      summary,\n    };\n  } finally {\n    // Clean up resources to prevent hanging connections\n    pricingService.destroy();\n  }\n}\n\n/**\n * Analyze a single CloudFormation template for estimated monthly costs\n * \n * @param options - Configuration options including template content, region, format, and analysis config\n * @returns Promise resolving to detailed cost analysis result\n * \n * @example\n * ```typescript\n * const result = await analyzeSingleTemplate({\n *   template: fs.readFileSync('template.json', 'utf-8'),\n *   region: 'us-east-1',\n *   format: 'text',\n *   config: {\n *     usageAssumptions: {\n *       lambda: { invocationsPerMonth: 1000000 }\n *     }\n *   }\n * });\n * console.log(result.summary);\n * ```\n */\nexport async function analyzeSingleTemplate(\n  options: AnalyzeSingleTemplateOptions,\n): Promise<SingleTemplateCostResult> {\n  const region = options.region || 'eu-central-1';\n  const format = options.format || 'text';\n\n  if (!options.template) {\n    throw new Error('Template content is required');\n  }\n\n  const analyzer = new SingleTemplateAnalyzer();\n  const reporter = new SingleTemplateReporter();\n\n  try {\n    // Analyze the template\n    const result = await analyzer.analyzeCosts(options.template, region, options.config);\n\n    // Generate formatted summary using the reporter\n    const summary = reporter.generateReport(result, format);\n\n    return {\n      ...result,\n      summary,\n    };\n  } catch (error) {\n    // Handle AWS credentials errors\n    if (error instanceof Error && error.message.includes('Could not load credentials')) {\n      throw new Error(\n        'AWS credentials not configured. Please set AWS credentials using one of:\\n' +\n        '  - AWS_ACCESS_KEY_ID and AWS_SECRET_ACCESS_KEY environment variables\\n' +\n        '  - AWS_PROFILE environment variable\\n' +\n        '  - AWS credentials file (~/.aws/credentials)\\n' +\n        '  - For CI/CD, configure AWS credentials in your pipeline',\n      );\n    }\n    throw error;\n  }\n}\n\n"]}
@@ -1,5 +1,6 @@
1
1
  import { ResourceCost } from '../pricing/types';
2
2
  import { UsageAssumptionsConfig, CacheConfig } from '../config/types';
3
+ import { OptimizationResult } from '../optimization/types';
3
4
  /**
4
5
  * Configuration options for single template analysis
5
6
  */
@@ -7,6 +8,8 @@ export interface AnalysisConfig {
7
8
  usageAssumptions?: UsageAssumptionsConfig;
8
9
  excludedResourceTypes?: string[];
9
10
  cacheConfig?: CacheConfig;
11
+ recommendations?: boolean;
12
+ minimumSavingsThreshold?: number;
10
13
  }
11
14
  /**
12
15
  * Metadata about the analysis
@@ -62,6 +65,7 @@ export interface SingleTemplateCostResult {
62
65
  costBreakdown: CostBreakdown;
63
66
  summary: string;
64
67
  metadata: AnalysisMetadata;
68
+ recommendations?: OptimizationResult;
65
69
  }
66
70
  /**
67
71
  * Options for analyzing a single template
@@ -1,3 +1,3 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2luZ2xlLXRlbXBsYXRlLXR5cGVzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2FwaS9zaW5nbGUtdGVtcGxhdGUtdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IFJlc291cmNlQ29zdCB9IGZyb20gJy4uL3ByaWNpbmcvdHlwZXMnO1xuaW1wb3J0IHsgVXNhZ2VBc3N1bXB0aW9uc0NvbmZpZywgQ2FjaGVDb25maWcgfSBmcm9tICcuLi9jb25maWcvdHlwZXMnO1xuXG4vKipcbiAqIENvbmZpZ3VyYXRpb24gb3B0aW9ucyBmb3Igc2luZ2xlIHRlbXBsYXRlIGFuYWx5c2lzXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgQW5hbHlzaXNDb25maWcge1xuICB1c2FnZUFzc3VtcHRpb25zPzogVXNhZ2VBc3N1bXB0aW9uc0NvbmZpZztcbiAgZXhjbHVkZWRSZXNvdXJjZVR5cGVzPzogc3RyaW5nW107XG4gIGNhY2hlQ29uZmlnPzogQ2FjaGVDb25maWc7XG59XG5cbi8qKlxuICogTWV0YWRhdGEgYWJvdXQgdGhlIGFuYWx5c2lzXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgQW5hbHlzaXNNZXRhZGF0YSB7XG4gIHRlbXBsYXRlSGFzaDogc3RyaW5nO1xuICByZWdpb246IHN0cmluZztcbiAgYW5hbHl6ZWRBdDogRGF0ZTtcbiAgcmVzb3VyY2VDb3VudDogbnVtYmVyO1xuICBzdXBwb3J0ZWRSZXNvdXJjZUNvdW50OiBudW1iZXI7XG4gIHVuc3VwcG9ydGVkUmVzb3VyY2VDb3VudDogbnVtYmVyO1xufVxuXG4vKipcbiAqIENvc3QgaW5mb3JtYXRpb24gZ3JvdXBlZCBieSByZXNvdXJjZSB0eXBlXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgUmVzb3VyY2VUeXBlQ29zdCB7XG4gIHJlc291cmNlVHlwZTogc3RyaW5nO1xuICBjb3VudDogbnVtYmVyO1xuICB0b3RhbENvc3Q6IG51bWJlcjtcbiAgcmVzb3VyY2VzOiBSZXNvdXJjZUNvc3RbXTtcbn1cblxuLyoqXG4gKiBDb3N0IGluZm9ybWF0aW9uIGdyb3VwZWQgYnkgY29uZmlkZW5jZSBsZXZlbFxuICovXG5leHBvcnQgaW50ZXJmYWNlIENvbmZpZGVuY2VMZXZlbENvc3Qge1xuICBjb25maWRlbmNlOiAnaGlnaCcgfCAnbWVkaXVtJyB8ICdsb3cnIHwgJ3Vua25vd24nO1xuICBjb3VudDogbnVtYmVyO1xuICB0b3RhbENvc3Q6IG51bWJlcjtcbn1cblxuLyoqXG4gKiBEZXRhaWxlZCBjb3N0IGJyZWFrZG93biBmb3IgYSB0ZW1wbGF0ZVxuICovXG5leHBvcnQgaW50ZXJmYWNlIENvc3RCcmVha2Rvd24ge1xuICBieVJlc291cmNlVHlwZTogUmVzb3VyY2VUeXBlQ29zdFtdO1xuICBieUNvbmZpZGVuY2VMZXZlbDogQ29uZmlkZW5jZUxldmVsQ29zdFtdO1xuICBhc3N1bXB0aW9uczogc3RyaW5nW107XG59XG5cbi8qKlxuICogRW5oYW5jZWQgcmVzb3VyY2UgY29zdCB3aXRoIGFkZGl0aW9uYWwgbWV0YWRhdGFcbiAqL1xuZXhwb3J0IGludGVyZmFjZSBFbmhhbmNlZFJlc291cmNlQ29zdCBleHRlbmRzIFJlc291cmNlQ29zdCB7XG4gIHByb3BlcnRpZXM/OiBSZWNvcmQ8c3RyaW5nLCBhbnk+O1xuICByZWdpb246IHN0cmluZztcbiAgY2FsY3VsYXRlZEF0OiBEYXRlO1xufVxuXG4vKipcbiAqIFJlc3VsdCBvZiBzaW5nbGUgdGVtcGxhdGUgY29zdCBhbmFseXNpc1xuICovXG5leHBvcnQgaW50ZXJmYWNlIFNpbmdsZVRlbXBsYXRlQ29zdFJlc3VsdCB7XG4gIHRvdGFsTW9udGhseUNvc3Q6IG51bWJlcjtcbiAgY3VycmVuY3k6IHN0cmluZztcbiAgcmVzb3VyY2VDb3N0czogRW5oYW5jZWRSZXNvdXJjZUNvc3RbXTtcbiAgY29zdEJyZWFrZG93bjogQ29zdEJyZWFrZG93bjtcbiAgc3VtbWFyeTogc3RyaW5nO1xuICBtZXRhZGF0YTogQW5hbHlzaXNNZXRhZGF0YTtcbn1cblxuLyoqXG4gKiBPcHRpb25zIGZvciBhbmFseXppbmcgYSBzaW5nbGUgdGVtcGxhdGVcbiAqL1xuZXhwb3J0IGludGVyZmFjZSBBbmFseXplU2luZ2xlVGVtcGxhdGVPcHRpb25zIHtcbiAgdGVtcGxhdGU6IHN0cmluZztcbiAgcmVnaW9uPzogc3RyaW5nO1xuICBmb3JtYXQ/OiAndGV4dCcgfCAnanNvbicgfCAnbWFya2Rvd24nO1xuICBjb25maWc/OiBBbmFseXNpc0NvbmZpZztcbn1cblxuLyoqXG4gKiBPcHRpb25zIGZvciBnZW5lcmF0aW5nIHNpbmdsZSB0ZW1wbGF0ZSByZXBvcnRzXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgU2luZ2xlVGVtcGxhdGVSZXBvcnRPcHRpb25zIHtcbiAgc2hvd0JyZWFrZG93bj86IGJvb2xlYW47XG4gIHNob3dBc3N1bXB0aW9ucz86IGJvb2xlYW47XG4gIGdyb3VwQnlUeXBlPzogYm9vbGVhbjtcbiAgc29ydEJ5PzogJ2Nvc3QnIHwgJ25hbWUnIHwgJ3R5cGUnO1xufVxuIl19
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2luZ2xlLXRlbXBsYXRlLXR5cGVzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2FwaS9zaW5nbGUtdGVtcGxhdGUtdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IFJlc291cmNlQ29zdCB9IGZyb20gJy4uL3ByaWNpbmcvdHlwZXMnO1xuaW1wb3J0IHsgVXNhZ2VBc3N1bXB0aW9uc0NvbmZpZywgQ2FjaGVDb25maWcgfSBmcm9tICcuLi9jb25maWcvdHlwZXMnO1xuaW1wb3J0IHsgT3B0aW1pemF0aW9uUmVzdWx0IH0gZnJvbSAnLi4vb3B0aW1pemF0aW9uL3R5cGVzJztcblxuLyoqXG4gKiBDb25maWd1cmF0aW9uIG9wdGlvbnMgZm9yIHNpbmdsZSB0ZW1wbGF0ZSBhbmFseXNpc1xuICovXG5leHBvcnQgaW50ZXJmYWNlIEFuYWx5c2lzQ29uZmlnIHtcbiAgdXNhZ2VBc3N1bXB0aW9ucz86IFVzYWdlQXNzdW1wdGlvbnNDb25maWc7XG4gIGV4Y2x1ZGVkUmVzb3VyY2VUeXBlcz86IHN0cmluZ1tdO1xuICBjYWNoZUNvbmZpZz86IENhY2hlQ29uZmlnO1xuICByZWNvbW1lbmRhdGlvbnM/OiBib29sZWFuO1xuICBtaW5pbXVtU2F2aW5nc1RocmVzaG9sZD86IG51bWJlcjtcbn1cblxuLyoqXG4gKiBNZXRhZGF0YSBhYm91dCB0aGUgYW5hbHlzaXNcbiAqL1xuZXhwb3J0IGludGVyZmFjZSBBbmFseXNpc01ldGFkYXRhIHtcbiAgdGVtcGxhdGVIYXNoOiBzdHJpbmc7XG4gIHJlZ2lvbjogc3RyaW5nO1xuICBhbmFseXplZEF0OiBEYXRlO1xuICByZXNvdXJjZUNvdW50OiBudW1iZXI7XG4gIHN1cHBvcnRlZFJlc291cmNlQ291bnQ6IG51bWJlcjtcbiAgdW5zdXBwb3J0ZWRSZXNvdXJjZUNvdW50OiBudW1iZXI7XG59XG5cbi8qKlxuICogQ29zdCBpbmZvcm1hdGlvbiBncm91cGVkIGJ5IHJlc291cmNlIHR5cGVcbiAqL1xuZXhwb3J0IGludGVyZmFjZSBSZXNvdXJjZVR5cGVDb3N0IHtcbiAgcmVzb3VyY2VUeXBlOiBzdHJpbmc7XG4gIGNvdW50OiBudW1iZXI7XG4gIHRvdGFsQ29zdDogbnVtYmVyO1xuICByZXNvdXJjZXM6IFJlc291cmNlQ29zdFtdO1xufVxuXG4vKipcbiAqIENvc3QgaW5mb3JtYXRpb24gZ3JvdXBlZCBieSBjb25maWRlbmNlIGxldmVsXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgQ29uZmlkZW5jZUxldmVsQ29zdCB7XG4gIGNvbmZpZGVuY2U6ICdoaWdoJyB8ICdtZWRpdW0nIHwgJ2xvdycgfCAndW5rbm93bic7XG4gIGNvdW50OiBudW1iZXI7XG4gIHRvdGFsQ29zdDogbnVtYmVyO1xufVxuXG4vKipcbiAqIERldGFpbGVkIGNvc3QgYnJlYWtkb3duIGZvciBhIHRlbXBsYXRlXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgQ29zdEJyZWFrZG93biB7XG4gIGJ5UmVzb3VyY2VUeXBlOiBSZXNvdXJjZVR5cGVDb3N0W107XG4gIGJ5Q29uZmlkZW5jZUxldmVsOiBDb25maWRlbmNlTGV2ZWxDb3N0W107XG4gIGFzc3VtcHRpb25zOiBzdHJpbmdbXTtcbn1cblxuLyoqXG4gKiBFbmhhbmNlZCByZXNvdXJjZSBjb3N0IHdpdGggYWRkaXRpb25hbCBtZXRhZGF0YVxuICovXG5leHBvcnQgaW50ZXJmYWNlIEVuaGFuY2VkUmVzb3VyY2VDb3N0IGV4dGVuZHMgUmVzb3VyY2VDb3N0IHtcbiAgcHJvcGVydGllcz86IFJlY29yZDxzdHJpbmcsIGFueT47XG4gIHJlZ2lvbjogc3RyaW5nO1xuICBjYWxjdWxhdGVkQXQ6IERhdGU7XG59XG5cbi8qKlxuICogUmVzdWx0IG9mIHNpbmdsZSB0ZW1wbGF0ZSBjb3N0IGFuYWx5c2lzXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgU2luZ2xlVGVtcGxhdGVDb3N0UmVzdWx0IHtcbiAgdG90YWxNb250aGx5Q29zdDogbnVtYmVyO1xuICBjdXJyZW5jeTogc3RyaW5nO1xuICByZXNvdXJjZUNvc3RzOiBFbmhhbmNlZFJlc291cmNlQ29zdFtdO1xuICBjb3N0QnJlYWtkb3duOiBDb3N0QnJlYWtkb3duO1xuICBzdW1tYXJ5OiBzdHJpbmc7XG4gIG1ldGFkYXRhOiBBbmFseXNpc01ldGFkYXRhO1xuICByZWNvbW1lbmRhdGlvbnM/OiBPcHRpbWl6YXRpb25SZXN1bHQ7XG59XG5cbi8qKlxuICogT3B0aW9ucyBmb3IgYW5hbHl6aW5nIGEgc2luZ2xlIHRlbXBsYXRlXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgQW5hbHl6ZVNpbmdsZVRlbXBsYXRlT3B0aW9ucyB7XG4gIHRlbXBsYXRlOiBzdHJpbmc7XG4gIHJlZ2lvbj86IHN0cmluZztcbiAgZm9ybWF0PzogJ3RleHQnIHwgJ2pzb24nIHwgJ21hcmtkb3duJztcbiAgY29uZmlnPzogQW5hbHlzaXNDb25maWc7XG59XG5cbi8qKlxuICogT3B0aW9ucyBmb3IgZ2VuZXJhdGluZyBzaW5nbGUgdGVtcGxhdGUgcmVwb3J0c1xuICovXG5leHBvcnQgaW50ZXJmYWNlIFNpbmdsZVRlbXBsYXRlUmVwb3J0T3B0aW9ucyB7XG4gIHNob3dCcmVha2Rvd24/OiBib29sZWFuO1xuICBzaG93QXNzdW1wdGlvbnM/OiBib29sZWFuO1xuICBncm91cEJ5VHlwZT86IGJvb2xlYW47XG4gIHNvcnRCeT86ICdjb3N0JyB8ICduYW1lJyB8ICd0eXBlJztcbn1cbiJdfQ==
@@ -1,9 +1,11 @@
1
1
  import { ResourceCost, ModifiedResourceCost } from '../pricing/types';
2
+ import { OptimizationResult } from '../optimization/types';
2
3
  export interface AnalyzeOptions {
3
4
  baseTemplate: string;
4
5
  targetTemplate: string;
5
6
  region?: string;
6
7
  format?: 'text' | 'json' | 'markdown';
8
+ recommendations?: boolean;
7
9
  }
8
10
  export interface CostAnalysisResult {
9
11
  totalDelta: number;
@@ -12,4 +14,5 @@ export interface CostAnalysisResult {
12
14
  removedResources: ResourceCost[];
13
15
  modifiedResources: ModifiedResourceCost[];
14
16
  summary: string;
17
+ recommendations?: OptimizationResult;
15
18
  }
package/dist/api/types.js CHANGED
@@ -1,3 +1,3 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvYXBpL3R5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBSZXNvdXJjZUNvc3QsIE1vZGlmaWVkUmVzb3VyY2VDb3N0IH0gZnJvbSAnLi4vcHJpY2luZy90eXBlcyc7XG5cbmV4cG9ydCBpbnRlcmZhY2UgQW5hbHl6ZU9wdGlvbnMge1xuICBiYXNlVGVtcGxhdGU6IHN0cmluZztcbiAgdGFyZ2V0VGVtcGxhdGU6IHN0cmluZztcbiAgcmVnaW9uPzogc3RyaW5nO1xuICBmb3JtYXQ/OiAndGV4dCcgfCAnanNvbicgfCAnbWFya2Rvd24nO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIENvc3RBbmFseXNpc1Jlc3VsdCB7XG4gIHRvdGFsRGVsdGE6IG51bWJlcjtcbiAgY3VycmVuY3k6IHN0cmluZztcbiAgYWRkZWRSZXNvdXJjZXM6IFJlc291cmNlQ29zdFtdO1xuICByZW1vdmVkUmVzb3VyY2VzOiBSZXNvdXJjZUNvc3RbXTtcbiAgbW9kaWZpZWRSZXNvdXJjZXM6IE1vZGlmaWVkUmVzb3VyY2VDb3N0W107XG4gIHN1bW1hcnk6IHN0cmluZztcbn1cbiJdfQ==
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvYXBpL3R5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBSZXNvdXJjZUNvc3QsIE1vZGlmaWVkUmVzb3VyY2VDb3N0IH0gZnJvbSAnLi4vcHJpY2luZy90eXBlcyc7XG5pbXBvcnQgeyBPcHRpbWl6YXRpb25SZXN1bHQgfSBmcm9tICcuLi9vcHRpbWl6YXRpb24vdHlwZXMnO1xuXG5leHBvcnQgaW50ZXJmYWNlIEFuYWx5emVPcHRpb25zIHtcbiAgYmFzZVRlbXBsYXRlOiBzdHJpbmc7XG4gIHRhcmdldFRlbXBsYXRlOiBzdHJpbmc7XG4gIHJlZ2lvbj86IHN0cmluZztcbiAgZm9ybWF0PzogJ3RleHQnIHwgJ2pzb24nIHwgJ21hcmtkb3duJztcbiAgcmVjb21tZW5kYXRpb25zPzogYm9vbGVhbjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBDb3N0QW5hbHlzaXNSZXN1bHQge1xuICB0b3RhbERlbHRhOiBudW1iZXI7XG4gIGN1cnJlbmN5OiBzdHJpbmc7XG4gIGFkZGVkUmVzb3VyY2VzOiBSZXNvdXJjZUNvc3RbXTtcbiAgcmVtb3ZlZFJlc291cmNlczogUmVzb3VyY2VDb3N0W107XG4gIG1vZGlmaWVkUmVzb3VyY2VzOiBNb2RpZmllZFJlc291cmNlQ29zdFtdO1xuICBzdW1tYXJ5OiBzdHJpbmc7XG4gIHJlY29tbWVuZGF0aW9ucz86IE9wdGltaXphdGlvblJlc3VsdDtcbn1cbiJdfQ==