cdk-simplewebsite-deploy 2.1.10 → 2.2.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
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,57 +3,34 @@
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
  }
32
33
  ```
33
- ##### C#
34
- ```console
35
- dotnet add package ThonBecker.CDK.SimpleWebsiteDeploy
36
- ```
37
- ```cs
38
- using Amazon.CDK;
39
- using ThonBecker.CDK.SimpleWebsiteDeploy;
40
-
41
- namespace SimpleWebsiteDeploy
42
- {
43
- public class PipelineStack : Stack
44
- {
45
- internal PipelineStack(Construct scope, string id, IStackProps props = null) : base(scope, id, props)
46
- {
47
- new CreateBasicSite(scope, "test-website", new BasicSiteConfiguration()
48
- {
49
- WebsiteFolder = "./src/build",
50
- IndexDoc = "index.html",
51
- HostedZone = "example.com",
52
- });
53
- }
54
- }
55
- }
56
- ```
57
34
  ##### Java
58
35
  ```xml
59
36
  <dependency>
@@ -65,10 +42,10 @@ namespace SimpleWebsiteDeploy
65
42
  ```java
66
43
  package com.myorg;
67
44
 
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;
45
+ import com.thonbecker.simplewebsitedeploy.CreateCloudfrontSite;
46
+ import software.amazon.awscdk.Stack;
47
+ import software.amazon.awscdk.StackProps;
48
+ import software.constructs.Construct;
72
49
 
73
50
  public class MyProjectStack extends Stack {
74
51
  public MyProjectStack(final Construct scope, final String id) {
@@ -77,11 +54,13 @@ public class MyProjectStack extends Stack {
77
54
 
78
55
  public MyProjectStack(final Construct scope, final String id, final StackProps props) {
79
56
  super(scope, id, props);
80
-
81
- CreateBasicSite.Builder.create(this, "test-website")
82
- .websiteFolder("./src/build")
83
- .indexDoc("index.html")
84
- .hostedZone("example.com");
57
+
58
+ CreateCloudfrontSite.Builder.create(this, "test-website")
59
+ .websiteFolder("./src/build")
60
+ .indexDoc("index.html")
61
+ .hostedZone("example.com")
62
+ .subDomain("www.example.com")
63
+ .build();
85
64
  }
86
65
  }
87
66
  ```
@@ -91,67 +70,45 @@ pip install cdk-simplewebsite-deploy
91
70
  ```
92
71
  ```python
93
72
  from aws_cdk import Stack
94
- from cdk_simplewebsite_deploy import CreateBasicSite
73
+ from cdk_simplewebsite_deploy import CreateCloudfrontSite
95
74
  from constructs import Construct
96
75
 
76
+
97
77
  class MyProjectStack(Stack):
98
78
 
99
79
  def __init__(self, scope: Construct, construct_id: str, **kwargs) -> None:
100
80
  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')
81
+
82
+ CreateCloudfrontSite(self, 'test-website', website_folder='./src/build',
83
+ index_doc='index.html',
84
+ hosted_zone='example.com',
85
+ sub_domain='www.example.com')
105
86
  ```
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.
87
+
88
+ ### [CreateBasicSite](https://github.com/snappetal/cdk-simplewebsite-deploy/blob/main/API.md#cdk-cloudfront-deploy-createbasicsite)
89
+ #### Deprecated. Creates a website using public S3 website endpoints with a domain hosted in Route 53.
90
+ Use `CreateCloudfrontSite` for new sites. `CreateBasicSite` configures public bucket access so Route 53 can alias directly to the S3 website endpoint.
108
91
  ##### Typescript
109
92
  ```console
110
93
  yarn add cdk-simplewebsite-deploy
111
94
  ```
112
95
  ```typescript
113
96
  import * as cdk from 'aws-cdk-lib';
114
- import { CreateCloudfrontSite } from 'cdk-simplewebsite-deploy';
97
+ import { CreateBasicSite } from 'cdk-simplewebsite-deploy';
115
98
  import { Construct } from 'constructs';
116
99
 
