1 | ;
|
2 | var _a, _b;
|
3 | Object.defineProperty(exports, "__esModule", { value: true });
|
4 | exports.CloudFrontWebDistribution = exports.ViewerCertificate = exports.CloudFrontAllowedCachedMethods = exports.CloudFrontAllowedMethods = exports.OriginSslPolicy = exports.FailoverStatusCode = void 0;
|
5 | const jsiiDeprecationWarnings = require("../.warnings.jsii.js");
|
6 | const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
7 | const certificatemanager = require("@aws-cdk/aws-certificatemanager");
|
8 | const iam = require("@aws-cdk/aws-iam");
|
9 | const s3 = require("@aws-cdk/aws-s3");
|
10 | const cdk = require("@aws-cdk/core");
|
11 | const cloudfront_generated_1 = require("./cloudfront.generated");
|
12 | const distribution_1 = require("./distribution");
|
13 | /**
|
14 | * HTTP status code to failover to second origin
|
15 | */
|
16 | var FailoverStatusCode;
|
17 | (function (FailoverStatusCode) {
|
18 | /**
|
19 | * Forbidden (403)
|
20 | */
|
21 | FailoverStatusCode[FailoverStatusCode["FORBIDDEN"] = 403] = "FORBIDDEN";
|
22 | /**
|
23 | * Not found (404)
|
24 | */
|
25 | FailoverStatusCode[FailoverStatusCode["NOT_FOUND"] = 404] = "NOT_FOUND";
|
26 | /**
|
27 | * Internal Server Error (500)
|
28 | */
|
29 | FailoverStatusCode[FailoverStatusCode["INTERNAL_SERVER_ERROR"] = 500] = "INTERNAL_SERVER_ERROR";
|
30 | /**
|
31 | * Bad Gateway (502)
|
32 | */
|
33 | FailoverStatusCode[FailoverStatusCode["BAD_GATEWAY"] = 502] = "BAD_GATEWAY";
|
34 | /**
|
35 | * Service Unavailable (503)
|
36 | */
|
37 | FailoverStatusCode[FailoverStatusCode["SERVICE_UNAVAILABLE"] = 503] = "SERVICE_UNAVAILABLE";
|
38 | /**
|
39 | * Gateway Timeout (504)
|
40 | */
|
41 | FailoverStatusCode[FailoverStatusCode["GATEWAY_TIMEOUT"] = 504] = "GATEWAY_TIMEOUT";
|
42 | })(FailoverStatusCode = exports.FailoverStatusCode || (exports.FailoverStatusCode = {}));
|
43 | var OriginSslPolicy;
|
44 | (function (OriginSslPolicy) {
|
45 | OriginSslPolicy["SSL_V3"] = "SSLv3";
|
46 | OriginSslPolicy["TLS_V1"] = "TLSv1";
|
47 | OriginSslPolicy["TLS_V1_1"] = "TLSv1.1";
|
48 | OriginSslPolicy["TLS_V1_2"] = "TLSv1.2";
|
49 | })(OriginSslPolicy = exports.OriginSslPolicy || (exports.OriginSslPolicy = {}));
|
50 | /**
|
51 | * An enum for the supported methods to a CloudFront distribution.
|
52 | */
|
53 | var CloudFrontAllowedMethods;
|
54 | (function (CloudFrontAllowedMethods) {
|
55 | CloudFrontAllowedMethods["GET_HEAD"] = "GH";
|
56 | CloudFrontAllowedMethods["GET_HEAD_OPTIONS"] = "GHO";
|
57 | CloudFrontAllowedMethods["ALL"] = "ALL";
|
58 | })(CloudFrontAllowedMethods = exports.CloudFrontAllowedMethods || (exports.CloudFrontAllowedMethods = {}));
|
59 | /**
|
60 | * Enums for the methods CloudFront can cache.
|
61 | */
|
62 | var CloudFrontAllowedCachedMethods;
|
63 | (function (CloudFrontAllowedCachedMethods) {
|
64 | CloudFrontAllowedCachedMethods["GET_HEAD"] = "GH";
|
65 | CloudFrontAllowedCachedMethods["GET_HEAD_OPTIONS"] = "GHO";
|
66 | })(CloudFrontAllowedCachedMethods = exports.CloudFrontAllowedCachedMethods || (exports.CloudFrontAllowedCachedMethods = {}));
|
67 | /**
|
68 | * Viewer certificate configuration class
|
69 | */
|
70 | class ViewerCertificate {
|
71 | constructor(props, aliases = []) {
|
72 | this.props = props;
|
73 | this.aliases = aliases;
|
74 | }
|
75 | /**
|
76 | * Generate an AWS Certificate Manager (ACM) viewer certificate configuration
|
77 | *
|
78 | * @param certificate AWS Certificate Manager (ACM) certificate.
|
79 | * Your certificate must be located in the us-east-1 (US East (N. Virginia)) region to be accessed by CloudFront
|
80 | * @param options certificate configuration options
|
81 | */
|
82 | static fromAcmCertificate(certificate, options = {}) {
|
83 | try {
|
84 | jsiiDeprecationWarnings._aws_cdk_aws_cloudfront_ViewerCertificateOptions(options);
|
85 | }
|
86 | catch (error) {
|
87 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
88 | Error.captureStackTrace(error, this.fromAcmCertificate);
|
89 | }
|
90 | throw error;
|
91 | }
|
92 | const { sslMethod: sslSupportMethod = distribution_1.SSLMethod.SNI, securityPolicy: minimumProtocolVersion, aliases, } = options;
|
93 | return new ViewerCertificate({
|
94 | acmCertificateArn: certificate.certificateArn, sslSupportMethod, minimumProtocolVersion,
|
95 | }, aliases);
|
96 | }
|
97 | /**
|
98 | * Generate an IAM viewer certificate configuration
|
99 | *
|
100 | * @param iamCertificateId Identifier of the IAM certificate
|
101 | * @param options certificate configuration options
|
102 | */
|
103 | static fromIamCertificate(iamCertificateId, options = {}) {
|
104 | try {
|
105 | jsiiDeprecationWarnings._aws_cdk_aws_cloudfront_ViewerCertificateOptions(options);
|
106 | }
|
107 | catch (error) {
|
108 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
109 | Error.captureStackTrace(error, this.fromIamCertificate);
|
110 | }
|
111 | throw error;
|
112 | }
|
113 | const { sslMethod: sslSupportMethod = distribution_1.SSLMethod.SNI, securityPolicy: minimumProtocolVersion, aliases, } = options;
|
114 | return new ViewerCertificate({
|
115 | iamCertificateId, sslSupportMethod, minimumProtocolVersion,
|
116 | }, aliases);
|
117 | }
|
118 | /**
|
119 | * Generate a viewer certifcate configuration using
|
120 | * the CloudFront default certificate (e.g. d111111abcdef8.cloudfront.net)
|
121 | * and a {@link SecurityPolicyProtocol.TLS_V1} security policy.
|
122 | *
|
123 | * @param aliases Alternative CNAME aliases
|
124 | * You also must create a CNAME record with your DNS service to route queries
|
125 | */
|
126 | static fromCloudFrontDefaultCertificate(...aliases) {
|
127 | return new ViewerCertificate({ cloudFrontDefaultCertificate: true }, aliases);
|
128 | }
|
129 | }
|
130 | exports.ViewerCertificate = ViewerCertificate;
|
131 | _a = JSII_RTTI_SYMBOL_1;
|
132 | ViewerCertificate[_a] = { fqn: "@aws-cdk/aws-cloudfront.ViewerCertificate", version: "1.157.0" };
|
133 | /**
|
134 | * Amazon CloudFront is a global content delivery network (CDN) service that securely delivers data, videos,
|
135 | * applications, and APIs to your viewers with low latency and high transfer speeds.
|
136 | * CloudFront fronts user provided content and caches it at edge locations across the world.
|
137 | *
|
138 | * Here's how you can use this construct:
|
139 | *
|
140 | * ```ts
|
141 | * const sourceBucket = new s3.Bucket(this, 'Bucket');
|
142 | *
|
143 | * const distribution = new cloudfront.CloudFrontWebDistribution(this, 'MyDistribution', {
|
144 | * originConfigs: [
|
145 | * {
|
146 | * s3OriginSource: {
|
147 | * s3BucketSource: sourceBucket,
|
148 | * },
|
149 | * behaviors : [ {isDefaultBehavior: true}],
|
150 | * },
|
151 | * ],
|
152 | * });
|
153 | * ```
|
154 | *
|
155 | * This will create a CloudFront distribution that uses your S3Bucket as it's origin.
|
156 | *
|
157 | * You can customize the distribution using additional properties from the CloudFrontWebDistributionProps interface.
|
158 | *
|
159 | * @resource AWS::CloudFront::Distribution
|
160 | */
|
161 | class CloudFrontWebDistribution extends cdk.Resource {
|
162 | constructor(scope, id, props) {
|
163 | var _c, _d, _e, _f;
|
164 | super(scope, id);
|
165 | /**
|
166 | * Maps our methods to the string arrays they are
|
167 | */
|
168 | this.METHOD_LOOKUP_MAP = {
|
169 | GH: ['GET', 'HEAD'],
|
170 | GHO: ['GET', 'HEAD', 'OPTIONS'],
|
171 | ALL: ['DELETE', 'GET', 'HEAD', 'OPTIONS', 'PATCH', 'POST', 'PUT'],
|
172 | };
|
173 | /**
|
174 | * Maps for which SecurityPolicyProtocol are available to which SSLMethods
|
175 | */
|
176 | this.VALID_SSL_PROTOCOLS = {
|
177 | [distribution_1.SSLMethod.SNI]: [
|
178 | distribution_1.SecurityPolicyProtocol.TLS_V1, distribution_1.SecurityPolicyProtocol.TLS_V1_1_2016,
|
179 | distribution_1.SecurityPolicyProtocol.TLS_V1_2016, distribution_1.SecurityPolicyProtocol.TLS_V1_2_2018,
|
180 | distribution_1.SecurityPolicyProtocol.TLS_V1_2_2019, distribution_1.SecurityPolicyProtocol.TLS_V1_2_2021,
|
181 | ],
|
182 | [distribution_1.SSLMethod.VIP]: [distribution_1.SecurityPolicyProtocol.SSL_V3, distribution_1.SecurityPolicyProtocol.TLS_V1],
|
183 | };
|
184 | try {
|
185 | jsiiDeprecationWarnings._aws_cdk_aws_cloudfront_CloudFrontWebDistributionProps(props);
|
186 | }
|
187 | catch (error) {
|
188 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
189 | Error.captureStackTrace(error, this.constructor);
|
190 | }
|
191 | throw error;
|
192 | }
|
193 | // Comments have an undocumented limit of 128 characters
|
194 | const trimmedComment = props.comment && props.comment.length > 128
|
195 | ? `${props.comment.slice(0, 128 - 3)}...`
|
196 | : props.comment;
|
197 | let distributionConfig = {
|
198 | comment: trimmedComment,
|
199 | enabled: (_c = props.enabled) !== null && _c !== void 0 ? _c : true,
|
200 | defaultRootObject: (_d = props.defaultRootObject) !== null && _d !== void 0 ? _d : 'index.html',
|
201 | httpVersion: props.httpVersion || distribution_1.HttpVersion.HTTP2,
|
202 | priceClass: props.priceClass || distribution_1.PriceClass.PRICE_CLASS_100,
|
203 | ipv6Enabled: (_e = props.enableIpV6) !== null && _e !== void 0 ? _e : true,
|
204 | // eslint-disable-next-line max-len
|
205 | customErrorResponses: props.errorConfigurations,
|
206 | webAclId: props.webACLId,
|
207 | };
|
208 | const behaviors = [];
|
209 | const origins = [];
|
210 | const originGroups = [];
|
211 | let originIndex = 1;
|
212 | for (const originConfig of props.originConfigs) {
|
213 | let originId = `origin${originIndex}`;
|
214 | const originProperty = this.toOriginProperty(originConfig, originId);
|
215 | if (originConfig.failoverCustomOriginSource || originConfig.failoverS3OriginSource) {
|
216 | const originSecondaryId = `originSecondary${originIndex}`;
|
217 | const originSecondaryProperty = this.toOriginProperty({
|
218 | s3OriginSource: originConfig.failoverS3OriginSource,
|
219 | customOriginSource: originConfig.failoverCustomOriginSource,
|
220 | originPath: originConfig.originPath,
|
221 | originHeaders: originConfig.originHeaders,
|
222 | originShieldRegion: originConfig.originShieldRegion,
|
223 | }, originSecondaryId);
|
224 | const originGroupsId = `OriginGroup${originIndex}`;
|
225 | const failoverCodes = (_f = originConfig.failoverCriteriaStatusCodes) !== null && _f !== void 0 ? _f : [500, 502, 503, 504];
|
226 | originGroups.push({
|
227 | id: originGroupsId,
|
228 | members: {
|
229 | items: [{ originId }, { originId: originSecondaryId }],
|
230 | quantity: 2,
|
231 | },
|
232 | failoverCriteria: {
|
233 | statusCodes: {
|
234 | items: failoverCodes,
|
235 | quantity: failoverCodes.length,
|
236 | },
|
237 | },
|
238 | });
|
239 | originId = originGroupsId;
|
240 | origins.push(originSecondaryProperty);
|
241 | }
|
242 | for (const behavior of originConfig.behaviors) {
|
243 | behaviors.push({ ...behavior, targetOriginId: originId });
|
244 | }
|
245 | origins.push(originProperty);
|
246 | originIndex++;
|
247 | }
|
248 | origins.forEach(origin => {
|
249 | if (!origin.s3OriginConfig && !origin.customOriginConfig) {
|
250 | throw new Error(`Origin ${origin.domainName} is missing either S3OriginConfig or CustomOriginConfig. At least 1 must be specified.`);
|
251 | }
|
252 | });
|
253 | const originGroupsDistConfig = originGroups.length > 0
|
254 | ? {
|
255 | items: originGroups,
|
256 | quantity: originGroups.length,
|
257 | }
|
258 | : undefined;
|
259 | distributionConfig = {
|
260 | ...distributionConfig,
|
261 | origins,
|
262 | originGroups: originGroupsDistConfig,
|
263 | };
|
264 | const defaultBehaviors = behaviors.filter(behavior => behavior.isDefaultBehavior);
|
265 | if (defaultBehaviors.length !== 1) {
|
266 | throw new Error('There can only be one default behavior across all sources. [ One default behavior per distribution ].');
|
267 | }
|
268 | distributionConfig = { ...distributionConfig, defaultCacheBehavior: this.toBehavior(defaultBehaviors[0], props.viewerProtocolPolicy) };
|
269 | const otherBehaviors = [];
|
270 | for (const behavior of behaviors.filter(b => !b.isDefaultBehavior)) {
|
271 | if (!behavior.pathPattern) {
|
272 | throw new Error('pathPattern is required for all non-default behaviors');
|
273 | }
|
274 | otherBehaviors.push(this.toBehavior(behavior, props.viewerProtocolPolicy));
|
275 | }
|
276 | distributionConfig = { ...distributionConfig, cacheBehaviors: otherBehaviors.length > 0 ? otherBehaviors : undefined };
|
277 | if (props.aliasConfiguration && props.viewerCertificate) {
|
278 | throw new Error([
|
279 | 'You cannot set both aliasConfiguration and viewerCertificate properties.',
|
280 | 'Please only use viewerCertificate, as aliasConfiguration is deprecated.',
|
281 | ].join(' '));
|
282 | }
|
283 | let _viewerCertificate = props.viewerCertificate;
|
284 | if (props.aliasConfiguration) {
|
285 | const { acmCertRef, securityPolicy, sslMethod, names: aliases } = props.aliasConfiguration;
|
286 | _viewerCertificate = ViewerCertificate.fromAcmCertificate(certificatemanager.Certificate.fromCertificateArn(this, 'AliasConfigurationCert', acmCertRef), { securityPolicy, sslMethod, aliases });
|
287 | }
|
288 | if (_viewerCertificate) {
|
289 | const { props: viewerCertificate, aliases } = _viewerCertificate;
|
290 | Object.assign(distributionConfig, { aliases, viewerCertificate });
|
291 | const { minimumProtocolVersion, sslSupportMethod } = viewerCertificate;
|
292 | if (minimumProtocolVersion != null && sslSupportMethod != null) {
|
293 | const validProtocols = this.VALID_SSL_PROTOCOLS[sslSupportMethod];
|
294 | if (validProtocols.indexOf(minimumProtocolVersion.toString()) === -1) {
|
295 | // eslint-disable-next-line max-len
|
296 | throw new Error(`${minimumProtocolVersion} is not compabtible with sslMethod ${sslSupportMethod}.\n\tValid Protocols are: ${validProtocols.join(', ')}`);
|
297 | }
|
298 | }
|
299 | }
|
300 | else {
|
301 | distributionConfig = {
|
302 | ...distributionConfig,
|
303 | viewerCertificate: { cloudFrontDefaultCertificate: true },
|
304 | };
|
305 | }
|
306 | if (props.loggingConfig) {
|
307 | this.loggingBucket = props.loggingConfig.bucket || new s3.Bucket(this, 'LoggingBucket', {
|
308 | encryption: s3.BucketEncryption.S3_MANAGED,
|
309 | });
|
310 | distributionConfig = {
|
311 | ...distributionConfig,
|
312 | logging: {
|
313 | bucket: this.loggingBucket.bucketRegionalDomainName,
|
314 | includeCookies: props.loggingConfig.includeCookies || false,
|
315 | prefix: props.loggingConfig.prefix,
|
316 | },
|
317 | };
|
318 | }
|
319 | if (props.geoRestriction) {
|
320 | distributionConfig = {
|
321 | ...distributionConfig,
|
322 | restrictions: {
|
323 | geoRestriction: {
|
324 | restrictionType: props.geoRestriction.restrictionType,
|
325 | locations: props.geoRestriction.locations,
|
326 | },
|
327 | },
|
328 | };
|
329 | }
|
330 | const distribution = new cloudfront_generated_1.CfnDistribution(this, 'CFDistribution', { distributionConfig });
|
331 | this.node.defaultChild = distribution;
|
332 | this.domainName = distribution.attrDomainName;
|
333 | this.distributionDomainName = distribution.attrDomainName;
|
334 | this.distributionId = distribution.ref;
|
335 | }
|
336 | /**
|
337 | * Creates a construct that represents an external (imported) distribution.
|
338 | */
|
339 | static fromDistributionAttributes(scope, id, attrs) {
|
340 | try {
|
341 | jsiiDeprecationWarnings._aws_cdk_aws_cloudfront_CloudFrontWebDistributionAttributes(attrs);
|
342 | }
|
343 | catch (error) {
|
344 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
345 | Error.captureStackTrace(error, this.fromDistributionAttributes);
|
346 | }
|
347 | throw error;
|
348 | }
|
349 | return new class extends cdk.Resource {
|
350 | constructor() {
|
351 | super(scope, id);
|
352 | this.domainName = attrs.domainName;
|
353 | this.distributionDomainName = attrs.domainName;
|
354 | this.distributionId = attrs.distributionId;
|
355 | }
|
356 | }();
|
357 | }
|
358 | toBehavior(input, protoPolicy) {
|
359 | var _c;
|
360 | let toReturn = {
|
361 | allowedMethods: this.METHOD_LOOKUP_MAP[input.allowedMethods || CloudFrontAllowedMethods.GET_HEAD],
|
362 | cachedMethods: this.METHOD_LOOKUP_MAP[input.cachedMethods || CloudFrontAllowedCachedMethods.GET_HEAD],
|
363 | compress: input.compress !== false,
|
364 | defaultTtl: input.defaultTtl && input.defaultTtl.toSeconds(),
|
365 | forwardedValues: input.forwardedValues || { queryString: false, cookies: { forward: 'none' } },
|
366 | maxTtl: input.maxTtl && input.maxTtl.toSeconds(),
|
367 | minTtl: input.minTtl && input.minTtl.toSeconds(),
|
368 | trustedKeyGroups: (_c = input.trustedKeyGroups) === null || _c === void 0 ? void 0 : _c.map(key => key.keyGroupId),
|
369 | trustedSigners: input.trustedSigners,
|
370 | targetOriginId: input.targetOriginId,
|
371 | viewerProtocolPolicy: input.viewerProtocolPolicy || protoPolicy || distribution_1.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,
|
372 | };
|
373 | if (!input.isDefaultBehavior) {
|
374 | toReturn = Object.assign(toReturn, { pathPattern: input.pathPattern });
|
375 | }
|
376 | if (input.functionAssociations) {
|
377 | toReturn = Object.assign(toReturn, {
|
378 | functionAssociations: input.functionAssociations.map(association => ({
|
379 | functionArn: association.function.functionArn,
|
380 | eventType: association.eventType.toString(),
|
381 | })),
|
382 | });
|
383 | }
|
384 | if (input.lambdaFunctionAssociations) {
|
385 | const includeBodyEventTypes = [distribution_1.LambdaEdgeEventType.ORIGIN_REQUEST, distribution_1.LambdaEdgeEventType.VIEWER_REQUEST];
|
386 | if (input.lambdaFunctionAssociations.some(fna => fna.includeBody && !includeBodyEventTypes.includes(fna.eventType))) {
|
387 | throw new Error('\'includeBody\' can only be true for ORIGIN_REQUEST or VIEWER_REQUEST event types.');
|
388 | }
|
389 | toReturn = Object.assign(toReturn, {
|
390 | lambdaFunctionAssociations: input.lambdaFunctionAssociations
|
391 | .map(fna => ({
|
392 | eventType: fna.eventType,
|
393 | lambdaFunctionArn: fna.lambdaFunction && fna.lambdaFunction.edgeArn,
|
394 | includeBody: fna.includeBody,
|
395 | })),
|
396 | });
|
397 | // allow edgelambda.amazonaws.com to assume the functions' execution role.
|
398 | for (const a of input.lambdaFunctionAssociations) {
|
399 | if (a.lambdaFunction.role && a.lambdaFunction.role instanceof iam.Role && a.lambdaFunction.role.assumeRolePolicy) {
|
400 | a.lambdaFunction.role.assumeRolePolicy.addStatements(new iam.PolicyStatement({
|
401 | actions: ['sts:AssumeRole'],
|
402 | principals: [new iam.ServicePrincipal('edgelambda.amazonaws.com')],
|
403 | }));
|
404 | }
|
405 | }
|
406 | }
|
407 | return toReturn;
|
408 | }
|
409 | toOriginProperty(originConfig, originId) {
|
410 | var _c, _d, _e, _f, _g, _h, _j, _k, _l, _m, _o, _p, _q, _r, _s;
|
411 | if (!originConfig.s3OriginSource &&
|
412 | !originConfig.customOriginSource) {
|
413 | throw new Error('There must be at least one origin source - either an s3OriginSource, a customOriginSource');
|
414 | }
|
415 | if (originConfig.customOriginSource && originConfig.s3OriginSource) {
|
416 | throw new Error('There cannot be both an s3OriginSource and a customOriginSource in the same SourceConfiguration.');
|
417 | }
|
418 | if ([
|
419 | originConfig.originHeaders,
|
420 | (_c = originConfig.s3OriginSource) === null || _c === void 0 ? void 0 : _c.originHeaders,
|
421 | (_d = originConfig.customOriginSource) === null || _d === void 0 ? void 0 : _d.originHeaders,
|
422 | ].filter(x => x).length > 1) {
|
423 | throw new Error('Only one originHeaders field allowed across origin and failover origins');
|
424 | }
|
425 | if ([
|
426 | originConfig.originPath,
|
427 | (_e = originConfig.s3OriginSource) === null || _e === void 0 ? void 0 : _e.originPath,
|
428 | (_f = originConfig.customOriginSource) === null || _f === void 0 ? void 0 : _f.originPath,
|
429 | ].filter(x => x).length > 1) {
|
430 | throw new Error('Only one originPath field allowed across origin and failover origins');
|
431 | }
|
432 | if ([
|
433 | originConfig.originShieldRegion,
|
434 | (_g = originConfig.s3OriginSource) === null || _g === void 0 ? void 0 : _g.originShieldRegion,
|
435 | (_h = originConfig.customOriginSource) === null || _h === void 0 ? void 0 : _h.originShieldRegion,
|
436 | ].filter(x => x).length > 1) {
|
437 | throw new Error('Only one originShieldRegion field allowed across origin and failover origins');
|
438 | }
|
439 | const headers = (_l = (_j = originConfig.originHeaders) !== null && _j !== void 0 ? _j : (_k = originConfig.s3OriginSource) === null || _k === void 0 ? void 0 : _k.originHeaders) !== null && _l !== void 0 ? _l : (_m = originConfig.customOriginSource) === null || _m === void 0 ? void 0 : _m.originHeaders;
|
440 | const originHeaders = [];
|
441 | if (headers) {
|
442 | Object.keys(headers).forEach((key) => {
|
443 | const oHeader = {
|
444 | headerName: key,
|
445 | headerValue: headers[key],
|
446 | };
|
447 | originHeaders.push(oHeader);
|
448 | });
|
449 | }
|
450 | let s3OriginConfig;
|
451 | if (originConfig.s3OriginSource) {
|
452 | // first case for backwards compatibility
|
453 | if (originConfig.s3OriginSource.originAccessIdentity) {
|
454 | // grant CloudFront OriginAccessIdentity read access to S3 bucket
|
455 | // Used rather than `grantRead` because `grantRead` will grant overly-permissive policies.
|
456 | // Only GetObject is needed to retrieve objects for the distribution.
|
457 | // This also excludes KMS permissions; currently, OAI only supports SSE-S3 for buckets.
|
458 | // Source: https://aws.amazon.com/blogs/networking-and-content-delivery/serving-sse-kms-encrypted-content-from-s3-using-cloudfront/
|
459 | originConfig.s3OriginSource.s3BucketSource.addToResourcePolicy(new iam.PolicyStatement({
|
460 | resources: [originConfig.s3OriginSource.s3BucketSource.arnForObjects('*')],
|
461 | actions: ['s3:GetObject'],
|
462 | principals: [originConfig.s3OriginSource.originAccessIdentity.grantPrincipal],
|
463 | }));
|
464 | s3OriginConfig = {
|
465 | originAccessIdentity: `origin-access-identity/cloudfront/${originConfig.s3OriginSource.originAccessIdentity.originAccessIdentityName}`,
|
466 | };
|
467 | }
|
468 | else {
|
469 | s3OriginConfig = {};
|
470 | }
|
471 | }
|
472 | const connectionAttempts = (_o = originConfig.connectionAttempts) !== null && _o !== void 0 ? _o : 3;
|
473 | if (connectionAttempts < 1 || 3 < connectionAttempts || !Number.isInteger(connectionAttempts)) {
|
474 | throw new Error('connectionAttempts: You can specify 1, 2, or 3 as the number of attempts.');
|
475 | }
|
476 | const connectionTimeout = (originConfig.connectionTimeout || cdk.Duration.seconds(10)).toSeconds();
|
477 | if (connectionTimeout < 1 || 10 < connectionTimeout || !Number.isInteger(connectionTimeout)) {
|
478 | throw new Error('connectionTimeout: You can specify a number of seconds between 1 and 10 (inclusive).');
|
479 | }
|
480 | const originProperty = {
|
481 | id: originId,
|
482 | domainName: originConfig.s3OriginSource
|
483 | ? originConfig.s3OriginSource.s3BucketSource.bucketRegionalDomainName
|
484 | : originConfig.customOriginSource.domainName,
|
485 | originPath: (_r = (_p = originConfig.originPath) !== null && _p !== void 0 ? _p : (_q = originConfig.customOriginSource) === null || _q === void 0 ? void 0 : _q.originPath) !== null && _r !== void 0 ? _r : (_s = originConfig.s3OriginSource) === null || _s === void 0 ? void 0 : _s.originPath,
|
486 | originCustomHeaders: originHeaders.length > 0 ? originHeaders : undefined,
|
487 | s3OriginConfig,
|
488 | originShield: this.toOriginShieldProperty(originConfig),
|
489 | customOriginConfig: originConfig.customOriginSource
|
490 | ? {
|
491 | httpPort: originConfig.customOriginSource.httpPort || 80,
|
492 | httpsPort: originConfig.customOriginSource.httpsPort || 443,
|
493 | originKeepaliveTimeout: (originConfig.customOriginSource.originKeepaliveTimeout &&
|
494 | originConfig.customOriginSource.originKeepaliveTimeout.toSeconds()) ||
|
495 | 5,
|
496 | originReadTimeout: (originConfig.customOriginSource.originReadTimeout &&
|
497 | originConfig.customOriginSource.originReadTimeout.toSeconds()) ||
|
498 | 30,
|
499 | originProtocolPolicy: originConfig.customOriginSource.originProtocolPolicy ||
|
500 | distribution_1.OriginProtocolPolicy.HTTPS_ONLY,
|
501 | originSslProtocols: originConfig.customOriginSource
|
502 | .allowedOriginSSLVersions || [OriginSslPolicy.TLS_V1_2],
|
503 | }
|
504 | : undefined,
|
505 | connectionAttempts,
|
506 | connectionTimeout,
|
507 | };
|
508 | return originProperty;
|
509 | }
|
510 | /**
|
511 | * Takes origin shield region from props and converts to CfnDistribution.OriginShieldProperty
|
512 | */
|
513 | toOriginShieldProperty(originConfig) {
|
514 | var _c, _d, _e, _f;
|
515 | const originShieldRegion = (_e = (_c = originConfig.originShieldRegion) !== null && _c !== void 0 ? _c : (_d = originConfig.customOriginSource) === null || _d === void 0 ? void 0 : _d.originShieldRegion) !== null && _e !== void 0 ? _e : (_f = originConfig.s3OriginSource) === null || _f === void 0 ? void 0 : _f.originShieldRegion;
|
516 | return originShieldRegion
|
517 | ? { enabled: true, originShieldRegion }
|
518 | : undefined;
|
519 | }
|
520 | }
|
521 | exports.CloudFrontWebDistribution = CloudFrontWebDistribution;
|
522 | _b = JSII_RTTI_SYMBOL_1;
|
523 | CloudFrontWebDistribution[_b] = { fqn: "@aws-cdk/aws-cloudfront.CloudFrontWebDistribution", version: "1.157.0" };
|
524 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"web-distribution.js","sourceRoot":"","sources":["web-distribution.ts"],"names":[],"mappings":"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","sourcesContent":["import * as certificatemanager from '@aws-cdk/aws-certificatemanager';\nimport * as iam from '@aws-cdk/aws-iam';\nimport * as lambda from '@aws-cdk/aws-lambda';\nimport * as s3 from '@aws-cdk/aws-s3';\nimport * as cdk from '@aws-cdk/core';\nimport { Construct } from 'constructs';\nimport { CfnDistribution } from './cloudfront.generated';\nimport { HttpVersion, IDistribution, LambdaEdgeEventType, OriginProtocolPolicy, PriceClass, ViewerProtocolPolicy, SSLMethod, SecurityPolicyProtocol } from './distribution';\nimport { FunctionAssociation } from './function';\nimport { GeoRestriction } from './geo-restriction';\nimport { IKeyGroup } from './key-group';\nimport { IOriginAccessIdentity } from './origin-access-identity';\n\n/**\n * HTTP status code to failover to second origin\n */\nexport enum FailoverStatusCode {\n  /**\n   * Forbidden (403)\n   */\n  FORBIDDEN = 403,\n\n  /**\n   * Not found (404)\n   */\n  NOT_FOUND = 404,\n\n  /**\n   * Internal Server Error (500)\n   */\n  INTERNAL_SERVER_ERROR = 500,\n\n  /**\n   * Bad Gateway (502)\n   */\n  BAD_GATEWAY = 502,\n\n  /**\n   * Service Unavailable (503)\n   */\n  SERVICE_UNAVAILABLE = 503,\n\n  /**\n   * Gateway Timeout (504)\n   */\n  GATEWAY_TIMEOUT = 504,\n}\n\n/**\n * Configuration for custom domain names\n *\n * CloudFront can use a custom domain that you provide instead of a\n * \"cloudfront.net\" domain. To use this feature you must provide the list of\n * additional domains, and the ACM Certificate that CloudFront should use for\n * these additional domains.\n * @deprecated see {@link CloudFrontWebDistributionProps#viewerCertificate} with {@link ViewerCertificate#acmCertificate}\n */\nexport interface AliasConfiguration {\n  /**\n   * ARN of an AWS Certificate Manager (ACM) certificate.\n   */\n  readonly acmCertRef: string;\n\n  /**\n   * Domain names on the certificate\n   *\n   * Both main domain name and Subject Alternative Names.\n   */\n  readonly names: string[];\n\n  /**\n   * How CloudFront should serve HTTPS requests.\n   *\n   * See the notes on SSLMethod if you wish to use other SSL termination types.\n   *\n   * @default SSLMethod.SNI\n   * @see https://docs.aws.amazon.com/cloudfront/latest/APIReference/API_ViewerCertificate.html\n   */\n  readonly sslMethod?: SSLMethod;\n\n  /**\n   * The minimum version of the SSL protocol that you want CloudFront to use for HTTPS connections.\n   *\n   * CloudFront serves your objects only to browsers or devices that support at\n   * least the SSL version that you specify.\n   *\n   * @default - SSLv3 if sslMethod VIP, TLSv1 if sslMethod SNI\n   */\n  readonly securityPolicy?: SecurityPolicyProtocol;\n}\n\n/**\n * Logging configuration for incoming requests\n */\nexport interface LoggingConfiguration {\n  /**\n   * Bucket to log requests to\n   *\n   * @default - A logging bucket is automatically created.\n   */\n  readonly bucket?: s3.IBucket,\n\n  /**\n   * Whether to include the cookies in the logs\n   *\n   * @default false\n   */\n  readonly includeCookies?: boolean,\n\n  /**\n   * Where in the bucket to store logs\n   *\n   * @default - No prefix.\n   */\n  readonly prefix?: string\n}\n\n// Subset of SourceConfiguration for rendering properties internally\ninterface SourceConfigurationRender {\n  readonly connectionAttempts?: number;\n  readonly connectionTimeout?: cdk.Duration;\n  readonly s3OriginSource?: S3OriginConfig;\n  readonly customOriginSource?: CustomOriginConfig;\n  readonly originPath?: string;\n  readonly originHeaders?: { [key: string]: string };\n  readonly originShieldRegion?: string\n}\n\n/**\n * A source configuration is a wrapper for CloudFront origins and behaviors.\n * An origin is what CloudFront will \"be in front of\" - that is, CloudFront will pull it's assets from an origin.\n *\n * If you're using s3 as a source - pass the `s3Origin` property, otherwise, pass the `customOriginSource` property.\n *\n * One or the other must be passed, and it is invalid to pass both in the same SourceConfiguration.\n */\nexport interface SourceConfiguration {\n  /**\n   * The number of times that CloudFront attempts to connect to the origin.\n   * You can specify 1, 2, or 3 as the number of attempts.\n   *\n   * @default 3\n   */\n  readonly connectionAttempts?: number;\n\n  /**\n   * The number of seconds that CloudFront waits when trying to establish a connection to the origin.\n   * You can specify a number of seconds between 1 and 10 (inclusive).\n   *\n   * @default cdk.Duration.seconds(10)\n   */\n  readonly connectionTimeout?: cdk.Duration;\n\n  /**\n   * An s3 origin source - if you're using s3 for your assets\n   */\n  readonly s3OriginSource?: S3OriginConfig;\n\n  /**\n   * A custom origin source - for all non-s3 sources.\n   */\n  readonly customOriginSource?: CustomOriginConfig;\n\n  /**\n   * An s3 origin source for failover in case the s3OriginSource returns invalid status code\n   *\n   * @default - no failover configuration\n   */\n  readonly failoverS3OriginSource?: S3OriginConfig;\n\n  /**\n   * A custom origin source for failover in case the s3OriginSource returns invalid status code\n   *\n   * @default - no failover configuration\n   */\n  readonly failoverCustomOriginSource?: CustomOriginConfig;\n\n  /**\n   * HTTP status code to failover to second origin\n   *\n   * @default [500, 502, 503, 504]\n   */\n  readonly failoverCriteriaStatusCodes?: FailoverStatusCode[];\n\n  /**\n   * The behaviors associated with this source.\n   * At least one (default) behavior must be included.\n   */\n  readonly behaviors: Behavior[];\n\n  /**\n   * The relative path to the origin root to use for sources.\n   *\n   * @default /\n   * @deprecated Use originPath on s3OriginSource or customOriginSource\n   */\n  readonly originPath?: string;\n\n  /**\n   * Any additional headers to pass to the origin\n   *\n   * @default - No additional headers are passed.\n   * @deprecated Use originHeaders on s3OriginSource or customOriginSource\n   */\n  readonly originHeaders?: { [key: string]: string };\n\n  /**\n   * When you enable Origin Shield in the AWS Region that has the lowest latency to your origin, you can get better network performance\n   *\n   * @see https://docs.aws.amazon.com/AmazonCloudFront/latest/DeveloperGuide/origin-shield.html\n   *\n   * @default - origin shield not enabled\n   */\n  readonly originShieldRegion?: string;\n}\n\n/**\n * A custom origin configuration\n */\nexport interface CustomOriginConfig {\n  /**\n   * The domain name of the custom origin. Should not include the path - that should be in the parent SourceConfiguration\n   */\n  readonly domainName: string,\n\n  /**\n   * The origin HTTP port\n   *\n   * @default 80\n   */\n  readonly httpPort?: number,\n\n  /**\n   * The origin HTTPS port\n   *\n   * @default 443\n   */\n  readonly httpsPort?: number,\n\n  /**\n   * The keep alive timeout when making calls in seconds.\n   *\n   * @default Duration.seconds(5)\n   */\n  readonly originKeepaliveTimeout?: cdk.Duration,\n\n  /**\n   * The protocol (http or https) policy to use when interacting with the origin.\n   *\n   * @default OriginProtocolPolicy.HttpsOnly\n   */\n  readonly originProtocolPolicy?: OriginProtocolPolicy,\n\n  /**\n   * The read timeout when calling the origin in seconds\n   *\n   * @default Duration.seconds(30)\n   */\n  readonly originReadTimeout?: cdk.Duration\n\n  /**\n   * The SSL versions to use when interacting with the origin.\n   *\n   * @default OriginSslPolicy.TLS_V1_2\n   */\n  readonly allowedOriginSSLVersions?: OriginSslPolicy[];\n\n  /**\n   * The relative path to the origin root to use for sources.\n   *\n   * @default /\n   */\n  readonly originPath?: string;\n\n  /**\n   * Any additional headers to pass to the origin\n   *\n   * @default - No additional headers are passed.\n   */\n  readonly originHeaders?: { [key: string]: string };\n\n  /**\n   * When you enable Origin Shield in the AWS Region that has the lowest latency to your origin, you can get better network performance\n   *\n   * @default - origin shield not enabled\n   */\n  readonly originShieldRegion?: string;\n}\n\nexport enum OriginSslPolicy {\n  SSL_V3 = 'SSLv3',\n  TLS_V1 = 'TLSv1',\n  TLS_V1_1 = 'TLSv1.1',\n  TLS_V1_2 = 'TLSv1.2',\n}\n\n/**\n * S3 origin configuration for CloudFront\n */\nexport interface S3OriginConfig {\n  /**\n   * The source bucket to serve content from\n   */\n  readonly s3BucketSource: s3.IBucket;\n\n  /**\n   * The optional Origin Access Identity of the origin identity cloudfront will use when calling your s3 bucket.\n   *\n   * @default No Origin Access Identity which requires the S3 bucket to be public accessible\n   */\n  readonly originAccessIdentity?: IOriginAccessIdentity;\n\n  /**\n   * The relative path to the origin root to use for sources.\n   *\n   * @default /\n   */\n  readonly originPath?: string;\n\n  /**\n   * Any additional headers to pass to the origin\n   *\n   * @default - No additional headers are passed.\n   */\n  readonly originHeaders?: { [key: string]: string };\n\n  /**\n   * When you enable Origin Shield in the AWS Region that has the lowest latency to your origin, you can get better network performance\n   *\n   * @default - origin shield not enabled\n   */\n  readonly originShieldRegion?: string;\n}\n\n/**\n * An enum for the supported methods to a CloudFront distribution.\n */\nexport enum CloudFrontAllowedMethods {\n  GET_HEAD = 'GH',\n  GET_HEAD_OPTIONS = 'GHO',\n  ALL = 'ALL'\n}\n\n/**\n * Enums for the methods CloudFront can cache.\n */\nexport enum CloudFrontAllowedCachedMethods {\n  GET_HEAD = 'GH',\n  GET_HEAD_OPTIONS = 'GHO',\n}\n\n/**\n * A CloudFront behavior wrapper.\n */\nexport interface Behavior {\n\n  /**\n   * If CloudFront should automatically compress some content types.\n   *\n   * @default true\n   */\n  readonly compress?: boolean;\n\n  /**\n   * If this behavior is the default behavior for the distribution.\n   *\n   * You must specify exactly one default distribution per CloudFront distribution.\n   * The default behavior is allowed to omit the \"path\" property.\n   */\n  readonly isDefaultBehavior?: boolean;\n\n  /**\n   * Trusted signers is how CloudFront allows you to serve private content.\n   * The signers are the account IDs that are allowed to sign cookies/presigned URLs for this distribution.\n   *\n   * If you pass a non empty value, all requests for this behavior must be signed (no public access will be allowed)\n   * @deprecated - We recommend using trustedKeyGroups instead of trustedSigners.\n   */\n  readonly trustedSigners?: string[];\n\n  /**\n   * A list of Key Groups that CloudFront can use to validate signed URLs or signed cookies.\n   *\n   * @default - no KeyGroups are associated with cache behavior\n   * @see https://docs.aws.amazon.com/AmazonCloudFront/latest/DeveloperGuide/PrivateContent.html\n   */\n  readonly trustedKeyGroups?: IKeyGroup[];\n\n  /**\n   *\n   * The default amount of time CloudFront will cache an object.\n   *\n   * This value applies only when your custom origin does not add HTTP headers,\n   * such as Cache-Control max-age, Cache-Control s-maxage, and Expires to objects.\n   * @default 86400 (1 day)\n   *\n   */\n  readonly defaultTtl?: cdk.Duration;\n\n  /**\n   * The method this CloudFront distribution responds do.\n   *\n   * @default GET_HEAD\n   */\n  readonly allowedMethods?: CloudFrontAllowedMethods;\n\n  /**\n   * The path this behavior responds to.\n   * Required for all non-default behaviors. (The default behavior implicitly has \"*\" as the path pattern. )\n   *\n   */\n  readonly pathPattern?: string;\n\n  /**\n   * Which methods are cached by CloudFront by default.\n   *\n   * @default GET_HEAD\n   */\n  readonly cachedMethods?: CloudFrontAllowedCachedMethods;\n\n  /**\n   * The values CloudFront will forward to the origin when making a request.\n   *\n   * @default none (no cookies - no headers)\n   *\n   */\n  readonly forwardedValues?: CfnDistribution.ForwardedValuesProperty;\n\n  /**\n   * The minimum amount of time that you want objects to stay in the cache\n   * before CloudFront queries your origin.\n   */\n  readonly minTtl?: cdk.Duration;\n\n  /**\n   * The max amount of time you want objects to stay in the cache\n   * before CloudFront queries your origin.\n   *\n   * @default Duration.seconds(31536000) (one year)\n   */\n  readonly maxTtl?: cdk.Duration;\n\n  /**\n   * Declares associated lambda@edge functions for this distribution behaviour.\n   *\n   * @default No lambda function associated\n   */\n  readonly lambdaFunctionAssociations?: LambdaFunctionAssociation[];\n\n  /**\n   * The CloudFront functions to invoke before serving the contents.\n   *\n   * @default - no functions will be invoked\n   */\n  readonly functionAssociations?: FunctionAssociation[];\n\n  /**\n   * The viewer policy for this behavior.\n   *\n   * @default - the distribution wide viewer protocol policy will be used\n   */\n  readonly viewerProtocolPolicy?: ViewerProtocolPolicy;\n}\n\nexport interface LambdaFunctionAssociation {\n\n  /**\n   * The lambda event type defines at which event the lambda\n   * is called during the request lifecycle\n   */\n  readonly eventType: LambdaEdgeEventType;\n\n  /**\n   * A version of the lambda to associate\n   */\n  readonly lambdaFunction: lambda.IVersion;\n\n  /**\n   * Allows a Lambda function to have read access to the body content.\n   * Only valid for \"request\" event types (`ORIGIN_REQUEST` or `VIEWER_REQUEST`).\n   * See https://docs.aws.amazon.com/AmazonCloudFront/latest/DeveloperGuide/lambda-include-body-access.html\n   *\n   * @default false\n   */\n  readonly includeBody?: boolean;\n}\n\nexport interface ViewerCertificateOptions {\n  /**\n   * How CloudFront should serve HTTPS requests.\n   *\n   * See the notes on SSLMethod if you wish to use other SSL termination types.\n   *\n   * @default SSLMethod.SNI\n   * @see https://docs.aws.amazon.com/cloudfront/latest/APIReference/API_ViewerCertificate.html\n   */\n  readonly sslMethod?: SSLMethod;\n\n  /**\n   * The minimum version of the SSL protocol that you want CloudFront to use for HTTPS connections.\n   *\n   * CloudFront serves your objects only to browsers or devices that support at\n   * least the SSL version that you specify.\n   *\n   * @default - SSLv3 if sslMethod VIP, TLSv1 if sslMethod SNI\n   */\n  readonly securityPolicy?: SecurityPolicyProtocol;\n\n  /**\n   * Domain names on the certificate (both main domain name and Subject Alternative names)\n   */\n  readonly aliases?: string[];\n}\n\n/**\n * Viewer certificate configuration class\n */\nexport class ViewerCertificate {\n  /**\n   * Generate an AWS Certificate Manager (ACM) viewer certificate configuration\n   *\n   * @param certificate AWS Certificate Manager (ACM) certificate.\n   *                    Your certificate must be located in the us-east-1 (US East (N. Virginia)) region to be accessed by CloudFront\n   * @param options certificate configuration options\n   */\n  public static fromAcmCertificate(certificate: certificatemanager.ICertificate, options: ViewerCertificateOptions = {}) {\n    const {\n      sslMethod: sslSupportMethod = SSLMethod.SNI,\n      securityPolicy: minimumProtocolVersion,\n      aliases,\n    } = options;\n\n    return new ViewerCertificate({\n      acmCertificateArn: certificate.certificateArn, sslSupportMethod, minimumProtocolVersion,\n    }, aliases);\n  }\n\n  /**\n   * Generate an IAM viewer certificate configuration\n   *\n   * @param iamCertificateId Identifier of the IAM certificate\n   * @param options certificate configuration options\n   */\n  public static fromIamCertificate(iamCertificateId: string, options: ViewerCertificateOptions = {}) {\n    const {\n      sslMethod: sslSupportMethod = SSLMethod.SNI,\n      securityPolicy: minimumProtocolVersion,\n      aliases,\n    } = options;\n\n    return new ViewerCertificate({\n      iamCertificateId, sslSupportMethod, minimumProtocolVersion,\n    }, aliases);\n  }\n\n  /**\n   * Generate a viewer certifcate configuration using\n   * the CloudFront default certificate (e.g. d111111abcdef8.cloudfront.net)\n   * and a {@link SecurityPolicyProtocol.TLS_V1} security policy.\n   *\n   * @param aliases Alternative CNAME aliases\n   *                You also must create a CNAME record with your DNS service to route queries\n   */\n  public static fromCloudFrontDefaultCertificate(...aliases: string[]) {\n    return new ViewerCertificate({ cloudFrontDefaultCertificate: true }, aliases);\n  }\n\n  private constructor(\n    public readonly props: CfnDistribution.ViewerCertificateProperty,\n    public readonly aliases: string[] = []) { }\n}\n\nexport interface CloudFrontWebDistributionProps {\n\n  /**\n   * AliasConfiguration is used to configured CloudFront to respond to requests on custom domain names.\n   *\n   * @default - None.\n   * @deprecated see {@link CloudFrontWebDistributionProps#viewerCertificate} with {@link ViewerCertificate#acmCertificate}\n   */\n  readonly aliasConfiguration?: AliasConfiguration;\n\n  /**\n   * A comment for this distribution in the CloudFront console.\n   *\n   * @default - No comment is added to distribution.\n   */\n  readonly comment?: string;\n\n  /**\n   * Enable or disable the distribution.\n   *\n   * @default true\n   */\n  readonly enabled?: boolean;\n\n  /**\n   * The default object to serve.\n   *\n   * @default - \"index.html\" is served.\n   */\n  readonly defaultRootObject?: string;\n\n  /**\n   * If your distribution should have IPv6 enabled.\n   *\n   * @default true\n   */\n  readonly enableIpV6?: boolean;\n\n  /**\n   * The max supported HTTP Versions.\n   *\n   * @default HttpVersion.HTTP2\n   */\n  readonly httpVersion?: HttpVersion;\n\n  /**\n   * The price class for the distribution (this impacts how many locations CloudFront uses for your distribution, and billing)\n   *\n   * @default PriceClass.PRICE_CLASS_100 the cheapest option for CloudFront is picked by default.\n   */\n  readonly priceClass?: PriceClass;\n\n  /**\n   * The default viewer policy for incoming clients.\n   *\n   * @default RedirectToHTTPs\n   */\n  readonly viewerProtocolPolicy?: ViewerProtocolPolicy;\n\n  /**\n   * The origin configurations for this distribution. Behaviors are a part of the origin.\n   */\n  readonly originConfigs: SourceConfiguration[];\n\n  /**\n   * Optional - if we should enable logging.\n   * You can pass an empty object ({}) to have us auto create a bucket for logging.\n   * Omission of this property indicates no logging is to be enabled.\n   *\n   * @default - no logging is enabled by default.\n   */\n  readonly loggingConfig?: LoggingConfiguration;\n\n  /**\n   * How CloudFront should handle requests that are not successful (eg PageNotFound)\n   *\n   * By default, CloudFront does not replace HTTP status codes in the 4xx and 5xx range\n   * with custom error messages. CloudFront does not cache HTTP status codes.\n   *\n   * @default - No custom error configuration.\n   */\n  readonly errorConfigurations?: CfnDistribution.CustomErrorResponseProperty[];\n\n  /**\n   * Unique identifier that specifies the AWS WAF web ACL to associate with this CloudFront distribution.\n   *\n   * To specify a web ACL created using the latest version of AWS WAF, use the ACL ARN, for example\n   * `arn:aws:wafv2:us-east-1:123456789012:global/webacl/ExampleWebACL/473e64fd-f30b-4765-81a0-62ad96dd167a`.\n   *\n   * To specify a web ACL created using AWS WAF Classic, use the ACL ID, for example `473e64fd-f30b-4765-81a0-62ad96dd167a`.\n   *\n   * @see https://docs.aws.amazon.com/waf/latest/developerguide/what-is-aws-waf.html\n   * @see https://docs.aws.amazon.com/cloudfront/latest/APIReference/API_CreateDistribution.html#API_CreateDistribution_RequestParameters.\n   *\n   * @default - No AWS Web Application Firewall web access control list (web ACL).\n   */\n  readonly webACLId?: string;\n\n  /**\n   * Specifies whether you want viewers to use HTTP or HTTPS to request your objects,\n   * whether you're using an alternate domain name with HTTPS, and if so,\n   * if you're using AWS Certificate Manager (ACM) or a third-party certificate authority.\n   *\n   * @default ViewerCertificate.fromCloudFrontDefaultCertificate()\n   *\n   * @see https://aws.amazon.com/premiumsupport/knowledge-center/custom-ssl-certificate-cloudfront/\n   */\n  readonly viewerCertificate?: ViewerCertificate;\n\n  /**\n   * Controls the countries in which your content is distributed.\n   *\n   * @default No geo restriction\n   */\n  readonly geoRestriction?: GeoRestriction;\n}\n\n/**\n * Internal only - just adds the originId string to the Behavior\n */\ninterface BehaviorWithOrigin extends Behavior {\n  readonly targetOriginId: string;\n}\n\n/**\n * Attributes used to import a Distribution.\n */\nexport interface CloudFrontWebDistributionAttributes {\n  /**\n   * The generated domain name of the Distribution, such as d111111abcdef8.cloudfront.net.\n   *\n   * @attribute\n   */\n  readonly domainName: string;\n\n  /**\n   * The distribution ID for this distribution.\n   *\n   * @attribute\n   */\n  readonly distributionId: string;\n}\n\n/**\n * Amazon CloudFront is a global content delivery network (CDN) service that securely delivers data, videos,\n * applications, and APIs to your viewers with low latency and high transfer speeds.\n * CloudFront fronts user provided content and caches it at edge locations across the world.\n *\n * Here's how you can use this construct:\n *\n * ```ts\n * const sourceBucket = new s3.Bucket(this, 'Bucket');\n *\n * const distribution = new cloudfront.CloudFrontWebDistribution(this, 'MyDistribution', {\n *   originConfigs: [\n *     {\n *       s3OriginSource: {\n *       s3BucketSource: sourceBucket,\n *       },\n *       behaviors : [ {isDefaultBehavior: true}],\n *     },\n *   ],\n * });\n * ```\n *\n * This will create a CloudFront distribution that uses your S3Bucket as it's origin.\n *\n * You can customize the distribution using additional properties from the CloudFrontWebDistributionProps interface.\n *\n * @resource AWS::CloudFront::Distribution\n */\nexport class CloudFrontWebDistribution extends cdk.Resource implements IDistribution {\n\n  /**\n   * Creates a construct that represents an external (imported) distribution.\n   */\n  public static fromDistributionAttributes(scope: Construct, id: string, attrs: CloudFrontWebDistributionAttributes): IDistribution {\n    return new class extends cdk.Resource implements IDistribution {\n      public readonly domainName: string;\n      public readonly distributionDomainName: string;\n      public readonly distributionId: string;\n\n      constructor() {\n        super(scope, id);\n        this.domainName = attrs.domainName;\n        this.distributionDomainName = attrs.domainName;\n        this.distributionId = attrs.distributionId;\n      }\n    }();\n  }\n\n  /**\n   * The logging bucket for this CloudFront distribution.\n   * If logging is not enabled for this distribution - this property will be undefined.\n   */\n  public readonly loggingBucket?: s3.IBucket;\n\n  /**\n   * The domain name created by CloudFront for this distribution.\n   * If you are using aliases for your distribution, this is the domainName your DNS records should point to.\n   * (In Route53, you could create an ALIAS record to this value, for example.)\n   *\n   * @deprecated - Use `distributionDomainName` instead.\n   */\n  public readonly domainName: string;\n\n  /**\n   * The domain name created by CloudFront for this distribution.\n   * If you are using aliases for your distribution, this is the domainName your DNS records should point to.\n   * (In Route53, you could create an ALIAS record to this value, for example.)\n   */\n  public readonly distributionDomainName: string;\n\n  /**\n   * The distribution ID for this distribution.\n   */\n  public readonly distributionId: string;\n\n  /**\n   * Maps our methods to the string arrays they are\n   */\n  private readonly METHOD_LOOKUP_MAP = {\n    GH: ['GET', 'HEAD'],\n    GHO: ['GET', 'HEAD', 'OPTIONS'],\n    ALL: ['DELETE', 'GET', 'HEAD', 'OPTIONS', 'PATCH', 'POST', 'PUT'],\n  };\n\n  /**\n   * Maps for which SecurityPolicyProtocol are available to which SSLMethods\n   */\n  private readonly VALID_SSL_PROTOCOLS: { [method in SSLMethod]: string[] } = {\n    [SSLMethod.SNI]: [\n      SecurityPolicyProtocol.TLS_V1, SecurityPolicyProtocol.TLS_V1_1_2016,\n      SecurityPolicyProtocol.TLS_V1_2016, SecurityPolicyProtocol.TLS_V1_2_2018,\n      SecurityPolicyProtocol.TLS_V1_2_2019, SecurityPolicyProtocol.TLS_V1_2_2021,\n    ],\n    [SSLMethod.VIP]: [SecurityPolicyProtocol.SSL_V3, SecurityPolicyProtocol.TLS_V1],\n  };\n\n  constructor(scope: Construct, id: string, props: CloudFrontWebDistributionProps) {\n    super(scope, id);\n\n    // Comments have an undocumented limit of 128 characters\n    const trimmedComment =\n      props.comment && props.comment.length > 128\n        ? `${props.comment.slice(0, 128 - 3)}...`\n        : props.comment;\n\n    let distributionConfig: CfnDistribution.DistributionConfigProperty = {\n      comment: trimmedComment,\n      enabled: props.enabled ?? true,\n      defaultRootObject: props.defaultRootObject ?? 'index.html',\n      httpVersion: props.httpVersion || HttpVersion.HTTP2,\n      priceClass: props.priceClass || PriceClass.PRICE_CLASS_100,\n      ipv6Enabled: props.enableIpV6 ?? true,\n      // eslint-disable-next-line max-len\n      customErrorResponses: props.errorConfigurations, // TODO: validation : https://docs.aws.amazon.com/AWSCloudFormation/latest/UserGuide/aws-properties-cloudfront-distribution-customerrorresponse.html#cfn-cloudfront-distribution-customerrorresponse-errorcachingminttl\n      webAclId: props.webACLId,\n    };\n\n    const behaviors: BehaviorWithOrigin[] = [];\n\n    const origins: CfnDistribution.OriginProperty[] = [];\n\n    const originGroups: CfnDistribution.OriginGroupProperty[] = [];\n\n    let originIndex = 1;\n    for (const originConfig of props.originConfigs) {\n      let originId = `origin${originIndex}`;\n      const originProperty = this.toOriginProperty(originConfig, originId);\n\n      if (originConfig.failoverCustomOriginSource || originConfig.failoverS3OriginSource) {\n        const originSecondaryId = `originSecondary${originIndex}`;\n        const originSecondaryProperty = this.toOriginProperty(\n          {\n            s3OriginSource: originConfig.failoverS3OriginSource,\n            customOriginSource: originConfig.failoverCustomOriginSource,\n            originPath: originConfig.originPath,\n            originHeaders: originConfig.originHeaders,\n            originShieldRegion: originConfig.originShieldRegion,\n          },\n          originSecondaryId,\n        );\n        const originGroupsId = `OriginGroup${originIndex}`;\n        const failoverCodes = originConfig.failoverCriteriaStatusCodes ?? [500, 502, 503, 504];\n        originGroups.push({\n          id: originGroupsId,\n          members: {\n            items: [{ originId }, { originId: originSecondaryId }],\n            quantity: 2,\n          },\n          failoverCriteria: {\n            statusCodes: {\n              items: failoverCodes,\n              quantity: failoverCodes.length,\n            },\n          },\n        });\n        originId = originGroupsId;\n        origins.push(originSecondaryProperty);\n      }\n\n      for (const behavior of originConfig.behaviors) {\n        behaviors.push({ ...behavior, targetOriginId: originId });\n      }\n\n      origins.push(originProperty);\n      originIndex++;\n    }\n\n    origins.forEach(origin => {\n      if (!origin.s3OriginConfig && !origin.customOriginConfig) {\n        throw new Error(`Origin ${origin.domainName} is missing either S3OriginConfig or CustomOriginConfig. At least 1 must be specified.`);\n      }\n    });\n    const originGroupsDistConfig =\n      originGroups.length > 0\n        ? {\n          items: originGroups,\n          quantity: originGroups.length,\n        }\n        : undefined;\n    distributionConfig = {\n      ...distributionConfig,\n      origins,\n      originGroups: originGroupsDistConfig,\n    };\n\n    const defaultBehaviors = behaviors.filter(behavior => behavior.isDefaultBehavior);\n    if (defaultBehaviors.length !== 1) {\n      throw new Error('There can only be one default behavior across all sources. [ One default behavior per distribution ].');\n    }\n\n    distributionConfig = { ...distributionConfig, defaultCacheBehavior: this.toBehavior(defaultBehaviors[0], props.viewerProtocolPolicy) };\n\n    const otherBehaviors: CfnDistribution.CacheBehaviorProperty[] = [];\n    for (const behavior of behaviors.filter(b => !b.isDefaultBehavior)) {\n      if (!behavior.pathPattern) {\n        throw new Error('pathPattern is required for all non-default behaviors');\n      }\n      otherBehaviors.push(this.toBehavior(behavior, props.viewerProtocolPolicy) as CfnDistribution.CacheBehaviorProperty);\n    }\n\n    distributionConfig = { ...distributionConfig, cacheBehaviors: otherBehaviors.length > 0 ? otherBehaviors : undefined };\n\n    if (props.aliasConfiguration && props.viewerCertificate) {\n      throw new Error([\n        'You cannot set both aliasConfiguration and viewerCertificate properties.',\n        'Please only use viewerCertificate, as aliasConfiguration is deprecated.',\n      ].join(' '));\n    }\n\n    let _viewerCertificate = props.viewerCertificate;\n    if (props.aliasConfiguration) {\n      const { acmCertRef, securityPolicy, sslMethod, names: aliases } = props.aliasConfiguration;\n\n      _viewerCertificate = ViewerCertificate.fromAcmCertificate(\n        certificatemanager.Certificate.fromCertificateArn(this, 'AliasConfigurationCert', acmCertRef),\n        { securityPolicy, sslMethod, aliases },\n      );\n    }\n\n    if (_viewerCertificate) {\n      const { props: viewerCertificate, aliases } = _viewerCertificate;\n      Object.assign(distributionConfig, { aliases, viewerCertificate });\n\n      const { minimumProtocolVersion, sslSupportMethod } = viewerCertificate;\n\n      if (minimumProtocolVersion != null && sslSupportMethod != null) {\n        const validProtocols = this.VALID_SSL_PROTOCOLS[sslSupportMethod as SSLMethod];\n\n        if (validProtocols.indexOf(minimumProtocolVersion.toString()) === -1) {\n          // eslint-disable-next-line max-len\n          throw new Error(`${minimumProtocolVersion} is not compabtible with sslMethod ${sslSupportMethod}.\\n\\tValid Protocols are: ${validProtocols.join(', ')}`);\n        }\n      }\n    } else {\n      distributionConfig = {\n        ...distributionConfig,\n        viewerCertificate: { cloudFrontDefaultCertificate: true },\n      };\n    }\n\n    if (props.loggingConfig) {\n      this.loggingBucket = props.loggingConfig.bucket || new s3.Bucket(this, 'LoggingBucket', {\n        encryption: s3.BucketEncryption.S3_MANAGED,\n      });\n      distributionConfig = {\n        ...distributionConfig,\n        logging: {\n          bucket: this.loggingBucket.bucketRegionalDomainName,\n          includeCookies: props.loggingConfig.includeCookies || false,\n          prefix: props.loggingConfig.prefix,\n        },\n      };\n    }\n\n    if (props.geoRestriction) {\n      distributionConfig = {\n        ...distributionConfig,\n        restrictions: {\n          geoRestriction: {\n            restrictionType: props.geoRestriction.restrictionType,\n            locations: props.geoRestriction.locations,\n          },\n        },\n      };\n    }\n\n    const distribution = new CfnDistribution(this, 'CFDistribution', { distributionConfig });\n    this.node.defaultChild = distribution;\n    this.domainName = distribution.attrDomainName;\n    this.distributionDomainName = distribution.attrDomainName;\n    this.distributionId = distribution.ref;\n  }\n\n  private toBehavior(input: BehaviorWithOrigin, protoPolicy?: ViewerProtocolPolicy) {\n    let toReturn = {\n      allowedMethods: this.METHOD_LOOKUP_MAP[input.allowedMethods || CloudFrontAllowedMethods.GET_HEAD],\n      cachedMethods: this.METHOD_LOOKUP_MAP[input.cachedMethods || CloudFrontAllowedCachedMethods.GET_HEAD],\n      compress: input.compress !== false,\n      defaultTtl: input.defaultTtl && input.defaultTtl.toSeconds(),\n      forwardedValues: input.forwardedValues || { queryString: false, cookies: { forward: 'none' } },\n      maxTtl: input.maxTtl && input.maxTtl.toSeconds(),\n      minTtl: input.minTtl && input.minTtl.toSeconds(),\n      trustedKeyGroups: input.trustedKeyGroups?.map(key => key.keyGroupId),\n      trustedSigners: input.trustedSigners,\n      targetOriginId: input.targetOriginId,\n      viewerProtocolPolicy: input.viewerProtocolPolicy || protoPolicy || ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n    };\n    if (!input.isDefaultBehavior) {\n      toReturn = Object.assign(toReturn, { pathPattern: input.pathPattern });\n    }\n    if (input.functionAssociations) {\n      toReturn = Object.assign(toReturn, {\n        functionAssociations: input.functionAssociations.map(association => ({\n          functionArn: association.function.functionArn,\n          eventType: association.eventType.toString(),\n        })),\n      });\n    }\n    if (input.lambdaFunctionAssociations) {\n      const includeBodyEventTypes = [LambdaEdgeEventType.ORIGIN_REQUEST, LambdaEdgeEventType.VIEWER_REQUEST];\n      if (input.lambdaFunctionAssociations.some(fna => fna.includeBody && !includeBodyEventTypes.includes(fna.eventType))) {\n        throw new Error('\\'includeBody\\' can only be true for ORIGIN_REQUEST or VIEWER_REQUEST event types.');\n      }\n\n      toReturn = Object.assign(toReturn, {\n        lambdaFunctionAssociations: input.lambdaFunctionAssociations\n          .map(fna => ({\n            eventType: fna.eventType,\n            lambdaFunctionArn: fna.lambdaFunction && fna.lambdaFunction.edgeArn,\n            includeBody: fna.includeBody,\n          })),\n      });\n\n      // allow edgelambda.amazonaws.com to assume the functions' execution role.\n      for (const a of input.lambdaFunctionAssociations) {\n        if (a.lambdaFunction.role && a.lambdaFunction.role instanceof iam.Role && a.lambdaFunction.role.assumeRolePolicy) {\n          a.lambdaFunction.role.assumeRolePolicy.addStatements(new iam.PolicyStatement({\n            actions: ['sts:AssumeRole'],\n            principals: [new iam.ServicePrincipal('edgelambda.amazonaws.com')],\n          }));\n        }\n      }\n    }\n    return toReturn;\n  }\n\n  private toOriginProperty(originConfig: SourceConfigurationRender, originId: string): CfnDistribution.OriginProperty {\n    if (\n      !originConfig.s3OriginSource &&\n      !originConfig.customOriginSource\n    ) {\n      throw new Error(\n        'There must be at least one origin source - either an s3OriginSource, a customOriginSource',\n      );\n    }\n    if (originConfig.customOriginSource && originConfig.s3OriginSource) {\n      throw new Error(\n        'There cannot be both an s3OriginSource and a customOriginSource in the same SourceConfiguration.',\n      );\n    }\n\n    if ([\n      originConfig.originHeaders,\n      originConfig.s3OriginSource?.originHeaders,\n      originConfig.customOriginSource?.originHeaders,\n    ].filter(x => x).length > 1) {\n      throw new Error('Only one originHeaders field allowed across origin and failover origins');\n    }\n\n    if ([\n      originConfig.originPath,\n      originConfig.s3OriginSource?.originPath,\n      originConfig.customOriginSource?.originPath,\n    ].filter(x => x).length > 1) {\n      throw new Error('Only one originPath field allowed across origin and failover origins');\n    }\n\n    if ([\n      originConfig.originShieldRegion,\n      originConfig.s3OriginSource?.originShieldRegion,\n      originConfig.customOriginSource?.originShieldRegion,\n    ].filter(x => x).length > 1) {\n      throw new Error('Only one originShieldRegion field allowed across origin and failover origins');\n    }\n\n    const headers = originConfig.originHeaders ?? originConfig.s3OriginSource?.originHeaders ?? originConfig.customOriginSource?.originHeaders;\n\n    const originHeaders: CfnDistribution.OriginCustomHeaderProperty[] = [];\n    if (headers) {\n      Object.keys(headers).forEach((key) => {\n        const oHeader: CfnDistribution.OriginCustomHeaderProperty = {\n          headerName: key,\n          headerValue: headers[key],\n        };\n        originHeaders.push(oHeader);\n      });\n    }\n\n    let s3OriginConfig: CfnDistribution.S3OriginConfigProperty | undefined;\n    if (originConfig.s3OriginSource) {\n      // first case for backwards compatibility\n      if (originConfig.s3OriginSource.originAccessIdentity) {\n        // grant CloudFront OriginAccessIdentity read access to S3 bucket\n        // Used rather than `grantRead` because `grantRead` will grant overly-permissive policies.\n        // Only GetObject is needed to retrieve objects for the distribution.\n        // This also excludes KMS permissions; currently, OAI only supports SSE-S3 for buckets.\n        // Source: https://aws.amazon.com/blogs/networking-and-content-delivery/serving-sse-kms-encrypted-content-from-s3-using-cloudfront/\n        originConfig.s3OriginSource.s3BucketSource.addToResourcePolicy(new iam.PolicyStatement({\n          resources: [originConfig.s3OriginSource.s3BucketSource.arnForObjects('*')],\n          actions: ['s3:GetObject'],\n          principals: [originConfig.s3OriginSource.originAccessIdentity.grantPrincipal],\n        }));\n\n        s3OriginConfig = {\n          originAccessIdentity: `origin-access-identity/cloudfront/${originConfig.s3OriginSource.originAccessIdentity.originAccessIdentityName}`,\n        };\n      } else {\n        s3OriginConfig = {};\n      }\n    }\n\n    const connectionAttempts = originConfig.connectionAttempts ?? 3;\n    if (connectionAttempts < 1 || 3 < connectionAttempts || !Number.isInteger(connectionAttempts)) {\n      throw new Error('connectionAttempts: You can specify 1, 2, or 3 as the number of attempts.');\n    }\n\n    const connectionTimeout = (originConfig.connectionTimeout || cdk.Duration.seconds(10)).toSeconds();\n    if (connectionTimeout < 1 || 10 < connectionTimeout || !Number.isInteger(connectionTimeout)) {\n      throw new Error('connectionTimeout: You can specify a number of seconds between 1 and 10 (inclusive).');\n    }\n\n    const originProperty: CfnDistribution.OriginProperty = {\n      id: originId,\n      domainName: originConfig.s3OriginSource\n        ? originConfig.s3OriginSource.s3BucketSource.bucketRegionalDomainName\n        : originConfig.customOriginSource!.domainName,\n      originPath: originConfig.originPath ?? originConfig.customOriginSource?.originPath ?? originConfig.s3OriginSource?.originPath,\n      originCustomHeaders:\n        originHeaders.length > 0 ? originHeaders : undefined,\n      s3OriginConfig,\n      originShield: this.toOriginShieldProperty(originConfig),\n      customOriginConfig: originConfig.customOriginSource\n        ? {\n          httpPort: originConfig.customOriginSource.httpPort || 80,\n          httpsPort: originConfig.customOriginSource.httpsPort || 443,\n          originKeepaliveTimeout:\n            (originConfig.customOriginSource.originKeepaliveTimeout &&\n              originConfig.customOriginSource.originKeepaliveTimeout.toSeconds()) ||\n            5,\n          originReadTimeout:\n            (originConfig.customOriginSource.originReadTimeout &&\n              originConfig.customOriginSource.originReadTimeout.toSeconds()) ||\n            30,\n          originProtocolPolicy:\n            originConfig.customOriginSource.originProtocolPolicy ||\n            OriginProtocolPolicy.HTTPS_ONLY,\n          originSslProtocols: originConfig.customOriginSource\n            .allowedOriginSSLVersions || [OriginSslPolicy.TLS_V1_2],\n        }\n        : undefined,\n      connectionAttempts,\n      connectionTimeout,\n    };\n\n    return originProperty;\n  }\n\n  /**\n   * Takes origin shield region from props and converts to CfnDistribution.OriginShieldProperty\n   */\n  private toOriginShieldProperty(originConfig:SourceConfigurationRender): CfnDistribution.OriginShieldProperty | undefined {\n    const originShieldRegion = originConfig.originShieldRegion ??\n    originConfig.customOriginSource?.originShieldRegion ??\n    originConfig.s3OriginSource?.originShieldRegion;\n    return originShieldRegion\n      ? { enabled: true, originShieldRegion }\n      : undefined;\n  }\n}\n"]} |
\ | No newline at end of file |