cdk-cost-analyzer 0.1.32 → 0.1.33

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.
@@ -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": 1770972967203
5
+ "timestamp": 1770977254582
6
6
  },
7
7
  "AmazonDynamoDB:US East (N. Virginia):group:DDB-ReadUnits|productFamily:Amazon DynamoDB PayPerRequest Throughput": {
8
8
  "price": 0.023,
9
- "timestamp": 1770972967209
9
+ "timestamp": 1770977254588
10
10
  },
11
11
  "AmazonDynamoDB:US East (N. Virginia):group:DDB-WriteUnits|productFamily:Amazon DynamoDB PayPerRequest Throughput": {
12
12
  "price": 0.023,
13
- "timestamp": 1770972967210
13
+ "timestamp": 1770977254588
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": 1770972967219
17
+ "timestamp": 1770977254597
18
18
  },
19
19
  "AWSLambda:US East (N. Virginia):group:AWS-Lambda-Requests": {
20
20
  "price": 0.023,
21
- "timestamp": 1770972967229
21
+ "timestamp": 1770977254605
22
22
  },
23
23
  "AWSLambda:US East (N. Virginia):group:AWS-Lambda-Duration": {
24
24
  "price": 0.023,
25
- "timestamp": 1770972967229
25
+ "timestamp": 1770977254605
26
26
  },
27
27
  "AmazonS3:EU (Frankfurt):storageClass:General Purpose|volumeType:Standard": {
28
28
  "price": 0.023,
29
- "timestamp": 1770972974708
29
+ "timestamp": 1770977262339
30
30
  },
31
31
  "AmazonS3:invalid-region-123:storageClass:General Purpose|volumeType:Standard": {
32
32
  "price": 0.023,
33
- "timestamp": 1770972974758
33
+ "timestamp": 1770977262394
34
34
  }
35
35
  }
36
36
  }
@@ -62,6 +62,18 @@ export interface UsageAssumptionsConfig {
62
62
  vpcEndpoint?: {
63
63
  dataProcessedGB?: number;
64
64
  };
65
+ /**
66
+ * EFS (Elastic File System) usage assumptions for cost estimation.
67
+ * These values are used to estimate monthly costs for EFS file systems.
68
+ *
69
+ * @see https://aws.amazon.com/efs/pricing/
70
+ */
71
+ efs?: {
72
+ /** Total storage size in GB (default: 100) */
73
+ storageSizeGb?: number;
74
+ /** Percentage of storage in Infrequent Access class (default: 0, range: 0-100) */
75
+ infrequentAccessPercentage?: number;
76
+ };
65
77
  /**
66
78
  * SNS (Simple Notification Service) usage assumptions.
67
79
  * These values are used to estimate monthly costs for SNS topics.
@@ -106,6 +118,15 @@ export interface UsageAssumptionsConfig {
106
118
  /** Average execution duration in milliseconds (default: 1000, for Express workflows) */
107
119
  averageDurationMs?: number;
108
120
  };
121
+ /**
122
+ * Secrets Manager usage assumptions for cost estimation.
123
+ *
124
+ * @see https://aws.amazon.com/secrets-manager/pricing/
125
+ */
126
+ secretsManager?: {
127
+ /** Number of API calls per month (default: 10,000) */
128
+ monthlyApiCalls?: number;
129
+ };
109
130
  }
110
131
  export interface SynthesisConfig {
111
132
  appPath?: string;
@@ -12,4 +12,4 @@ class ConfigurationError extends Error {
12
12
  }
13
13
  }
14
14
  exports.ConfigurationError = ConfigurationError;
