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/.jsii +5144 -264
- package/.tool-versions +3 -0
- package/API.md +34 -4
- package/README.md +114 -96
- package/lib/cdk-simplewebsite-deploy.d.ts +13 -0
- package/lib/cdk-simplewebsite-deploy.js +10 -4
- package/package.json +15 -13
package/.tool-versions
CHANGED
package/API.md
CHANGED
|
@@ -47,7 +47,7 @@ new CreateBasicSite(scope: Construct, id: string, props: BasicSiteConfiguration)
|
|
|
47
47
|
|
|
48
48
|
---
|
|
49
49
|
|
|
50
|
-
#####
|
|
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
|
-
#####
|
|
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
|
-
#####
|
|
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
|
-
#####
|
|
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
|

|
|
4
4
|
|
|
5
5
|
# cdk-simplewebsite-deploy
|
|
6
|
-
This is an AWS CDK
|
|
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
|
-
### [
|
|
11
|
-
#### Creates a
|
|
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 {
|
|
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
|
|
26
|
-
websiteFolder: './src/
|
|
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
|
|
69
|
-
import software.amazon.awscdk.
|
|
70
|
-
import software.amazon.awscdk.
|
|
71
|
-
import
|
|
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
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
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
|
|
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
|
-
|
|
103
|
-
|
|
104
|
-
|
|
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
|
-
|
|
107
|
-
|
|
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 {
|
|
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
|
|
122
|
-
websiteFolder: './src/
|
|
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
|
|
167
|
-
import software.amazon.awscdk.
|
|
168
|
-
import software.amazon.awscdk.
|
|
169
|
-
import
|
|
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
|
-
|
|
180
|
-
|
|
181
|
-
|
|
182
|
-
|
|
183
|
-
|
|
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
|
|
193
|
-
from cdk_simplewebsite_deploy import
|
|
194
|
-
|
|
149
|
+
from aws_cdk import Stack
|
|
150
|
+
from cdk_simplewebsite_deploy import CreateBasicSite
|
|
151
|
+
from constructs import Construct
|
|
195
152
|
|
|
196
|
-
class MyProjectStack(
|
|
153
|
+
class MyProjectStack(Stack):
|
|
197
154
|
|
|
198
|
-
def __init__(self, scope:
|
|
155
|
+
def __init__(self, scope: Construct, construct_id: str, **kwargs) -> None:
|
|
199
156
|
super().__init__(scope, construct_id, **kwargs)
|
|
200
157
|
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
|
|
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
|
|
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,
|
|
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
|
|
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
|
|
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"]}
|