cdk-simplewebsite-deploy 2.1.10 → 2.2.0

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.
package/.tool-versions CHANGED
@@ -1 +1,4 @@
1
1
  python 3.13.1t
2
+ nodejs 24.14.1
3
+ maven 3.9.14
4
+ java corretto-21.0.7.6.1
package/API.md CHANGED
@@ -47,7 +47,7 @@ new CreateBasicSite(scope: Construct, id: string, props: BasicSiteConfiguration)
47
47
 
48
48
  ---
49
49
 
50
- ##### `toString` <a name="toString" id="cdk-simplewebsite-deploy.CreateBasicSite.toString"></a>
50
+ ##### ~~`toString`~~ <a name="toString" id="cdk-simplewebsite-deploy.CreateBasicSite.toString"></a>
51
51
 
52
52
  ```typescript
53
53
  public toString(): string
@@ -55,7 +55,7 @@ public toString(): string
55
55
 
56
56
  Returns a string representation of this construct.
57
57
 
58
- ##### `with` <a name="with" id="cdk-simplewebsite-deploy.CreateBasicSite.with"></a>
58
+ ##### ~~`with`~~ <a name="with" id="cdk-simplewebsite-deploy.CreateBasicSite.with"></a>
59
59
 
60
60
  ```typescript
61
61
  public with(mixins: ...IMixin[]): IConstruct
@@ -84,7 +84,7 @@ The mixins to apply.
84
84
 
85
85
  ---
86
86
 
87
- ##### `isConstruct` <a name="isConstruct" id="cdk-simplewebsite-deploy.CreateBasicSite.isConstruct"></a>
87
+ ##### ~~`isConstruct`~~ <a name="isConstruct" id="cdk-simplewebsite-deploy.CreateBasicSite.isConstruct"></a>
88
88
 
89
89
  ```typescript
90
90
  import { CreateBasicSite } from 'cdk-simplewebsite-deploy'
@@ -124,7 +124,9 @@ Any object.
124
124
 
125
125
  ---
126
126
 
127
- ##### `node`<sup>Required</sup> <a name="node" id="cdk-simplewebsite-deploy.CreateBasicSite.property.node"></a>
127
+ ##### ~~`node`~~<sup>Required</sup> <a name="node" id="cdk-simplewebsite-deploy.CreateBasicSite.property.node"></a>
128
+
129
+ - *Deprecated:* Use CreateCloudfrontSite instead. CreateBasicSite configures a public S3 website endpoint.
128
130
 
129
131
  ```typescript
130
132
  public readonly node: Node;
@@ -368,7 +370,9 @@ const cloudfrontSiteConfiguration: CloudfrontSiteConfiguration = { ... }
368
370
  | <code><a href="#cdk-simplewebsite-deploy.CloudfrontSiteConfiguration.property.enableLogging">enableLogging</a></code> | <code>boolean</code> | Enable CloudFront access logging. |
369
371
  | <code><a href="#cdk-simplewebsite-deploy.CloudfrontSiteConfiguration.property.enableSecurityHeaders">enableSecurityHeaders</a></code> | <code>boolean</code> | Enable response headers policy for security headers. |
370
372
  | <code><a href="#cdk-simplewebsite-deploy.CloudfrontSiteConfiguration.property.errorDoc">errorDoc</a></code> | <code>string</code> | The error document of the website. |
373
+ | <code><a href="#cdk-simplewebsite-deploy.CloudfrontSiteConfiguration.property.functionAssociations">functionAssociations</a></code> | <code>aws-cdk-lib.aws_cloudfront.FunctionAssociation[]</code> | CloudFront Functions to associate with the default behavior. |
371
374
  | <code><a href="#cdk-simplewebsite-deploy.CloudfrontSiteConfiguration.property.logsBucket">logsBucket</a></code> | <code>aws-cdk-lib.aws_s3.IBucket</code> | S3 bucket for CloudFront access logs. |
375
+ | <code><a href="#cdk-simplewebsite-deploy.CloudfrontSiteConfiguration.property.originAccessLevels">originAccessLevels</a></code> | <code>aws-cdk-lib.aws_cloudfront.AccessLevel[]</code> | Additional permissions granted to the CloudFront Origin Access Control for the website bucket. |
372
376
  | <code><a href="#cdk-simplewebsite-deploy.CloudfrontSiteConfiguration.property.priceClass">priceClass</a></code> | <code>aws-cdk-lib.aws_cloudfront.PriceClass</code> | The price class determines how many edge locations CloudFront will use for your distribution. |
373
377
  | <code><a href="#cdk-simplewebsite-deploy.CloudfrontSiteConfiguration.property.subDomain">subDomain</a></code> | <code>string</code> | The subdomain name you want to deploy. |
374
378
  | <code><a href="#cdk-simplewebsite-deploy.CloudfrontSiteConfiguration.property.webAclId">webAclId</a></code> | <code>string</code> | Optional WAF Web ACL ARN for enhanced security. |
@@ -506,6 +510,19 @@ The error document of the website.
506
510
 
507
511
  ---
508
512
 
513
+ ##### `functionAssociations`<sup>Optional</sup> <a name="functionAssociations" id="cdk-simplewebsite-deploy.CloudfrontSiteConfiguration.property.functionAssociations"></a>
514
+
515
+ ```typescript
516
+ public readonly functionAssociations: FunctionAssociation[];
517
+ ```
518
+
519
+ - *Type:* aws-cdk-lib.aws_cloudfront.FunctionAssociation[]
520
+ - *Default:* No CloudFront Function associations.
521
+
522
+ CloudFront Functions to associate with the default behavior.
523
+
524
+ ---
525
+
509
526
  ##### `logsBucket`<sup>Optional</sup> <a name="logsBucket" id="cdk-simplewebsite-deploy.CloudfrontSiteConfiguration.property.logsBucket"></a>
510
527
 
511
528
  ```typescript
@@ -521,6 +538,19 @@ If not provided and logging is enabled, a new bucket will be created.
521
538
 
522
539
  ---
523
540
 
541
+ ##### `originAccessLevels`<sup>Optional</sup> <a name="originAccessLevels" id="cdk-simplewebsite-deploy.CloudfrontSiteConfiguration.property.originAccessLevels"></a>
542
+
543
+ ```typescript
544
+ public readonly originAccessLevels: AccessLevel[];
545
+ ```
546
+
547
+ - *Type:* aws-cdk-lib.aws_cloudfront.AccessLevel[]
548
+ - *Default:* [cloudfront.AccessLevel.READ]
549
+
550
+ Additional permissions granted to the CloudFront Origin Access Control for the website bucket.
551
+
552
+ ---
553
+
524
554
  ##### `priceClass`<sup>Optional</sup> <a name="priceClass" id="cdk-simplewebsite-deploy.CloudfrontSiteConfiguration.property.priceClass"></a>
525
555
 
526
556
  ```typescript