15
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvY29uZmlnL3R5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQXVJQSxNQUFhLGtCQUFtQixTQUFRLEtBQUs7SUFHbEM7SUFDQTtJQUhULFlBQ0UsT0FBZSxFQUNSLFVBQWtCLEVBQ2xCLGdCQUEwQjtRQUVqQyxLQUFLLENBQUMsT0FBTyxDQUFDLENBQUM7UUFIUixlQUFVLEdBQVYsVUFBVSxDQUFRO1FBQ2xCLHFCQUFnQixHQUFoQixnQkFBZ0IsQ0FBVTtRQUdqQyxJQUFJLENBQUMsSUFBSSxHQUFHLG9CQUFvQixDQUFDO0lBQ25DLENBQUM7Q0FDRjtBQVRELGdEQVNDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IGludGVyZmFjZSBDb3N0QW5hbHl6ZXJDb25maWcge1xuICB0aHJlc2hvbGRzPzogVGhyZXNob2xkQ29uZmlnO1xuICB1c2FnZUFzc3VtcHRpb25zPzogVXNhZ2VBc3N1bXB0aW9uc0NvbmZpZztcbiAgc3ludGhlc2lzPzogU3ludGhlc2lzQ29uZmlnO1xuICBleGNsdXNpb25zPzogRXhjbHVzaW9uc0NvbmZpZztcbiAgY2FjaGU/OiBDYWNoZUNvbmZpZztcbn1cblxuZXhwb3J0IGludGVyZmFjZSBUaHJlc2hvbGRDb25maWcge1xuICBkZWZhdWx0PzogVGhyZXNob2xkTGV2ZWxzO1xuICBlbnZpcm9ubWVudHM/OiBSZWNvcmQ8c3RyaW5nLCBUaHJlc2hvbGRMZXZlbHM+O1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFRocmVzaG9sZExldmVscyB7XG4gIHdhcm5pbmc/OiBudW1iZXI7XG4gIGVycm9yPzogbnVtYmVyO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFVzYWdlQXNzdW1wdGlvbnNDb25maWcge1xuICBzMz86IHtcbiAgICBzdG9yYWdlR0I/OiBudW1iZXI7XG4gICAgZ2V0UmVxdWVzdHM/OiBudW1iZXI7XG4gICAgcHV0UmVxdWVzdHM/OiBudW1iZXI7XG4gIH07XG4gIGxhbWJkYT86IHtcbiAgICBpbnZvY2F0aW9uc1Blck1vbnRoPzogbnVtYmVyO1xuICAgIGF2ZXJhZ2VEdXJhdGlvbk1zPzogbnVtYmVyO1xuICB9O1xuICAvKipcbiAgICogRHluYW1vREIgdXNhZ2UgYXNzdW1wdGlvbnMgZm9yIG9uLWRlbWFuZCAocGF5LXBlci1yZXF1ZXN0KSBiaWxsaW5nIG1vZGUuXG4gICAqIFRoZXNlIHZhbHVlcyBhcmUgdXNlZCB0byBlc3RpbWF0ZSBtb250aGx5IGNvc3RzIGZvciBEeW5hbW9EQiB0YWJsZXNcbiAgICogY29uZmlndXJlZCB3aXRoIEJpbGxpbmdNb2RlOiBQQVlfUEVSX1JFUVVFU1QuXG4gICAqXG4gICAqIEZvciBwcm92aXNpb25lZCBiaWxsaW5nIG1vZGUsIGNvc3RzIGFyZSBjYWxjdWxhdGVkIGJhc2VkIG9uIHRoZVxuICAgKiBSZWFkQ2FwYWNpdHlVbml0cyBhbmQgV3JpdGVDYXBhY2l0eVVuaXRzIHNwZWNpZmllZCBpbiB0aGUgdGVtcGxhdGUuXG4gICAqXG4gICAqIEBzZWUgaHR0cHM6Ly9hd3MuYW1hem9uLmNvbS9keW5hbW9kYi9wcmljaW5nL1xuICAgKi9cbiAgZHluYW1vZGI/OiB7XG4gICAgLyoqIE51bWJlciBvZiByZWFkIHJlcXVlc3RzIHBlciBtb250aCAoZGVmYXVsdDogMTAsMDAwLDAwMCkgKi9cbiAgICByZWFkUmVxdWVzdHNQZXJNb250aD86IG51bWJlcjtcbiAgICAvKiogTnVtYmVyIG9mIHdyaXRlIHJlcXVlc3RzIHBlciBtb250aCAoZGVmYXVsdDogMSwwMDAsMDAwKSAqL1xuICAgIHdyaXRlUmVxdWVzdHNQZXJNb250aD86IG51bWJlcjtcbiAgfTtcbiAgbmF0R2F0ZXdheT86IHtcbiAgICBkYXRhUHJvY2Vzc2VkR0I/OiBudW1iZXI7XG4gIH07XG4gIGFsYj86IHtcbiAgICBuZXdDb25uZWN0aW9uc1BlclNlY29uZD86IG51bWJlcjtcbiAgICBhY3RpdmVDb25uZWN0aW9uc1Blck1pbnV0ZT86IG51bWJlcjtcbiAgICBwcm9jZXNzZWRCeXRlc0dCPzogbnVtYmVyO1xuICB9O1xuICBubGI/OiB7XG4gICAgbmV3Q29ubmVjdGlvbnNQZXJTZWNvbmQ/OiBudW1iZXI7XG4gICAgYWN0aXZlQ29ubmVjdGlvbnNQZXJNaW51dGU/OiBudW1iZXI7XG4gICAgcHJvY2Vzc2VkQnl0ZXNHQj86IG51bWJlcjtcbiAgfTtcbiAgY2xvdWRmcm9udD86IHtcbiAgICBkYXRhVHJhbnNmZXJHQj86IG51bWJlcjtcbiAgICByZXF1ZXN0cz86IG51bWJlcjtcbiAgfTtcbiAgYXBpR2F0ZXdheT86IHtcbiAgICByZXF1ZXN0c1Blck1vbnRoPzogbnVtYmVyO1xuICB9O1xuICB2cGNFbmRwb2ludD86IHtcbiAgICBkYXRhUHJvY2Vzc2VkR0I/OiBudW1iZXI7XG4gIH07XG4gIC8qKlxuICAgKiBTTlMgKFNpbXBsZSBOb3RpZmljYXRpb24gU2VydmljZSkgdXNhZ2UgYXNzdW1wdGlvbnMuXG4gICAqIFRoZXNlIHZhbHVlcyBhcmUgdXNlZCB0byBlc3RpbWF0ZSBtb250aGx5IGNvc3RzIGZvciBTTlMgdG9waWNzLlxuICAgKlxuICAgKiBAc2VlIGh0dHBzOi8vYXdzLmFtYXpvbi5jb20vc25zL3ByaWNpbmcvXG4gICAqL1xuICBzbnM/OiB7XG4gICAgLyoqIE51bWJlciBvZiBwdWJsaXNoIHJlcXVlc3RzIHBlciBtb250aCAoZGVmYXVsdDogMSwwMDAsMDAwKSAqL1xuICAgIG1vbnRobHlQdWJsaXNoZXM/OiBudW1iZXI7XG4gICAgLyoqIE51bWJlciBvZiBIVFRQL1MgZGVsaXZlcmllcyBwZXIgbW9udGggKGRlZmF1bHQ6IDEsMDAwLDAwMCkgKi9cbiAgICBodHRwRGVsaXZlcmllcz86IG51bWJlcjtcbiAgICAvKiogTnVtYmVyIG9mIGVtYWlsIGRlbGl2ZXJpZXMgcGVyIG1vbnRoIChkZWZhdWx0OiAwKSAqL1xuICAgIGVtYWlsRGVsaXZlcmllcz86IG51bWJlcjtcbiAgICAvKiogTnVtYmVyIG9mIFNNUyBkZWxpdmVyaWVzIHBlciBtb250aCAoZGVmYXVsdDogMCkgKi9cbiAgICBzbXNEZWxpdmVyaWVzPzogbnVtYmVyO1xuICAgIC8qKiBOdW1iZXIgb2YgbW9iaWxlIHB1c2ggZGVsaXZlcmllcyBwZXIgbW9udGggKGRlZmF1bHQ6IDApICovXG4gICAgbW9iaWxlUHVzaERlbGl2ZXJpZXM/OiBudW1iZXI7XG4gIH07XG4gIC8qKlxuICAgKiBTUVMgdXNhZ2UgYXNzdW1wdGlvbnMgZm9yIGNvc3QgZXN0aW1hdGlvbi5cbiAgICogQXBwbGllcyB0byBib3RoIFN0YW5kYXJkIGFuZCBGSUZPIHF1ZXVlcy5cbiAgICpcbiAgICogQHNlZSBodHRwczovL2F3cy5hbWF6b24uY29tL3Nxcy9wcmljaW5nL1xuICAgKi9cbiAgc3FzPzoge1xuICAgIC8qKiBOdW1iZXIgb2YgcmVxdWVzdHMgcGVyIG1vbnRoIChkZWZhdWx0OiAxLDAwMCwwMDApICovXG4gICAgbW9udGhseVJlcXVlc3RzPzogbnVtYmVyO1xuICB9O1xuICAvKipcbiAgICogU3RlcCBGdW5jdGlvbnMgdXNhZ2UgYXNzdW1wdGlvbnMgZm9yIGNvc3QgZXN0aW1hdGlvbi5cbiAgICpcbiAgICogU3RhbmRhcmQgd29ya2Zsb3dzIGFyZSBjaGFyZ2VkIHBlciBzdGF0ZSB0cmFuc2l0aW9uLlxuICAgKiBFeHByZXNzIHdvcmtmbG93cyBhcmUgY2hhcmdlZCBwZXIgcmVxdWVzdCBhbmQgcGVyIEdCLXNlY29uZCBvZiBkdXJhdGlvbi5cbiAgICpcbiAgICogQHNlZSBodHRwczovL2F3cy5hbWF6b24uY29tL3N0ZXAtZnVuY3Rpb25zL3ByaWNpbmcvXG4gICAqL1xuICBzdGVwRnVuY3Rpb25zPzoge1xuICAgIC8qKiBOdW1iZXIgb2Ygd29ya2Zsb3cgZXhlY3V0aW9ucyBwZXIgbW9udGggKGRlZmF1bHQ6IDEwLDAwMCkgKi9cbiAgICBtb250aGx5RXhlY3V0aW9ucz86IG51bWJlcjtcbiAgICAvKiogTnVtYmVyIG9mIHN0YXRlIHRyYW5zaXRpb25zIHBlciBleGVjdXRpb24gKGRlZmF1bHQ6IDEwLCBmb3IgU3RhbmRhcmQgd29ya2Zsb3dzKSAqL1xuICAgIHN0YXRlVHJhbnNpdGlvbnNQZXJFeGVjdXRpb24/OiBudW1iZXI7XG4gICAgLyoqIEF2ZXJhZ2UgZXhlY3V0aW9uIGR1cmF0aW9uIGluIG1pbGxpc2Vjb25kcyAoZGVmYXVsdDogMTAwMCwgZm9yIEV4cHJlc3Mgd29ya2Zsb3dzKSAqL1xuICAgIGF2ZXJhZ2VEdXJhdGlvbk1zPzogbnVtYmVyO1xuICB9O1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFN5bnRoZXNpc0NvbmZpZyB7XG4gIGFwcFBhdGg/OiBzdHJpbmc7XG4gIG91dHB1dFBhdGg/OiBzdHJpbmc7XG4gIGN1c3RvbUNvbW1hbmQ/OiBzdHJpbmc7XG4gIGNvbnRleHQ/OiBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+O1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIEV4Y2x1c2lvbnNDb25maWcge1xuICByZXNvdXJjZVR5cGVzPzogc3RyaW5nW107XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQ2FjaGVDb25maWcge1xuICBlbmFibGVkPzogYm9vbGVhbjtcbiAgZHVyYXRpb25Ib3Vycz86IG51bWJlcjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBWYWxpZGF0aW9uUmVzdWx0IHtcbiAgdmFsaWQ6IGJvb2xlYW47XG4gIGVycm9yczogc3RyaW5nW107XG4gIHdhcm5pbmdzOiBzdHJpbmdbXTtcbn1cblxuZXhwb3J0IGNsYXNzIENvbmZpZ3VyYXRpb25FcnJvciBleHRlbmRzIEVycm9yIHtcbiAgY29uc3RydWN0b3IoXG4gICAgbWVzc2FnZTogc3RyaW5nLFxuICAgIHB1YmxpYyBjb25maWdQYXRoOiBzdHJpbmcsXG4gICAgcHVibGljIHZhbGlkYXRpb25FcnJvcnM6IHN0cmluZ1tdLFxuICApIHtcbiAgICBzdXBlcihtZXNzYWdlKTtcbiAgICB0aGlzLm5hbWUgPSAnQ29uZmlndXJhdGlvbkVycm9yJztcbiAgfVxufVxuIl19
15
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvY29uZmlnL3R5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQTRKQSxNQUFhLGtCQUFtQixTQUFRLEtBQUs7SUFHbEM7SUFDQTtJQUhULFlBQ0UsT0FBZSxFQUNSLFVBQWtCLEVBQ2xCLGdCQUEwQjtRQUVqQyxLQUFLLENBQUMsT0FBTyxDQUFDLENBQUM7UUFIUixlQUFVLEdBQVYsVUFBVSxDQUFRO1FBQ2xCLHFCQUFnQixHQUFoQixnQkFBZ0IsQ0FBVTtRQUdqQyxJQUFJLENBQUMsSUFBSSxHQUFHLG9CQUFvQixDQUFDO0lBQ25DLENBQUM7Q0FDRjtBQVRELGdEQVNDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IGludGVyZmFjZSBDb3N0QW5hbHl6ZXJDb25maWcge1xuICB0aHJlc2hvbGRzPzogVGhyZXNob2xkQ29uZmlnO1xuICB1c2FnZUFzc3VtcHRpb25zPzogVXNhZ2VBc3N1bXB0aW9uc0NvbmZpZztcbiAgc3ludGhlc2lzPzogU3ludGhlc2lzQ29uZmlnO1xuICBleGNsdXNpb25zPzogRXhjbHVzaW9uc0NvbmZpZztcbiAgY2FjaGU/OiBDYWNoZUNvbmZpZztcbn1cblxuZXhwb3J0IGludGVyZmFjZSBUaHJlc2hvbGRDb25maWcge1xuICBkZWZhdWx0PzogVGhyZXNob2xkTGV2ZWxzO1xuICBlbnZpcm9ubWVudHM/OiBSZWNvcmQ8c3RyaW5nLCBUaHJlc2hvbGRMZXZlbHM+O1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFRocmVzaG9sZExldmVscyB7XG4gIHdhcm5pbmc/OiBudW1iZXI7XG4gIGVycm9yPzogbnVtYmVyO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFVzYWdlQXNzdW1wdGlvbnNDb25maWcge1xuICBzMz86IHtcbiAgICBzdG9yYWdlR0I/OiBudW1iZXI7XG4gICAgZ2V0UmVxdWVzdHM/OiBudW1iZXI7XG4gICAgcHV0UmVxdWVzdHM/OiBudW1iZXI7XG4gIH07XG4gIGxhbWJkYT86IHtcbiAgICBpbnZvY2F0aW9uc1Blck1vbnRoPzogbnVtYmVyO1xuICAgIGF2ZXJhZ2VEdXJhdGlvbk1zPzogbnVtYmVyO1xuICB9O1xuICAvKipcbiAgICogRHluYW1vREIgdXNhZ2UgYXNzdW1wdGlvbnMgZm9yIG9uLWRlbWFuZCAocGF5LXBlci1yZXF1ZXN0KSBiaWxsaW5nIG1vZGUuXG4gICAqIFRoZXNlIHZhbHVlcyBhcmUgdXNlZCB0byBlc3RpbWF0ZSBtb250aGx5IGNvc3RzIGZvciBEeW5hbW9EQiB0YWJsZXNcbiAgICogY29uZmlndXJlZCB3aXRoIEJpbGxpbmdNb2RlOiBQQVlfUEVSX1JFUVVFU1QuXG4gICAqXG4gICAqIEZvciBwcm92aXNpb25lZCBiaWxsaW5nIG1vZGUsIGNvc3RzIGFyZSBjYWxjdWxhdGVkIGJhc2VkIG9uIHRoZVxuICAgKiBSZWFkQ2FwYWNpdHlVbml0cyBhbmQgV3JpdGVDYXBhY2l0eVVuaXRzIHNwZWNpZmllZCBpbiB0aGUgdGVtcGxhdGUuXG4gICAqXG4gICAqIEBzZWUgaHR0cHM6Ly9hd3MuYW1hem9uLmNvbS9keW5hbW9kYi9wcmljaW5nL1xuICAgKi9cbiAgZHluYW1vZGI/OiB7XG4gICAgLyoqIE51bWJlciBvZiByZWFkIHJlcXVlc3RzIHBlciBtb250aCAoZGVmYXVsdDogMTAsMDAwLDAwMCkgKi9cbiAgICByZWFkUmVxdWVzdHNQZXJNb250aD86IG51bWJlcjtcbiAgICAvKiogTnVtYmVyIG9mIHdyaXRlIHJlcXVlc3RzIHBlciBtb250aCAoZGVmYXVsdDogMSwwMDAsMDAwKSAqL1xuICAgIHdyaXRlUmVxdWVzdHNQZXJNb250aD86IG51bWJlcjtcbiAgfTtcbiAgbmF0R2F0ZXdheT86IHtcbiAgICBkYXRhUHJvY2Vzc2VkR0I/OiBudW1iZXI7XG4gIH07XG4gIGFsYj86IHtcbiAgICBuZXdDb25uZWN0aW9uc1BlclNlY29uZD86IG51bWJlcjtcbiAgICBhY3RpdmVDb25uZWN0aW9uc1Blck1pbnV0ZT86IG51bWJlcjtcbiAgICBwcm9jZXNzZWRCeXRlc0dCPzogbnVtYmVyO1xuICB9O1xuICBubGI/OiB7XG4gICAgbmV3Q29ubmVjdGlvbnNQZXJTZWNvbmQ/OiBudW1iZXI7XG4gICAgYWN0aXZlQ29ubmVjdGlvbnNQZXJNaW51dGU/OiBudW1iZXI7XG4gICAgcHJvY2Vzc2VkQnl0ZXNHQj86IG51bWJlcjtcbiAgfTtcbiAgY2xvdWRmcm9udD86IHtcbiAgICBkYXRhVHJhbnNmZXJHQj86IG51bWJlcjtcbiAgICByZXF1ZXN0cz86IG51bWJlcjtcbiAgfTtcbiAgYXBpR2F0ZXdheT86IHtcbiAgICByZXF1ZXN0c1Blck1vbnRoPzogbnVtYmVyO1xuICB9O1xuICB2cGNFbmRwb2ludD86IHtcbiAgICBkYXRhUHJvY2Vzc2VkR0I/OiBudW1iZXI7XG4gIH07XG4gIC8qKlxuICAgKiBFRlMgKEVsYXN0aWMgRmlsZSBTeXN0ZW0pIHVzYWdlIGFzc3VtcHRpb25zIGZvciBjb3N0IGVzdGltYXRpb24uXG4gICAqIFRoZXNlIHZhbHVlcyBhcmUgdXNlZCB0byBlc3RpbWF0ZSBtb250aGx5IGNvc3RzIGZvciBFRlMgZmlsZSBzeXN0ZW1zLlxuICAgKlxuICAgKiBAc2VlIGh0dHBzOi8vYXdzLmFtYXpvbi5jb20vZWZzL3ByaWNpbmcvXG4gICAqL1xuICBlZnM/OiB7XG4gICAgLyoqIFRvdGFsIHN0b3JhZ2Ugc2l6ZSBpbiBHQiAoZGVmYXVsdDogMTAwKSAqL1xuICAgIHN0b3JhZ2VTaXplR2I/OiBudW1iZXI7XG4gICAgLyoqIFBlcmNlbnRhZ2Ugb2Ygc3RvcmFnZSBpbiBJbmZyZXF1ZW50IEFjY2VzcyBjbGFzcyAoZGVmYXVsdDogMCwgcmFuZ2U6IDAtMTAwKSAqL1xuICAgIGluZnJlcXVlbnRBY2Nlc3NQZXJjZW50YWdlPzogbnVtYmVyO1xuICB9O1xuICAvKipcbiAgICogU05TIChTaW1wbGUgTm90aWZpY2F0aW9uIFNlcnZpY2UpIHVzYWdlIGFzc3VtcHRpb25zLlxuICAgKiBUaGVzZSB2YWx1ZXMgYXJlIHVzZWQgdG8gZXN0aW1hdGUgbW9udGhseSBjb3N0cyBmb3IgU05TIHRvcGljcy5cbiAgICpcbiAgICogQHNlZSBodHRwczovL2F3cy5hbWF6b24uY29tL3Nucy9wcmljaW5nL1xuICAgKi9cbiAgc25zPzoge1xuICAgIC8qKiBOdW1iZXIgb2YgcHVibGlzaCByZXF1ZXN0cyBwZXIgbW9udGggKGRlZmF1bHQ6IDEsMDAwLDAwMCkgKi9cbiAgICBtb250aGx5UHVibGlzaGVzPzogbnVtYmVyO1xuICAgIC8qKiBOdW1iZXIgb2YgSFRUUC9TIGRlbGl2ZXJpZXMgcGVyIG1vbnRoIChkZWZhdWx0OiAxLDAwMCwwMDApICovXG4gICAgaHR0cERlbGl2ZXJpZXM/OiBudW1iZXI7XG4gICAgLyoqIE51bWJlciBvZiBlbWFpbCBkZWxpdmVyaWVzIHBlciBtb250aCAoZGVmYXVsdDogMCkgKi9cbiAgICBlbWFpbERlbGl2ZXJpZXM/OiBudW1iZXI7XG4gICAgLyoqIE51bWJlciBvZiBTTVMgZGVsaXZlcmllcyBwZXIgbW9udGggKGRlZmF1bHQ6IDApICovXG4gICAgc21zRGVsaXZlcmllcz86IG51bWJlcjtcbiAgICAvKiogTnVtYmVyIG9mIG1vYmlsZSBwdXNoIGRlbGl2ZXJpZXMgcGVyIG1vbnRoIChkZWZhdWx0OiAwKSAqL1xuICAgIG1vYmlsZVB1c2hEZWxpdmVyaWVzPzogbnVtYmVyO1xuICB9O1xuICAvKipcbiAgICogU1FTIHVzYWdlIGFzc3VtcHRpb25zIGZvciBjb3N0IGVzdGltYXRpb24uXG4gICAqIEFwcGxpZXMgdG8gYm90aCBTdGFuZGFyZCBhbmQgRklGTyBxdWV1ZXMuXG4gICAqXG4gICAqIEBzZWUgaHR0cHM6Ly9hd3MuYW1hem9uLmNvbS9zcXMvcHJpY2luZy9cbiAgICovXG4gIHNxcz86IHtcbiAgICAvKiogTnVtYmVyIG9mIHJlcXVlc3RzIHBlciBtb250aCAoZGVmYXVsdDogMSwwMDAsMDAwKSAqL1xuICAgIG1vbnRobHlSZXF1ZXN0cz86IG51bWJlcjtcbiAgfTtcbiAgLyoqXG4gICAqIFN0ZXAgRnVuY3Rpb25zIHVzYWdlIGFzc3VtcHRpb25zIGZvciBjb3N0IGVzdGltYXRpb24uXG4gICAqXG4gICAqIFN0YW5kYXJkIHdvcmtmbG93cyBhcmUgY2hhcmdlZCBwZXIgc3RhdGUgdHJhbnNpdGlvbi5cbiAgICogRXhwcmVzcyB3b3JrZmxvd3MgYXJlIGNoYXJnZWQgcGVyIHJlcXVlc3QgYW5kIHBlciBHQi1zZWNvbmQgb2YgZHVyYXRpb24uXG4gICAqXG4gICAqIEBzZWUgaHR0cHM6Ly9hd3MuYW1hem9uLmNvbS9zdGVwLWZ1bmN0aW9ucy9wcmljaW5nL1xuICAgKi9cbiAgc3RlcEZ1bmN0aW9ucz86IHtcbiAgICAvKiogTnVtYmVyIG9mIHdvcmtmbG93IGV4ZWN1dGlvbnMgcGVyIG1vbnRoIChkZWZhdWx0OiAxMCwwMDApICovXG4gICAgbW9udGhseUV4ZWN1dGlvbnM/OiBudW1iZXI7XG4gICAgLyoqIE51bWJlciBvZiBzdGF0ZSB0cmFuc2l0aW9ucyBwZXIgZXhlY3V0aW9uIChkZWZhdWx0OiAxMCwgZm9yIFN0YW5kYXJkIHdvcmtmbG93cykgKi9cbiAgICBzdGF0ZVRyYW5zaXRpb25zUGVyRXhlY3V0aW9uPzogbnVtYmVyO1xuICAgIC8qKiBBdmVyYWdlIGV4ZWN1dGlvbiBkdXJhdGlvbiBpbiBtaWxsaXNlY29uZHMgKGRlZmF1bHQ6IDEwMDAsIGZvciBFeHByZXNzIHdvcmtmbG93cykgKi9cbiAgICBhdmVyYWdlRHVyYXRpb25Ncz86IG51bWJlcjtcbiAgfTtcbiAgLyoqXG4gICAqIFNlY3JldHMgTWFuYWdlciB1c2FnZSBhc3N1bXB0aW9ucyBmb3IgY29zdCBlc3RpbWF0aW9uLlxuICAgKlxuICAgKiBAc2VlIGh0dHBzOi8vYXdzLmFtYXpvbi5jb20vc2VjcmV0cy1tYW5hZ2VyL3ByaWNpbmcvXG4gICAqL1xuICBzZWNyZXRzTWFuYWdlcj86IHtcbiAgICAvKiogTnVtYmVyIG9mIEFQSSBjYWxscyBwZXIgbW9udGggKGRlZmF1bHQ6IDEwLDAwMCkgKi9cbiAgICBtb250aGx5QXBpQ2FsbHM/OiBudW1iZXI7XG4gIH07XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgU3ludGhlc2lzQ29uZmlnIHtcbiAgYXBwUGF0aD86IHN0cmluZztcbiAgb3V0cHV0UGF0aD86IHN0cmluZztcbiAgY3VzdG9tQ29tbWFuZD86IHN0cmluZztcbiAgY29udGV4dD86IFJlY29yZDxzdHJpbmcsIHN0cmluZz47XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgRXhjbHVzaW9uc0NvbmZpZyB7XG4gIHJlc291cmNlVHlwZXM/OiBzdHJpbmdbXTtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBDYWNoZUNvbmZpZyB7XG4gIGVuYWJsZWQ/OiBib29sZWFuO1xuICBkdXJhdGlvbkhvdXJzPzogbnVtYmVyO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFZhbGlkYXRpb25SZXN1bHQge1xuICB2YWxpZDogYm9vbGVhbjtcbiAgZXJyb3JzOiBzdHJpbmdbXTtcbiAgd2FybmluZ3M6IHN0cmluZ1tdO1xufVxuXG5leHBvcnQgY2xhc3MgQ29uZmlndXJhdGlvbkVycm9yIGV4dGVuZHMgRXJyb3Ige1xuICBjb25zdHJ1Y3RvcihcbiAgICBtZXNzYWdlOiBzdHJpbmcsXG4gICAgcHVibGljIGNvbmZpZ1BhdGg6IHN0cmluZyxcbiAgICBwdWJsaWMgdmFsaWRhdGlvbkVycm9yczogc3RyaW5nW10sXG4gICkge1xuICAgIHN1cGVyKG1lc3NhZ2UpO1xuICAgIHRoaXMubmFtZSA9ICdDb25maWd1cmF0aW9uRXJyb3InO1xuICB9XG59XG4iXX0=
@@ -9,6 +9,7 @@ const CloudFrontCalculator_1 = require("./calculators/CloudFrontCalculator");
9
9
  const DynamoDBCalculator_1 = require("./calculators/DynamoDBCalculator");
10
10
  const EC2Calculator_1 = require("./calculators/EC2Calculator");
11
11
  const ECSCalculator_1 = require("./calculators/ECSCalculator");
12
+ const EFSCalculator_1 = require("./calculators/EFSCalculator");
12
13
  const ElastiCacheCalculator_1 = require("./calculators/ElastiCacheCalculator");
13
14
  const LambdaCalculator_1 = require("./calculators/LambdaCalculator");
14
15
  const LaunchTemplateCalculator_1 = require("./calculators/LaunchTemplateCalculator");
@@ -16,6 +17,7 @@ const NatGatewayCalculator_1 = require("./calculators/NatGatewayCalculator");
16
17
  const NLBCalculator_1 = require("./calculators/NLBCalculator");
17
18
  const RDSCalculator_1 = require("./calculators/RDSCalculator");
18
19
  const S3Calculator_1 = require("./calculators/S3Calculator");
20
+ const SecretsManagerCalculator_1 = require("./calculators/SecretsManagerCalculator");
19
21
  const SNSCalculator_1 = require("./calculators/SNSCalculator");
20
22
  const SQSCalculator_1 = require("./calculators/SQSCalculator");
21
23
  const StepFunctionsCalculator_1 = require("./calculators/StepFunctionsCalculator");
@@ -58,9 +60,11 @@ class PricingService {
58
60
  new ElastiCacheCalculator_1.ElastiCacheCalculator(),
59
61
  new AutoScalingGroupCalculator_1.AutoScalingGroupCalculator(),
60
62
  new LaunchTemplateCalculator_1.LaunchTemplateCalculator(),
63
+ new EFSCalculator_1.EFSCalculator(usageAssumptions?.efs?.storageSizeGb, usageAssumptions?.efs?.infrequentAccessPercentage),
61
64
  new SNSCalculator_1.SNSCalculator(usageAssumptions?.sns?.monthlyPublishes, usageAssumptions?.sns?.httpDeliveries, usageAssumptions?.sns?.emailDeliveries, usageAssumptions?.sns?.smsDeliveries, usageAssumptions?.sns?.mobilePushDeliveries),
62
65
  new SQSCalculator_1.SQSCalculator(config),
63
66
  new StepFunctionsCalculator_1.StepFunctionsCalculator(usageAssumptions?.stepFunctions?.monthlyExecutions, usageAssumptions?.stepFunctions?.stateTransitionsPerExecution, usageAssumptions?.stepFunctions?.averageDurationMs),
67
+ new SecretsManagerCalculator_1.SecretsManagerCalculator(usageAssumptions?.secretsManager?.monthlyApiCalls),
64
68
  ];