117
100
  export class PipelineStack extends cdk.Stack {
118
101
  constructor(scope: Construct, id: string, props?: cdk.StackProps) {
119
102
  super(scope, id, props);
120
103
 
121
- new CreateCloudfrontSite(this, 'test-website', {
122
- websiteFolder: './src/dist',
104
+ new CreateBasicSite(this, 'test-website', {
105
+ websiteFolder: './src/build',
123
106
  indexDoc: 'index.html',
124
107
  hostedZone: 'example.com',
125
- subDomain: 'www.example.com',
126
108
  });
127
109
  }
128
110
  }
129
111
  ```
130
- ##### C#
131
- ```console
132
- dotnet add package ThonBecker.CDK.SimpleWebsiteDeploy
133
- ```
134
- ```cs
135
- using Amazon.CDK;
136
- using ThonBecker.CDK.SimpleWebsiteDeploy;
137
-
138
- namespace SimpleWebsiteDeploy
139
- {
140
- public class PipelineStack : Stack
141
- {
142
- internal PipelineStack(Construct scope, string id, IStackProps props = null) : base(scope, id, props)
143
- {
144
- new CreateCloudfrontSite(scope, "test-website", new CloudfrontSiteConfiguration()
145
- {
146
- WebsiteFolder = "./src/build",
147
- IndexDoc = "index.html",
148
- HostedZone = "example.com",
149
- SubDomain = "www.example.com",
150
- });
151
- }
152
- }
153
- }
154
- ```
155
112
  ##### Java
156
113
  ```xml
157
114
  <dependency>
@@ -163,10 +120,10 @@ namespace SimpleWebsiteDeploy
163
120
  ```java
164
121
  package com.myorg;
165
122
 
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;
123
+ import com.thonbecker.simplewebsitedeploy.CreateBasicSite;
124
+ import software.amazon.awscdk.Stack;
125
+ import software.amazon.awscdk.StackProps;
126
+ import software.constructs.Construct;
170
127
 
171
128
  public class MyProjectStack extends Stack {
172
129
  public MyProjectStack(final Construct scope, final String id) {
@@ -175,12 +132,12 @@ public class MyProjectStack extends Stack {
175
132
 
176
133
  public MyProjectStack(final Construct scope, final String id, final StackProps props) {
177
134
  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");
135
+
136
+ CreateBasicSite.Builder.create(this, "test-website")
137
+ .websiteFolder("./src/build")
138
+ .indexDoc("index.html")
139
+ .hostedZone("example.com")
140
+ .build();
184
141
  }
185
142
  }
186
143
  ```
@@ -189,24 +146,23 @@ public class MyProjectStack extends Stack {
189
146
  pip install cdk-simplewebsite-deploy
190
147
  ```
191
148
  ```python
192
- from aws_cdk import core
193
- from cdk_simplewebsite_deploy import CreateCloudfrontSite
194
-
149
+ from aws_cdk import Stack
150
+ from cdk_simplewebsite_deploy import CreateBasicSite
151
+ from constructs import Construct
195
152
 
196
- class MyProjectStack(core.Stack):
153
+ class MyProjectStack(Stack):
197
154
 
198
- def __init__(self, scope: core.Construct, construct_id: str, **kwargs) -> None:
155
+ def __init__(self, scope: Construct, construct_id: str, **kwargs) -> None:
199
156
  super().__init__(scope, construct_id, **kwargs)
200
157
 
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')
158
+ CreateBasicSite(self, 'test-website', website_folder='./src/build',
159
+ index_doc='index.html',
160
+ hosted_zone='example.com')
205
161
  ```
206
162
 
207
163
  ## 🚀 Enhanced Features
208
164
 
209
- The `CreateCloudfrontSite` construct now includes several optional advanced features for improved security, performance, and monitoring:
165
+ The `CreateCloudfrontSite` construct includes optional advanced features for security, performance, and monitoring.
210
166
 
211
167
  ### Security Headers
212
168
  Enable comprehensive security headers including HSTS, X-Frame-Options, Content-Type-Options, and XSS protection:
@@ -257,6 +213,48 @@ new CreateCloudfrontSite(this, 'waf-protected-website', {
257
213
  });
258
214
  ```
259
215
 
216
+ ### Origin Access Levels
217
+ Grant additional OAC permissions to the website bucket. This can be useful when you need CloudFront to distinguish missing objects from access-denied responses.
218
+
219
+ ```typescript
220
+ import * as cloudfront from 'aws-cdk-lib/aws-cloudfront';
221
+
222
+ new CreateCloudfrontSite(this, 'website-with-list-access', {
223
+ websiteFolder: './src/dist',
224
+ indexDoc: 'index.html',
225
+ hostedZone: 'example.com',
226
+ originAccessLevels: [
227
+ cloudfront.AccessLevel.READ,
228
+ cloudfront.AccessLevel.LIST,
229
+ ],
230
+ });
231
+ ```
232
+
233
+ ### CloudFront Function Associations
234
+ Attach CloudFront Functions to the default behavior for lightweight viewer request or viewer response logic.
235
+
236
+ ```typescript
237
+ import * as cloudfront from 'aws-cdk-lib/aws-cloudfront';
238
+
239
+ const rewriteFunction = new cloudfront.Function(this, 'RewriteFunction', {
240
+ code: cloudfront.FunctionCode.fromInline(
241
+ 'function handler(event) { return event.request; }',
242
+ ),
243
+ });
244
+
245
+ new CreateCloudfrontSite(this, 'website-with-function', {
246
+ websiteFolder: './src/dist',
247
+ indexDoc: 'index.html',
248
+ hostedZone: 'example.com',
249
+ functionAssociations: [
250
+ {
251
+ eventType: cloudfront.FunctionEventType.VIEWER_REQUEST,
252
+ function: rewriteFunction,
253
+ },
254
+ ],
255
+ });
256
+ ```
257
+
260
258
  ### Custom Cache Behaviors
261
259
  Add custom cache behaviors for different content types:
262
260
 
@@ -314,6 +312,12 @@ export class AdvancedWebsiteStack extends cdk.Stack {
314
312
  constructor(scope: Construct, id: string, props?: cdk.StackProps) {
315
313
  super(scope, id, props);
316
314
 
315
+ const rewriteFunction = new cloudfront.Function(this, 'RewriteFunction', {
316
+ code: cloudfront.FunctionCode.fromInline(
317
+ 'function handler(event) { return event.request; }',
318
+ ),
319
+ });
320
+
317
321
  new CreateCloudfrontSite(this, 'advanced-website', {
318
322
  websiteFolder: './dist',
319
323
  indexDoc: 'index.html',
@@ -325,6 +329,10 @@ export class AdvancedWebsiteStack extends cdk.Stack {
325
329
  priceClass: cloudfront.PriceClass.PRICE_CLASS_ALL,
326
330
  enableSecurityHeaders: true,
327
331
  enableIpv6: true,
332
+ originAccessLevels: [
333
+ cloudfront.AccessLevel.READ,
334
+ cloudfront.AccessLevel.LIST,
335
+ ],
328
336
 
329
337
  // Monitoring & Protection
330
338
  enableLogging: true,
@@ -337,6 +345,14 @@ export class AdvancedWebsiteStack extends cdk.Stack {
337
345
  cachePolicy: cloudfront.CachePolicy.CACHING_DISABLED,
338
346
  },
339
347
  },
348
+
349
+ // Edge Logic
350
+ functionAssociations: [
351
+ {
352
+ eventType: cloudfront.FunctionEventType.VIEWER_REQUEST,
353
+ function: rewriteFunction,
354
+ },
355
+ ],
340
356
 
341
357
  // SPA Error Handling
342
358
  customErrorResponses: [
@@ -357,12 +373,14 @@ export class AdvancedWebsiteStack extends cdk.Stack {
357
373
  - **Security Headers**: Automatic HSTS, X-Frame-Options, Content-Type-Options, and XSS protection
358
374
  - **WAF Integration**: Support for AWS WAF Web ACLs for advanced threat protection
359
375
  - **Origin Access Control**: Modern S3 bucket protection (replaces deprecated OAI)
376
+ - **Configurable OAC Permissions**: Optional origin access levels for the website bucket
360
377
 
361
378
  ### ⚡ **Optimized Performance**
362
379
  - **Smart Caching**: Optimized cache policies for better performance
363
380
  - **HTTP/2 & HTTP/3**: Latest protocol support for faster loading
364
381
  - **Global Edge Locations**: Configurable price classes for worldwide distribution
365
382
  - **IPv6 Support**: Dual-stack networking for better connectivity
383
+ - **CloudFront Functions**: Optional viewer request and response function associations
366
384
 
367
385
  ### 📊 **Comprehensive Monitoring**
368
386
  - **Access Logging**: CloudFront access logs for analytics
@@ -373,7 +391,7 @@ export class AdvancedWebsiteStack extends cdk.Stack {
373
391
  - **Backward Compatible**: All existing configurations continue to work
374
392
  - **Type Safe**: Full TypeScript support with comprehensive interfaces
375
393
  - **CDK v2 Ready**: Built for the latest AWS CDK version
376
- - **Multi-Language**: Support for TypeScript, Python, Java, and C#
394
+ - **Multi-Language**: Support for TypeScript, Python, and Java
377
395
 
378
396
  ## License
379
397
 
@@ -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.1" };
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":"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","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.1" };
213
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cdk-simplewebsite-deploy.js","sourceRoot":"","sources":["../src/cdk-simplewebsite-deploy.ts"],"names":[],"mappings":"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","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"]}