package/README.md CHANGED
@@ -3,29 +3,30 @@
3
3
  ![Release](https://github.com/SnapPetal/cdk-simplewebsite-deploy/workflows/release/badge.svg?branch=main)
4
4
 
5
5
  # cdk-simplewebsite-deploy
6
- This is an AWS CDK Construct to simplify deploying a single-page website using either S3 buckets or CloudFront distributions with enhanced security, performance, and monitoring capabilities.
6
+ This is an AWS CDK v2 construct library for deploying a single-page website with S3, CloudFront, Route 53, and ACM. `CreateCloudfrontSite` is the recommended construct because it uses a private S3 origin with CloudFront Origin Access Control (OAC), while `CreateBasicSite` is deprecated because it creates a public S3 website endpoint.
7
7
 
8
8
  ## Installation and Usage
9
9
 
10
- ### [CreateBasicSite](https://github.com/snappetal/cdk-simplewebsite-deploy/blob/main/API.md#cdk-cloudfront-deploy-createbasicsite)
11
- #### Creates a simple website using S3 buckets with a domain hosted in Route 53.
10
+ ### [CreateCloudfrontSite](https://github.com/snappetal/cdk-simplewebsite-deploy/blob/main/API.md#cdk-cloudfront-deploy-createcloudfrontsite)
11
+ #### Creates a website using a private S3 bucket, a CloudFront distribution, and DNS records in Route 53.
12
12
  ##### Typescript
13
13
  ```console
14
14
  yarn add cdk-simplewebsite-deploy
15
15
  ```
16
16
  ```typescript
17
17
  import * as cdk from 'aws-cdk-lib';
18
- import { CreateBasicSite } from 'cdk-simplewebsite-deploy';
18
+ import { CreateCloudfrontSite } from 'cdk-simplewebsite-deploy';
19
19
  import { Construct } from 'constructs';
20
20
 
21
21
  export class PipelineStack extends cdk.Stack {
22
22
  constructor(scope: Construct, id: string, props?: cdk.StackProps) {
23
23
  super(scope, id, props);
24
24
 
25
- new CreateBasicSite(this, 'test-website', {
26
- websiteFolder: './src/build',
25
+ new CreateCloudfrontSite(this, 'test-website', {
26
+ websiteFolder: './src/dist',
27
27
  indexDoc: 'index.html',
28
28
  hostedZone: 'example.com',
29
+ subDomain: 'www.example.com',
29
30
  });
30
31
  }
31
32
  }
@@ -36,6 +37,7 @@ dotnet add package ThonBecker.CDK.SimpleWebsiteDeploy
36
37
  ```
37
38
  ```cs
38
39
  using Amazon.CDK;
40
+ using Constructs;
39
41
  using ThonBecker.CDK.SimpleWebsiteDeploy;
40
42
 
41
43
  namespace SimpleWebsiteDeploy
@@ -44,11 +46,12 @@ namespace SimpleWebsiteDeploy
44
46
  {
45
47
  internal PipelineStack(Construct scope, string id, IStackProps props = null) : base(scope, id, props)
46
48
  {
47
- new CreateBasicSite(scope, "test-website", new BasicSiteConfiguration()
49
+ new CreateCloudfrontSite(scope, "test-website", new CloudfrontSiteConfiguration()
48
50
  {
49
51
  WebsiteFolder = "./src/build",
50
52
  IndexDoc = "index.html",
51
53
  HostedZone = "example.com",
54
+ SubDomain = "www.example.com",
52
55
  });
53
56
  }
54
57
  }
@@ -65,10 +68,10 @@ namespace SimpleWebsiteDeploy
65
68
  ```java
66
69
  package com.myorg;
67
70
 
68
- import software.amazon.awscdk.core.Construct;
69
- import software.amazon.awscdk.core.Stack;
70
- import software.amazon.awscdk.core.StackProps;
71
- import com.thonbecker.simplewebsitedeploy.CreateBasicSite;
71
+ import com.thonbecker.simplewebsitedeploy.CreateCloudfrontSite;
72
+ import software.amazon.awscdk.Stack;
73
+ import software.amazon.awscdk.StackProps;
74
+ import software.constructs.Construct;
72
75
 
73
76
  public class MyProjectStack extends Stack {
74
77
  public MyProjectStack(final Construct scope, final String id) {
@@ -77,11 +80,13 @@ public class MyProjectStack extends Stack {
77
80
 
78
81
  public MyProjectStack(final Construct scope, final String id, final StackProps props) {
79
82
  super(scope, id, props);
80
-
81
- CreateBasicSite.Builder.create(this, "test-website")
82
- .websiteFolder("./src/build")
83
- .indexDoc("index.html")
84
- .hostedZone("example.com");
83
+
84
+ CreateCloudfrontSite.Builder.create(this, "test-website")
85
+ .websiteFolder("./src/build")
86
+ .indexDoc("index.html")
87
+ .hostedZone("example.com")
88
+ .subDomain("www.example.com")
89
+ .build();
85
90
  }
86
91
  }
87
92
  ```
@@ -91,38 +96,41 @@ pip install cdk-simplewebsite-deploy
91
96
  ```
92
97
  ```python
93
98
  from aws_cdk import Stack
94
- from cdk_simplewebsite_deploy import CreateBasicSite
99
+ from cdk_simplewebsite_deploy import CreateCloudfrontSite
95
100
  from constructs import Construct
96
101
 
102
+
97
103
  class MyProjectStack(Stack):
98
104
 
99
105
  def __init__(self, scope: Construct, construct_id: str, **kwargs) -> None:
100
106
  super().__init__(scope, construct_id, **kwargs)
101
-
102
- CreateBasicSite(self, 'test-website', website_folder='./src/build',
103
- index_doc='index.html',
104
- hosted_zone='example.com')
107
+
108
+ CreateCloudfrontSite(self, 'test-website', website_folder='./src/build',
109
+ index_doc='index.html',
110
+ hosted_zone='example.com',
111
+ sub_domain='www.example.com')
105
112
  ```
106
- ### [CreateCloudfrontSite](https://github.com/snappetal/cdk-simplewebsite-deploy/blob/main/API.md#cdk-cloudfront-deploy-createcloudfrontsite)
107
- #### Creates a simple website using a CloudFront distribution with a domain hosted in Route 53.
113
+
114
+ ### [CreateBasicSite](https://github.com/snappetal/cdk-simplewebsite-deploy/blob/main/API.md#cdk-cloudfront-deploy-createbasicsite)
115
+ #### Deprecated. Creates a website using public S3 website endpoints with a domain hosted in Route 53.
116
+ Use `CreateCloudfrontSite` for new sites. `CreateBasicSite` configures public bucket access so Route 53 can alias directly to the S3 website endpoint.
108
117
  ##### Typescript
109
118
  ```console
110
119
  yarn add cdk-simplewebsite-deploy
111
120
  ```
112
121
  ```typescript
113
122
  import * as cdk from 'aws-cdk-lib';
114
- import { CreateCloudfrontSite } from 'cdk-simplewebsite-deploy';
123
+ import { CreateBasicSite } from 'cdk-simplewebsite-deploy';
115
124
  import { Construct } from 'constructs';
116
125
 
117
126
  export class PipelineStack extends cdk.Stack {
118
127
  constructor(scope: Construct, id: string, props?: cdk.StackProps) {
119
128
  super(scope, id, props);
120
129
 
121
- new CreateCloudfrontSite(this, 'test-website', {
122
- websiteFolder: './src/dist',
130
+ new CreateBasicSite(this, 'test-website', {
131
+ websiteFolder: './src/build',
123
132
  indexDoc: 'index.html',
124
133
  hostedZone: 'example.com',
125
- subDomain: 'www.example.com',
126
134
  });
127
135
  }
128
136
  }
@@ -133,6 +141,7 @@ dotnet add package ThonBecker.CDK.SimpleWebsiteDeploy
133
141
  ```
134
142
  ```cs
135
143
  using Amazon.CDK;
144
+ using Constructs;
136
145
  using ThonBecker.CDK.SimpleWebsiteDeploy;
137
146
 
138
147
  namespace SimpleWebsiteDeploy
@@ -141,12 +150,11 @@ namespace SimpleWebsiteDeploy
141
150
  {
142
151
  internal PipelineStack(Construct scope, string id, IStackProps props = null) : base(scope, id, props)
143
152
  {
144
- new CreateCloudfrontSite(scope, "test-website", new CloudfrontSiteConfiguration()
153
+ new CreateBasicSite(scope, "test-website", new BasicSiteConfiguration()
145
154
  {
146
155
  WebsiteFolder = "./src/build",
147
156
  IndexDoc = "index.html",
148
157
  HostedZone = "example.com",
149
- SubDomain = "www.example.com",
150
158
  });
151
159
  }
152
160
  }
@@ -163,10 +171,10 @@ namespace SimpleWebsiteDeploy
163
171
  ```java
164
172
  package com.myorg;
165
173
 
166
- import software.amazon.awscdk.core.Construct;
167
- import software.amazon.awscdk.core.Stack;
168
- import software.amazon.awscdk.core.StackProps;
169
- import com.thonbecker.simplewebsitedeploy.CreateCloudfrontSite;
174
+ import com.thonbecker.simplewebsitedeploy.CreateBasicSite;
175
+ import software.amazon.awscdk.Stack;
176
+ import software.amazon.awscdk.StackProps;
177
+ import software.constructs.Construct;
170
178
 
171
179
  public class MyProjectStack extends Stack {
172
180
  public MyProjectStack(final Construct scope, final String id) {
@@ -175,12 +183,12 @@ public class MyProjectStack extends Stack {
175
183
 
176
184
  public MyProjectStack(final Construct scope, final String id, final StackProps props) {
177
185
  super(scope, id, props);
178
-
179
- CreateCloudfrontSite.Builder.create(this, "test-website")
180
- .websiteFolder("./src/build")
181
- .indexDoc("index.html")
182
- .hostedZone("example.com")
183
- .subDomain("www.example.com");
186
+
187
+ CreateBasicSite.Builder.create(this, "test-website")
188
+ .websiteFolder("./src/build")
189
+ .indexDoc("index.html")
190
+ .hostedZone("example.com")
191
+ .build();
184
192
  }
185
193
  }
186
194
  ```
@@ -189,24 +197,23 @@ public class MyProjectStack extends Stack {
189
197
  pip install cdk-simplewebsite-deploy
190
198
  ```
191
199
  ```python
192
- from aws_cdk import core
193
- from cdk_simplewebsite_deploy import CreateCloudfrontSite
194
-
200
+ from aws_cdk import Stack
201
+ from cdk_simplewebsite_deploy import CreateBasicSite
202
+ from constructs import Construct
195
203
 
196
- class MyProjectStack(core.Stack):
204
+ class MyProjectStack(Stack):
197
205
 
198
- def __init__(self, scope: core.Construct, construct_id: str, **kwargs) -> None:
206
+ def __init__(self, scope: Construct, construct_id: str, **kwargs) -> None:
199
207
  super().__init__(scope, construct_id, **kwargs)
200
208
 
201
- CreateCloudfrontSite(self, 'test-website', website_folder='./src/build',
202
- index_doc='index.html',
203
- hosted_zone='example.com',
204
- sub_domain='www.example.com')
209
+ CreateBasicSite(self, 'test-website', website_folder='./src/build',
210
+ index_doc='index.html',
211
+ hosted_zone='example.com')
205
212
  ```
206
213
 
207
214
  ## 🚀 Enhanced Features
208
215
 
209
- The `CreateCloudfrontSite` construct now includes several optional advanced features for improved security, performance, and monitoring:
216
+ The `CreateCloudfrontSite` construct includes optional advanced features for security, performance, and monitoring.
210
217
 
211
218
  ### Security Headers
212
219
  Enable comprehensive security headers including HSTS, X-Frame-Options, Content-Type-Options, and XSS protection:
@@ -257,6 +264,48 @@ new CreateCloudfrontSite(this, 'waf-protected-website', {
257
264
  });
258
265
  ```
259
266
 
267
+ ### Origin Access Levels
268
+ Grant additional OAC permissions to the website bucket. This can be useful when you need CloudFront to distinguish missing objects from access-denied responses.
269
+
270
+ ```typescript
271
+ import * as cloudfront from 'aws-cdk-lib/aws-cloudfront';
272
+
273
+ new CreateCloudfrontSite(this, 'website-with-list-access', {
274
+ websiteFolder: './src/dist',
275
+ indexDoc: 'index.html',
276
+ hostedZone: 'example.com',
277
+ originAccessLevels: [
278
+ cloudfront.AccessLevel.READ,
279
+ cloudfront.AccessLevel.LIST,
280
+ ],
281
+ });
282
+ ```
283
+
284
+ ### CloudFront Function Associations
285
+ Attach CloudFront Functions to the default behavior for lightweight viewer request or viewer response logic.
286
+
287
+ ```typescript
288
+ import * as cloudfront from 'aws-cdk-lib/aws-cloudfront';
289
+
290
+ const rewriteFunction = new cloudfront.Function(this, 'RewriteFunction', {
291
+ code: cloudfront.FunctionCode.fromInline(
292
+ 'function handler(event) { return event.request; }',
293
+ ),
294
+ });
295
+
296
+ new CreateCloudfrontSite(this, 'website-with-function', {
297
+ websiteFolder: './src/dist',
298
+ indexDoc: 'index.html',
299
+ hostedZone: 'example.com',
300
+ functionAssociations: [
301
+ {
302
+ eventType: cloudfront.FunctionEventType.VIEWER_REQUEST,
303
+ function: rewriteFunction,
304
+ },
305
+ ],
306
+ });
307
+ ```
308
+
260
309
  ### Custom Cache Behaviors
261
310
  Add custom cache behaviors for different content types:
262
311
 
@@ -314,6 +363,12 @@ export class AdvancedWebsiteStack extends cdk.Stack {
314
363
  constructor(scope: Construct, id: string, props?: cdk.StackProps) {
315
364
  super(scope, id, props);
316
365
 
366
+ const rewriteFunction = new cloudfront.Function(this, 'RewriteFunction', {
367
+ code: cloudfront.FunctionCode.fromInline(
368
+ 'function handler(event) { return event.request; }',
369
+ ),
370
+ });
371
+
317
372
  new CreateCloudfrontSite(this, 'advanced-website', {
318
373
  websiteFolder: './dist',
319
374
  indexDoc: 'index.html',
@@ -325,6 +380,10 @@ export class AdvancedWebsiteStack extends cdk.Stack {
325
380
  priceClass: cloudfront.PriceClass.PRICE_CLASS_ALL,
326
381
  enableSecurityHeaders: true,
327
382
  enableIpv6: true,
383
+ originAccessLevels: [
384
+ cloudfront.AccessLevel.READ,
385
+ cloudfront.AccessLevel.LIST,
386
+ ],
328
387
 
329
388
  // Monitoring & Protection
330
389
  enableLogging: true,
@@ -337,6 +396,14 @@ export class AdvancedWebsiteStack extends cdk.Stack {
337
396
  cachePolicy: cloudfront.CachePolicy.CACHING_DISABLED,
338
397
  },
339
398
  },
399
+
400
+ // Edge Logic
401
+ functionAssociations: [
402
+ {
403
+ eventType: cloudfront.FunctionEventType.VIEWER_REQUEST,
404
+ function: rewriteFunction,
405
+ },
406
+ ],
340
407
 
341
408
  // SPA Error Handling
342
409
  customErrorResponses: [
@@ -357,12 +424,14 @@ export class AdvancedWebsiteStack extends cdk.Stack {
357
424
  - **Security Headers**: Automatic HSTS, X-Frame-Options, Content-Type-Options, and XSS protection
358
425
  - **WAF Integration**: Support for AWS WAF Web ACLs for advanced threat protection
359
426
  - **Origin Access Control**: Modern S3 bucket protection (replaces deprecated OAI)
427
+ - **Configurable OAC Permissions**: Optional origin access levels for the website bucket
360
428
 
361
429
  ### ⚡ **Optimized Performance**
362
430
  - **Smart Caching**: Optimized cache policies for better performance
363
431
  - **HTTP/2 & HTTP/3**: Latest protocol support for faster loading
364
432
  - **Global Edge Locations**: Configurable price classes for worldwide distribution
365
433
  - **IPv6 Support**: Dual-stack networking for better connectivity
434
+ - **CloudFront Functions**: Optional viewer request and response function associations
366
435
 
367
436
  ### 📊 **Comprehensive Monitoring**
368
437
  - **Access Logging**: CloudFront access logs for analytics
@@ -65,6 +65,16 @@ export interface CloudfrontSiteConfiguration {
65
65
  * @default - No additional cache behaviors.
66
66
  */
67
67
  readonly additionalBehaviors?: Record<string, cloudfront.BehaviorOptions>;
68
+ /**
69
+ * Additional permissions granted to the CloudFront Origin Access Control for the website bucket.
70
+ * @default [cloudfront.AccessLevel.READ]
71
+ */
72
+ readonly originAccessLevels?: cloudfront.AccessLevel[];
73
+ /**
74
+ * CloudFront Functions to associate with the default behavior.
75
+ * @default - No CloudFront Function associations.
76
+ */
77
+ readonly functionAssociations?: cloudfront.FunctionAssociation[];
68
78
  /**
69
79
  * Enable response headers policy for security headers.
70
80
  * @default false - No security headers policy applied.
@@ -96,6 +106,9 @@ export interface CloudfrontSiteConfiguration {
96
106
  */
97
107
  readonly webAclId?: string;
98
108
  }
109
+ /**
110
+ * @deprecated Use CreateCloudfrontSite instead. CreateBasicSite configures a public S3 website endpoint.
111
+ */
99
112
  export declare class CreateBasicSite extends Construct {
100
113
  constructor(scope: Construct, id: string, props: BasicSiteConfiguration);
101
114
  }
@@ -12,6 +12,9 @@ const targets = require("aws-cdk-lib/aws-route53-targets");
12
12
  const s3 = require("aws-cdk-lib/aws-s3");
13
13
  const s3deploy = require("aws-cdk-lib/aws-s3-deployment");
14
14
  const constructs_1 = require("constructs");
15
+ /**
16
+ * @deprecated Use CreateCloudfrontSite instead. CreateBasicSite configures a public S3 website endpoint.
17
+ */
15
18
  class CreateBasicSite extends constructs_1.Construct {
16
19
  constructor(scope, id, props) {
17
20
  super(scope, id);
@@ -60,7 +63,7 @@ class CreateBasicSite extends constructs_1.Construct {
60
63
  }
61
64
  exports.CreateBasicSite = CreateBasicSite;
62
65
  _a = JSII_RTTI_SYMBOL_1;
63
- CreateBasicSite[_a] = { fqn: "cdk-simplewebsite-deploy.CreateBasicSite", version: "2.1.10" };
66
+ CreateBasicSite[_a] = { fqn: "cdk-simplewebsite-deploy.CreateBasicSite", version: "2.2.0" };
64
67
  class CreateCloudfrontSite extends constructs_1.Construct {
65
68
  constructor(scope, id, props) {
66
69
  super(scope, id);
@@ -147,12 +150,15 @@ class CreateCloudfrontSite extends constructs_1.Construct {
147
150
  : undefined;
148
151
  const websiteDist = new cloudfront.Distribution(scope, 'WebsiteDist', {
149
152
  defaultBehavior: {
150
- origin: origins.S3BucketOrigin.withOriginAccessControl(websiteBucket),
153
+ origin: origins.S3BucketOrigin.withOriginAccessControl(websiteBucket, {
154
+ originAccessLevels: props.originAccessLevels,
155
+ }),
151
156
  allowedMethods: cloudfront.AllowedMethods.ALLOW_GET_HEAD_OPTIONS,
152
157
  viewerProtocolPolicy: cloudfront.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,
153
158
  cachePolicy: cloudfront.CachePolicy.CACHING_OPTIMIZED,
154
159
  originRequestPolicy: cloudfront.OriginRequestPolicy.CORS_S3_ORIGIN,
155
160
  responseHeadersPolicy,
161
+ functionAssociations: props.functionAssociations,
156
162
  },
157
163
  additionalBehaviors: props.additionalBehaviors,
158
164
  minimumProtocolVersion: cloudfront.SecurityPolicyProtocol.TLS_V1_2_2021,
@@ -203,5 +209,5 @@ class CreateCloudfrontSite extends constructs_1.Construct {
203
209
  }
204
210
  exports.CreateCloudfrontSite = CreateCloudfrontSite;
205
211
  _b = JSII_RTTI_SYMBOL_1;
206
- CreateCloudfrontSite[_b] = { fqn: "cdk-simplewebsite-deploy.CreateCloudfrontSite", version: "2.1.10" };
207
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cdk-simplewebsite-deploy.js","sourceRoot":"","sources":["../src/cdk-simplewebsite-deploy.ts"],"names":[],"mappings":";;;;;AAAA,6CAAsD;AACtD,0DAA0D;AAC1D,yDAAyD;AACzD,8DAA8D;AAC9D,mDAAmD;AACnD,2DAA2D;AAC3D,yCAAyC;AACzC,0DAA0D;AAC1D,2CAAuC;AAmGvC,MAAa,eAAgB,SAAQ,sBAAS;IAC5C,YAAY,KAAgB,EAAE,EAAU,EAAE,KAA6B;QACrE,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEjB,MAAM,gBAAgB,GAAG,OAAO,CAAC,UAAU,CAAC,UAAU,CACpD,IAAI,EACJ,mBAAmB,EACnB;YACE,UAAU,EAAE,KAAK,CAAC,UAAU;SAC7B,CACF,CAAC;QAEF,MAAM,qBAAqB,GAAG,IAAI,EAAE,CAAC,MAAM,CACzC,KAAK,EACL,uBAAuB,EACvB;YACE,UAAU,EAAE,OAAO,KAAK,CAAC,UAAU,EAAE;YACrC,aAAa,EAAE,2BAAa,CAAC,OAAO;YACpC,iBAAiB,EAAE,IAAI;YACvB,eAAe,EAAE;gBACf,QAAQ,EAAE,KAAK,CAAC,UAAU;gBAC1B,QAAQ,EAAE,EAAE,CAAC,gBAAgB,CAAC,IAAI;aACnC;SACF,CACF,CAAC;QAEF,MAAM,aAAa,GAAG,IAAI,EAAE,CAAC,MAAM,CAAC,KAAK,EAAE,eAAe,EAAE;YAC1D,UAAU,EAAE,KAAK,CAAC,UAAU;YAC5B,aAAa,EAAE,2BAAa,CAAC,OAAO;YACpC,iBAAiB,EAAE,IAAI;YACvB,oBAAoB,EAAE,KAAK,CAAC,QAAQ;YACpC,oBAAoB,EAAE,KAAK,CAAC,QAAQ;YACpC,gBAAgB,EAAE,IAAI;YACtB,iBAAiB,EAAE;gBACjB,iBAAiB,EAAE,KAAK;gBACxB,eAAe,EAAE,KAAK;gBACtB,gBAAgB,EAAE,KAAK;gBACvB,qBAAqB,EAAE,KAAK;aAC7B;YACD,UAAU,EAAE,EAAE,CAAC,gBAAgB,CAAC,UAAU;SAC3C,CAAC,CAAC;QAEH,IAAI,QAAQ,CAAC,gBAAgB,CAAC,KAAK,EAAE,eAAe,EAAE;YACpD,OAAO,EAAE,CAAC,QAAQ,CAAC,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC;YACrD,iBAAiB,EAAE,aAAa;SACjC,CAAC,CAAC;QAEH,IAAI,OAAO,CAAC,OAAO,CAAC,KAAK,EAAE,cAAc,EAAE;YACzC,IAAI,EAAE,gBAAgB;YACtB,UAAU,EAAE,KAAK,CAAC,UAAU;YAC5B,MAAM,EAAE,OAAO,CAAC,YAAY,CAAC,SAAS,CACpC,IAAI,OAAO,CAAC,mBAAmB,CAAC,aAAa,CAAC,CAC/C;SACF,CAAC,CAAC;QAEH,IAAI,OAAO,CAAC,OAAO,CAAC,KAAK,EAAE,sBAAsB,EAAE;YACjD,IAAI,EAAE,gBAAgB;YACtB,UAAU,EAAE,OAAO,KAAK,CAAC,UAAU,EAAE;YACrC,MAAM,EAAE,OAAO,CAAC,YAAY,CAAC,SAAS,CACpC,IAAI,OAAO,CAAC,mBAAmB,CAAC,qBAAqB,CAAC,CACvD;SACF,CAAC,CAAC;IACL,CAAC;;AA9DH,0CA+DC;;;AAED,MAAa,oBAAqB,SAAQ,sBAAS;IACjD,YACE,KAAgB,EAChB,EAAU,EACV,KAAkC;QAElC,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEjB,IAAI,KAAK,CAAC,MAAM,IAAI,KAAK,CAAC,SAAS,EAAE,CAAC;YACpC,MAAM,IAAI,KAAK,CACb,yDAAyD,CAC1D,CAAC;QACJ,CAAC;QAED,MAAM,gBAAgB,GAAG,OAAO,CAAC,UAAU,CAAC,UAAU,CACpD,IAAI,EACJ,mBAAmB,EACnB;YACE,UAAU,EAAE,KAAK,CAAC,UAAU;SAC7B,CACF,CAAC;QAEF,sEAAsE;QACtE,MAAM,cAAc,GAA+B,KAAK,CAAC,oBAAoB,IAAI,EAAE,CAAC;QACpF,IAAI,CAAC,KAAK,CAAC,oBAAoB,EAAE,CAAC;YAChC,IAAI,KAAK,CAAC,QAAQ,EAAE,CAAC;gBACnB,cAAc,CAAC,IAAI,CAAC;oBAClB,UAAU,EAAE,GAAG;oBACf,gBAAgB,EAAE,IAAI,KAAK,CAAC,QAAQ,EAAE;iBACvC,CAAC,CAAC;gBACH,cAAc,CAAC,IAAI,CAAC;oBAClB,UAAU,EAAE,GAAG;oBACf,gBAAgB,EAAE,IAAI,KAAK,CAAC,QAAQ,EAAE;iBACvC,CAAC,CAAC;YACL,CAAC;iBAAM,CAAC;gBACN,cAAc,CAAC,IAAI,CAAC;oBAClB,UAAU,EAAE,GAAG;oBACf,kBAAkB,EAAE,GAAG;oBACvB,gBAAgB,EAAE,IAAI,KAAK,CAAC,QAAQ,EAAE;iBACvC,CAAC,CAAC;gBACH,cAAc,CAAC,IAAI,CAAC;oBAClB,UAAU,EAAE,GAAG;oBACf,kBAAkB,EAAE,GAAG;oBACvB,gBAAgB,EAAE,IAAI,KAAK,CAAC,QAAQ,EAAE;iBACvC,CAAC,CAAC;YACL,CAAC;QACH,CAAC;QAED,MAAM,uBAAuB,GAAa,EAAE,CAAC;QAC7C,IAAI,KAAK,CAAC,MAAM;YAAE,uBAAuB,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;QAC7D,IAAI,KAAK,CAAC,SAAS;YAAE,uBAAuB,CAAC,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;QAEnE,MAAM,WAAW,GAAG,IAAI,GAAG,CAAC,WAAW,CAAC,IAAI,EAAE,aAAa,EAAE;YAC3D,UAAU,EAAE,KAAK,CAAC,UAAU;YAC5B,uBAAuB;YACvB,UAAU,EAAE,GAAG,CAAC,qBAAqB,CAAC,OAAO,CAAC,gBAAgB,CAAC;SAChE,CAAC,CAAC;QAEH,MAAM,aAAa,GAAG,IAAI,EAAE,CAAC,MAAM,CAAC,KAAK,EAAE,eAAe,EAAE;YAC1D,aAAa,EAAE,2BAAa,CAAC,OAAO;YACpC,iBAAiB,EAAE,IAAI;YACvB,UAAU,EAAE,EAAE,CAAC,gBAAgB,CAAC,UAAU;SAC3C,CAAC,CAAC;QAEH,MAAM,WAAW,GAAa,EAAE,CAAC;QACjC,IAAI,KAAK,CAAC,MAAM,EAAE,CAAC;YACjB,WAAW,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;QACjC,CAAC;aAAM,CAAC;YACN,WAAW,CAAC,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC;QACrC,CAAC;QAED,IAAI,KAAK,CAAC,SAAS;YAAE,WAAW,CAAC,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;QAEvD,yEAAyE;QACzE,MAAM,UAAU,GAAG,KAAK,CAAC,aAAa;YACpC,CAAC,CAAC,CAAC,KAAK,CAAC,UAAU,IAAI,IAAI,EAAE,CAAC,MAAM,CAAC,IAAI,EAAE,kBAAkB,EAAE;gBAC7D,aAAa,EAAE,2BAAa,CAAC,OAAO;gBACpC,iBAAiB,EAAE,IAAI;gBACvB,UAAU,EAAE,EAAE,CAAC,gBAAgB,CAAC,UAAU;aAC3C,CAAC,CAAC;YACH,CAAC,CAAC,SAAS,CAAC;QAEd,4CAA4C;QAC5C,MAAM,qBAAqB,GAAG,KAAK,CAAC,qBAAqB;YACvD,CAAC,CAAC,IAAI,UAAU,CAAC,qBAAqB,CAAC,IAAI,EAAE,uBAAuB,EAAE;gBACpE,uBAAuB,EAAE;oBACvB,kBAAkB,EAAE,EAAE,QAAQ,EAAE,IAAI,EAAE;oBACtC,YAAY,EAAE,EAAE,WAAW,EAAE,UAAU,CAAC,kBAAkB,CAAC,IAAI,EAAE,QAAQ,EAAE,IAAI,EAAE;oBACjF,cAAc,EAAE,EAAE,cAAc,EAAE,UAAU,CAAC,qBAAqB,CAAC,+BAA+B,EAAE,QAAQ,EAAE,IAAI,EAAE;oBACpH,uBAAuB,EAAE;wBACvB,mBAAmB,EAAE,sBAAQ,CAAC,OAAO,CAAC,QAAQ,CAAC;wBAC/C,iBAAiB,EAAE,IAAI;wBACvB,OAAO,EAAE,IAAI;wBACb,QAAQ,EAAE,IAAI;qBACf;oBACD,aAAa,EAAE,EAAE,UAAU,EAAE,IAAI,EAAE,SAAS,EAAE,IAAI,EAAE,QAAQ,EAAE,IAAI,EAAE;iBACrE;aACF,CAAC;YACF,CAAC,CAAC,SAAS,CAAC;QAEd,MAAM,WAAW,GAAG,IAAI,UAAU,CAAC,YAAY,CAAC,KAAK,EAAE,aAAa,EAAE;YACpE,eAAe,EAAE;gBACf,MAAM,EAAE,OAAO,CAAC,cAAc,CAAC,uBAAuB,CAAC,aAAa,CAAC;gBACrE,cAAc,EAAE,UAAU,CAAC,cAAc,CAAC,sBAAsB;gBAChE,oBAAoB,EAAE,UAAU,CAAC,oBAAoB,CAAC,iBAAiB;gBACvE,WAAW,EAAE,UAAU,CAAC,WAAW,CAAC,iBAAiB;gBACrD,mBAAmB,EAAE,UAAU,CAAC,mBAAmB,CAAC,cAAc;gBAClE,qBAAqB;aACtB;YACD,mBAAmB,EAAE,KAAK,CAAC,mBAAmB;YAC9C,sBAAsB,EAAE,UAAU,CAAC,sBAAsB,CAAC,aAAa;YACvE,WAAW,EAAE,UAAU,CAAC,WAAW,CAAC,WAAW;YAC/C,UAAU,EAAE,KAAK,CAAC,UAAU,IAAI,UAAU,CAAC,UAAU,CAAC,eAAe;YACrE,cAAc;YACd,iBAAiB,EAAE,KAAK,CAAC,QAAQ;YACjC,WAAW;YACX,WAAW,EAAE,WAAW;YACxB,SAAS,EAAE,UAAU;YACrB,kBAAkB,EAAE,KAAK;YACzB,QAAQ,EAAE,KAAK,CAAC,QAAQ;SACzB,CAAC,CAAC;QAEH,IAAI,QAAQ,CAAC,gBAAgB,CAAC,KAAK,EAAE,eAAe,EAAE;YACpD,OAAO,EAAE,CAAC,QAAQ,CAAC,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC;YACrD,iBAAiB,EAAE,aAAa;YAChC,YAAY,EAAE,WAAW;SAC1B,CAAC,CAAC;QAEH,IAAI,OAAO,CAAC,OAAO,CAAC,KAAK,EAAE,oBAAoB,EAAE;YAC/C,IAAI,EAAE,gBAAgB;YACtB,UAAU,EAAE,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,KAAK,CAAC,UAAU;YAC1D,MAAM,EAAE,OAAO,CAAC,YAAY,CAAC,SAAS,CACpC,IAAI,OAAO,CAAC,gBAAgB,CAAC,WAAW,CAAC,CAC1C;SACF,CAAC,CAAC;QAEH,8BAA8B;QAC9B,IAAI,KAAK,CAAC,UAAU,EAAE,CAAC;YACrB,IAAI,OAAO,CAAC,UAAU,CAAC,KAAK,EAAE,wBAAwB,EAAE;gBACtD,IAAI,EAAE,gBAAgB;gBACtB,UAAU,EAAE,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,KAAK,CAAC,UAAU;gBAC1D,MAAM,EAAE,OAAO,CAAC,YAAY,CAAC,SAAS,CACpC,IAAI,OAAO,CAAC,gBAAgB,CAAC,WAAW,CAAC,CAC1C;aACF,CAAC,CAAC;QACL,CAAC;QAED,IAAI,KAAK,CAAC,SAAS,EAAE,CAAC;YACpB,IAAI,OAAO,CAAC,OAAO,CAAC,KAAK,EAAE,uBAAuB,EAAE;gBAClD,IAAI,EAAE,gBAAgB;gBACtB,UAAU,EAAE,KAAK,CAAC,SAAS;gBAC3B,MAAM,EAAE,OAAO,CAAC,YAAY,CAAC,SAAS,CACpC,IAAI,OAAO,CAAC,gBAAgB,CAAC,WAAW,CAAC,CAC1C;aACF,CAAC,CAAC;YAEH,4CAA4C;YAC5C,IAAI,KAAK,CAAC,UAAU,EAAE,CAAC;gBACrB,IAAI,OAAO,CAAC,UAAU,CAAC,KAAK,EAAE,2BAA2B,EAAE;oBACzD,IAAI,EAAE,gBAAgB;oBACtB,UAAU,EAAE,KAAK,CAAC,SAAS;oBAC3B,MAAM,EAAE,OAAO,CAAC,YAAY,CAAC,SAAS,CACpC,IAAI,OAAO,CAAC,gBAAgB,CAAC,WAAW,CAAC,CAC1C;iBACF,CAAC,CAAC;YACL,CAAC;QACH,CAAC;IACH,CAAC;;AAvKH,oDAwKC","sourcesContent":["import { RemovalPolicy, Duration } from 'aws-cdk-lib';\nimport * as acm from 'aws-cdk-lib/aws-certificatemanager';\nimport * as cloudfront from 'aws-cdk-lib/aws-cloudfront';\nimport * as origins from 'aws-cdk-lib/aws-cloudfront-origins';\nimport * as route53 from 'aws-cdk-lib/aws-route53';\nimport * as targets from 'aws-cdk-lib/aws-route53-targets';\nimport * as s3 from 'aws-cdk-lib/aws-s3';\nimport * as s3deploy from 'aws-cdk-lib/aws-s3-deployment';\nimport { Construct } from 'constructs';\n\nexport interface BasicSiteConfiguration {\n  /**\n   * Local path to the website folder you want to deploy on S3.\n   */\n  readonly websiteFolder: string;\n  /**\n   * The index document of the website.\n   */\n  readonly indexDoc: string;\n  /**\n   * The error document of the website.\n   * @default - No error document.\n   */\n  readonly errorDoc?: string;\n  /**\n   * Hosted Zone used to create the DNS record for the website.\n   */\n  readonly hostedZone: string;\n}\n\nexport interface CloudfrontSiteConfiguration {\n  /**\n   * Local path to the website folder you want to deploy on S3.\n   */\n  readonly websiteFolder: string;\n  /**\n   * The index document of the website.\n   */\n  readonly indexDoc: string;\n  /**\n   * The error document of the website.\n   * @default - No error document.\n   */\n  readonly errorDoc?: string;\n  /**\n   * Hosted Zone used to create the DNS record for the website.\n   */\n  readonly hostedZone: string;\n  /**\n   * Used to deploy a Cloudfront site with a single domain. e.g. sample.example.com\n   * If you include a value for both domain and subDomain,\n   * an error will be thrown.\n   *\n   * @default - no value\n   */\n  readonly domain?: string;\n  /**\n   * The subdomain name you want to deploy. e.g. www.example.com\n   * If you include a value for both domain and subDomain,\n   * an error will be thrown.\n   *\n   * @default - no value\n   */\n  readonly subDomain?: string;\n  /**\n   * The price class determines how many edge locations CloudFront will use for your distribution.\n   * @default PriceClass.PRICE_CLASS_100.\n   * @see https://aws.amazon.com/cloudfront/pricing/.\n   */\n  readonly priceClass?: cloudfront.PriceClass;\n  /**\n   * Optional cache behaviors for different path patterns.\n   * @default - No additional cache behaviors.\n   */\n  readonly additionalBehaviors?: Record<string, cloudfront.BehaviorOptions>;\n  /**\n   * Enable response headers policy for security headers.\n   * @default false - No security headers policy applied.\n   */\n  readonly enableSecurityHeaders?: boolean;\n  /**\n   * Enable IPv6 support with AAAA records.\n   * @default false - No IPv6 support.\n   */\n  readonly enableIpv6?: boolean;\n  /**\n   * Custom error responses for different HTTP status codes.\n   * @default - Default error responses based on errorDoc setting.\n   */\n  readonly customErrorResponses?: cloudfront.ErrorResponse[];\n  /**\n   * Enable CloudFront access logging.\n   * @default false - No access logging.\n   */\n  readonly enableLogging?: boolean;\n  /**\n   * S3 bucket for CloudFront access logs. If not provided and logging is enabled, a new bucket will be created.\n   * @default - New bucket created if logging is enabled.\n   */\n  readonly logsBucket?: s3.IBucket;\n  /**\n   * Optional WAF Web ACL ARN for enhanced security.\n   * @default - No WAF integration.\n   */\n  readonly webAclId?: string;\n}\n\nexport class CreateBasicSite extends Construct {\n  constructor(scope: Construct, id: string, props: BasicSiteConfiguration) {\n    super(scope, id);\n\n    const hostedZoneLookup = route53.HostedZone.fromLookup(\n      this,\n      'WebsiteHostedZone',\n      {\n        domainName: props.hostedZone,\n      },\n    );\n\n    const websiteRedirectBucket = new s3.Bucket(\n      scope,\n      'WebsiteRedirectBucket',\n      {\n        bucketName: `www.${props.hostedZone}`,\n        removalPolicy: RemovalPolicy.DESTROY,\n        autoDeleteObjects: true,\n        websiteRedirect: {\n          hostName: props.hostedZone,\n          protocol: s3.RedirectProtocol.HTTP,\n        },\n      },\n    );\n\n    const websiteBucket = new s3.Bucket(scope, 'WebsiteBucket', {\n      bucketName: props.hostedZone,\n      removalPolicy: RemovalPolicy.DESTROY,\n      autoDeleteObjects: true,\n      websiteIndexDocument: props.indexDoc,\n      websiteErrorDocument: props.errorDoc,\n      publicReadAccess: true,\n      blockPublicAccess: {\n        blockPublicPolicy: false,\n        blockPublicAcls: false,\n        ignorePublicAcls: false,\n        restrictPublicBuckets: false,\n      },\n      encryption: s3.BucketEncryption.S3_MANAGED,\n    });\n\n    new s3deploy.BucketDeployment(scope, 'WebsiteDeploy', {\n      sources: [s3deploy.Source.asset(props.websiteFolder)],\n      destinationBucket: websiteBucket,\n    });\n\n    new route53.ARecord(scope, 'WebsiteAlias', {\n      zone: hostedZoneLookup,\n      recordName: props.hostedZone,\n      target: route53.RecordTarget.fromAlias(\n        new targets.BucketWebsiteTarget(websiteBucket),\n      ),\n    });\n\n    new route53.ARecord(scope, 'WebsiteRedirectAlias', {\n      zone: hostedZoneLookup,\n      recordName: `www.${props.hostedZone}`,\n      target: route53.RecordTarget.fromAlias(\n        new targets.BucketWebsiteTarget(websiteRedirectBucket),\n      ),\n    });\n  }\n}\n\nexport class CreateCloudfrontSite extends Construct {\n  constructor(\n    scope: Construct,\n    id: string,\n    props: CloudfrontSiteConfiguration,\n  ) {\n    super(scope, id);\n\n    if (props.domain && props.subDomain) {\n      throw new Error(\n        'Domain and sub domain parameters cannot both be defined',\n      );\n    }\n\n    const hostedZoneLookup = route53.HostedZone.fromLookup(\n      this,\n      'WebsiteHostedZone',\n      {\n        domainName: props.hostedZone,\n      },\n    );\n\n    // Use custom error responses if provided, otherwise use default logic\n    const errorResponses: cloudfront.ErrorResponse[] = props.customErrorResponses || [];\n    if (!props.customErrorResponses) {\n      if (props.errorDoc) {\n        errorResponses.push({\n          httpStatus: 404,\n          responsePagePath: `/${props.errorDoc}`,\n        });\n        errorResponses.push({\n          httpStatus: 403,\n          responsePagePath: `/${props.errorDoc}`,\n        });\n      } else {\n        errorResponses.push({\n          httpStatus: 404,\n          responseHttpStatus: 200,\n          responsePagePath: `/${props.indexDoc}`,\n        });\n        errorResponses.push({\n          httpStatus: 403,\n          responseHttpStatus: 200,\n          responsePagePath: `/${props.indexDoc}`,\n        });\n      }\n    }\n\n    const subjectAlternativeNames: string[] = [];\n    if (props.domain) subjectAlternativeNames.push(props.domain);\n    if (props.subDomain) subjectAlternativeNames.push(props.subDomain);\n\n    const websiteCert = new acm.Certificate(this, 'WebsiteCert', {\n      domainName: props.hostedZone,\n      subjectAlternativeNames,\n      validation: acm.CertificateValidation.fromDns(hostedZoneLookup),\n    });\n\n    const websiteBucket = new s3.Bucket(scope, 'WebsiteBucket', {\n      removalPolicy: RemovalPolicy.DESTROY,\n      autoDeleteObjects: true,\n      encryption: s3.BucketEncryption.S3_MANAGED,\n    });\n\n    const domainNames: string[] = [];\n    if (props.domain) {\n      domainNames.push(props.domain);\n    } else {\n      domainNames.push(props.hostedZone);\n    }\n\n    if (props.subDomain) domainNames.push(props.subDomain);\n\n    // Create access logs bucket if logging is enabled but no bucket provided\n    const logsBucket = props.enableLogging\n      ? (props.logsBucket || new s3.Bucket(this, 'AccessLogsBucket', {\n        removalPolicy: RemovalPolicy.DESTROY,\n        autoDeleteObjects: true,\n        encryption: s3.BucketEncryption.S3_MANAGED,\n      }))\n      : undefined;\n\n    // Create security headers policy if enabled\n    const responseHeadersPolicy = props.enableSecurityHeaders\n      ? new cloudfront.ResponseHeadersPolicy(this, 'SecurityHeadersPolicy', {\n        securityHeadersBehavior: {\n          contentTypeOptions: { override: true },\n          frameOptions: { frameOption: cloudfront.HeadersFrameOption.DENY, override: true },\n          referrerPolicy: { referrerPolicy: cloudfront.HeadersReferrerPolicy.STRICT_ORIGIN_WHEN_CROSS_ORIGIN, override: true },\n          strictTransportSecurity: {\n            accessControlMaxAge: Duration.seconds(31536000),\n            includeSubdomains: true,\n            preload: true,\n            override: true,\n          },\n          xssProtection: { protection: true, modeBlock: true, override: true },\n        },\n      })\n      : undefined;\n\n    const websiteDist = new cloudfront.Distribution(scope, 'WebsiteDist', {\n      defaultBehavior: {\n        origin: origins.S3BucketOrigin.withOriginAccessControl(websiteBucket),\n        allowedMethods: cloudfront.AllowedMethods.ALLOW_GET_HEAD_OPTIONS,\n        viewerProtocolPolicy: cloudfront.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n        cachePolicy: cloudfront.CachePolicy.CACHING_OPTIMIZED,\n        originRequestPolicy: cloudfront.OriginRequestPolicy.CORS_S3_ORIGIN,\n        responseHeadersPolicy,\n      },\n      additionalBehaviors: props.additionalBehaviors,\n      minimumProtocolVersion: cloudfront.SecurityPolicyProtocol.TLS_V1_2_2021,\n      httpVersion: cloudfront.HttpVersion.HTTP2_AND_3,\n      priceClass: props.priceClass ?? cloudfront.PriceClass.PRICE_CLASS_100,\n      errorResponses,\n      defaultRootObject: props.indexDoc,\n      domainNames,\n      certificate: websiteCert,\n      logBucket: logsBucket,\n      logIncludesCookies: false,\n      webAclId: props.webAclId,\n    });\n\n    new s3deploy.BucketDeployment(scope, 'WebsiteDeploy', {\n      sources: [s3deploy.Source.asset(props.websiteFolder)],\n      destinationBucket: websiteBucket,\n      distribution: websiteDist,\n    });\n\n    new route53.ARecord(scope, 'WebsiteDomainAlias', {\n      zone: hostedZoneLookup,\n      recordName: props.domain ? props.domain : props.hostedZone,\n      target: route53.RecordTarget.fromAlias(\n        new targets.CloudFrontTarget(websiteDist),\n      ),\n    });\n\n    // Add IPv6 support if enabled\n    if (props.enableIpv6) {\n      new route53.AaaaRecord(scope, 'WebsiteDomainAliasIpv6', {\n        zone: hostedZoneLookup,\n        recordName: props.domain ? props.domain : props.hostedZone,\n        target: route53.RecordTarget.fromAlias(\n          new targets.CloudFrontTarget(websiteDist),\n        ),\n      });\n    }\n\n    if (props.subDomain) {\n      new route53.ARecord(scope, 'WebsiteSubDomainAlias', {\n        zone: hostedZoneLookup,\n        recordName: props.subDomain,\n        target: route53.RecordTarget.fromAlias(\n          new targets.CloudFrontTarget(websiteDist),\n        ),\n      });\n\n      // Add IPv6 support for subdomain if enabled\n      if (props.enableIpv6) {\n        new route53.AaaaRecord(scope, 'WebsiteSubDomainAliasIpv6', {\n          zone: hostedZoneLookup,\n          recordName: props.subDomain,\n          target: route53.RecordTarget.fromAlias(\n            new targets.CloudFrontTarget(websiteDist),\n          ),\n        });\n      }\n    }\n  }\n}\n"]}
212
+ CreateCloudfrontSite[_b] = { fqn: "cdk-simplewebsite-deploy.CreateCloudfrontSite", version: "2.2.0" };
213
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cdk-simplewebsite-deploy.js","sourceRoot":"","sources":["../src/cdk-simplewebsite-deploy.ts"],"names":[],"mappings":";;;;;AAAA,6CAAsD;AACtD,0DAA0D;AAC1D,yDAAyD;AACzD,8DAA8D;AAC9D,mDAAmD;AACnD,2DAA2D;AAC3D,yCAAyC;AACzC,0DAA0D;AAC1D,2CAAuC;AA6GvC;;GAEG;AACH,MAAa,eAAgB,SAAQ,sBAAS;IAC5C,YAAY,KAAgB,EAAE,EAAU,EAAE,KAA6B;QACrE,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEjB,MAAM,gBAAgB,GAAG,OAAO,CAAC,UAAU,CAAC,UAAU,CACpD,IAAI,EACJ,mBAAmB,EACnB;YACE,UAAU,EAAE,KAAK,CAAC,UAAU;SAC7B,CACF,CAAC;QAEF,MAAM,qBAAqB,GAAG,IAAI,EAAE,CAAC,MAAM,CACzC,KAAK,EACL,uBAAuB,EACvB;YACE,UAAU,EAAE,OAAO,KAAK,CAAC,UAAU,EAAE;YACrC,aAAa,EAAE,2BAAa,CAAC,OAAO;YACpC,iBAAiB,EAAE,IAAI;YACvB,eAAe,EAAE;gBACf,QAAQ,EAAE,KAAK,CAAC,UAAU;gBAC1B,QAAQ,EAAE,EAAE,CAAC,gBAAgB,CAAC,IAAI;aACnC;SACF,CACF,CAAC;QAEF,MAAM,aAAa,GAAG,IAAI,EAAE,CAAC,MAAM,CAAC,KAAK,EAAE,eAAe,EAAE;YAC1D,UAAU,EAAE,KAAK,CAAC,UAAU;YAC5B,aAAa,EAAE,2BAAa,CAAC,OAAO;YACpC,iBAAiB,EAAE,IAAI;YACvB,oBAAoB,EAAE,KAAK,CAAC,QAAQ;YACpC,oBAAoB,EAAE,KAAK,CAAC,QAAQ;YACpC,gBAAgB,EAAE,IAAI;YACtB,iBAAiB,EAAE;gBACjB,iBAAiB,EAAE,KAAK;gBACxB,eAAe,EAAE,KAAK;gBACtB,gBAAgB,EAAE,KAAK;gBACvB,qBAAqB,EAAE,KAAK;aAC7B;YACD,UAAU,EAAE,EAAE,CAAC,gBAAgB,CAAC,UAAU;SAC3C,CAAC,CAAC;QAEH,IAAI,QAAQ,CAAC,gBAAgB,CAAC,KAAK,EAAE,eAAe,EAAE;YACpD,OAAO,EAAE,CAAC,QAAQ,CAAC,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC;YACrD,iBAAiB,EAAE,aAAa;SACjC,CAAC,CAAC;QAEH,IAAI,OAAO,CAAC,OAAO,CAAC,KAAK,EAAE,cAAc,EAAE;YACzC,IAAI,EAAE,gBAAgB;YACtB,UAAU,EAAE,KAAK,CAAC,UAAU;YAC5B,MAAM,EAAE,OAAO,CAAC,YAAY,CAAC,SAAS,CACpC,IAAI,OAAO,CAAC,mBAAmB,CAAC,aAAa,CAAC,CAC/C;SACF,CAAC,CAAC;QAEH,IAAI,OAAO,CAAC,OAAO,CAAC,KAAK,EAAE,sBAAsB,EAAE;YACjD,IAAI,EAAE,gBAAgB;YACtB,UAAU,EAAE,OAAO,KAAK,CAAC,UAAU,EAAE;YACrC,MAAM,EAAE,OAAO,CAAC,YAAY,CAAC,SAAS,CACpC,IAAI,OAAO,CAAC,mBAAmB,CAAC,qBAAqB,CAAC,CACvD;SACF,CAAC,CAAC;IACL,CAAC;;AA9DH,0CA+DC;;;AAED,MAAa,oBAAqB,SAAQ,sBAAS;IACjD,YACE,KAAgB,EAChB,EAAU,EACV,KAAkC;QAElC,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEjB,IAAI,KAAK,CAAC,MAAM,IAAI,KAAK,CAAC,SAAS,EAAE,CAAC;YACpC,MAAM,IAAI,KAAK,CACb,yDAAyD,CAC1D,CAAC;QACJ,CAAC;QAED,MAAM,gBAAgB,GAAG,OAAO,CAAC,UAAU,CAAC,UAAU,CACpD,IAAI,EACJ,mBAAmB,EACnB;YACE,UAAU,EAAE,KAAK,CAAC,UAAU;SAC7B,CACF,CAAC;QAEF,sEAAsE;QACtE,MAAM,cAAc,GAA+B,KAAK,CAAC,oBAAoB,IAAI,EAAE,CAAC;QACpF,IAAI,CAAC,KAAK,CAAC,oBAAoB,EAAE,CAAC;YAChC,IAAI,KAAK,CAAC,QAAQ,EAAE,CAAC;gBACnB,cAAc,CAAC,IAAI,CAAC;oBAClB,UAAU,EAAE,GAAG;oBACf,gBAAgB,EAAE,IAAI,KAAK,CAAC,QAAQ,EAAE;iBACvC,CAAC,CAAC;gBACH,cAAc,CAAC,IAAI,CAAC;oBAClB,UAAU,EAAE,GAAG;oBACf,gBAAgB,EAAE,IAAI,KAAK,CAAC,QAAQ,EAAE;iBACvC,CAAC,CAAC;YACL,CAAC;iBAAM,CAAC;gBACN,cAAc,CAAC,IAAI,CAAC;oBAClB,UAAU,EAAE,GAAG;oBACf,kBAAkB,EAAE,GAAG;oBACvB,gBAAgB,EAAE,IAAI,KAAK,CAAC,QAAQ,EAAE;iBACvC,CAAC,CAAC;gBACH,cAAc,CAAC,IAAI,CAAC;oBAClB,UAAU,EAAE,GAAG;oBACf,kBAAkB,EAAE,GAAG;oBACvB,gBAAgB,EAAE,IAAI,KAAK,CAAC,QAAQ,EAAE;iBACvC,CAAC,CAAC;YACL,CAAC;QACH,CAAC;QAED,MAAM,uBAAuB,GAAa,EAAE,CAAC;QAC7C,IAAI,KAAK,CAAC,MAAM;YAAE,uBAAuB,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;QAC7D,IAAI,KAAK,CAAC,SAAS;YAAE,uBAAuB,CAAC,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;QAEnE,MAAM,WAAW,GAAG,IAAI,GAAG,CAAC,WAAW,CAAC,IAAI,EAAE,aAAa,EAAE;YAC3D,UAAU,EAAE,KAAK,CAAC,UAAU;YAC5B,uBAAuB;YACvB,UAAU,EAAE,GAAG,CAAC,qBAAqB,CAAC,OAAO,CAAC,gBAAgB,CAAC;SAChE,CAAC,CAAC;QAEH,MAAM,aAAa,GAAG,IAAI,EAAE,CAAC,MAAM,CAAC,KAAK,EAAE,eAAe,EAAE;YAC1D,aAAa,EAAE,2BAAa,CAAC,OAAO;YACpC,iBAAiB,EAAE,IAAI;YACvB,UAAU,EAAE,EAAE,CAAC,gBAAgB,CAAC,UAAU;SAC3C,CAAC,CAAC;QAEH,MAAM,WAAW,GAAa,EAAE,CAAC;QACjC,IAAI,KAAK,CAAC,MAAM,EAAE,CAAC;YACjB,WAAW,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;QACjC,CAAC;aAAM,CAAC;YACN,WAAW,CAAC,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC;QACrC,CAAC;QAED,IAAI,KAAK,CAAC,SAAS;YAAE,WAAW,CAAC,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;QAEvD,yEAAyE;QACzE,MAAM,UAAU,GAAG,KAAK,CAAC,aAAa;YACpC,CAAC,CAAC,CAAC,KAAK,CAAC,UAAU,IAAI,IAAI,EAAE,CAAC,MAAM,CAAC,IAAI,EAAE,kBAAkB,EAAE;gBAC7D,aAAa,EAAE,2BAAa,CAAC,OAAO;gBACpC,iBAAiB,EAAE,IAAI;gBACvB,UAAU,EAAE,EAAE,CAAC,gBAAgB,CAAC,UAAU;aAC3C,CAAC,CAAC;YACH,CAAC,CAAC,SAAS,CAAC;QAEd,4CAA4C;QAC5C,MAAM,qBAAqB,GAAG,KAAK,CAAC,qBAAqB;YACvD,CAAC,CAAC,IAAI,UAAU,CAAC,qBAAqB,CAAC,IAAI,EAAE,uBAAuB,EAAE;gBACpE,uBAAuB,EAAE;oBACvB,kBAAkB,EAAE,EAAE,QAAQ,EAAE,IAAI,EAAE;oBACtC,YAAY,EAAE,EAAE,WAAW,EAAE,UAAU,CAAC,kBAAkB,CAAC,IAAI,EAAE,QAAQ,EAAE,IAAI,EAAE;oBACjF,cAAc,EAAE,EAAE,cAAc,EAAE,UAAU,CAAC,qBAAqB,CAAC,+BAA+B,EAAE,QAAQ,EAAE,IAAI,EAAE;oBACpH,uBAAuB,EAAE;wBACvB,mBAAmB,EAAE,sBAAQ,CAAC,OAAO,CAAC,QAAQ,CAAC;wBAC/C,iBAAiB,EAAE,IAAI;wBACvB,OAAO,EAAE,IAAI;wBACb,QAAQ,EAAE,IAAI;qBACf;oBACD,aAAa,EAAE,EAAE,UAAU,EAAE,IAAI,EAAE,SAAS,EAAE,IAAI,EAAE,QAAQ,EAAE,IAAI,EAAE;iBACrE;aACF,CAAC;YACF,CAAC,CAAC,SAAS,CAAC;QAEd,MAAM,WAAW,GAAG,IAAI,UAAU,CAAC,YAAY,CAAC,KAAK,EAAE,aAAa,EAAE;YACpE,eAAe,EAAE;gBACf,MAAM,EAAE,OAAO,CAAC,cAAc,CAAC,uBAAuB,CAAC,aAAa,EAAE;oBACpE,kBAAkB,EAAE,KAAK,CAAC,kBAAkB;iBAC7C,CAAC;gBACF,cAAc,EAAE,UAAU,CAAC,cAAc,CAAC,sBAAsB;gBAChE,oBAAoB,EAAE,UAAU,CAAC,oBAAoB,CAAC,iBAAiB;gBACvE,WAAW,EAAE,UAAU,CAAC,WAAW,CAAC,iBAAiB;gBACrD,mBAAmB,EAAE,UAAU,CAAC,mBAAmB,CAAC,cAAc;gBAClE,qBAAqB;gBACrB,oBAAoB,EAAE,KAAK,CAAC,oBAAoB;aACjD;YACD,mBAAmB,EAAE,KAAK,CAAC,mBAAmB;YAC9C,sBAAsB,EAAE,UAAU,CAAC,sBAAsB,CAAC,aAAa;YACvE,WAAW,EAAE,UAAU,CAAC,WAAW,CAAC,WAAW;YAC/C,UAAU,EAAE,KAAK,CAAC,UAAU,IAAI,UAAU,CAAC,UAAU,CAAC,eAAe;YACrE,cAAc;YACd,iBAAiB,EAAE,KAAK,CAAC,QAAQ;YACjC,WAAW;YACX,WAAW,EAAE,WAAW;YACxB,SAAS,EAAE,UAAU;YACrB,kBAAkB,EAAE,KAAK;YACzB,QAAQ,EAAE,KAAK,CAAC,QAAQ;SACzB,CAAC,CAAC;QAEH,IAAI,QAAQ,CAAC,gBAAgB,CAAC,KAAK,EAAE,eAAe,EAAE;YACpD,OAAO,EAAE,CAAC,QAAQ,CAAC,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC;YACrD,iBAAiB,EAAE,aAAa;YAChC,YAAY,EAAE,WAAW;SAC1B,CAAC,CAAC;QAEH,IAAI,OAAO,CAAC,OAAO,CAAC,KAAK,EAAE,oBAAoB,EAAE;YAC/C,IAAI,EAAE,gBAAgB;YACtB,UAAU,EAAE,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,KAAK,CAAC,UAAU;YAC1D,MAAM,EAAE,OAAO,CAAC,YAAY,CAAC,SAAS,CACpC,IAAI,OAAO,CAAC,gBAAgB,CAAC,WAAW,CAAC,CAC1C;SACF,CAAC,CAAC;QAEH,8BAA8B;QAC9B,IAAI,KAAK,CAAC,UAAU,EAAE,CAAC;YACrB,IAAI,OAAO,CAAC,UAAU,CAAC,KAAK,EAAE,wBAAwB,EAAE;gBACtD,IAAI,EAAE,gBAAgB;gBACtB,UAAU,EAAE,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,KAAK,CAAC,UAAU;gBAC1D,MAAM,EAAE,OAAO,CAAC,YAAY,CAAC,SAAS,CACpC,IAAI,OAAO,CAAC,gBAAgB,CAAC,WAAW,CAAC,CAC1C;aACF,CAAC,CAAC;QACL,CAAC;QAED,IAAI,KAAK,CAAC,SAAS,EAAE,CAAC;YACpB,IAAI,OAAO,CAAC,OAAO,CAAC,KAAK,EAAE,uBAAuB,EAAE;gBAClD,IAAI,EAAE,gBAAgB;gBACtB,UAAU,EAAE,KAAK,CAAC,SAAS;gBAC3B,MAAM,EAAE,OAAO,CAAC,YAAY,CAAC,SAAS,CACpC,IAAI,OAAO,CAAC,gBAAgB,CAAC,WAAW,CAAC,CAC1C;aACF,CAAC,CAAC;YAEH,4CAA4C;YAC5C,IAAI,KAAK,CAAC,UAAU,EAAE,CAAC;gBACrB,IAAI,OAAO,CAAC,UAAU,CAAC,KAAK,EAAE,2BAA2B,EAAE;oBACzD,IAAI,EAAE,gBAAgB;oBACtB,UAAU,EAAE,KAAK,CAAC,SAAS;oBAC3B,MAAM,EAAE,OAAO,CAAC,YAAY,CAAC,SAAS,CACpC,IAAI,OAAO,CAAC,gBAAgB,CAAC,WAAW,CAAC,CAC1C;iBACF,CAAC,CAAC;YACL,CAAC;QACH,CAAC;IACH,CAAC;;AA1KH,oDA2KC","sourcesContent":["import { RemovalPolicy, Duration } from 'aws-cdk-lib';\nimport * as acm from 'aws-cdk-lib/aws-certificatemanager';\nimport * as cloudfront from 'aws-cdk-lib/aws-cloudfront';\nimport * as origins from 'aws-cdk-lib/aws-cloudfront-origins';\nimport * as route53 from 'aws-cdk-lib/aws-route53';\nimport * as targets from 'aws-cdk-lib/aws-route53-targets';\nimport * as s3 from 'aws-cdk-lib/aws-s3';\nimport * as s3deploy from 'aws-cdk-lib/aws-s3-deployment';\nimport { Construct } from 'constructs';\n\nexport interface BasicSiteConfiguration {\n  /**\n   * Local path to the website folder you want to deploy on S3.\n   */\n  readonly websiteFolder: string;\n  /**\n   * The index document of the website.\n   */\n  readonly indexDoc: string;\n  /**\n   * The error document of the website.\n   * @default - No error document.\n   */\n  readonly errorDoc?: string;\n  /**\n   * Hosted Zone used to create the DNS record for the website.\n   */\n  readonly hostedZone: string;\n}\n\nexport interface CloudfrontSiteConfiguration {\n  /**\n   * Local path to the website folder you want to deploy on S3.\n   */\n  readonly websiteFolder: string;\n  /**\n   * The index document of the website.\n   */\n  readonly indexDoc: string;\n  /**\n   * The error document of the website.\n   * @default - No error document.\n   */\n  readonly errorDoc?: string;\n  /**\n   * Hosted Zone used to create the DNS record for the website.\n   */\n  readonly hostedZone: string;\n  /**\n   * Used to deploy a Cloudfront site with a single domain. e.g. sample.example.com\n   * If you include a value for both domain and subDomain,\n   * an error will be thrown.\n   *\n   * @default - no value\n   */\n  readonly domain?: string;\n  /**\n   * The subdomain name you want to deploy. e.g. www.example.com\n   * If you include a value for both domain and subDomain,\n   * an error will be thrown.\n   *\n   * @default - no value\n   */\n  readonly subDomain?: string;\n  /**\n   * The price class determines how many edge locations CloudFront will use for your distribution.\n   * @default PriceClass.PRICE_CLASS_100.\n   * @see https://aws.amazon.com/cloudfront/pricing/.\n   */\n  readonly priceClass?: cloudfront.PriceClass;\n  /**\n   * Optional cache behaviors for different path patterns.\n   * @default - No additional cache behaviors.\n   */\n  readonly additionalBehaviors?: Record<string, cloudfront.BehaviorOptions>;\n  /**\n   * Additional permissions granted to the CloudFront Origin Access Control for the website bucket.\n   * @default [cloudfront.AccessLevel.READ]\n   */\n  readonly originAccessLevels?: cloudfront.AccessLevel[];\n  /**\n   * CloudFront Functions to associate with the default behavior.\n   * @default - No CloudFront Function associations.\n   */\n  readonly functionAssociations?: cloudfront.FunctionAssociation[];\n  /**\n   * Enable response headers policy for security headers.\n   * @default false - No security headers policy applied.\n   */\n  readonly enableSecurityHeaders?: boolean;\n  /**\n   * Enable IPv6 support with AAAA records.\n   * @default false - No IPv6 support.\n   */\n  readonly enableIpv6?: boolean;\n  /**\n   * Custom error responses for different HTTP status codes.\n   * @default - Default error responses based on errorDoc setting.\n   */\n  readonly customErrorResponses?: cloudfront.ErrorResponse[];\n  /**\n   * Enable CloudFront access logging.\n   * @default false - No access logging.\n   */\n  readonly enableLogging?: boolean;\n  /**\n   * S3 bucket for CloudFront access logs. If not provided and logging is enabled, a new bucket will be created.\n   * @default - New bucket created if logging is enabled.\n   */\n  readonly logsBucket?: s3.IBucket;\n  /**\n   * Optional WAF Web ACL ARN for enhanced security.\n   * @default - No WAF integration.\n   */\n  readonly webAclId?: string;\n}\n\n/**\n * @deprecated Use CreateCloudfrontSite instead. CreateBasicSite configures a public S3 website endpoint.\n */\nexport class CreateBasicSite extends Construct {\n  constructor(scope: Construct, id: string, props: BasicSiteConfiguration) {\n    super(scope, id);\n\n    const hostedZoneLookup = route53.HostedZone.fromLookup(\n      this,\n      'WebsiteHostedZone',\n      {\n        domainName: props.hostedZone,\n      },\n    );\n\n    const websiteRedirectBucket = new s3.Bucket(\n      scope,\n      'WebsiteRedirectBucket',\n      {\n        bucketName: `www.${props.hostedZone}`,\n        removalPolicy: RemovalPolicy.DESTROY,\n        autoDeleteObjects: true,\n        websiteRedirect: {\n          hostName: props.hostedZone,\n          protocol: s3.RedirectProtocol.HTTP,\n        },\n      },\n    );\n\n    const websiteBucket = new s3.Bucket(scope, 'WebsiteBucket', {\n      bucketName: props.hostedZone,\n      removalPolicy: RemovalPolicy.DESTROY,\n      autoDeleteObjects: true,\n      websiteIndexDocument: props.indexDoc,\n      websiteErrorDocument: props.errorDoc,\n      publicReadAccess: true,\n      blockPublicAccess: {\n        blockPublicPolicy: false,\n        blockPublicAcls: false,\n        ignorePublicAcls: false,\n        restrictPublicBuckets: false,\n      },\n      encryption: s3.BucketEncryption.S3_MANAGED,\n    });\n\n    new s3deploy.BucketDeployment(scope, 'WebsiteDeploy', {\n      sources: [s3deploy.Source.asset(props.websiteFolder)],\n      destinationBucket: websiteBucket,\n    });\n\n    new route53.ARecord(scope, 'WebsiteAlias', {\n      zone: hostedZoneLookup,\n      recordName: props.hostedZone,\n      target: route53.RecordTarget.fromAlias(\n        new targets.BucketWebsiteTarget(websiteBucket),\n      ),\n    });\n\n    new route53.ARecord(scope, 'WebsiteRedirectAlias', {\n      zone: hostedZoneLookup,\n      recordName: `www.${props.hostedZone}`,\n      target: route53.RecordTarget.fromAlias(\n        new targets.BucketWebsiteTarget(websiteRedirectBucket),\n      ),\n    });\n  }\n}\n\nexport class CreateCloudfrontSite extends Construct {\n  constructor(\n    scope: Construct,\n    id: string,\n    props: CloudfrontSiteConfiguration,\n  ) {\n    super(scope, id);\n\n    if (props.domain && props.subDomain) {\n      throw new Error(\n        'Domain and sub domain parameters cannot both be defined',\n      );\n    }\n\n    const hostedZoneLookup = route53.HostedZone.fromLookup(\n      this,\n      'WebsiteHostedZone',\n      {\n        domainName: props.hostedZone,\n      },\n    );\n\n    // Use custom error responses if provided, otherwise use default logic\n    const errorResponses: cloudfront.ErrorResponse[] = props.customErrorResponses || [];\n    if (!props.customErrorResponses) {\n      if (props.errorDoc) {\n        errorResponses.push({\n          httpStatus: 404,\n          responsePagePath: `/${props.errorDoc}`,\n        });\n        errorResponses.push({\n          httpStatus: 403,\n          responsePagePath: `/${props.errorDoc}`,\n        });\n      } else {\n        errorResponses.push({\n          httpStatus: 404,\n          responseHttpStatus: 200,\n          responsePagePath: `/${props.indexDoc}`,\n        });\n        errorResponses.push({\n          httpStatus: 403,\n          responseHttpStatus: 200,\n          responsePagePath: `/${props.indexDoc}`,\n        });\n      }\n    }\n\n    const subjectAlternativeNames: string[] = [];\n    if (props.domain) subjectAlternativeNames.push(props.domain);\n    if (props.subDomain) subjectAlternativeNames.push(props.subDomain);\n\n    const websiteCert = new acm.Certificate(this, 'WebsiteCert', {\n      domainName: props.hostedZone,\n      subjectAlternativeNames,\n      validation: acm.CertificateValidation.fromDns(hostedZoneLookup),\n    });\n\n    const websiteBucket = new s3.Bucket(scope, 'WebsiteBucket', {\n      removalPolicy: RemovalPolicy.DESTROY,\n      autoDeleteObjects: true,\n      encryption: s3.BucketEncryption.S3_MANAGED,\n    });\n\n    const domainNames: string[] = [];\n    if (props.domain) {\n      domainNames.push(props.domain);\n    } else {\n      domainNames.push(props.hostedZone);\n    }\n\n    if (props.subDomain) domainNames.push(props.subDomain);\n\n    // Create access logs bucket if logging is enabled but no bucket provided\n    const logsBucket = props.enableLogging\n      ? (props.logsBucket || new s3.Bucket(this, 'AccessLogsBucket', {\n        removalPolicy: RemovalPolicy.DESTROY,\n        autoDeleteObjects: true,\n        encryption: s3.BucketEncryption.S3_MANAGED,\n      }))\n      : undefined;\n\n    // Create security headers policy if enabled\n    const responseHeadersPolicy = props.enableSecurityHeaders\n      ? new cloudfront.ResponseHeadersPolicy(this, 'SecurityHeadersPolicy', {\n        securityHeadersBehavior: {\n          contentTypeOptions: { override: true },\n          frameOptions: { frameOption: cloudfront.HeadersFrameOption.DENY, override: true },\n          referrerPolicy: { referrerPolicy: cloudfront.HeadersReferrerPolicy.STRICT_ORIGIN_WHEN_CROSS_ORIGIN, override: true },\n          strictTransportSecurity: {\n            accessControlMaxAge: Duration.seconds(31536000),\n            includeSubdomains: true,\n            preload: true,\n            override: true,\n          },\n          xssProtection: { protection: true, modeBlock: true, override: true },\n        },\n      })\n      : undefined;\n\n    const websiteDist = new cloudfront.Distribution(scope, 'WebsiteDist', {\n      defaultBehavior: {\n        origin: origins.S3BucketOrigin.withOriginAccessControl(websiteBucket, {\n          originAccessLevels: props.originAccessLevels,\n        }),\n        allowedMethods: cloudfront.AllowedMethods.ALLOW_GET_HEAD_OPTIONS,\n        viewerProtocolPolicy: cloudfront.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n        cachePolicy: cloudfront.CachePolicy.CACHING_OPTIMIZED,\n        originRequestPolicy: cloudfront.OriginRequestPolicy.CORS_S3_ORIGIN,\n        responseHeadersPolicy,\n        functionAssociations: props.functionAssociations,\n      },\n      additionalBehaviors: props.additionalBehaviors,\n      minimumProtocolVersion: cloudfront.SecurityPolicyProtocol.TLS_V1_2_2021,\n      httpVersion: cloudfront.HttpVersion.HTTP2_AND_3,\n      priceClass: props.priceClass ?? cloudfront.PriceClass.PRICE_CLASS_100,\n      errorResponses,\n      defaultRootObject: props.indexDoc,\n      domainNames,\n      certificate: websiteCert,\n      logBucket: logsBucket,\n      logIncludesCookies: false,\n      webAclId: props.webAclId,\n    });\n\n    new s3deploy.BucketDeployment(scope, 'WebsiteDeploy', {\n      sources: [s3deploy.Source.asset(props.websiteFolder)],\n      destinationBucket: websiteBucket,\n      distribution: websiteDist,\n    });\n\n    new route53.ARecord(scope, 'WebsiteDomainAlias', {\n      zone: hostedZoneLookup,\n      recordName: props.domain ? props.domain : props.hostedZone,\n      target: route53.RecordTarget.fromAlias(\n        new targets.CloudFrontTarget(websiteDist),\n      ),\n    });\n\n    // Add IPv6 support if enabled\n    if (props.enableIpv6) {\n      new route53.AaaaRecord(scope, 'WebsiteDomainAliasIpv6', {\n        zone: hostedZoneLookup,\n        recordName: props.domain ? props.domain : props.hostedZone,\n        target: route53.RecordTarget.fromAlias(\n          new targets.CloudFrontTarget(websiteDist),\n        ),\n      });\n    }\n\n    if (props.subDomain) {\n      new route53.ARecord(scope, 'WebsiteSubDomainAlias', {\n        zone: hostedZoneLookup,\n        recordName: props.subDomain,\n        target: route53.RecordTarget.fromAlias(\n          new targets.CloudFrontTarget(websiteDist),\n        ),\n      });\n\n      // Add IPv6 support for subdomain if enabled\n      if (props.enableIpv6) {\n        new route53.AaaaRecord(scope, 'WebsiteSubDomainAliasIpv6', {\n          zone: hostedZoneLookup,\n          recordName: props.subDomain,\n          target: route53.RecordTarget.fromAlias(\n            new targets.CloudFrontTarget(websiteDist),\n          ),\n        });\n      }\n    }\n  }\n}\n"]}