65
69
  }
66
70
  async getResourceCost(resource, region, templateResources) {
@@ -174,4 +178,4 @@ class PricingService {
174
178
  }
175
179
  }
176
180
  exports.PricingService = PricingService;
177
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"PricingService.js","sourceRoot":"","sources":["../../src/pricing/PricingService.ts"],"names":[],"mappings":";;;AAAA,iDAA8C;AAC9C,+DAA4D;AAC5D,6EAA0E;AAC1E,yFAAsF;AACtF,6EAA0E;AAC1E,yEAAsE;AACtE,+DAA4D;AAC5D,+DAA4D;AAC5D,+EAA4E;AAC5E,qEAAkE;AAClE,qFAAkF;AAClF,6EAA0E;AAC1E,+DAA4D;AAC5D,+DAA4D;AAC5D,6DAA0D;AAC1D,+DAA4D;AAC5D,+DAA4D;AAC5D,mFAAgF;AAChF,+EAA4E;AAC5E,mDAAgD;AAUhD,MAAa,cAAc;IACjB,WAAW,CAA2B;IACtC,aAAa,CAAgB;IAC7B,qBAAqB,CAAc;IAE3C,YACE,SAAiB,WAAW,EAC5B,gBAAyC,EACzC,qBAAgC,EAChC,WAAyB,EACzB,aAA6B;QAE7B,kDAAkD;QAClD,IAAI,aAAa,EAAE,CAAC;YAClB,IAAI,CAAC,aAAa,GAAG,aAAa,CAAC;QACrC,CAAC;aAAM,CAAC;YACN,iDAAiD;YACjD,IAAI,YAAsC,CAAC;YAC3C,IAAI,WAAW,EAAE,OAAO,KAAK,KAAK,EAAE,CAAC;gBACnC,MAAM,aAAa,GAAG,WAAW,EAAE,aAAa,IAAI,EAAE,CAAC;gBACvD,YAAY,GAAG,IAAI,2BAAY,CAAC,0BAA0B,EAAE,aAAa,CAAC,CAAC;YAC7E,CAAC;YAED,IAAI,CAAC,aAAa,GAAG,IAAI,6BAAa,CAAC,MAAM,EAAE,YAAY,CAAC,CAAC;QAC/D,CAAC;QACD,IAAI,CAAC,qBAAqB,GAAG,IAAI,GAAG,CAAC,qBAAqB,IAAI,EAAE,CAAC,CAAC;QAElE,mDAAmD;QACnD,MAAM,MAAM,GAAmC,gBAAgB,CAAC,CAAC,CAAC,EAAE,gBAAgB,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC;QAEnG,IAAI,CAAC,WAAW,GAAG;YACjB,IAAI,6BAAa,EAAE;YACnB,IAAI,2BAAY,EAAE;YAClB,IAAI,mCAAgB,CAClB,gBAAgB,EAAE,MAAM,EAAE,mBAAmB,EAC7C,gBAAgB,EAAE,MAAM,EAAE,iBAAiB,CAC5C;YACD,IAAI,6BAAa,EAAE;YACnB,IAAI,uCAAkB,CAAC,MAAM,CAAC;YAC9B,IAAI,6BAAa,EAAE;YACnB,IAAI,2CAAoB,EAAE;YAC1B,IAAI,2CAAoB,CAAC,gBAAgB,EAAE,UAAU,EAAE,eAAe,CAAC;YACvE,IAAI,6BAAa,CACf,gBAAgB,EAAE,GAAG,EAAE,uBAAuB,EAC9C,gBAAgB,EAAE,GAAG,EAAE,0BAA0B,EACjD,gBAAgB,EAAE,GAAG,EAAE,gBAAgB,CACxC;YACD,IAAI,6BAAa,CACf,gBAAgB,EAAE,GAAG,EAAE,uBAAuB,EAC9C,gBAAgB,EAAE,GAAG,EAAE,0BAA0B,EACjD,gBAAgB,EAAE,GAAG,EAAE,gBAAgB,CACxC;YACD,IAAI,6CAAqB,CAAC,gBAAgB,EAAE,WAAW,EAAE,eAAe,CAAC;YACzE,IAAI,2CAAoB,CACtB,gBAAgB,EAAE,UAAU,EAAE,cAAc,EAC5C,gBAAgB,EAAE,UAAU,EAAE,QAAQ,CACvC;YACD,IAAI,6CAAqB,EAAE;YAC3B,IAAI,uDAA0B,EAAE;YAChC,IAAI,mDAAwB,EAAE;YAC9B,IAAI,6BAAa,CACf,gBAAgB,EAAE,GAAG,EAAE,gBAAgB,EACvC,gBAAgB,EAAE,GAAG,EAAE,cAAc,EACrC,gBAAgB,EAAE,GAAG,EAAE,eAAe,EACtC,gBAAgB,EAAE,GAAG,EAAE,aAAa,EACpC,gBAAgB,EAAE,GAAG,EAAE,oBAAoB,CAC5C;YACD,IAAI,6BAAa,CAAC,MAAM,CAAC;YACzB,IAAI,iDAAuB,CACzB,gBAAgB,EAAE,aAAa,EAAE,iBAAiB,EAClD,gBAAgB,EAAE,aAAa,EAAE,4BAA4B,EAC7D,gBAAgB,EAAE,aAAa,EAAE,iBAAiB,CACnD;SACF,CAAC;IACJ,CAAC;IAED,KAAK,CAAC,eAAe,CAAC,QAAwB,EAAE,MAAc,EAAE,iBAAoC;QAClG,qCAAqC;QACrC,IAAI,IAAI,CAAC,qBAAqB,CAAC,GAAG,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC;YAClD,OAAO;gBACL,MAAM,EAAE,CAAC;gBACT,QAAQ,EAAE,KAAK;gBACf,UAAU,EAAE,MAAM;gBAClB,WAAW,EAAE,CAAC,iBAAiB,QAAQ,CAAC,IAAI,iCAAiC,CAAC;aAC/E,CAAC;QACJ,CAAC;QAED,mFAAmF;QACnF,MAAM,UAAU,GAAG,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE;YAC9C,IAAI,IAAI,CAAC,YAAY,EAAE,CAAC;gBACtB,OAAO,IAAI,CAAC,YAAY,CAAC,QAAQ,CAAC,CAAC;YACrC,CAAC;YACD,OAAO,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;QACtC,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,UAAU,EAAE,CAAC;YAChB,OAAO;gBACL,MAAM,EAAE,CAAC;gBACT,QAAQ,EAAE,KAAK;gBACf,UAAU,EAAE,SAAS;gBACrB,WAAW,EAAE,CAAC,iBAAiB,QAAQ,CAAC,IAAI,mBAAmB,CAAC;aACjE,CAAC;QACJ,CAAC;QAED,IAAI,CAAC;YACH,OAAO,MAAM,UAAU,CAAC,aAAa,CAAC,QAAQ,EAAE,MAAM,EAAE,IAAI,CAAC,aAAa,EAAE,iBAAiB,CAAC,CAAC;QACjG,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO;gBACL,MAAM,EAAE,CAAC;gBACT,QAAQ,EAAE,KAAK;gBACf,UAAU,EAAE,SAAS;gBACrB,WAAW,EAAE,CAAC,6BAA6B,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC;aACrG,CAAC;QACJ,CAAC;IACH,CAAC;IAED,KAAK,CAAC,YAAY,CAAC,IAAkB,EAAE,MAAc;QACnD,kDAAkD;QAClD,MAAM,aAAa,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,IAAI,CAAC,qBAAqB,CAAC,GAAG,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC;QACtF,MAAM,eAAe,GAAG,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,IAAI,CAAC,qBAAqB,CAAC,GAAG,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC;QAC1F,MAAM,gBAAgB,GAAG,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,IAAI,CAAC,qBAAqB,CAAC,GAAG,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC;QAE5F,iEAAiE;QACjE,MAAM,YAAY,GAAqB;YACrC,GAAG,IAAI,CAAC,KAAK;YACb,GAAG,IAAI,CAAC,OAAO;YACf,GAAG,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;gBACzB,SAAS,EAAE,CAAC,CAAC,SAAS;gBACtB,IAAI,EAAE,CAAC,CAAC,IAAI;gBACZ,UAAU,EAAE,CAAC,CAAC,aAAa;aAC5B,CAAC,CAAC;SACJ,CAAC;QAEF,MAAM,UAAU,GAAG,MAAM,OAAO,CAAC,GAAG,CAClC,aAAa,CAAC,GAAG,CAAC,KAAK,EAAE,QAAQ,EAAE,EAAE;YACnC,MAAM,WAAW,GAAG,MAAM,IAAI,CAAC,eAAe,CAAC,QAAQ,EAAE,MAAM,EAAE,YAAY,CAAC,CAAC;YAC/E,OAAO;gBACL,SAAS,EAAE,QAAQ,CAAC,SAAS;gBAC7B,IAAI,EAAE,QAAQ,CAAC,IAAI;gBACnB,WAAW;aACZ,CAAC;QACJ,CAAC,CAAC,CACH,CAAC;QAEF,MAAM,YAAY,GAAG,MAAM,OAAO,CAAC,GAAG,CACpC,eAAe,CAAC,GAAG,CAAC,KAAK,EAAE,QAAQ,EAAE,EAAE;YACrC,MAAM,WAAW,GAAG,MAAM,IAAI,CAAC,eAAe,CAAC,QAAQ,EAAE,MAAM,EAAE,YAAY,CAAC,CAAC;YAC/E,OAAO;gBACL,SAAS,EAAE,QAAQ,CAAC,SAAS;gBAC7B,IAAI,EAAE,QAAQ,CAAC,IAAI;gBACnB,WAAW;aACZ,CAAC;QACJ,CAAC,CAAC,CACH,CAAC;QAEF,MAAM,aAAa,GAAG,MAAM,OAAO,CAAC,GAAG,CACrC,gBAAgB,CAAC,GAAG,CAAC,KAAK,EAAE,QAAQ,EAAE,EAAE;YACtC,MAAM,WAAW,GAAmB;gBAClC,SAAS,EAAE,QAAQ,CAAC,SAAS;gBAC7B,IAAI,EAAE,QAAQ,CAAC,IAAI;gBACnB,UAAU,EAAE,QAAQ,CAAC,aAAa;aACnC,CAAC;YACF,MAAM,WAAW,GAAmB;gBAClC,SAAS,EAAE,QAAQ,CAAC,SAAS;gBAC7B,IAAI,EAAE,QAAQ,CAAC,IAAI;gBACnB,UAAU,EAAE,QAAQ,CAAC,aAAa;aACnC,CAAC;YAEF,MAAM,cAAc,GAAG,MAAM,IAAI,CAAC,eAAe,CAAC,WAAW,EAAE,MAAM,EAAE,YAAY,CAAC,CAAC;YACrF,MAAM,cAAc,GAAG,MAAM,IAAI,CAAC,eAAe,CAAC,WAAW,EAAE,MAAM,EAAE,YAAY,CAAC,CAAC;YACrF,MAAM,SAAS,GAAG,cAAc,CAAC,MAAM,GAAG,cAAc,CAAC,MAAM,CAAC;YAEhE,OAAO;gBACL,SAAS,EAAE,QAAQ,CAAC,SAAS;gBAC7B,IAAI,EAAE,QAAQ,CAAC,IAAI;gBACnB,WAAW,EAAE,cAAc;gBAC3B,cAAc;gBACd,cAAc;gBACd,SAAS;aACV,CAAC;QACJ,CAAC,CAAC,CACH,CAAC;QAEF,MAAM,cAAc,GAAG,UAAU,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC,GAAG,GAAG,CAAC,CAAC,WAAW,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;QACpF,MAAM,gBAAgB,GAAG,YAAY,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC,GAAG,GAAG,CAAC,CAAC,WAAW,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;QACxF,MAAM,kBAAkB,GAAG,aAAa,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC,GAAG,GAAG,CAAC,CAAC,SAAS,EAAE,CAAC,CAAC,CAAC;QAElF,MAAM,UAAU,GAAG,cAAc,GAAG,gBAAgB,GAAG,kBAAkB,CAAC;QAE1E,OAAO;YACL,UAAU;YACV,QAAQ,EAAE,KAAK;YACf,UAAU;YACV,YAAY;YACZ,aAAa;SACd,CAAC;IACJ,CAAC;IAED;;OAEG;IACH,OAAO;QACL,IAAI,CAAC,aAAa,CAAC,OAAO,EAAE,CAAC;IAC/B,CAAC;CACF;AA5MD,wCA4MC","sourcesContent":["import { CacheManager } from './CacheManager';\nimport { ALBCalculator } from './calculators/ALBCalculator';\nimport { APIGatewayCalculator } from './calculators/APIGatewayCalculator';\nimport { AutoScalingGroupCalculator } from './calculators/AutoScalingGroupCalculator';\nimport { CloudFrontCalculator } from './calculators/CloudFrontCalculator';\nimport { DynamoDBCalculator } from './calculators/DynamoDBCalculator';\nimport { EC2Calculator } from './calculators/EC2Calculator';\nimport { ECSCalculator } from './calculators/ECSCalculator';\nimport { ElastiCacheCalculator } from './calculators/ElastiCacheCalculator';\nimport { LambdaCalculator } from './calculators/LambdaCalculator';\nimport { LaunchTemplateCalculator } from './calculators/LaunchTemplateCalculator';\nimport { NatGatewayCalculator } from './calculators/NatGatewayCalculator';\nimport { NLBCalculator } from './calculators/NLBCalculator';\nimport { RDSCalculator } from './calculators/RDSCalculator';\nimport { S3Calculator } from './calculators/S3Calculator';\nimport { SNSCalculator } from './calculators/SNSCalculator';\nimport { SQSCalculator } from './calculators/SQSCalculator';\nimport { StepFunctionsCalculator } from './calculators/StepFunctionsCalculator';\nimport { VPCEndpointCalculator } from './calculators/VPCEndpointCalculator';\nimport { PricingClient } from './PricingClient';\nimport {\n  PricingService as IPricingService,\n  MonthlyCost,\n  CostDelta,\n  ResourceCostCalculator,\n} from './types';\nimport { UsageAssumptionsConfig, CacheConfig, CostAnalyzerConfig } from '../config/types';\nimport { ResourceWithId, ResourceDiff } from '../diff/types';\n\nexport class PricingService implements IPricingService {\n  private calculators: ResourceCostCalculator[];\n  private pricingClient: PricingClient;\n  private excludedResourceTypes: Set<string>;\n\n  constructor(\n    region: string = 'us-east-1',\n    usageAssumptions?: UsageAssumptionsConfig,\n    excludedResourceTypes?: string[],\n    cacheConfig?: CacheConfig,\n    pricingClient?: PricingClient,\n  ) {\n    // Use provided pricing client or create a new one\n    if (pricingClient) {\n      this.pricingClient = pricingClient;\n    } else {\n      // Initialize cache manager if caching is enabled\n      let cacheManager: CacheManager | undefined;\n      if (cacheConfig?.enabled !== false) {\n        const cacheDuration = cacheConfig?.durationHours ?? 24;\n        cacheManager = new CacheManager('.cdk-cost-analyzer-cache', cacheDuration);\n      }\n\n      this.pricingClient = new PricingClient(region, cacheManager);\n    }\n    this.excludedResourceTypes = new Set(excludedResourceTypes || []);\n    \n    // Build config object for calculators that need it\n    const config: CostAnalyzerConfig | undefined = usageAssumptions ? { usageAssumptions } : undefined;\n    \n    this.calculators = [\n      new EC2Calculator(),\n      new S3Calculator(),\n      new LambdaCalculator(\n        usageAssumptions?.lambda?.invocationsPerMonth,\n        usageAssumptions?.lambda?.averageDurationMs,\n      ),\n      new RDSCalculator(),\n      new DynamoDBCalculator(config),\n      new ECSCalculator(),\n      new APIGatewayCalculator(),\n      new NatGatewayCalculator(usageAssumptions?.natGateway?.dataProcessedGB),\n      new ALBCalculator(\n        usageAssumptions?.alb?.newConnectionsPerSecond,\n        usageAssumptions?.alb?.activeConnectionsPerMinute,\n        usageAssumptions?.alb?.processedBytesGB,\n      ),\n      new NLBCalculator(\n        usageAssumptions?.nlb?.newConnectionsPerSecond,\n        usageAssumptions?.nlb?.activeConnectionsPerMinute,\n        usageAssumptions?.nlb?.processedBytesGB,\n      ),\n      new VPCEndpointCalculator(usageAssumptions?.vpcEndpoint?.dataProcessedGB),\n      new CloudFrontCalculator(\n        usageAssumptions?.cloudfront?.dataTransferGB,\n        usageAssumptions?.cloudfront?.requests,\n      ),\n      new ElastiCacheCalculator(),\n      new AutoScalingGroupCalculator(),\n      new LaunchTemplateCalculator(),\n      new SNSCalculator(\n        usageAssumptions?.sns?.monthlyPublishes,\n        usageAssumptions?.sns?.httpDeliveries,\n        usageAssumptions?.sns?.emailDeliveries,\n        usageAssumptions?.sns?.smsDeliveries,\n        usageAssumptions?.sns?.mobilePushDeliveries,\n      ),\n      new SQSCalculator(config),\n      new StepFunctionsCalculator(\n        usageAssumptions?.stepFunctions?.monthlyExecutions,\n        usageAssumptions?.stepFunctions?.stateTransitionsPerExecution,\n        usageAssumptions?.stepFunctions?.averageDurationMs,\n      ),\n    ];\n  }\n\n  async getResourceCost(resource: ResourceWithId, region: string, templateResources?: ResourceWithId[]): Promise<MonthlyCost> {\n    // Check if resource type is excluded\n    if (this.excludedResourceTypes.has(resource.type)) {\n      return {\n        amount: 0,\n        currency: 'USD',\n        confidence: 'high',\n        assumptions: [`Resource type ${resource.type} is excluded from cost analysis`],\n      };\n    }\n\n    // Find calculator using canCalculate if available, otherwise fall back to supports\n    const calculator = this.calculators.find(calc => {\n      if (calc.canCalculate) {\n        return calc.canCalculate(resource);\n      }\n      return calc.supports(resource.type);\n    });\n\n    if (!calculator) {\n      return {\n        amount: 0,\n        currency: 'USD',\n        confidence: 'unknown',\n        assumptions: [`Resource type ${resource.type} is not supported`],\n      };\n    }\n\n    try {\n      return await calculator.calculateCost(resource, region, this.pricingClient, templateResources);\n    } catch (error) {\n      return {\n        amount: 0,\n        currency: 'USD',\n        confidence: 'unknown',\n        assumptions: [`Failed to calculate cost: ${error instanceof Error ? error.message : String(error)}`],\n      };\n    }\n  }\n\n  async getCostDelta(diff: ResourceDiff, region: string): Promise<CostDelta> {\n    // Filter out excluded resources before processing\n    const filteredAdded = diff.added.filter(r => !this.excludedResourceTypes.has(r.type));\n    const filteredRemoved = diff.removed.filter(r => !this.excludedResourceTypes.has(r.type));\n    const filteredModified = diff.modified.filter(r => !this.excludedResourceTypes.has(r.type));\n\n    // Build template resource context from all resources in the diff\n    const allResources: ResourceWithId[] = [\n      ...diff.added,\n      ...diff.removed,\n      ...diff.modified.map(r => ({\n        logicalId: r.logicalId,\n        type: r.type,\n        properties: r.newProperties,\n      })),\n    ];\n\n    const addedCosts = await Promise.all(\n      filteredAdded.map(async (resource) => {\n        const monthlyCost = await this.getResourceCost(resource, region, allResources);\n        return {\n          logicalId: resource.logicalId,\n          type: resource.type,\n          monthlyCost,\n        };\n      }),\n    );\n\n    const removedCosts = await Promise.all(\n      filteredRemoved.map(async (resource) => {\n        const monthlyCost = await this.getResourceCost(resource, region, allResources);\n        return {\n          logicalId: resource.logicalId,\n          type: resource.type,\n          monthlyCost,\n        };\n      }),\n    );\n\n    const modifiedCosts = await Promise.all(\n      filteredModified.map(async (resource) => {\n        const oldResource: ResourceWithId = {\n          logicalId: resource.logicalId,\n          type: resource.type,\n          properties: resource.oldProperties,\n        };\n        const newResource: ResourceWithId = {\n          logicalId: resource.logicalId,\n          type: resource.type,\n          properties: resource.newProperties,\n        };\n\n        const oldMonthlyCost = await this.getResourceCost(oldResource, region, allResources);\n        const newMonthlyCost = await this.getResourceCost(newResource, region, allResources);\n        const costDelta = newMonthlyCost.amount - oldMonthlyCost.amount;\n\n        return {\n          logicalId: resource.logicalId,\n          type: resource.type,\n          monthlyCost: newMonthlyCost,\n          oldMonthlyCost,\n          newMonthlyCost,\n          costDelta,\n        };\n      }),\n    );\n\n    const totalAddedCost = addedCosts.reduce((sum, r) => sum + r.monthlyCost.amount, 0);\n    const totalRemovedCost = removedCosts.reduce((sum, r) => sum + r.monthlyCost.amount, 0);\n    const totalModifiedDelta = modifiedCosts.reduce((sum, r) => sum + r.costDelta, 0);\n\n    const totalDelta = totalAddedCost - totalRemovedCost + totalModifiedDelta;\n\n    return {\n      totalDelta,\n      currency: 'USD',\n      addedCosts,\n      removedCosts,\n      modifiedCosts,\n    };\n  }\n\n  /**\n   * Clean up resources and connections\n   */\n  destroy(): void {\n    this.pricingClient.destroy();\n  }\n}\n"]}
181
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"PricingService.js","sourceRoot":"","sources":["../../src/pricing/PricingService.ts"],"names":[],"mappings":";;;AAAA,iDAA8C;AAC9C,+DAA4D;AAC5D,6EAA0E;AAC1E,yFAAsF;AACtF,6EAA0E;AAC1E,yEAAsE;AACtE,+DAA4D;AAC5D,+DAA4D;AAC5D,+DAA4D;AAC5D,+EAA4E;AAC5E,qEAAkE;AAClE,qFAAkF;AAClF,6EAA0E;AAC1E,+DAA4D;AAC5D,+DAA4D;AAC5D,6DAA0D;AAC1D,qFAAkF;AAClF,+DAA4D;AAC5D,+DAA4D;AAC5D,mFAAgF;AAChF,+EAA4E;AAC5E,mDAAgD;AAUhD,MAAa,cAAc;IACjB,WAAW,CAA2B;IACtC,aAAa,CAAgB;IAC7B,qBAAqB,CAAc;IAE3C,YACE,SAAiB,WAAW,EAC5B,gBAAyC,EACzC,qBAAgC,EAChC,WAAyB,EACzB,aAA6B;QAE7B,kDAAkD;QAClD,IAAI,aAAa,EAAE,CAAC;YAClB,IAAI,CAAC,aAAa,GAAG,aAAa,CAAC;QACrC,CAAC;aAAM,CAAC;YACN,iDAAiD;YACjD,IAAI,YAAsC,CAAC;YAC3C,IAAI,WAAW,EAAE,OAAO,KAAK,KAAK,EAAE,CAAC;gBACnC,MAAM,aAAa,GAAG,WAAW,EAAE,aAAa,IAAI,EAAE,CAAC;gBACvD,YAAY,GAAG,IAAI,2BAAY,CAAC,0BAA0B,EAAE,aAAa,CAAC,CAAC;YAC7E,CAAC;YAED,IAAI,CAAC,aAAa,GAAG,IAAI,6BAAa,CAAC,MAAM,EAAE,YAAY,CAAC,CAAC;QAC/D,CAAC;QACD,IAAI,CAAC,qBAAqB,GAAG,IAAI,GAAG,CAAC,qBAAqB,IAAI,EAAE,CAAC,CAAC;QAElE,mDAAmD;QACnD,MAAM,MAAM,GAAmC,gBAAgB,CAAC,CAAC,CAAC,EAAE,gBAAgB,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC;QAEnG,IAAI,CAAC,WAAW,GAAG;YACjB,IAAI,6BAAa,EAAE;YACnB,IAAI,2BAAY,EAAE;YAClB,IAAI,mCAAgB,CAClB,gBAAgB,EAAE,MAAM,EAAE,mBAAmB,EAC7C,gBAAgB,EAAE,MAAM,EAAE,iBAAiB,CAC5C;YACD,IAAI,6BAAa,EAAE;YACnB,IAAI,uCAAkB,CAAC,MAAM,CAAC;YAC9B,IAAI,6BAAa,EAAE;YACnB,IAAI,2CAAoB,EAAE;YAC1B,IAAI,2CAAoB,CAAC,gBAAgB,EAAE,UAAU,EAAE,eAAe,CAAC;YACvE,IAAI,6BAAa,CACf,gBAAgB,EAAE,GAAG,EAAE,uBAAuB,EAC9C,gBAAgB,EAAE,GAAG,EAAE,0BAA0B,EACjD,gBAAgB,EAAE,GAAG,EAAE,gBAAgB,CACxC;YACD,IAAI,6BAAa,CACf,gBAAgB,EAAE,GAAG,EAAE,uBAAuB,EAC9C,gBAAgB,EAAE,GAAG,EAAE,0BAA0B,EACjD,gBAAgB,EAAE,GAAG,EAAE,gBAAgB,CACxC;YACD,IAAI,6CAAqB,CAAC,gBAAgB,EAAE,WAAW,EAAE,eAAe,CAAC;YACzE,IAAI,2CAAoB,CACtB,gBAAgB,EAAE,UAAU,EAAE,cAAc,EAC5C,gBAAgB,EAAE,UAAU,EAAE,QAAQ,CACvC;YACD,IAAI,6CAAqB,EAAE;YAC3B,IAAI,uDAA0B,EAAE;YAChC,IAAI,mDAAwB,EAAE;YAC9B,IAAI,6BAAa,CACf,gBAAgB,EAAE,GAAG,EAAE,aAAa,EACpC,gBAAgB,EAAE,GAAG,EAAE,0BAA0B,CAClD;YACD,IAAI,6BAAa,CACf,gBAAgB,EAAE,GAAG,EAAE,gBAAgB,EACvC,gBAAgB,EAAE,GAAG,EAAE,cAAc,EACrC,gBAAgB,EAAE,GAAG,EAAE,eAAe,EACtC,gBAAgB,EAAE,GAAG,EAAE,aAAa,EACpC,gBAAgB,EAAE,GAAG,EAAE,oBAAoB,CAC5C;YACD,IAAI,6BAAa,CAAC,MAAM,CAAC;YACzB,IAAI,iDAAuB,CACzB,gBAAgB,EAAE,aAAa,EAAE,iBAAiB,EAClD,gBAAgB,EAAE,aAAa,EAAE,4BAA4B,EAC7D,gBAAgB,EAAE,aAAa,EAAE,iBAAiB,CACnD;YACD,IAAI,mDAAwB,CAC1B,gBAAgB,EAAE,cAAc,EAAE,eAAe,CAClD;SACF,CAAC;IACJ,CAAC;IAED,KAAK,CAAC,eAAe,CAAC,QAAwB,EAAE,MAAc,EAAE,iBAAoC;QAClG,qCAAqC;QACrC,IAAI,IAAI,CAAC,qBAAqB,CAAC,GAAG,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC;YAClD,OAAO;gBACL,MAAM,EAAE,CAAC;gBACT,QAAQ,EAAE,KAAK;gBACf,UAAU,EAAE,MAAM;gBAClB,WAAW,EAAE,CAAC,iBAAiB,QAAQ,CAAC,IAAI,iCAAiC,CAAC;aAC/E,CAAC;QACJ,CAAC;QAED,mFAAmF;QACnF,MAAM,UAAU,GAAG,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE;YAC9C,IAAI,IAAI,CAAC,YAAY,EAAE,CAAC;gBACtB,OAAO,IAAI,CAAC,YAAY,CAAC,QAAQ,CAAC,CAAC;YACrC,CAAC;YACD,OAAO,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;QACtC,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,UAAU,EAAE,CAAC;YAChB,OAAO;gBACL,MAAM,EAAE,CAAC;gBACT,QAAQ,EAAE,KAAK;gBACf,UAAU,EAAE,SAAS;gBACrB,WAAW,EAAE,CAAC,iBAAiB,QAAQ,CAAC,IAAI,mBAAmB,CAAC;aACjE,CAAC;QACJ,CAAC;QAED,IAAI,CAAC;YACH,OAAO,MAAM,UAAU,CAAC,aAAa,CAAC,QAAQ,EAAE,MAAM,EAAE,IAAI,CAAC,aAAa,EAAE,iBAAiB,CAAC,CAAC;QACjG,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO;gBACL,MAAM,EAAE,CAAC;gBACT,QAAQ,EAAE,KAAK;gBACf,UAAU,EAAE,SAAS;gBACrB,WAAW,EAAE,CAAC,6BAA6B,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC;aACrG,CAAC;QACJ,CAAC;IACH,CAAC;IAED,KAAK,CAAC,YAAY,CAAC,IAAkB,EAAE,MAAc;QACnD,kDAAkD;QAClD,MAAM,aAAa,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,IAAI,CAAC,qBAAqB,CAAC,GAAG,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC;QACtF,MAAM,eAAe,GAAG,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,IAAI,CAAC,qBAAqB,CAAC,GAAG,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC;QAC1F,MAAM,gBAAgB,GAAG,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,IAAI,CAAC,qBAAqB,CAAC,GAAG,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC;QAE5F,iEAAiE;QACjE,MAAM,YAAY,GAAqB;YACrC,GAAG,IAAI,CAAC,KAAK;YACb,GAAG,IAAI,CAAC,OAAO;YACf,GAAG,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;gBACzB,SAAS,EAAE,CAAC,CAAC,SAAS;gBACtB,IAAI,EAAE,CAAC,CAAC,IAAI;gBACZ,UAAU,EAAE,CAAC,CAAC,aAAa;aAC5B,CAAC,CAAC;SACJ,CAAC;QAEF,MAAM,UAAU,GAAG,MAAM,OAAO,CAAC,GAAG,CAClC,aAAa,CAAC,GAAG,CAAC,KAAK,EAAE,QAAQ,EAAE,EAAE;YACnC,MAAM,WAAW,GAAG,MAAM,IAAI,CAAC,eAAe,CAAC,QAAQ,EAAE,MAAM,EAAE,YAAY,CAAC,CAAC;YAC/E,OAAO;gBACL,SAAS,EAAE,QAAQ,CAAC,SAAS;gBAC7B,IAAI,EAAE,QAAQ,CAAC,IAAI;gBACnB,WAAW;aACZ,CAAC;QACJ,CAAC,CAAC,CACH,CAAC;QAEF,MAAM,YAAY,GAAG,MAAM,OAAO,CAAC,GAAG,CACpC,eAAe,CAAC,GAAG,CAAC,KAAK,EAAE,QAAQ,EAAE,EAAE;YACrC,MAAM,WAAW,GAAG,MAAM,IAAI,CAAC,eAAe,CAAC,QAAQ,EAAE,MAAM,EAAE,YAAY,CAAC,CAAC;YAC/E,OAAO;gBACL,SAAS,EAAE,QAAQ,CAAC,SAAS;gBAC7B,IAAI,EAAE,QAAQ,CAAC,IAAI;gBACnB,WAAW;aACZ,CAAC;QACJ,CAAC,CAAC,CACH,CAAC;QAEF,MAAM,aAAa,GAAG,MAAM,OAAO,CAAC,GAAG,CACrC,gBAAgB,CAAC,GAAG,CAAC,KAAK,EAAE,QAAQ,EAAE,EAAE;YACtC,MAAM,WAAW,GAAmB;gBAClC,SAAS,EAAE,QAAQ,CAAC,SAAS;gBAC7B,IAAI,EAAE,QAAQ,CAAC,IAAI;gBACnB,UAAU,EAAE,QAAQ,CAAC,aAAa;aACnC,CAAC;YACF,MAAM,WAAW,GAAmB;gBAClC,SAAS,EAAE,QAAQ,CAAC,SAAS;gBAC7B,IAAI,EAAE,QAAQ,CAAC,IAAI;gBACnB,UAAU,EAAE,QAAQ,CAAC,aAAa;aACnC,CAAC;YAEF,MAAM,cAAc,GAAG,MAAM,IAAI,CAAC,eAAe,CAAC,WAAW,EAAE,MAAM,EAAE,YAAY,CAAC,CAAC;YACrF,MAAM,cAAc,GAAG,MAAM,IAAI,CAAC,eAAe,CAAC,WAAW,EAAE,MAAM,EAAE,YAAY,CAAC,CAAC;YACrF,MAAM,SAAS,GAAG,cAAc,CAAC,MAAM,GAAG,cAAc,CAAC,MAAM,CAAC;YAEhE,OAAO;gBACL,SAAS,EAAE,QAAQ,CAAC,SAAS;gBAC7B,IAAI,EAAE,QAAQ,CAAC,IAAI;gBACnB,WAAW,EAAE,cAAc;gBAC3B,cAAc;gBACd,cAAc;gBACd,SAAS;aACV,CAAC;QACJ,CAAC,CAAC,CACH,CAAC;QAEF,MAAM,cAAc,GAAG,UAAU,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC,GAAG,GAAG,CAAC,CAAC,WAAW,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;QACpF,MAAM,gBAAgB,GAAG,YAAY,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC,GAAG,GAAG,CAAC,CAAC,WAAW,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;QACxF,MAAM,kBAAkB,GAAG,aAAa,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC,GAAG,GAAG,CAAC,CAAC,SAAS,EAAE,CAAC,CAAC,CAAC;QAElF,MAAM,UAAU,GAAG,cAAc,GAAG,gBAAgB,GAAG,kBAAkB,CAAC;QAE1E,OAAO;YACL,UAAU;YACV,QAAQ,EAAE,KAAK;YACf,UAAU;YACV,YAAY;YACZ,aAAa;SACd,CAAC;IACJ,CAAC;IAED;;OAEG;IACH,OAAO;QACL,IAAI,CAAC,aAAa,CAAC,OAAO,EAAE,CAAC;IAC/B,CAAC;CACF;AAnND,wCAmNC","sourcesContent":["import { CacheManager } from './CacheManager';\nimport { ALBCalculator } from './calculators/ALBCalculator';\nimport { APIGatewayCalculator } from './calculators/APIGatewayCalculator';\nimport { AutoScalingGroupCalculator } from './calculators/AutoScalingGroupCalculator';\nimport { CloudFrontCalculator } from './calculators/CloudFrontCalculator';\nimport { DynamoDBCalculator } from './calculators/DynamoDBCalculator';\nimport { EC2Calculator } from './calculators/EC2Calculator';\nimport { ECSCalculator } from './calculators/ECSCalculator';\nimport { EFSCalculator } from './calculators/EFSCalculator';\nimport { ElastiCacheCalculator } from './calculators/ElastiCacheCalculator';\nimport { LambdaCalculator } from './calculators/LambdaCalculator';\nimport { LaunchTemplateCalculator } from './calculators/LaunchTemplateCalculator';\nimport { NatGatewayCalculator } from './calculators/NatGatewayCalculator';\nimport { NLBCalculator } from './calculators/NLBCalculator';\nimport { RDSCalculator } from './calculators/RDSCalculator';\nimport { S3Calculator } from './calculators/S3Calculator';\nimport { SecretsManagerCalculator } from './calculators/SecretsManagerCalculator';\nimport { SNSCalculator } from './calculators/SNSCalculator';\nimport { SQSCalculator } from './calculators/SQSCalculator';\nimport { StepFunctionsCalculator } from './calculators/StepFunctionsCalculator';\nimport { VPCEndpointCalculator } from './calculators/VPCEndpointCalculator';\nimport { PricingClient } from './PricingClient';\nimport {\n  PricingService as IPricingService,\n  MonthlyCost,\n  CostDelta,\n  ResourceCostCalculator,\n} from './types';\nimport { UsageAssumptionsConfig, CacheConfig, CostAnalyzerConfig } from '../config/types';\nimport { ResourceWithId, ResourceDiff } from '../diff/types';\n\nexport class PricingService implements IPricingService {\n  private calculators: ResourceCostCalculator[];\n  private pricingClient: PricingClient;\n  private excludedResourceTypes: Set<string>;\n\n  constructor(\n    region: string = 'us-east-1',\n    usageAssumptions?: UsageAssumptionsConfig,\n    excludedResourceTypes?: string[],\n    cacheConfig?: CacheConfig,\n    pricingClient?: PricingClient,\n  ) {\n    // Use provided pricing client or create a new one\n    if (pricingClient) {\n      this.pricingClient = pricingClient;\n    } else {\n      // Initialize cache manager if caching is enabled\n      let cacheManager: CacheManager | undefined;\n      if (cacheConfig?.enabled !== false) {\n        const cacheDuration = cacheConfig?.durationHours ?? 24;\n        cacheManager = new CacheManager('.cdk-cost-analyzer-cache', cacheDuration);\n      }\n\n      this.pricingClient = new PricingClient(region, cacheManager);\n    }\n    this.excludedResourceTypes = new Set(excludedResourceTypes || []);\n    \n    // Build config object for calculators that need it\n    const config: CostAnalyzerConfig | undefined = usageAssumptions ? { usageAssumptions } : undefined;\n    \n    this.calculators = [\n      new EC2Calculator(),\n      new S3Calculator(),\n      new LambdaCalculator(\n        usageAssumptions?.lambda?.invocationsPerMonth,\n        usageAssumptions?.lambda?.averageDurationMs,\n      ),\n      new RDSCalculator(),\n      new DynamoDBCalculator(config),\n      new ECSCalculator(),\n      new APIGatewayCalculator(),\n      new NatGatewayCalculator(usageAssumptions?.natGateway?.dataProcessedGB),\n      new ALBCalculator(\n        usageAssumptions?.alb?.newConnectionsPerSecond,\n        usageAssumptions?.alb?.activeConnectionsPerMinute,\n        usageAssumptions?.alb?.processedBytesGB,\n      ),\n      new NLBCalculator(\n        usageAssumptions?.nlb?.newConnectionsPerSecond,\n        usageAssumptions?.nlb?.activeConnectionsPerMinute,\n        usageAssumptions?.nlb?.processedBytesGB,\n      ),\n      new VPCEndpointCalculator(usageAssumptions?.vpcEndpoint?.dataProcessedGB),\n      new CloudFrontCalculator(\n        usageAssumptions?.cloudfront?.dataTransferGB,\n        usageAssumptions?.cloudfront?.requests,\n      ),\n      new ElastiCacheCalculator(),\n      new AutoScalingGroupCalculator(),\n      new LaunchTemplateCalculator(),\n      new EFSCalculator(\n        usageAssumptions?.efs?.storageSizeGb,\n        usageAssumptions?.efs?.infrequentAccessPercentage,\n      ),\n      new SNSCalculator(\n        usageAssumptions?.sns?.monthlyPublishes,\n        usageAssumptions?.sns?.httpDeliveries,\n        usageAssumptions?.sns?.emailDeliveries,\n        usageAssumptions?.sns?.smsDeliveries,\n        usageAssumptions?.sns?.mobilePushDeliveries,\n      ),\n      new SQSCalculator(config),\n      new StepFunctionsCalculator(\n        usageAssumptions?.stepFunctions?.monthlyExecutions,\n        usageAssumptions?.stepFunctions?.stateTransitionsPerExecution,\n        usageAssumptions?.stepFunctions?.averageDurationMs,\n      ),\n      new SecretsManagerCalculator(\n        usageAssumptions?.secretsManager?.monthlyApiCalls,\n      ),\n    ];\n  }\n\n  async getResourceCost(resource: ResourceWithId, region: string, templateResources?: ResourceWithId[]): Promise<MonthlyCost> {\n    // Check if resource type is excluded\n    if (this.excludedResourceTypes.has(resource.type)) {\n      return {\n        amount: 0,\n        currency: 'USD',\n        confidence: 'high',\n        assumptions: [`Resource type ${resource.type} is excluded from cost analysis`],\n      };\n    }\n\n    // Find calculator using canCalculate if available, otherwise fall back to supports\n    const calculator = this.calculators.find(calc => {\n      if (calc.canCalculate) {\n        return calc.canCalculate(resource);\n      }\n      return calc.supports(resource.type);\n    });\n\n    if (!calculator) {\n      return {\n        amount: 0,\n        currency: 'USD',\n        confidence: 'unknown',\n        assumptions: [`Resource type ${resource.type} is not supported`],\n      };\n    }\n\n    try {\n      return await calculator.calculateCost(resource, region, this.pricingClient, templateResources);\n    } catch (error) {\n      return {\n        amount: 0,\n        currency: 'USD',\n        confidence: 'unknown',\n        assumptions: [`Failed to calculate cost: ${error instanceof Error ? error.message : String(error)}`],\n      };\n    }\n  }\n\n  async getCostDelta(diff: ResourceDiff, region: string): Promise<CostDelta> {\n    // Filter out excluded resources before processing\n    const filteredAdded = diff.added.filter(r => !this.excludedResourceTypes.has(r.type));\n    const filteredRemoved = diff.removed.filter(r => !this.excludedResourceTypes.has(r.type));\n    const filteredModified = diff.modified.filter(r => !this.excludedResourceTypes.has(r.type));\n\n    // Build template resource context from all resources in the diff\n    const allResources: ResourceWithId[] = [\n      ...diff.added,\n      ...diff.removed,\n      ...diff.modified.map(r => ({\n        logicalId: r.logicalId,\n        type: r.type,\n        properties: r.newProperties,\n      })),\n    ];\n\n    const addedCosts = await Promise.all(\n      filteredAdded.map(async (resource) => {\n        const monthlyCost = await this.getResourceCost(resource, region, allResources);\n        return {\n          logicalId: resource.logicalId,\n          type: resource.type,\n          monthlyCost,\n        };\n      }),\n    );\n\n    const removedCosts = await Promise.all(\n      filteredRemoved.map(async (resource) => {\n        const monthlyCost = await this.getResourceCost(resource, region, allResources);\n        return {\n          logicalId: resource.logicalId,\n          type: resource.type,\n          monthlyCost,\n        };\n      }),\n    );\n\n    const modifiedCosts = await Promise.all(\n      filteredModified.map(async (resource) => {\n        const oldResource: ResourceWithId = {\n          logicalId: resource.logicalId,\n          type: resource.type,\n          properties: resource.oldProperties,\n        };\n        const newResource: ResourceWithId = {\n          logicalId: resource.logicalId,\n          type: resource.type,\n          properties: resource.newProperties,\n        };\n\n        const oldMonthlyCost = await this.getResourceCost(oldResource, region, allResources);\n        const newMonthlyCost = await this.getResourceCost(newResource, region, allResources);\n        const costDelta = newMonthlyCost.amount - oldMonthlyCost.amount;\n\n        return {\n          logicalId: resource.logicalId,\n          type: resource.type,\n          monthlyCost: newMonthlyCost,\n          oldMonthlyCost,\n          newMonthlyCost,\n          costDelta,\n        };\n      }),\n    );\n\n    const totalAddedCost = addedCosts.reduce((sum, r) => sum + r.monthlyCost.amount, 0);\n    const totalRemovedCost = removedCosts.reduce((sum, r) => sum + r.monthlyCost.amount, 0);\n    const totalModifiedDelta = modifiedCosts.reduce((sum, r) => sum + r.costDelta, 0);\n\n    const totalDelta = totalAddedCost - totalRemovedCost + totalModifiedDelta;\n\n    return {\n      totalDelta,\n      currency: 'USD',\n      addedCosts,\n      removedCosts,\n      modifiedCosts,\n    };\n  }\n\n  /**\n   * Clean up resources and connections\n   */\n  destroy(): void {\n    this.pricingClient.destroy();\n  }\n}\n"]}
@@ -0,0 +1,19 @@
1
+ import { ResourceWithId } from '../../diff/types';
2
+ import { ResourceCostCalculator, MonthlyCost, PricingClient } from '../types';
3
+ export interface EFSUsageAssumptions {
4
+ storageSizeGb?: number;
5
+ infrequentAccessPercentage?: number;
6
+ }
7
+ export declare class EFSCalculator implements ResourceCostCalculator {
8
+ private readonly customStorageSizeGb?;
9
+ private readonly customInfrequentAccessPercentage?;
10
+ private readonly DEFAULT_STORAGE_SIZE_GB;
11
+ private readonly DEFAULT_IA_PERCENTAGE;
12
+ private readonly FALLBACK_STANDARD_PRICE;
13
+ private readonly FALLBACK_IA_STORAGE_PRICE;
14
+ private readonly FALLBACK_IA_REQUEST_PRICE;
15
+ private readonly FALLBACK_PROVISIONED_THROUGHPUT_PRICE;
16
+ constructor(customStorageSizeGb?: number | undefined, customInfrequentAccessPercentage?: number | undefined);
17
+ supports(resourceType: string): boolean;
18
+ calculateCost(resource: ResourceWithId, region: string, pricingClient: PricingClient): Promise<MonthlyCost>;
19
+ }
@@ -0,0 +1,195 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.EFSCalculator = void 0;
4
+ const RegionMapper_1 = require("../RegionMapper");
5
+ const Logger_1 = require("../../utils/Logger");
6
+ class EFSCalculator {
7
+ customStorageSizeGb;
8
+ customInfrequentAccessPercentage;
9
+ DEFAULT_STORAGE_SIZE_GB = 100;
10
+ DEFAULT_IA_PERCENTAGE = 0;
11
+ // Fallback pricing rates (AWS EFS us-east-1 pricing as of 2024)
12
+ FALLBACK_STANDARD_PRICE = 0.30; // Per GB-month
13
+ FALLBACK_IA_STORAGE_PRICE = 0.016; // Per GB-month
14
+ FALLBACK_IA_REQUEST_PRICE = 0.01; // Per GB transferred
15
+ FALLBACK_PROVISIONED_THROUGHPUT_PRICE = 6.00; // Per MB/s-month
16
+ constructor(customStorageSizeGb, customInfrequentAccessPercentage) {
17
+ this.customStorageSizeGb = customStorageSizeGb;
18
+ this.customInfrequentAccessPercentage = customInfrequentAccessPercentage;
19
+ }
20
+ supports(resourceType) {
21
+ return resourceType === 'AWS::EFS::FileSystem';
22
+ }
23
+ async calculateCost(resource, region, pricingClient) {
24
+ try {
25
+ // Guard against undefined properties
26
+ if (!resource.properties) {
27
+ return {
28
+ amount: 0,
29
+ currency: 'USD',
30
+ confidence: 'unknown',
31
+ assumptions: ['Resource properties are undefined'],
32
+ };
33
+ }
34
+ const regionPrefix = (0, RegionMapper_1.getRegionPrefix)(region);
35
+ const storageSizeGb = this.customStorageSizeGb ?? this.DEFAULT_STORAGE_SIZE_GB;
36
+ const iaPercentage = this.customInfrequentAccessPercentage ?? this.DEFAULT_IA_PERCENTAGE;
37
+ Logger_1.Logger.debug('EFS pricing calculation started', {
38
+ region,
39
+ regionPrefix,
40
+ normalizedRegion: (0, RegionMapper_1.normalizeRegion)(region),
41
+ storageSizeGb,
42
+ iaPercentage,
43
+ });
44
+ // Check for lifecycle policies to determine if IA storage is used
45
+ const lifecyclePolicies = resource.properties?.LifecyclePolicies;
46
+ const hasIATransition = lifecyclePolicies?.some(policy => policy.TransitionToIA !== undefined) ?? false;
47
+ // Check for provisioned throughput
48
+ const throughputMode = resource.properties?.ThroughputMode;
49
+ const provisionedThroughputInMibps = resource.properties?.ProvisionedThroughputInMibps;
50
+ const isProvisionedThroughput = throughputMode === 'provisioned' && provisionedThroughputInMibps !== undefined;
51
+ // Calculate storage distribution
52
+ const effectiveIAPercentage = hasIATransition ? iaPercentage : 0;
53
+ const standardStorageGb = storageSizeGb * (1 - effectiveIAPercentage / 100);
54
+ const iaStorageGb = storageSizeGb * (effectiveIAPercentage / 100);
55
+ // Get Standard storage pricing
56
+ const standardStoragePrice = await pricingClient.getPrice({
57
+ serviceCode: 'AmazonEFS',
58
+ region: (0, RegionMapper_1.normalizeRegion)(region),
59
+ filters: [
60
+ { field: 'productFamily', value: 'Storage' },
61
+ { field: 'usagetype', value: `${regionPrefix}-TimedStorage-ByteHrs` },
62
+ ],
63
+ });
64
+ Logger_1.Logger.debug('EFS Standard storage price retrieved', {
65
+ standardStoragePrice,
66
+ usageType: `${regionPrefix}-TimedStorage-ByteHrs`,
67
+ });
68
+ // Get Infrequent Access storage pricing
69
+ let iaStoragePrice = null;
70
+ let iaRequestPrice = null;
71
+ if (hasIATransition && iaPercentage > 0) {
72
+ iaStoragePrice = await pricingClient.getPrice({
73
+ serviceCode: 'AmazonEFS',
74
+ region: (0, RegionMapper_1.normalizeRegion)(region),
75
+ filters: [
76
+ { field: 'productFamily', value: 'Storage' },
77
+ { field: 'usagetype', value: `${regionPrefix}-IATimedStorage-ByteHrs` },
78
+ ],
79
+ });
80
+ iaRequestPrice = await pricingClient.getPrice({
81
+ serviceCode: 'AmazonEFS',
82
+ region: (0, RegionMapper_1.normalizeRegion)(region),
83
+ filters: [
84
+ { field: 'productFamily', value: 'Storage' },
85
+ { field: 'usagetype', value: `${regionPrefix}-IARequests-Bytes` },
86
+ ],
87
+ });
88
+ Logger_1.Logger.debug('EFS IA storage prices retrieved', {
89
+ iaStoragePrice,
90
+ iaRequestPrice,
91
+ });
92
+ }
93
+ // Get Provisioned Throughput pricing
94
+ let provisionedThroughputPrice = null;
95
+ if (isProvisionedThroughput) {
96
+ provisionedThroughputPrice = await pricingClient.getPrice({
97
+ serviceCode: 'AmazonEFS',
98
+ region: (0, RegionMapper_1.normalizeRegion)(region),
99
+ filters: [
100
+ { field: 'productFamily', value: 'Provisioned Throughput' },
101
+ { field: 'usagetype', value: `${regionPrefix}-ProvisionedTP-MiBpsHrs` },
102
+ ],
103
+ });
104
+ Logger_1.Logger.debug('EFS Provisioned Throughput price retrieved', {
105
+ provisionedThroughputPrice,
106
+ usageType: `${regionPrefix}-ProvisionedTP-MiBpsHrs`,
107
+ });
108
+ }
109
+ const assumptions = [];
110
+ let totalCost = 0;
111
+ let confidence = 'medium';
112
+ // Calculate standard storage cost
113
+ const effectiveStandardPrice = standardStoragePrice ?? this.FALLBACK_STANDARD_PRICE;
114
+ const standardStorageCost = standardStorageGb * effectiveStandardPrice;
115
+ if (standardStoragePrice === null) {
116
+ assumptions.push('Using fallback Standard storage pricing (API unavailable)');
117
+ confidence = 'low';
118
+ }
119
+ totalCost += standardStorageCost;
120
+ assumptions.push(`Standard storage: ${standardStorageGb.toFixed(2)} GB × $${effectiveStandardPrice.toFixed(4)}/GB = $${standardStorageCost.toFixed(2)}/month`);
121
+ // Calculate IA storage cost
122
+ if (hasIATransition && iaPercentage > 0) {
123
+ const effectiveIAStoragePrice = iaStoragePrice ?? this.FALLBACK_IA_STORAGE_PRICE;
124
+ const iaStorageCost = iaStorageGb * effectiveIAStoragePrice;
125
+ if (iaStoragePrice === null) {
126
+ assumptions.push('Using fallback Infrequent Access storage pricing (API unavailable)');
127
+ confidence = 'low';
128
+ }
129
+ totalCost += iaStorageCost;
130
+ assumptions.push(`Infrequent Access storage: ${iaStorageGb.toFixed(2)} GB × $${effectiveIAStoragePrice.toFixed(4)}/GB = $${iaStorageCost.toFixed(2)}/month`);
131
+ // Estimate IA request cost (assume 10% of IA data is accessed per month)
132
+ const estimatedIAAccessGb = iaStorageGb * 0.10;
133
+ const effectiveIARequestPrice = iaRequestPrice ?? this.FALLBACK_IA_REQUEST_PRICE;
134
+ const iaRequestCost = estimatedIAAccessGb * effectiveIARequestPrice;
135
+ if (iaRequestPrice === null && iaPercentage > 0) {
136
+ assumptions.push('Using fallback Infrequent Access request pricing (API unavailable)');
137
+ confidence = 'low';
138
+ }
139
+ totalCost += iaRequestCost;
140
+ assumptions.push(`IA requests (estimated 10% access): ${estimatedIAAccessGb.toFixed(2)} GB × $${effectiveIARequestPrice.toFixed(4)}/GB = $${iaRequestCost.toFixed(2)}/month`);
141
+ }
142
+ // Calculate provisioned throughput cost
143
+ if (isProvisionedThroughput && provisionedThroughputInMibps !== undefined) {
144
+ const effectiveProvisionedPrice = provisionedThroughputPrice ?? this.FALLBACK_PROVISIONED_THROUGHPUT_PRICE;
145
+ const provisionedCost = provisionedThroughputInMibps * effectiveProvisionedPrice;
146
+ if (provisionedThroughputPrice === null) {
147
+ assumptions.push('Using fallback Provisioned Throughput pricing (API unavailable)');
148
+ confidence = 'low';
149
+ }
150
+ totalCost += provisionedCost;
151
+ assumptions.push(`Provisioned Throughput: ${provisionedThroughputInMibps} MB/s × $${effectiveProvisionedPrice.toFixed(2)}/MB/s = $${provisionedCost.toFixed(2)}/month`);
152
+ }
153
+ // Add summary assumptions
154
+ assumptions.push(`Total storage: ${storageSizeGb} GB`);
155
+ if (this.customStorageSizeGb !== undefined) {
156
+ assumptions.push('Using custom storage size assumption from configuration');
157
+ }
158
+ if (hasIATransition) {
159
+ assumptions.push(`Lifecycle policy detected: ${effectiveIAPercentage}% in Infrequent Access`);
160
+ if (this.customInfrequentAccessPercentage !== undefined) {
161
+ assumptions.push('Using custom IA percentage assumption from configuration');
162
+ }
163
+ }
164
+ if (throughputMode) {
165
+ assumptions.push(`Throughput mode: ${throughputMode}`);
166
+ }
167
+ assumptions.push(`Total: $${totalCost.toFixed(2)}/month`);
168
+ Logger_1.Logger.debug('EFS cost calculated', {
169
+ standardStorageCost,
170
+ totalCost,
171
+ confidence,
172
+ });
173
+ return {
174
+ amount: totalCost,
175
+ currency: 'USD',
176
+ confidence,
177
+ assumptions,
178
+ };
179
+ }
180
+ catch (error) {
181
+ Logger_1.Logger.debug('EFS pricing calculation failed', {
182
+ error: error instanceof Error ? error.message : String(error),
183
+ region,
184
+ });
185
+ return {
186
+ amount: 0,
187
+ currency: 'USD',
188
+ confidence: 'unknown',
189
+ assumptions: [`Failed to fetch pricing: ${error instanceof Error ? error.message : String(error)}`],
190
+ };
191
+ }
192
+ }
193
+ }
194
+ exports.EFSCalculator = EFSCalculator;
195
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"EFSCalculator.js","sourceRoot":"","sources":["../../../src/pricing/calculators/EFSCalculator.ts"],"names":[],"mappings":"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","sourcesContent":["import { ResourceWithId } from '../../diff/types';\nimport { ResourceCostCalculator, MonthlyCost, PricingClient } from '../types';\nimport { normalizeRegion, getRegionPrefix } from '../RegionMapper';\nimport { Logger } from '../../utils/Logger';\n\nexport interface EFSUsageAssumptions {\n  storageSizeGb?: number;\n  infrequentAccessPercentage?: number;\n}\n\nexport class EFSCalculator implements ResourceCostCalculator {\n  private readonly DEFAULT_STORAGE_SIZE_GB = 100;\n  private readonly DEFAULT_IA_PERCENTAGE = 0;\n  \n  // Fallback pricing rates (AWS EFS us-east-1 pricing as of 2024)\n  private readonly FALLBACK_STANDARD_PRICE = 0.30; // Per GB-month\n  private readonly FALLBACK_IA_STORAGE_PRICE = 0.016; // Per GB-month\n  private readonly FALLBACK_IA_REQUEST_PRICE = 0.01; // Per GB transferred\n  private readonly FALLBACK_PROVISIONED_THROUGHPUT_PRICE = 6.00; // Per MB/s-month\n\n  constructor(\n    private readonly customStorageSizeGb?: number,\n    private readonly customInfrequentAccessPercentage?: number,\n  ) {}\n\n  supports(resourceType: string): boolean {\n    return resourceType === 'AWS::EFS::FileSystem';\n  }\n\n  async calculateCost(\n    resource: ResourceWithId,\n    region: string,\n    pricingClient: PricingClient,\n  ): Promise<MonthlyCost> {\n    try {\n      // Guard against undefined properties\n      if (!resource.properties) {\n        return {\n          amount: 0,\n          currency: 'USD',\n          confidence: 'unknown',\n          assumptions: ['Resource properties are undefined'],\n        };\n      }\n\n      const regionPrefix = getRegionPrefix(region);\n      const storageSizeGb = this.customStorageSizeGb ?? this.DEFAULT_STORAGE_SIZE_GB;\n      const iaPercentage = this.customInfrequentAccessPercentage ?? this.DEFAULT_IA_PERCENTAGE;\n\n      Logger.debug('EFS pricing calculation started', {\n        region,\n        regionPrefix,\n        normalizedRegion: normalizeRegion(region),\n        storageSizeGb,\n        iaPercentage,\n      });\n\n      // Check for lifecycle policies to determine if IA storage is used\n      const lifecyclePolicies = resource.properties?.LifecyclePolicies as Array<{ TransitionToIA?: string }> | undefined;\n      const hasIATransition = lifecyclePolicies?.some(policy => policy.TransitionToIA !== undefined) ?? false;\n\n      // Check for provisioned throughput\n      const throughputMode = resource.properties?.ThroughputMode as string | undefined;\n      const provisionedThroughputInMibps = resource.properties?.ProvisionedThroughputInMibps as number | undefined;\n      const isProvisionedThroughput = throughputMode === 'provisioned' && provisionedThroughputInMibps !== undefined;\n\n      // Calculate storage distribution\n      const effectiveIAPercentage = hasIATransition ? iaPercentage : 0;\n      const standardStorageGb = storageSizeGb * (1 - effectiveIAPercentage / 100);\n      const iaStorageGb = storageSizeGb * (effectiveIAPercentage / 100);\n\n      // Get Standard storage pricing\n      const standardStoragePrice = await pricingClient.getPrice({\n        serviceCode: 'AmazonEFS',\n        region: normalizeRegion(region),\n        filters: [\n          { field: 'productFamily', value: 'Storage' },\n          { field: 'usagetype', value: `${regionPrefix}-TimedStorage-ByteHrs` },\n        ],\n      });\n\n      Logger.debug('EFS Standard storage price retrieved', {\n        standardStoragePrice,\n        usageType: `${regionPrefix}-TimedStorage-ByteHrs`,\n      });\n\n      // Get Infrequent Access storage pricing\n      let iaStoragePrice: number | null = null;\n      let iaRequestPrice: number | null = null;\n      if (hasIATransition && iaPercentage > 0) {\n        iaStoragePrice = await pricingClient.getPrice({\n          serviceCode: 'AmazonEFS',\n          region: normalizeRegion(region),\n          filters: [\n            { field: 'productFamily', value: 'Storage' },\n            { field: 'usagetype', value: `${regionPrefix}-IATimedStorage-ByteHrs` },\n          ],\n        });\n\n        iaRequestPrice = await pricingClient.getPrice({\n          serviceCode: 'AmazonEFS',\n          region: normalizeRegion(region),\n          filters: [\n            { field: 'productFamily', value: 'Storage' },\n            { field: 'usagetype', value: `${regionPrefix}-IARequests-Bytes` },\n          ],\n        });\n\n        Logger.debug('EFS IA storage prices retrieved', {\n          iaStoragePrice,\n          iaRequestPrice,\n        });\n      }\n\n      // Get Provisioned Throughput pricing\n      let provisionedThroughputPrice: number | null = null;\n      if (isProvisionedThroughput) {\n        provisionedThroughputPrice = await pricingClient.getPrice({\n          serviceCode: 'AmazonEFS',\n          region: normalizeRegion(region),\n          filters: [\n            { field: 'productFamily', value: 'Provisioned Throughput' },\n            { field: 'usagetype', value: `${regionPrefix}-ProvisionedTP-MiBpsHrs` },\n          ],\n        });\n\n        Logger.debug('EFS Provisioned Throughput price retrieved', {\n          provisionedThroughputPrice,\n          usageType: `${regionPrefix}-ProvisionedTP-MiBpsHrs`,\n        });\n      }\n\n      const assumptions: string[] = [];\n      let totalCost = 0;\n      let confidence: 'high' | 'medium' | 'low' | 'unknown' = 'medium';\n\n      // Calculate standard storage cost\n      const effectiveStandardPrice = standardStoragePrice ?? this.FALLBACK_STANDARD_PRICE;\n      const standardStorageCost = standardStorageGb * effectiveStandardPrice;\n      if (standardStoragePrice === null) {\n        assumptions.push('Using fallback Standard storage pricing (API unavailable)');\n        confidence = 'low';\n      }\n      totalCost += standardStorageCost;\n      assumptions.push(`Standard storage: ${standardStorageGb.toFixed(2)} GB × $${effectiveStandardPrice.toFixed(4)}/GB = $${standardStorageCost.toFixed(2)}/month`);\n\n      // Calculate IA storage cost\n      if (hasIATransition && iaPercentage > 0) {\n        const effectiveIAStoragePrice = iaStoragePrice ?? this.FALLBACK_IA_STORAGE_PRICE;\n        const iaStorageCost = iaStorageGb * effectiveIAStoragePrice;\n        if (iaStoragePrice === null) {\n          assumptions.push('Using fallback Infrequent Access storage pricing (API unavailable)');\n          confidence = 'low';\n        }\n        totalCost += iaStorageCost;\n        assumptions.push(`Infrequent Access storage: ${iaStorageGb.toFixed(2)} GB × $${effectiveIAStoragePrice.toFixed(4)}/GB = $${iaStorageCost.toFixed(2)}/month`);\n\n        // Estimate IA request cost (assume 10% of IA data is accessed per month)\n        const estimatedIAAccessGb = iaStorageGb * 0.10;\n        const effectiveIARequestPrice = iaRequestPrice ?? this.FALLBACK_IA_REQUEST_PRICE;\n        const iaRequestCost = estimatedIAAccessGb * effectiveIARequestPrice;\n        if (iaRequestPrice === null && iaPercentage > 0) {\n          assumptions.push('Using fallback Infrequent Access request pricing (API unavailable)');\n          confidence = 'low';\n        }\n        totalCost += iaRequestCost;\n        assumptions.push(`IA requests (estimated 10% access): ${estimatedIAAccessGb.toFixed(2)} GB × $${effectiveIARequestPrice.toFixed(4)}/GB = $${iaRequestCost.toFixed(2)}/month`);\n      }\n\n      // Calculate provisioned throughput cost\n      if (isProvisionedThroughput && provisionedThroughputInMibps !== undefined) {\n        const effectiveProvisionedPrice = provisionedThroughputPrice ?? this.FALLBACK_PROVISIONED_THROUGHPUT_PRICE;\n        const provisionedCost = provisionedThroughputInMibps * effectiveProvisionedPrice;\n        if (provisionedThroughputPrice === null) {\n          assumptions.push('Using fallback Provisioned Throughput pricing (API unavailable)');\n          confidence = 'low';\n        }\n        totalCost += provisionedCost;\n        assumptions.push(`Provisioned Throughput: ${provisionedThroughputInMibps} MB/s × $${effectiveProvisionedPrice.toFixed(2)}/MB/s = $${provisionedCost.toFixed(2)}/month`);\n      }\n\n      // Add summary assumptions\n      assumptions.push(`Total storage: ${storageSizeGb} GB`);\n      if (this.customStorageSizeGb !== undefined) {\n        assumptions.push('Using custom storage size assumption from configuration');\n      }\n      if (hasIATransition) {\n        assumptions.push(`Lifecycle policy detected: ${effectiveIAPercentage}% in Infrequent Access`);\n        if (this.customInfrequentAccessPercentage !== undefined) {\n          assumptions.push('Using custom IA percentage assumption from configuration');\n        }\n      }\n      if (throughputMode) {\n        assumptions.push(`Throughput mode: ${throughputMode}`);\n      }\n\n      assumptions.push(`Total: $${totalCost.toFixed(2)}/month`);\n\n      Logger.debug('EFS cost calculated', {\n        standardStorageCost,\n        totalCost,\n        confidence,\n      });\n\n      return {\n        amount: totalCost,\n        currency: 'USD',\n        confidence,\n        assumptions,\n      };\n    } catch (error) {\n      Logger.debug('EFS pricing calculation failed', {\n        error: error instanceof Error ? error.message : String(error),\n        region,\n      });\n\n      return {\n        amount: 0,\n        currency: 'USD',\n        confidence: 'unknown',\n        assumptions: [`Failed to fetch pricing: ${error instanceof Error ? error.message : String(error)}`],\n      };\n    }\n  }\n}\n"]}
@@ -0,0 +1,21 @@
1
+ import { ResourceWithId } from '../../diff/types';
2
+ import { ResourceCostCalculator, MonthlyCost, PricingClient } from '../types';
3
+ /**
4
+ * Calculator for AWS::SecretsManager::Secret resources.
5
+ *
6
+ * AWS Secrets Manager Pricing Model (as of 2024):
7
+ * - Secret storage: $0.40 per secret per month
8
+ * - API calls: $0.05 per 10,000 API calls
9
+ * - No free tier
10
+ *
11
+ * @see https://aws.amazon.com/secrets-manager/pricing/
12
+ */
13
+ export declare class SecretsManagerCalculator implements ResourceCostCalculator {
14
+ private readonly customMonthlyApiCalls?;
15
+ private readonly DEFAULT_MONTHLY_API_CALLS;
16
+ private readonly FALLBACK_SECRET_STORAGE_PRICE;
17
+ private readonly FALLBACK_API_CALL_PRICE_PER_10K;
18
+ constructor(customMonthlyApiCalls?: number | undefined);
19
+ supports(resourceType: string): boolean;
20
+ calculateCost(resource: ResourceWithId, region: string, pricingClient: PricingClient): Promise<MonthlyCost>;
21
+ }
@@ -0,0 +1,138 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.SecretsManagerCalculator = void 0;
4
+ const RegionMapper_1 = require("../RegionMapper");
5
+ const Logger_1 = require("../../utils/Logger");
6
+ /**
7
+ * Calculator for AWS::SecretsManager::Secret resources.
8
+ *
9
+ * AWS Secrets Manager Pricing Model (as of 2024):
10
+ * - Secret storage: $0.40 per secret per month
11
+ * - API calls: $0.05 per 10,000 API calls
12
+ * - No free tier
13
+ *
14
+ * @see https://aws.amazon.com/secrets-manager/pricing/
15
+ */
16
+ class SecretsManagerCalculator {
17
+ customMonthlyApiCalls;
18
+ // Default usage assumptions
19
+ DEFAULT_MONTHLY_API_CALLS = 10_000;
20
+ // Fallback pricing (AWS Secrets Manager pricing as of 2024)
21
+ FALLBACK_SECRET_STORAGE_PRICE = 0.40; // Per secret per month
22
+ FALLBACK_API_CALL_PRICE_PER_10K = 0.05;
23
+ constructor(customMonthlyApiCalls) {
24
+ this.customMonthlyApiCalls = customMonthlyApiCalls;
25
+ }
26
+ supports(resourceType) {
27
+ return resourceType === 'AWS::SecretsManager::Secret';
28
+ }
29
+ async calculateCost(resource, region, pricingClient) {
30
+ const monthlyApiCalls = this.customMonthlyApiCalls ?? this.DEFAULT_MONTHLY_API_CALLS;
31
+ Logger_1.Logger.debug('Secrets Manager pricing calculation started', {
32
+ region,
33
+ logicalId: resource.logicalId,
34
+ monthlyApiCalls,
35
+ });
36
+ try {
37
+ const normalizedRegion = (0, RegionMapper_1.normalizeRegion)(region);
38
+ // Query pricing for secret storage (per secret monthly cost)
39
+ const secretStoragePrice = await pricingClient.getPrice({
40
+ serviceCode: 'AWSSecretsManager',
41
+ region: normalizedRegion,
42
+ filters: [
43
+ { field: 'productFamily', value: 'Secret' },
44
+ { field: 'group', value: 'SecretStorage' },
45
+ ],
46
+ });
47
+ Logger_1.Logger.debug('Secrets Manager storage price retrieved', {
48
+ secretStoragePrice,
49
+ region: normalizedRegion,
50
+ });
51
+ // Query pricing for API calls
52
+ const apiCallPrice = await pricingClient.getPrice({
53
+ serviceCode: 'AWSSecretsManager',
54
+ region: normalizedRegion,
55
+ filters: [
56
+ { field: 'productFamily', value: 'Secret' },
57
+ { field: 'group', value: 'SecretRotation' },
58
+ ],
59
+ });
60
+ Logger_1.Logger.debug('Secrets Manager API call price retrieved', {
61
+ apiCallPrice,
62
+ region: normalizedRegion,
63
+ });
64
+ // If pricing data is completely unavailable
65
+ if (secretStoragePrice === null && apiCallPrice === null) {
66
+ Logger_1.Logger.debug('Secrets Manager pricing not available', {
67
+ region,
68
+ normalizedRegion,
69
+ });
70
+ return {
71
+ amount: 0,
72
+ currency: 'USD',
73
+ confidence: 'unknown',
74
+ assumptions: [
75
+ `Pricing data not available for Secrets Manager in region ${region}`,
76
+ `Would assume ${monthlyApiCalls.toLocaleString()} API calls per month`,
77
+ ],
78
+ };
79
+ }
80
+ // Handle pricing unavailability with fallback
81
+ const storageCost = secretStoragePrice ?? this.FALLBACK_SECRET_STORAGE_PRICE;
82
+ const callCostPer10K = apiCallPrice ?? this.FALLBACK_API_CALL_PRICE_PER_10K;
83
+ const apiCallCost = (monthlyApiCalls / 10_000) * callCostPer10K;
84
+ const totalCost = storageCost + apiCallCost;
85
+ const confidence = (secretStoragePrice === null || apiCallPrice === null) ? 'low' : 'medium';
86
+ const usedFallback = (secretStoragePrice === null || apiCallPrice === null);
87
+ Logger_1.Logger.debug('Secrets Manager cost calculated', {
88
+ storageCost,
89
+ callCostPer10K,
90
+ apiCallCost,
91
+ totalCost,
92
+ confidence,
93
+ usedFallback,
94
+ });
95
+ const assumptions = [];
96
+ // Add fallback warning first if used
97
+ if (usedFallback) {
98
+ if (secretStoragePrice === null) {
99
+ assumptions.push('Using fallback storage pricing (API unavailable)');
100
+ }
101
+ if (apiCallPrice === null) {
102
+ assumptions.push('Using fallback API call pricing (API unavailable)');
103
+ }
104
+ }
105
+ // Add cost breakdown
106
+ assumptions.push(`Secret storage: $${storageCost.toFixed(2)}/month`);
107
+ assumptions.push(`API calls: ${monthlyApiCalls.toLocaleString()} calls × $${callCostPer10K.toFixed(4)}/10K = $${apiCallCost.toFixed(2)}/month`);
108
+ assumptions.push(`Total: $${totalCost.toFixed(2)}/month`);
109
+ // Add custom assumption note
110
+ if (this.customMonthlyApiCalls !== undefined) {
111
+ assumptions.push('Using custom API call volume from configuration');
112
+ }
113
+ // Add informational notes
114
+ assumptions.push('No free tier for Secrets Manager');
115
+ assumptions.push('Cross-region replication incurs additional costs (not calculated)');
116
+ return {
117
+ amount: totalCost,
118
+ currency: 'USD',
119
+ confidence,
120
+ assumptions,
121
+ };
122
+ }
123
+ catch (error) {
124
+ Logger_1.Logger.debug('Secrets Manager pricing calculation failed', {
125
+ error: error instanceof Error ? error.message : String(error),
126
+ region,
127
+ });
128
+ return {
129
+ amount: 0,
130
+ currency: 'USD',
131
+ confidence: 'unknown',
132
+ assumptions: [`Failed to calculate Secrets Manager cost: ${error instanceof Error ? error.message : String(error)}`],
133
+ };
134
+ }
135
+ }
136
+ }
137
+ exports.SecretsManagerCalculator = SecretsManagerCalculator;
138
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"SecretsManagerCalculator.js","sourceRoot":"","sources":["../../../src/pricing/calculators/SecretsManagerCalculator.ts"],"names":[],"mappings":"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","sourcesContent":["import { ResourceWithId } from '../../diff/types';\nimport { ResourceCostCalculator, MonthlyCost, PricingClient } from '../types';\nimport { normalizeRegion } from '../RegionMapper';\nimport { Logger } from '../../utils/Logger';\n\n/**\n * Calculator for AWS::SecretsManager::Secret resources.\n *\n * AWS Secrets Manager Pricing Model (as of 2024):\n * - Secret storage: $0.40 per secret per month\n * - API calls: $0.05 per 10,000 API calls\n * - No free tier\n *\n * @see https://aws.amazon.com/secrets-manager/pricing/\n */\nexport class SecretsManagerCalculator implements ResourceCostCalculator {\n  // Default usage assumptions\n  private readonly DEFAULT_MONTHLY_API_CALLS = 10_000;\n\n  // Fallback pricing (AWS Secrets Manager pricing as of 2024)\n  private readonly FALLBACK_SECRET_STORAGE_PRICE = 0.40; // Per secret per month\n  private readonly FALLBACK_API_CALL_PRICE_PER_10K = 0.05;\n\n  constructor(private readonly customMonthlyApiCalls?: number) {}\n\n  supports(resourceType: string): boolean {\n    return resourceType === 'AWS::SecretsManager::Secret';\n  }\n\n  async calculateCost(\n    resource: ResourceWithId,\n    region: string,\n    pricingClient: PricingClient,\n  ): Promise<MonthlyCost> {\n    const monthlyApiCalls = this.customMonthlyApiCalls ?? this.DEFAULT_MONTHLY_API_CALLS;\n\n    Logger.debug('Secrets Manager pricing calculation started', {\n      region,\n      logicalId: resource.logicalId,\n      monthlyApiCalls,\n    });\n\n    try {\n      const normalizedRegion = normalizeRegion(region);\n\n      // Query pricing for secret storage (per secret monthly cost)\n      const secretStoragePrice = await pricingClient.getPrice({\n        serviceCode: 'AWSSecretsManager',\n        region: normalizedRegion,\n        filters: [\n          { field: 'productFamily', value: 'Secret' },\n          { field: 'group', value: 'SecretStorage' },\n        ],\n      });\n\n      Logger.debug('Secrets Manager storage price retrieved', {\n        secretStoragePrice,\n        region: normalizedRegion,\n      });\n\n      // Query pricing for API calls\n      const apiCallPrice = await pricingClient.getPrice({\n        serviceCode: 'AWSSecretsManager',\n        region: normalizedRegion,\n        filters: [\n          { field: 'productFamily', value: 'Secret' },\n          { field: 'group', value: 'SecretRotation' },\n        ],\n      });\n\n      Logger.debug('Secrets Manager API call price retrieved', {\n        apiCallPrice,\n        region: normalizedRegion,\n      });\n\n      // If pricing data is completely unavailable\n      if (secretStoragePrice === null && apiCallPrice === null) {\n        Logger.debug('Secrets Manager pricing not available', {\n          region,\n          normalizedRegion,\n        });\n\n        return {\n          amount: 0,\n          currency: 'USD',\n          confidence: 'unknown',\n          assumptions: [\n            `Pricing data not available for Secrets Manager in region ${region}`,\n            `Would assume ${monthlyApiCalls.toLocaleString()} API calls per month`,\n          ],\n        };\n      }\n\n      // Handle pricing unavailability with fallback\n      const storageCost = secretStoragePrice ?? this.FALLBACK_SECRET_STORAGE_PRICE;\n      const callCostPer10K = apiCallPrice ?? this.FALLBACK_API_CALL_PRICE_PER_10K;\n\n      const apiCallCost = (monthlyApiCalls / 10_000) * callCostPer10K;\n      const totalCost = storageCost + apiCallCost;\n\n      const confidence = (secretStoragePrice === null || apiCallPrice === null) ? 'low' : 'medium';\n      const usedFallback = (secretStoragePrice === null || apiCallPrice === null);\n\n      Logger.debug('Secrets Manager cost calculated', {\n        storageCost,\n        callCostPer10K,\n        apiCallCost,\n        totalCost,\n        confidence,\n        usedFallback,\n      });\n\n      const assumptions: string[] = [];\n\n      // Add fallback warning first if used\n      if (usedFallback) {\n        if (secretStoragePrice === null) {\n          assumptions.push('Using fallback storage pricing (API unavailable)');\n        }\n        if (apiCallPrice === null) {\n          assumptions.push('Using fallback API call pricing (API unavailable)');\n        }\n      }\n\n      // Add cost breakdown\n      assumptions.push(`Secret storage: $${storageCost.toFixed(2)}/month`);\n      assumptions.push(\n        `API calls: ${monthlyApiCalls.toLocaleString()} calls × $${callCostPer10K.toFixed(4)}/10K = $${apiCallCost.toFixed(2)}/month`,\n      );\n      assumptions.push(`Total: $${totalCost.toFixed(2)}/month`);\n\n      // Add custom assumption note\n      if (this.customMonthlyApiCalls !== undefined) {\n        assumptions.push('Using custom API call volume from configuration');\n      }\n\n      // Add informational notes\n      assumptions.push('No free tier for Secrets Manager');\n      assumptions.push('Cross-region replication incurs additional costs (not calculated)');\n\n      return {\n        amount: totalCost,\n        currency: 'USD',\n        confidence,\n        assumptions,\n      };\n    } catch (error) {\n      Logger.debug('Secrets Manager pricing calculation failed', {\n        error: error instanceof Error ? error.message : String(error),\n        region,\n      });\n\n      return {\n        amount: 0,\n        currency: 'USD',\n        confidence: 'unknown',\n        assumptions: [`Failed to calculate Secrets Manager cost: ${error instanceof Error ? error.message : String(error)}`],\n      };\n    }\n  }\n}\n"]}
@@ -6,7 +6,9 @@ export { S3Calculator } from './calculators/S3Calculator';
6
6
  export { LambdaCalculator } from './calculators/LambdaCalculator';
7
7
  export { RDSCalculator } from './calculators/RDSCalculator';
8
8
  export { CloudFrontCalculator } from './calculators/CloudFrontCalculator';
9
+ export { EFSCalculator, EFSUsageAssumptions } from './calculators/EFSCalculator';
9
10
  export { ElastiCacheCalculator } from './calculators/ElastiCacheCalculator';
11
+ export { SecretsManagerCalculator } from './calculators/SecretsManagerCalculator';
10
12
  export { SNSCalculator, SNSCostBreakdown } from './calculators/SNSCalculator';
11
13
  export { SQSCalculator } from './calculators/SQSCalculator';
12
14
  export { LaunchTemplateCalculator, LaunchTemplateConfig, EbsVolumeConfig, } from './calculators/LaunchTemplateCalculator';
@@ -14,7 +14,7 @@ var __exportStar = (this && this.__exportStar) || function(m, exports) {
14
14
  for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
15
15
  };
16
16
  Object.defineProperty(exports, "__esModule", { value: true });
17
- exports.StepFunctionsCalculator = exports.LaunchTemplateCalculator = exports.SQSCalculator = exports.SNSCalculator = exports.ElastiCacheCalculator = exports.CloudFrontCalculator = exports.RDSCalculator = exports.LambdaCalculator = exports.S3Calculator = exports.EC2Calculator = exports.CacheManager = exports.PricingClient = exports.PricingService = void 0;
17
+ exports.StepFunctionsCalculator = exports.LaunchTemplateCalculator = exports.SQSCalculator = exports.SNSCalculator = exports.SecretsManagerCalculator = exports.ElastiCacheCalculator = exports.EFSCalculator = exports.CloudFrontCalculator = exports.RDSCalculator = exports.LambdaCalculator = exports.S3Calculator = exports.EC2Calculator = exports.CacheManager = exports.PricingClient = exports.PricingService = void 0;
18
18
  var PricingService_1 = require("./PricingService");
19
19
  Object.defineProperty(exports, "PricingService", { enumerable: true, get: function () { return PricingService_1.PricingService; } });
20
20
  var PricingClient_1 = require("./PricingClient");
@@ -31,8 +31,12 @@ var RDSCalculator_1 = require("./calculators/RDSCalculator");
31
31
  Object.defineProperty(exports, "RDSCalculator", { enumerable: true, get: function () { return RDSCalculator_1.RDSCalculator; } });
32
32
  var CloudFrontCalculator_1 = require("./calculators/CloudFrontCalculator");
33
33
  Object.defineProperty(exports, "CloudFrontCalculator", { enumerable: true, get: function () { return CloudFrontCalculator_1.CloudFrontCalculator; } });
34
+ var EFSCalculator_1 = require("./calculators/EFSCalculator");
35
+ Object.defineProperty(exports, "EFSCalculator", { enumerable: true, get: function () { return EFSCalculator_1.EFSCalculator; } });
34
36
  var ElastiCacheCalculator_1 = require("./calculators/ElastiCacheCalculator");
35
37
  Object.defineProperty(exports, "ElastiCacheCalculator", { enumerable: true, get: function () { return ElastiCacheCalculator_1.ElastiCacheCalculator; } });
38
+ var SecretsManagerCalculator_1 = require("./calculators/SecretsManagerCalculator");
39
+ Object.defineProperty(exports, "SecretsManagerCalculator", { enumerable: true, get: function () { return SecretsManagerCalculator_1.SecretsManagerCalculator; } });
36
40
  var SNSCalculator_1 = require("./calculators/SNSCalculator");
37
41
  Object.defineProperty(exports, "SNSCalculator", { enumerable: true, get: function () { return SNSCalculator_1.SNSCalculator; } });
38
42
  var SQSCalculator_1 = require("./calculators/SQSCalculator");
@@ -42,4 +46,4 @@ Object.defineProperty(exports, "LaunchTemplateCalculator", { enumerable: true, g
42
46
  var StepFunctionsCalculator_1 = require("./calculators/StepFunctionsCalculator");
43
47
  Object.defineProperty(exports, "StepFunctionsCalculator", { enumerable: true, get: function () { return StepFunctionsCalculator_1.StepFunctionsCalculator; } });
44
48
  __exportStar(require("./types"), exports);
45
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvcHJpY2luZy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7OztBQUFBLG1EQUFrRDtBQUF6QyxnSEFBQSxjQUFjLE9BQUE7QUFDdkIsaURBQWdEO0FBQXZDLDhHQUFBLGFBQWEsT0FBQTtBQUN0QiwrQ0FBOEM7QUFBckMsNEdBQUEsWUFBWSxPQUFBO0FBQ3JCLDZEQUE0RDtBQUFuRCw4R0FBQSxhQUFhLE9BQUE7QUFDdEIsMkRBQTBEO0FBQWpELDRHQUFBLFlBQVksT0FBQTtBQUNyQixtRUFBa0U7QUFBekQsb0hBQUEsZ0JBQWdCLE9BQUE7QUFDekIsNkRBQTREO0FBQW5ELDhHQUFBLGFBQWEsT0FBQTtBQUN0QiwyRUFBMEU7QUFBakUsNEhBQUEsb0JBQW9CLE9BQUE7QUFDN0IsNkVBQTRFO0FBQW5FLDhIQUFBLHFCQUFxQixPQUFBO0FBQzlCLDZEQUE4RTtBQUFyRSw4R0FBQSxhQUFhLE9BQUE7QUFDdEIsNkRBQTREO0FBQW5ELDhHQUFBLGFBQWEsT0FBQTtBQUN0QixtRkFJZ0Q7QUFIOUMsb0lBQUEsd0JBQXdCLE9BQUE7QUFJMUIsaUZBRytDO0FBRjdDLGtJQUFBLHVCQUF1QixPQUFBO0FBR3pCLDBDQUF3QiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB7IFByaWNpbmdTZXJ2aWNlIH0gZnJvbSAnLi9QcmljaW5nU2VydmljZSc7XG5leHBvcnQgeyBQcmljaW5nQ2xpZW50IH0gZnJvbSAnLi9QcmljaW5nQ2xpZW50JztcbmV4cG9ydCB7IENhY2hlTWFuYWdlciB9IGZyb20gJy4vQ2FjaGVNYW5hZ2VyJztcbmV4cG9ydCB7IEVDMkNhbGN1bGF0b3IgfSBmcm9tICcuL2NhbGN1bGF0b3JzL0VDMkNhbGN1bGF0b3InO1xuZXhwb3J0IHsgUzNDYWxjdWxhdG9yIH0gZnJvbSAnLi9jYWxjdWxhdG9ycy9TM0NhbGN1bGF0b3InO1xuZXhwb3J0IHsgTGFtYmRhQ2FsY3VsYXRvciB9IGZyb20gJy4vY2FsY3VsYXRvcnMvTGFtYmRhQ2FsY3VsYXRvcic7XG5leHBvcnQgeyBSRFNDYWxjdWxhdG9yIH0gZnJvbSAnLi9jYWxjdWxhdG9ycy9SRFNDYWxjdWxhdG9yJztcbmV4cG9ydCB7IENsb3VkRnJvbnRDYWxjdWxhdG9yIH0gZnJvbSAnLi9jYWxjdWxhdG9ycy9DbG91ZEZyb250Q2FsY3VsYXRvcic7XG5leHBvcnQgeyBFbGFzdGlDYWNoZUNhbGN1bGF0b3IgfSBmcm9tICcuL2NhbGN1bGF0b3JzL0VsYXN0aUNhY2hlQ2FsY3VsYXRvcic7XG5leHBvcnQgeyBTTlNDYWxjdWxhdG9yLCBTTlNDb3N0QnJlYWtkb3duIH0gZnJvbSAnLi9jYWxjdWxhdG9ycy9TTlNDYWxjdWxhdG9yJztcbmV4cG9ydCB7IFNRU0NhbGN1bGF0b3IgfSBmcm9tICcuL2NhbGN1bGF0b3JzL1NRU0NhbGN1bGF0b3InO1xuZXhwb3J0IHtcbiAgTGF1bmNoVGVtcGxhdGVDYWxjdWxhdG9yLFxuICBMYXVuY2hUZW1wbGF0ZUNvbmZpZyxcbiAgRWJzVm9sdW1lQ29uZmlnLFxufSBmcm9tICcuL2NhbGN1bGF0b3JzL0xhdW5jaFRlbXBsYXRlQ2FsY3VsYXRvcic7XG5leHBvcnQge1xuICBTdGVwRnVuY3Rpb25zQ2FsY3VsYXRvcixcbiAgU3RlcEZ1bmN0aW9uc1dvcmtmbG93VHlwZSxcbn0gZnJvbSAnLi9jYWxjdWxhdG9ycy9TdGVwRnVuY3Rpb25zQ2FsY3VsYXRvcic7XG5leHBvcnQgKiBmcm9tICcuL3R5cGVzJztcbiJdfQ==
49
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvcHJpY2luZy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7OztBQUFBLG1EQUFrRDtBQUF6QyxnSEFBQSxjQUFjLE9BQUE7QUFDdkIsaURBQWdEO0FBQXZDLDhHQUFBLGFBQWEsT0FBQTtBQUN0QiwrQ0FBOEM7QUFBckMsNEdBQUEsWUFBWSxPQUFBO0FBQ3JCLDZEQUE0RDtBQUFuRCw4R0FBQSxhQUFhLE9BQUE7QUFDdEIsMkRBQTBEO0FBQWpELDRHQUFBLFlBQVksT0FBQTtBQUNyQixtRUFBa0U7QUFBekQsb0hBQUEsZ0JBQWdCLE9BQUE7QUFDekIsNkRBQTREO0FBQW5ELDhHQUFBLGFBQWEsT0FBQTtBQUN0QiwyRUFBMEU7QUFBakUsNEhBQUEsb0JBQW9CLE9BQUE7QUFDN0IsNkRBQWlGO0FBQXhFLDhHQUFBLGFBQWEsT0FBQTtBQUN0Qiw2RUFBNEU7QUFBbkUsOEhBQUEscUJBQXFCLE9BQUE7QUFDOUIsbUZBQWtGO0FBQXpFLG9JQUFBLHdCQUF3QixPQUFBO0FBQ2pDLDZEQUE4RTtBQUFyRSw4R0FBQSxhQUFhLE9BQUE7QUFDdEIsNkRBQTREO0FBQW5ELDhHQUFBLGFBQWEsT0FBQTtBQUN0QixtRkFJZ0Q7QUFIOUMsb0lBQUEsd0JBQXdCLE9BQUE7QUFJMUIsaUZBRytDO0FBRjdDLGtJQUFBLHVCQUF1QixPQUFBO0FBR3pCLDBDQUF3QiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB7IFByaWNpbmdTZXJ2aWNlIH0gZnJvbSAnLi9QcmljaW5nU2VydmljZSc7XG5leHBvcnQgeyBQcmljaW5nQ2xpZW50IH0gZnJvbSAnLi9QcmljaW5nQ2xpZW50JztcbmV4cG9ydCB7IENhY2hlTWFuYWdlciB9IGZyb20gJy4vQ2FjaGVNYW5hZ2VyJztcbmV4cG9ydCB7IEVDMkNhbGN1bGF0b3IgfSBmcm9tICcuL2NhbGN1bGF0b3JzL0VDMkNhbGN1bGF0b3InO1xuZXhwb3J0IHsgUzNDYWxjdWxhdG9yIH0gZnJvbSAnLi9jYWxjdWxhdG9ycy9TM0NhbGN1bGF0b3InO1xuZXhwb3J0IHsgTGFtYmRhQ2FsY3VsYXRvciB9IGZyb20gJy4vY2FsY3VsYXRvcnMvTGFtYmRhQ2FsY3VsYXRvcic7XG5leHBvcnQgeyBSRFNDYWxjdWxhdG9yIH0gZnJvbSAnLi9jYWxjdWxhdG9ycy9SRFNDYWxjdWxhdG9yJztcbmV4cG9ydCB7IENsb3VkRnJvbnRDYWxjdWxhdG9yIH0gZnJvbSAnLi9jYWxjdWxhdG9ycy9DbG91ZEZyb250Q2FsY3VsYXRvcic7XG5leHBvcnQgeyBFRlNDYWxjdWxhdG9yLCBFRlNVc2FnZUFzc3VtcHRpb25zIH0gZnJvbSAnLi9jYWxjdWxhdG9ycy9FRlNDYWxjdWxhdG9yJztcbmV4cG9ydCB7IEVsYXN0aUNhY2hlQ2FsY3VsYXRvciB9IGZyb20gJy4vY2FsY3VsYXRvcnMvRWxhc3RpQ2FjaGVDYWxjdWxhdG9yJztcbmV4cG9ydCB7IFNlY3JldHNNYW5hZ2VyQ2FsY3VsYXRvciB9IGZyb20gJy4vY2FsY3VsYXRvcnMvU2VjcmV0c01hbmFnZXJDYWxjdWxhdG9yJztcbmV4cG9ydCB7IFNOU0NhbGN1bGF0b3IsIFNOU0Nvc3RCcmVha2Rvd24gfSBmcm9tICcuL2NhbGN1bGF0b3JzL1NOU0NhbGN1bGF0b3InO1xuZXhwb3J0IHsgU1FTQ2FsY3VsYXRvciB9IGZyb20gJy4vY2FsY3VsYXRvcnMvU1FTQ2FsY3VsYXRvcic7XG5leHBvcnQge1xuICBMYXVuY2hUZW1wbGF0ZUNhbGN1bGF0b3IsXG4gIExhdW5jaFRlbXBsYXRlQ29uZmlnLFxuICBFYnNWb2x1bWVDb25maWcsXG59IGZyb20gJy4vY2FsY3VsYXRvcnMvTGF1bmNoVGVtcGxhdGVDYWxjdWxhdG9yJztcbmV4cG9ydCB7XG4gIFN0ZXBGdW5jdGlvbnNDYWxjdWxhdG9yLFxuICBTdGVwRnVuY3Rpb25zV29ya2Zsb3dUeXBlLFxufSBmcm9tICcuL2NhbGN1bGF0b3JzL1N0ZXBGdW5jdGlvbnNDYWxjdWxhdG9yJztcbmV4cG9ydCAqIGZyb20gJy4vdHlwZXMnO1xuIl19
@@ -1 +1 @@
1
- v0.1.32
1
+ v0.1.33
@@ -131,6 +131,99 @@ Total: $2.31/month
131
131
  - Cross-region replication costs not included
132
132
  - Lifecycle policies not considered
133
133
 
134
+ ### AWS::EFS::FileSystem
135
+
136
+ **Description:** Amazon Elastic File System for scalable, shared file storage
137
+
138
+ **Cost Components:**
139
+ - Standard Storage: GB-month for frequently accessed data
140
+ - Infrequent Access (IA) Storage: GB-month for infrequently accessed data
141
+ - IA Requests: Data transferred to/from IA storage
142
+ - Provisioned Throughput: MB/s-month (if configured)
143
+
144
+ **Default Assumptions:**
145
+ - 100 GB total storage
146
+ - 0% in Infrequent Access storage class
147
+ - Bursting throughput mode (no provisioned throughput cost)
148
+ - 10% of IA data accessed per month (for IA request cost estimation)
149
+
150
+ **Configuration:**
151
+ ```yaml
152
+ usageAssumptions:
153
+ efs:
154
+ storageSizeGb: 100
155
+ infrequentAccessPercentage: 0
156
+ ```
157
+
158
+ **Pricing Model:**
159
+ | Component | Price (us-east-1) |
160
+ |-----------|-------------------|
161
+ | Standard Storage | $0.30/GB-month |
162
+ | Infrequent Access Storage | $0.016/GB-month |
163
+ | IA Requests | $0.01/GB transferred |
164
+ | Provisioned Throughput | $6.00/MB/s-month |
165
+
166
+ **Example (Standard Storage Only):**
167
+ ```
168
+ Storage: 100 GB × $0.30/GB = $30.00/month
169
+ ```
170
+
171
+ **Example (With Infrequent Access - 50%):**
172
+ ```
173
+ Standard storage: 50 GB × $0.30/GB = $15.00
174
+ IA storage: 50 GB × $0.016/GB = $0.80
175
+ IA requests: 5 GB × $0.01/GB = $0.05
176
+ Total: $15.85/month
177
+ ```
178
+
179
+ **Example (With Provisioned Throughput):**
180
+ ```
181
+ Storage: 100 GB × $0.30/GB = $30.00
182
+ Provisioned: 10 MB/s × $6.00/MB/s = $60.00
183
+ Total: $90.00/month
184
+ ```
185
+
186
+ **Storage Class Detection:**
187
+
188
+ The calculator automatically detects Infrequent Access usage from CloudFormation template:
189
+
190
+ ```yaml
191
+ # CloudFormation template with lifecycle policy
192
+ Resources:
193
+ MyFileSystem:
194
+ Type: AWS::EFS::FileSystem
195
+ Properties:
196
+ LifecyclePolicies:
197
+ - TransitionToIA: AFTER_30_DAYS
198
+ ```
199
+
200
+ When `TransitionToIA` is detected, the calculator applies the configured `infrequentAccessPercentage` to estimate IA storage costs. Without a lifecycle policy, all storage is calculated at Standard rates.
201
+
202
+ **Throughput Mode Detection:**
203
+
204
+ The calculator detects provisioned throughput from CloudFormation template:
205
+
206
+ ```yaml
207
+ # CloudFormation template with provisioned throughput
208
+ Resources:
209
+ MyFileSystem:
210
+ Type: AWS::EFS::FileSystem
211
+ Properties:
212
+ ThroughputMode: provisioned
213
+ ProvisionedThroughputInMibps: 10
214
+ ```
215
+
216
+ When `ThroughputMode: provisioned` is detected with a `ProvisionedThroughputInMibps` value, the provisioned throughput cost is added to the total.
217
+
218
+ **Notes:**
219
+ - EFS One Zone storage class not currently supported (uses Standard pricing)
220
+ - Archive storage class not calculated
221
+ - Data transfer costs not included
222
+ - Mount target costs not included (no charge)
223
+ - Backup costs not included
224
+ - Actual storage used may vary from assumptions
225
+ - IA request cost assumes 10% of IA data is accessed monthly
226
+
134
227
  ## Database Resources
135
228
 
136
229
  ### AWS::RDS::DBInstance
@@ -761,6 +854,54 @@ Total: $0.02/month
761
854
  - First 4,000 state transitions per month are free tier eligible (Standard)
762
855
  - Activity polling and callbacks may incur additional charges
763
856
 
857
+ ## Security Resources
858
+
859
+ ### AWS::SecretsManager::Secret
860
+
861
+ **Description:** AWS Secrets Manager for credential and secret storage
862
+
863
+ **Cost Components:**
864
+ - Secret storage: $0.40 per secret per month
865
+ - API calls: $0.05 per 10,000 API calls
866
+
867
+ **Default Assumptions:**
868
+ - 10,000 API calls per month (GetSecretValue, DescribeSecret, etc.)
869
+
870
+ **Configuration:**
871
+ ```yaml
872
+ usageAssumptions:
873
+ secretsManager:
874
+ monthlyApiCalls: 10000
875
+ ```
876
+
877
+ **Example:**
878
+ ```
879
+ Secret storage: 1 secret × $0.40 = $0.40/month
880
+ API calls: 10,000 calls × $0.05/10K = $0.05/month
881
+ Total: $0.45/month
882
+ ```
883
+
884
+ **Example with higher API volume:**
885
+ ```yaml
886
+ usageAssumptions:
887
+ secretsManager:
888
+ monthlyApiCalls: 50000
889
+ ```
890
+
891
+ ```
892
+ Secret storage: 1 secret × $0.40 = $0.40/month
893
+ API calls: 50,000 calls × $0.05/10K = $0.25/month
894
+ Total: $0.65/month
895
+ ```
896
+
897
+ **Notes:**
898
+ - No free tier for Secrets Manager
899
+ - Automatic rotation is included in the base secret cost
900
+ - Cross-region replication incurs additional storage costs (not calculated)
901
+ - API call pricing includes GetSecretValue, DescribeSecret, ListSecrets, and other operations
902
+ - Deletion protection can prevent accidental secret deletion
903
+ - Secrets can be rotated automatically using Lambda functions (Lambda costs separate)
904
+
764
905
  ## Container Resources
765
906
 
766
907
  ### AWS::ECS::Service
@@ -823,6 +964,10 @@ usageAssumptions:
823
964
  getRequests: 100000
824
965
  putRequests: 10000
825
966
 
967
+ efs:
968
+ storageSizeGb: 500
969
+ infrequentAccessPercentage: 30 # 30% in Infrequent Access
970
+
826
971
  # Database
827
972
  rds:
828
973
  hoursPerMonth: 730
@@ -891,6 +1036,10 @@ usageAssumptions:
891
1036
  stateTransitionsPerExecution: 15 # State transitions per execution (Standard)
892
1037
  averageDurationMs: 2000 # Average duration in ms (Express)
893
1038
 
1039
+ # Security
1040
+ secretsManager:
1041
+ monthlyApiCalls: 50000 # API calls per month
1042
+
894
1043
  # Containers
895
1044
  ecs:
896
1045
  fargate:
@@ -917,11 +1066,18 @@ const result = await analyzeCosts({
917
1066
  invocationsPerMonth: 5000000,
918
1067
  averageDurationMs: 500,
919
1068
  },
1069
+ efs: {
1070
+ storageSizeGb: 200,
1071
+ infrequentAccessPercentage: 25,
1072
+ },
920
1073
  stepFunctions: {
921
1074
  monthlyExecutions: 50000,
922
1075
  stateTransitionsPerExecution: 15,
923
1076
  averageDurationMs: 2000,
924
1077
  },
1078
+ secretsManager: {
1079
+ monthlyApiCalls: 50000,
1080
+ },
925
1081
  },
926
1082
  });
927
1083
  ```
package/package.json CHANGED
@@ -83,7 +83,7 @@
83
83
  "publishConfig": {
84
84
  "access": "public"
85
85
  },
86
- "version": "0.1.32",
86
+ "version": "0.1.33",
87
87
  "bugs": {
88
88
  "url": "https://github.com/buildinginthecloud/cdk-cost-analyzer/issues"
89
89
  },