@cdklabs/cdk-appmod-catalog-blueprints 1.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.jsii +8644 -0
- package/LICENSE +202 -0
- package/README.md +212 -0
- package/lib/document-processing/agentic-document-processing.d.ts +16 -0
- package/lib/document-processing/agentic-document-processing.js +90 -0
- package/lib/document-processing/base-document-processing.d.ts +189 -0
- package/lib/document-processing/base-document-processing.js +509 -0
- package/lib/document-processing/bedrock-document-processing.d.ts +167 -0
- package/lib/document-processing/bedrock-document-processing.js +297 -0
- package/lib/document-processing/index.d.ts +3 -0
- package/lib/document-processing/index.js +20 -0
- package/lib/document-processing/resources/default-bedrock-invoke/index.py +63 -0
- package/lib/document-processing/resources/default-bedrock-invoke/requirements.txt +4 -0
- package/lib/document-processing/resources/default-doc-retrieval-lambda/index.mjs +92 -0
- package/lib/document-processing/resources/default-doc-retrieval-lambda/package.json +10 -0
- package/lib/document-processing/resources/default-error-handler/index.js +46 -0
- package/lib/document-processing/resources/default-error-handler/package.json +4 -0
- package/lib/document-processing/resources/default-image-processor/classifier.mjs +665 -0
- package/lib/document-processing/resources/default-image-processor/extractors.mjs +465 -0
- package/lib/document-processing/resources/default-image-processor/index.mjs +143 -0
- package/lib/document-processing/resources/default-image-processor/package-lock.json +12 -0
- package/lib/document-processing/resources/default-image-processor/package.json +4 -0
- package/lib/document-processing/resources/default-image-validator/index.mjs +76 -0
- package/lib/document-processing/resources/default-image-validator/package-lock.json +154 -0
- package/lib/document-processing/resources/default-image-validator/package.json +7 -0
- package/lib/document-processing/resources/default-pdf-processor/index.js +46 -0
- package/lib/document-processing/resources/default-pdf-validator/index.js +36 -0
- package/lib/document-processing/resources/default-sqs-consumer/index.py +111 -0
- package/lib/document-processing/resources/default-sqs-consumer/requirements.txt +4 -0
- package/lib/document-processing/resources/default-sqs-consumer/sample_payload.json +20 -0
- package/lib/document-processing/resources/default-sqs-consumer/sample_payload_multi.json +24 -0
- package/lib/document-processing/resources/default-strands-agent/index.py +111 -0
- package/lib/document-processing/resources/default-strands-agent/requirements.txt +6 -0
- package/lib/document-processing/tests/agentic-document-processing-nag.test.d.ts +1 -0
- package/lib/document-processing/tests/agentic-document-processing-nag.test.js +107 -0
- package/lib/document-processing/tests/agentic-document-processing.test.d.ts +1 -0
- package/lib/document-processing/tests/agentic-document-processing.test.js +125 -0
- package/lib/document-processing/tests/bedrock-document-processing-nag.test.d.ts +1 -0
- package/lib/document-processing/tests/bedrock-document-processing-nag.test.js +101 -0
- package/lib/document-processing/tests/bedrock-document-processing.test.d.ts +1 -0
- package/lib/document-processing/tests/bedrock-document-processing.test.js +79 -0
- package/lib/framework/custom-resource/default-runtimes.d.ts +21 -0
- package/lib/framework/custom-resource/default-runtimes.js +34 -0
- package/lib/framework/custom-resource/index.d.ts +1 -0
- package/lib/framework/custom-resource/index.js +18 -0
- package/lib/framework/foundation/access-log.d.ts +69 -0
- package/lib/framework/foundation/access-log.js +121 -0
- package/lib/framework/foundation/eventbridge-broker.d.ts +18 -0
- package/lib/framework/foundation/eventbridge-broker.js +42 -0
- package/lib/framework/foundation/index.d.ts +3 -0
- package/lib/framework/foundation/index.js +20 -0
- package/lib/framework/foundation/network.d.ts +19 -0
- package/lib/framework/foundation/network.js +83 -0
- package/lib/framework/index.d.ts +2 -0
- package/lib/framework/index.js +19 -0
- package/lib/framework/quickstart/base-quickstart.d.ts +30 -0
- package/lib/framework/quickstart/base-quickstart.js +30 -0
- package/lib/index.d.ts +4 -0
- package/lib/index.js +21 -0
- package/lib/tsconfig.tsbuildinfo +1 -0
- package/lib/utilities/cdk-nag-config.d.ts +42 -0
- package/lib/utilities/cdk-nag-config.js +194 -0
- package/lib/utilities/data-loader-lambda/index.py +282 -0
- package/lib/utilities/data-loader-lambda/requirements.txt +3 -0
- package/lib/utilities/data-loader.d.ts +173 -0
- package/lib/utilities/data-loader.js +447 -0
- package/lib/utilities/index.d.ts +3 -0
- package/lib/utilities/index.js +20 -0
- package/lib/utilities/lambda-iam-utils.d.ts +145 -0
- package/lib/utilities/lambda-iam-utils.js +235 -0
- package/lib/utilities/lambda_layers/data-masking/layer-construct.d.ts +42 -0
- package/lib/utilities/lambda_layers/data-masking/layer-construct.js +53 -0
- package/lib/utilities/lambda_layers/data-masking/layer-construct.ts +88 -0
- package/lib/utilities/observability/bedrock-observability.d.ts +18 -0
- package/lib/utilities/observability/bedrock-observability.js +131 -0
- package/lib/utilities/observability/cloudfront-distribution-observability-property-injector.d.ts +6 -0
- package/lib/utilities/observability/cloudfront-distribution-observability-property-injector.js +22 -0
- package/lib/utilities/observability/index.d.ts +6 -0
- package/lib/utilities/observability/index.js +25 -0
- package/lib/utilities/observability/lambda-observability-property-injector.d.ts +8 -0
- package/lib/utilities/observability/lambda-observability-property-injector.js +43 -0
- package/lib/utilities/observability/log-group-data-protection-props.d.ts +19 -0
- package/lib/utilities/observability/log-group-data-protection-props.js +5 -0
- package/lib/utilities/observability/observability.d.ts +83 -0
- package/lib/utilities/observability/observability.js +278 -0
- package/lib/utilities/observability/observable.d.ts +32 -0
- package/lib/utilities/observability/observable.js +3 -0
- package/lib/utilities/observability/powertools-config.d.ts +3 -0
- package/lib/utilities/observability/powertools-config.js +25 -0
- package/lib/utilities/observability/resources/bedrock-manage-logging-configuration/index.py +27 -0
- package/lib/utilities/observability/state-machine-observability-property-injector.d.ts +8 -0
- package/lib/utilities/observability/state-machine-observability-property-injector.js +49 -0
- package/lib/utilities/tests/data-loader-nag.test.d.ts +1 -0
- package/lib/utilities/tests/data-loader-nag.test.js +432 -0
- package/lib/utilities/tests/data-loader.test.d.ts +1 -0
- package/lib/utilities/tests/data-loader.test.js +284 -0
- package/lib/webapp/frontend-construct.d.ts +136 -0
- package/lib/webapp/frontend-construct.js +253 -0
- package/lib/webapp/index.d.ts +1 -0
- package/lib/webapp/index.js +18 -0
- package/lib/webapp/tests/frontend-construct-nag.test.d.ts +1 -0
- package/lib/webapp/tests/frontend-construct-nag.test.js +266 -0
- package/lib/webapp/tests/frontend-construct.test.d.ts +1 -0
- package/lib/webapp/tests/frontend-construct.test.js +385 -0
- package/package.json +183 -0
|
@@ -0,0 +1,253 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var _a;
|
|
3
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
4
|
+
exports.Frontend = exports.DEFAULT_SPA_ERROR_RESPONSES = void 0;
|
|
5
|
+
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
|
6
|
+
// Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
|
|
7
|
+
// SPDX-License-Identifier: Apache-2.0
|
|
8
|
+
const path = require("path");
|
|
9
|
+
const aws_cdk_lib_1 = require("aws-cdk-lib");
|
|
10
|
+
const aws_cloudfront_1 = require("aws-cdk-lib/aws-cloudfront");
|
|
11
|
+
const aws_cloudfront_origins_1 = require("aws-cdk-lib/aws-cloudfront-origins");
|
|
12
|
+
const aws_iam_1 = require("aws-cdk-lib/aws-iam");
|
|
13
|
+
const aws_route53_1 = require("aws-cdk-lib/aws-route53");
|
|
14
|
+
const aws_route53_targets_1 = require("aws-cdk-lib/aws-route53-targets");
|
|
15
|
+
const aws_s3_1 = require("aws-cdk-lib/aws-s3");
|
|
16
|
+
const aws_s3_assets_1 = require("aws-cdk-lib/aws-s3-assets");
|
|
17
|
+
const aws_s3_deployment_1 = require("aws-cdk-lib/aws-s3-deployment");
|
|
18
|
+
const constructs_1 = require("constructs");
|
|
19
|
+
const utilities_1 = require("../utilities");
|
|
20
|
+
/**
|
|
21
|
+
* Default CloudFront error responses for Single Page Applications
|
|
22
|
+
*/
|
|
23
|
+
exports.DEFAULT_SPA_ERROR_RESPONSES = [
|
|
24
|
+
{
|
|
25
|
+
httpStatus: 403,
|
|
26
|
+
responseHttpStatus: 200,
|
|
27
|
+
responsePagePath: '/index.html',
|
|
28
|
+
},
|
|
29
|
+
{
|
|
30
|
+
httpStatus: 404,
|
|
31
|
+
responseHttpStatus: 200,
|
|
32
|
+
responsePagePath: '/index.html',
|
|
33
|
+
},
|
|
34
|
+
];
|
|
35
|
+
/**
|
|
36
|
+
* Frontend construct that deploys a frontend application to S3 and CloudFront
|
|
37
|
+
*
|
|
38
|
+
* This construct provides a complete solution for hosting static frontend applications
|
|
39
|
+
* with the following features:
|
|
40
|
+
* - S3 bucket for hosting static assets with security best practices
|
|
41
|
+
* - CloudFront distribution for global content delivery
|
|
42
|
+
* - Optional custom domain with SSL certificate
|
|
43
|
+
* - Automatic build process execution
|
|
44
|
+
* - SPA-friendly error handling by default
|
|
45
|
+
* - Security configurations
|
|
46
|
+
*/
|
|
47
|
+
class Frontend extends constructs_1.Construct {
|
|
48
|
+
/**
|
|
49
|
+
* Creates a new Frontend
|
|
50
|
+
* @param scope The construct scope
|
|
51
|
+
* @param id The construct ID
|
|
52
|
+
* @param props The frontend properties
|
|
53
|
+
*/
|
|
54
|
+
constructor(scope, id, props) {
|
|
55
|
+
super(scope, id);
|
|
56
|
+
if (props.enableObservability) {
|
|
57
|
+
aws_cdk_lib_1.PropertyInjectors.of(this).add(new utilities_1.CloudfrontDistributionObservabilityPropertyInjector());
|
|
58
|
+
}
|
|
59
|
+
// Validate required parameters
|
|
60
|
+
this._validateProps(props);
|
|
61
|
+
// Get removal policy with default
|
|
62
|
+
const removalPolicy = props.removalPolicy || aws_cdk_lib_1.RemovalPolicy.DESTROY;
|
|
63
|
+
// Create asset for source code with optional bundling
|
|
64
|
+
if (!props.skipBuild) {
|
|
65
|
+
this.asset = this._createAsset(props);
|
|
66
|
+
}
|
|
67
|
+
// Create S3 bucket for hosting
|
|
68
|
+
this.bucket = new aws_s3_1.Bucket(this, 'FrontendBucket', {
|
|
69
|
+
encryption: aws_s3_1.BucketEncryption.S3_MANAGED,
|
|
70
|
+
blockPublicAccess: aws_s3_1.BlockPublicAccess.BLOCK_ALL,
|
|
71
|
+
removalPolicy: removalPolicy,
|
|
72
|
+
autoDeleteObjects: removalPolicy === aws_cdk_lib_1.RemovalPolicy.DESTROY,
|
|
73
|
+
});
|
|
74
|
+
// Create CloudFront distribution
|
|
75
|
+
this.distribution = this._createDistribution(props, removalPolicy);
|
|
76
|
+
// Deploy frontend assets to S3
|
|
77
|
+
const buildOutputDirectory = props.buildOutputDirectory || path.join(props.sourceDirectory, 'build');
|
|
78
|
+
this.bucketDeployment = new aws_s3_deployment_1.BucketDeployment(this, 'FrontendDeployment', {
|
|
79
|
+
sources: this.asset
|
|
80
|
+
? [aws_s3_deployment_1.Source.bucket(this.asset.bucket, this.asset.s3ObjectKey)]
|
|
81
|
+
: [aws_s3_deployment_1.Source.asset(buildOutputDirectory)],
|
|
82
|
+
destinationBucket: this.bucket,
|
|
83
|
+
distribution: this.distribution,
|
|
84
|
+
distributionPaths: ['/*'],
|
|
85
|
+
});
|
|
86
|
+
this.bucketDeployment.handlerRole.addToPrincipalPolicy(new aws_iam_1.PolicyStatement({
|
|
87
|
+
effect: aws_iam_1.Effect.ALLOW,
|
|
88
|
+
actions: [
|
|
89
|
+
'cloudfront:GetInvalidation',
|
|
90
|
+
'cloudfront:CreateInvalidation',
|
|
91
|
+
],
|
|
92
|
+
resources: ['*'],
|
|
93
|
+
}));
|
|
94
|
+
// Note: BucketDeployment doesn't support applyRemovalPolicy directly
|
|
95
|
+
// It will be cleaned up when the bucket is deleted due to autoDeleteObjects
|
|
96
|
+
// Setup custom domain if provided
|
|
97
|
+
if (props.customDomain) {
|
|
98
|
+
this.domainName = props.customDomain.domainName;
|
|
99
|
+
this._setupCustomDomain(props.customDomain, removalPolicy);
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
/**
|
|
103
|
+
* Validates the construct properties
|
|
104
|
+
* @param props The frontend properties
|
|
105
|
+
* @private
|
|
106
|
+
*/
|
|
107
|
+
_validateProps(props) {
|
|
108
|
+
if (!props.sourceDirectory) {
|
|
109
|
+
throw new Error('sourceDirectory is required');
|
|
110
|
+
}
|
|
111
|
+
if (props.customDomain?.domainName && !props.customDomain.certificate) {
|
|
112
|
+
throw new Error('certificate is required when domainName is provided');
|
|
113
|
+
}
|
|
114
|
+
}
|
|
115
|
+
/**
|
|
116
|
+
* Creates an Asset for the frontend source code with bundling
|
|
117
|
+
* @param props The frontend properties
|
|
118
|
+
* @returns The Asset containing the built frontend
|
|
119
|
+
* @private
|
|
120
|
+
*/
|
|
121
|
+
_createAsset(props) {
|
|
122
|
+
const buildCommand = props.buildCommand || 'npm run build';
|
|
123
|
+
const buildOutputDirectory = props.buildOutputDirectory || path.join(props.sourceDirectory, 'build');
|
|
124
|
+
// Extract the build directory name from the full path
|
|
125
|
+
const buildDirName = path.basename(buildOutputDirectory);
|
|
126
|
+
const asset = new aws_s3_assets_1.Asset(this, 'FrontendAsset', {
|
|
127
|
+
path: props.sourceDirectory,
|
|
128
|
+
bundling: {
|
|
129
|
+
image: aws_cdk_lib_1.DockerImage.fromRegistry('public.ecr.aws/docker/library/node:lts-alpine'),
|
|
130
|
+
command: [
|
|
131
|
+
'sh', '-c', [
|
|
132
|
+
'cd /asset-input',
|
|
133
|
+
'npm ci --only=production',
|
|
134
|
+
buildCommand,
|
|
135
|
+
`cp -r ./${buildDirName}/* /asset-output/`,
|
|
136
|
+
].join(' && '),
|
|
137
|
+
],
|
|
138
|
+
user: 'root',
|
|
139
|
+
},
|
|
140
|
+
});
|
|
141
|
+
// Note: Asset doesn't support applyRemovalPolicy directly
|
|
142
|
+
// The underlying S3 objects will be managed by the asset bucket's removal policy
|
|
143
|
+
return asset;
|
|
144
|
+
}
|
|
145
|
+
/**
|
|
146
|
+
* Creates the CloudFront distribution
|
|
147
|
+
* @param props The frontend properties
|
|
148
|
+
* @param removalPolicy The removal policy to apply
|
|
149
|
+
* @returns The CloudFront distribution
|
|
150
|
+
* @private
|
|
151
|
+
*/
|
|
152
|
+
_createDistribution(props, removalPolicy) {
|
|
153
|
+
const errorResponses = props.errorResponses || exports.DEFAULT_SPA_ERROR_RESPONSES;
|
|
154
|
+
// Create a CloudFront function for security headers
|
|
155
|
+
const securityHeadersFunction = new aws_cloudfront_1.Function(this, 'SecurityHeadersFunction', {
|
|
156
|
+
code: aws_cloudfront_1.FunctionCode.fromInline(`
|
|
157
|
+
function handler(event) {
|
|
158
|
+
var response = event.response;
|
|
159
|
+
var headers = response.headers;
|
|
160
|
+
|
|
161
|
+
// Add security headers
|
|
162
|
+
headers['strict-transport-security'] = { value: 'max-age=63072000; includeSubdomains; preload' };
|
|
163
|
+
headers['content-type-options'] = { value: 'nosniff' };
|
|
164
|
+
headers['x-frame-options'] = { value: 'DENY' };
|
|
165
|
+
headers['x-content-type-options'] = { value: 'nosniff' };
|
|
166
|
+
headers['referrer-policy'] = { value: 'strict-origin-when-cross-origin' };
|
|
167
|
+
headers['permissions-policy'] = { value: 'camera=(), microphone=(), geolocation=()' };
|
|
168
|
+
|
|
169
|
+
return response;
|
|
170
|
+
}
|
|
171
|
+
`),
|
|
172
|
+
});
|
|
173
|
+
// Apply removal policy to CloudFront function
|
|
174
|
+
securityHeadersFunction.applyRemovalPolicy(removalPolicy);
|
|
175
|
+
const distributionConfig = {
|
|
176
|
+
defaultBehavior: {
|
|
177
|
+
origin: aws_cloudfront_origins_1.S3BucketOrigin.withOriginAccessControl(this.bucket),
|
|
178
|
+
viewerProtocolPolicy: aws_cloudfront_1.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,
|
|
179
|
+
functionAssociations: [
|
|
180
|
+
{
|
|
181
|
+
function: securityHeadersFunction,
|
|
182
|
+
eventType: aws_cloudfront_1.FunctionEventType.VIEWER_RESPONSE,
|
|
183
|
+
},
|
|
184
|
+
],
|
|
185
|
+
},
|
|
186
|
+
defaultRootObject: 'index.html',
|
|
187
|
+
errorResponses,
|
|
188
|
+
comment: props.distributionProps?.comment,
|
|
189
|
+
enabled: props.distributionProps?.enabled,
|
|
190
|
+
priceClass: props.distributionProps?.priceClass,
|
|
191
|
+
webAclId: props.distributionProps?.webAclId,
|
|
192
|
+
};
|
|
193
|
+
// Add custom domain configuration if provided
|
|
194
|
+
let distribution;
|
|
195
|
+
if (props.customDomain) {
|
|
196
|
+
distribution = new aws_cloudfront_1.Distribution(this, 'FrontendDistribution', {
|
|
197
|
+
...distributionConfig,
|
|
198
|
+
domainNames: [props.customDomain.domainName],
|
|
199
|
+
certificate: props.customDomain.certificate,
|
|
200
|
+
});
|
|
201
|
+
}
|
|
202
|
+
else {
|
|
203
|
+
distribution = new aws_cloudfront_1.Distribution(this, 'FrontendDistribution', distributionConfig);
|
|
204
|
+
}
|
|
205
|
+
// Apply removal policy to distribution
|
|
206
|
+
distribution.applyRemovalPolicy(removalPolicy);
|
|
207
|
+
return distribution;
|
|
208
|
+
}
|
|
209
|
+
/**
|
|
210
|
+
* Sets up custom domain with Route53 record
|
|
211
|
+
* @param customDomain The custom domain configuration
|
|
212
|
+
* @param removalPolicy The removal policy to apply
|
|
213
|
+
* @private
|
|
214
|
+
*/
|
|
215
|
+
_setupCustomDomain(customDomain, removalPolicy) {
|
|
216
|
+
if (customDomain.hostedZone) {
|
|
217
|
+
const aliasRecord = new aws_route53_1.ARecord(this, 'FrontendAliasRecord', {
|
|
218
|
+
zone: customDomain.hostedZone,
|
|
219
|
+
recordName: customDomain.domainName,
|
|
220
|
+
target: aws_route53_1.RecordTarget.fromAlias(new aws_route53_targets_1.CloudFrontTarget(this.distribution)),
|
|
221
|
+
});
|
|
222
|
+
// Apply removal policy to Route53 record
|
|
223
|
+
aliasRecord.applyRemovalPolicy(removalPolicy);
|
|
224
|
+
}
|
|
225
|
+
}
|
|
226
|
+
/**
|
|
227
|
+
* Gets the URL of the frontend application
|
|
228
|
+
* @returns The frontend URL
|
|
229
|
+
*/
|
|
230
|
+
url() {
|
|
231
|
+
return this.domainName
|
|
232
|
+
? `https://${this.domainName}`
|
|
233
|
+
: `https://${this.distribution.distributionDomainName}`;
|
|
234
|
+
}
|
|
235
|
+
/**
|
|
236
|
+
* Gets the CloudFront distribution domain name
|
|
237
|
+
* @returns The CloudFront domain name
|
|
238
|
+
*/
|
|
239
|
+
distributionDomainName() {
|
|
240
|
+
return this.distribution.distributionDomainName;
|
|
241
|
+
}
|
|
242
|
+
/**
|
|
243
|
+
* Gets the S3 bucket name
|
|
244
|
+
* @returns The S3 bucket name
|
|
245
|
+
*/
|
|
246
|
+
bucketName() {
|
|
247
|
+
return this.bucket.bucketName;
|
|
248
|
+
}
|
|
249
|
+
}
|
|
250
|
+
exports.Frontend = Frontend;
|
|
251
|
+
_a = JSII_RTTI_SYMBOL_1;
|
|
252
|
+
Frontend[_a] = { fqn: "@cdklabs/cdk-appmod-catalog-blueprints.Frontend", version: "1.0.0" };
|
|
253
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"frontend-construct.js","sourceRoot":"","sources":["../../use-cases/webapp/frontend-construct.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.\n// SPDX-License-Identifier: Apache-2.0\n\nimport * as path from 'path';\nimport { DockerImage, PropertyInjectors, RemovalPolicy } from 'aws-cdk-lib';\nimport { ICertificate } from 'aws-cdk-lib/aws-certificatemanager';\nimport {\n  Distribution,\n  DistributionProps,\n  ErrorResponse,\n  Function as CloudFrontFunction,\n  FunctionCode,\n  FunctionEventType,\n  PriceClass,\n  ViewerProtocolPolicy,\n} from 'aws-cdk-lib/aws-cloudfront';\nimport { S3BucketOrigin } from 'aws-cdk-lib/aws-cloudfront-origins';\nimport { Effect, PolicyStatement } from 'aws-cdk-lib/aws-iam';\nimport { ARecord, IHostedZone, RecordTarget } from 'aws-cdk-lib/aws-route53';\nimport { CloudFrontTarget } from 'aws-cdk-lib/aws-route53-targets';\nimport { BlockPublicAccess, Bucket, BucketEncryption } from 'aws-cdk-lib/aws-s3';\nimport { Asset } from 'aws-cdk-lib/aws-s3-assets';\nimport { BucketDeployment, Source } from 'aws-cdk-lib/aws-s3-deployment';\nimport { Construct } from 'constructs';\nimport { CloudfrontDistributionObservabilityPropertyInjector } from '../utilities';\n\n/**\n * Default CloudFront error responses for Single Page Applications\n */\nexport const DEFAULT_SPA_ERROR_RESPONSES: ErrorResponse[] = [\n  {\n    httpStatus: 403,\n    responseHttpStatus: 200,\n    responsePagePath: '/index.html',\n  },\n  {\n    httpStatus: 404,\n    responseHttpStatus: 200,\n    responsePagePath: '/index.html',\n  },\n];\n\n/**\n * Custom domain configuration for the frontend\n */\nexport interface CustomDomainConfig {\n  /** Domain name for the frontend (e.g., 'app.example.com') */\n  readonly domainName: string;\n  /** SSL certificate for the domain (required when domainName is provided) */\n  readonly certificate: ICertificate;\n  /** Optional hosted zone for automatic DNS record creation */\n  readonly hostedZone?: IHostedZone;\n}\n\n/**\n * Additional CloudFront distribution properties\n */\nexport interface AdditionalDistributionProps {\n  /** Optional comment for the distribution */\n  readonly comment?: string;\n  /** Optional enabled flag for the distribution */\n  readonly enabled?: boolean;\n  /** Optional price class for the distribution */\n  readonly priceClass?: PriceClass;\n  /** Optional web ACL ID for the distribution */\n  readonly webAclId?: string;\n}\n\n/**\n * Properties for the Frontend construct\n */\nexport interface FrontendProps {\n  /** Base directory of the frontend source code */\n  readonly sourceDirectory: string;\n  /** Directory where build artifacts are located after build command completes (defaults to '{sourceDirectory}/build') */\n  readonly buildOutputDirectory?: string;\n  /** Optional build command (defaults to 'npm run build') */\n  readonly buildCommand?: string;\n  /** Optional custom domain configuration */\n  readonly customDomain?: CustomDomainConfig;\n  /** Optional CloudFront error responses (defaults to SPA-friendly responses) */\n  readonly errorResponses?: ErrorResponse[];\n  /** Optional additional CloudFront distribution properties */\n  readonly distributionProps?: AdditionalDistributionProps;\n  /** Optional flag to skip the build process (useful for pre-built artifacts) */\n  readonly skipBuild?: boolean;\n  /** Optional removal policy for all resources (defaults to DESTROY) */\n  readonly removalPolicy?: RemovalPolicy;\n  /**\n   * Enable logging and tracing for all supporting resource\n   * @default false\n   */\n  readonly enableObservability?: boolean;\n}\n\n/**\n * Frontend construct that deploys a frontend application to S3 and CloudFront\n *\n * This construct provides a complete solution for hosting static frontend applications\n * with the following features:\n * - S3 bucket for hosting static assets with security best practices\n * - CloudFront distribution for global content delivery\n * - Optional custom domain with SSL certificate\n * - Automatic build process execution\n * - SPA-friendly error handling by default\n * - Security configurations\n */\nexport class Frontend extends Construct {\n  /** The S3 bucket hosting the frontend assets */\n  public readonly bucket: Bucket;\n  /** The CloudFront distribution */\n  public readonly distribution: Distribution;\n  /** The bucket deployment that uploads the frontend assets */\n  public readonly bucketDeployment: BucketDeployment;\n  /** The custom domain name (if configured) */\n  public readonly domainName?: string;\n  /** The Asset containing the frontend source code */\n  public readonly asset?: Asset;\n\n  /**\n   * Creates a new Frontend\n   * @param scope The construct scope\n   * @param id The construct ID\n   * @param props The frontend properties\n   */\n  constructor(scope: Construct, id: string, props: FrontendProps) {\n    super(scope, id);\n\n    if (props.enableObservability) {\n      PropertyInjectors.of(this).add(\n        new CloudfrontDistributionObservabilityPropertyInjector(),\n      );\n    }\n\n    // Validate required parameters\n    this._validateProps(props);\n\n    // Get removal policy with default\n    const removalPolicy = props.removalPolicy || RemovalPolicy.DESTROY;\n\n    // Create asset for source code with optional bundling\n    if (!props.skipBuild) {\n      this.asset = this._createAsset(props);\n    }\n\n    // Create S3 bucket for hosting\n    this.bucket = new Bucket(this, 'FrontendBucket', {\n      encryption: BucketEncryption.S3_MANAGED,\n      blockPublicAccess: BlockPublicAccess.BLOCK_ALL,\n      removalPolicy: removalPolicy,\n      autoDeleteObjects: removalPolicy === RemovalPolicy.DESTROY,\n    });\n\n    // Create CloudFront distribution\n    this.distribution = this._createDistribution(props, removalPolicy);\n\n    // Deploy frontend assets to S3\n    const buildOutputDirectory = props.buildOutputDirectory || path.join(props.sourceDirectory, 'build');\n    this.bucketDeployment = new BucketDeployment(this, 'FrontendDeployment', {\n      sources: this.asset\n        ? [Source.bucket(this.asset.bucket, this.asset.s3ObjectKey)]\n        : [Source.asset(buildOutputDirectory)],\n      destinationBucket: this.bucket,\n      distribution: this.distribution,\n      distributionPaths: ['/*'],\n    });\n\n    this.bucketDeployment.handlerRole.addToPrincipalPolicy(new PolicyStatement({\n      effect: Effect.ALLOW,\n      actions: [\n        'cloudfront:GetInvalidation',\n        'cloudfront:CreateInvalidation',\n      ],\n      resources: ['*'],\n    }));\n\n    // Note: BucketDeployment doesn't support applyRemovalPolicy directly\n    // It will be cleaned up when the bucket is deleted due to autoDeleteObjects\n\n    // Setup custom domain if provided\n    if (props.customDomain) {\n      this.domainName = props.customDomain.domainName;\n      this._setupCustomDomain(props.customDomain, removalPolicy);\n    }\n  }\n\n  /**\n   * Validates the construct properties\n   * @param props The frontend properties\n   * @private\n   */\n  private _validateProps(props: FrontendProps): void {\n    if (!props.sourceDirectory) {\n      throw new Error('sourceDirectory is required');\n    }\n\n    if (props.customDomain?.domainName && !props.customDomain.certificate) {\n      throw new Error('certificate is required when domainName is provided');\n    }\n  }\n\n  /**\n   * Creates an Asset for the frontend source code with bundling\n   * @param props The frontend properties\n   * @returns The Asset containing the built frontend\n   * @private\n   */\n  private _createAsset(props: FrontendProps): Asset {\n    const buildCommand = props.buildCommand || 'npm run build';\n    const buildOutputDirectory = props.buildOutputDirectory || path.join(props.sourceDirectory, 'build');\n\n    // Extract the build directory name from the full path\n    const buildDirName = path.basename(buildOutputDirectory);\n\n    const asset = new Asset(this, 'FrontendAsset', {\n      path: props.sourceDirectory,\n      bundling: {\n        image: DockerImage.fromRegistry('public.ecr.aws/docker/library/node:lts-alpine'),\n        command: [\n          'sh', '-c', [\n            'cd /asset-input',\n            'npm ci --only=production',\n            buildCommand,\n            `cp -r ./${buildDirName}/* /asset-output/`,\n          ].join(' && '),\n        ],\n        user: 'root',\n      },\n    });\n\n    // Note: Asset doesn't support applyRemovalPolicy directly\n    // The underlying S3 objects will be managed by the asset bucket's removal policy\n\n    return asset;\n  }\n\n  /**\n   * Creates the CloudFront distribution\n   * @param props The frontend properties\n   * @param removalPolicy The removal policy to apply\n   * @returns The CloudFront distribution\n   * @private\n   */\n  private _createDistribution(props: FrontendProps, removalPolicy: RemovalPolicy): Distribution {\n    const errorResponses = props.errorResponses || DEFAULT_SPA_ERROR_RESPONSES;\n\n    // Create a CloudFront function for security headers\n    const securityHeadersFunction = new CloudFrontFunction(this, 'SecurityHeadersFunction', {\n      code: FunctionCode.fromInline(`\n        function handler(event) {\n          var response = event.response;\n          var headers = response.headers;\n          \n          // Add security headers\n          headers['strict-transport-security'] = { value: 'max-age=63072000; includeSubdomains; preload' };\n          headers['content-type-options'] = { value: 'nosniff' };\n          headers['x-frame-options'] = { value: 'DENY' };\n          headers['x-content-type-options'] = { value: 'nosniff' };\n          headers['referrer-policy'] = { value: 'strict-origin-when-cross-origin' };\n          headers['permissions-policy'] = { value: 'camera=(), microphone=(), geolocation=()' };\n          \n          return response;\n        }\n      `),\n    });\n\n    // Apply removal policy to CloudFront function\n    securityHeadersFunction.applyRemovalPolicy(removalPolicy);\n\n    const distributionConfig: DistributionProps = {\n      defaultBehavior: {\n        origin: S3BucketOrigin.withOriginAccessControl(this.bucket),\n        viewerProtocolPolicy: ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n        functionAssociations: [\n          {\n            function: securityHeadersFunction,\n            eventType: FunctionEventType.VIEWER_RESPONSE,\n          },\n        ],\n      },\n      defaultRootObject: 'index.html',\n      errorResponses,\n      comment: props.distributionProps?.comment,\n      enabled: props.distributionProps?.enabled,\n      priceClass: props.distributionProps?.priceClass,\n      webAclId: props.distributionProps?.webAclId,\n    };\n\n    // Add custom domain configuration if provided\n    let distribution: Distribution;\n    if (props.customDomain) {\n      distribution = new Distribution(this, 'FrontendDistribution', {\n        ...distributionConfig,\n        domainNames: [props.customDomain.domainName],\n        certificate: props.customDomain.certificate,\n      });\n    } else {\n      distribution = new Distribution(this, 'FrontendDistribution', distributionConfig);\n    }\n\n    // Apply removal policy to distribution\n    distribution.applyRemovalPolicy(removalPolicy);\n\n    return distribution;\n  }\n\n  /**\n   * Sets up custom domain with Route53 record\n   * @param customDomain The custom domain configuration\n   * @param removalPolicy The removal policy to apply\n   * @private\n   */\n  private _setupCustomDomain(customDomain: CustomDomainConfig, removalPolicy: RemovalPolicy): void {\n    if (customDomain.hostedZone) {\n      const aliasRecord = new ARecord(this, 'FrontendAliasRecord', {\n        zone: customDomain.hostedZone,\n        recordName: customDomain.domainName,\n        target: RecordTarget.fromAlias(new CloudFrontTarget(this.distribution)),\n      });\n\n      // Apply removal policy to Route53 record\n      aliasRecord.applyRemovalPolicy(removalPolicy);\n    }\n  }\n\n  /**\n   * Gets the URL of the frontend application\n   * @returns The frontend URL\n   */\n  public url(): string {\n    return this.domainName\n      ? `https://${this.domainName}`\n      : `https://${this.distribution.distributionDomainName}`;\n  }\n\n  /**\n   * Gets the CloudFront distribution domain name\n   * @returns The CloudFront domain name\n   */\n  public distributionDomainName(): string {\n    return this.distribution.distributionDomainName;\n  }\n\n  /**\n   * Gets the S3 bucket name\n   * @returns The S3 bucket name\n   */\n  public bucketName(): string {\n    return this.bucket.bucketName;\n  }\n}\n"]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export * from './frontend-construct';
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __exportStar = (this && this.__exportStar) || function(m, exports) {
|
|
14
|
+
for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
|
|
15
|
+
};
|
|
16
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
17
|
+
__exportStar(require("./frontend-construct"), exports);
|
|
18
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi91c2UtY2FzZXMvd2ViYXBwL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7QUFBQSx1REFBcUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL2Zyb250ZW5kLWNvbnN0cnVjdCc7Il19
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,266 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
// Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
|
|
3
|
+
// SPDX-License-Identifier: Apache-2.0
|
|
4
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
5
|
+
const fs = require("fs");
|
|
6
|
+
const path = require("path");
|
|
7
|
+
const aws_cdk_lib_1 = require("aws-cdk-lib");
|
|
8
|
+
const assertions_1 = require("aws-cdk-lib/assertions");
|
|
9
|
+
const aws_certificatemanager_1 = require("aws-cdk-lib/aws-certificatemanager");
|
|
10
|
+
const aws_route53_1 = require("aws-cdk-lib/aws-route53");
|
|
11
|
+
const cdk_nag_1 = require("cdk-nag");
|
|
12
|
+
const frontend_construct_1 = require("../frontend-construct");
|
|
13
|
+
// Create temporary build directory for tests
|
|
14
|
+
const testBuildDir = '/tmp/test-frontend-build-nag';
|
|
15
|
+
if (!fs.existsSync(testBuildDir)) {
|
|
16
|
+
fs.mkdirSync(testBuildDir, { recursive: true });
|
|
17
|
+
}
|
|
18
|
+
fs.writeFileSync(path.join(testBuildDir, 'index.html'), '<!DOCTYPE html><html><head><title>Test</title></head><body><h1>Test App</h1></body></html>');
|
|
19
|
+
// Create app and stack
|
|
20
|
+
const app = new aws_cdk_lib_1.App();
|
|
21
|
+
const stack = new aws_cdk_lib_1.Stack(app, 'TestStack', {
|
|
22
|
+
env: {
|
|
23
|
+
account: '123456789012',
|
|
24
|
+
region: 'us-east-1',
|
|
25
|
+
},
|
|
26
|
+
});
|
|
27
|
+
// Create SSL certificate for custom domain testing
|
|
28
|
+
const certificate = aws_certificatemanager_1.Certificate.fromCertificateArn(stack, 'Certificate', 'arn:aws:acm:us-east-1:123456789012:certificate/12345678-1234-1234-1234-123456789012');
|
|
29
|
+
// Create hosted zone for DNS testing
|
|
30
|
+
const hostedZone = aws_route53_1.HostedZone.fromHostedZoneAttributes(stack, 'HostedZone', {
|
|
31
|
+
hostedZoneId: 'Z123456789',
|
|
32
|
+
zoneName: 'example.com',
|
|
33
|
+
});
|
|
34
|
+
// Create the main Frontend construct
|
|
35
|
+
const frontend = new frontend_construct_1.Frontend(stack, 'TestFrontend', {
|
|
36
|
+
sourceDirectory: '/tmp/test-frontend-src',
|
|
37
|
+
buildOutputDirectory: testBuildDir,
|
|
38
|
+
customDomain: {
|
|
39
|
+
domainName: 'app.example.com',
|
|
40
|
+
certificate,
|
|
41
|
+
hostedZone,
|
|
42
|
+
},
|
|
43
|
+
skipBuild: true, // Skip build for testing
|
|
44
|
+
});
|
|
45
|
+
// Add CDK Nag suppressions for known acceptable violations
|
|
46
|
+
cdk_nag_1.NagSuppressions.addResourceSuppressions(stack, [
|
|
47
|
+
{
|
|
48
|
+
id: 'AwsSolutions-CFR1',
|
|
49
|
+
reason: 'CloudFront geo restrictions are configured based on application requirements',
|
|
50
|
+
},
|
|
51
|
+
{
|
|
52
|
+
id: 'AwsSolutions-CFR2',
|
|
53
|
+
reason: 'CloudFront WAF integration is configured based on security requirements',
|
|
54
|
+
},
|
|
55
|
+
{
|
|
56
|
+
id: 'AwsSolutions-CFR3',
|
|
57
|
+
reason: 'CloudFront access logging is configured based on compliance requirements',
|
|
58
|
+
},
|
|
59
|
+
{
|
|
60
|
+
id: 'AwsSolutions-CFR4',
|
|
61
|
+
reason: 'CloudFront viewer protocol policy is set to redirect-to-https for security',
|
|
62
|
+
},
|
|
63
|
+
{
|
|
64
|
+
id: 'AwsSolutions-S1',
|
|
65
|
+
reason: 'S3 bucket access logging is configured based on compliance requirements',
|
|
66
|
+
},
|
|
67
|
+
{
|
|
68
|
+
id: 'AwsSolutions-S2',
|
|
69
|
+
reason: 'S3 bucket public access is blocked and access is controlled via CloudFront OAC',
|
|
70
|
+
},
|
|
71
|
+
{
|
|
72
|
+
id: 'AwsSolutions-S3',
|
|
73
|
+
reason: 'S3 bucket SSL requests only policy is enforced via CloudFront HTTPS redirect',
|
|
74
|
+
},
|
|
75
|
+
{
|
|
76
|
+
id: 'AwsSolutions-S10',
|
|
77
|
+
reason: 'S3 bucket MFA delete is managed through organizational security policies',
|
|
78
|
+
},
|
|
79
|
+
{
|
|
80
|
+
id: 'AwsSolutions-IAM4',
|
|
81
|
+
reason: 'AWS managed policies are acceptable for standard Lambda execution roles',
|
|
82
|
+
appliesTo: ['Policy::arn:<AWS::Partition>:iam::aws:policy/service-role/AWSLambdaBasicExecutionRole'],
|
|
83
|
+
},
|
|
84
|
+
{
|
|
85
|
+
id: 'AwsSolutions-IAM5',
|
|
86
|
+
reason: 'BucketDeployment requires broad S3 permissions to manage deployment assets',
|
|
87
|
+
appliesTo: [
|
|
88
|
+
'Action::s3:GetObject*',
|
|
89
|
+
'Action::s3:GetBucket*',
|
|
90
|
+
'Action::s3:List*',
|
|
91
|
+
'Action::s3:DeleteObject*',
|
|
92
|
+
'Action::s3:Abort*',
|
|
93
|
+
'Resource::arn:<AWS::Partition>:s3:::cdk-hnb659fds-assets-<AWS::AccountId>-<AWS::Region>/*',
|
|
94
|
+
'Resource::arn:<AWS::Partition>:s3:::cdk-hnb659fds-assets-123456789012-us-east-1/*',
|
|
95
|
+
'Resource::<TestFrontendFrontendBucketD37D22DE.Arn>/*',
|
|
96
|
+
'Resource::*',
|
|
97
|
+
],
|
|
98
|
+
},
|
|
99
|
+
{
|
|
100
|
+
id: 'AwsSolutions-L1',
|
|
101
|
+
reason: 'Lambda runtime versions are managed at the application deployment level',
|
|
102
|
+
},
|
|
103
|
+
], true);
|
|
104
|
+
// Apply CDK Nag checks
|
|
105
|
+
aws_cdk_lib_1.Aspects.of(app).add(new cdk_nag_1.AwsSolutionsChecks({ verbose: true }));
|
|
106
|
+
// Synthesize the stack
|
|
107
|
+
assertions_1.Template.fromStack(stack);
|
|
108
|
+
// Check for unsuppressed warnings and errors
|
|
109
|
+
const warnings = assertions_1.Annotations.fromStack(stack).findWarning('*', assertions_1.Match.stringLikeRegexp('AwsSolutions-.*'));
|
|
110
|
+
const errors = assertions_1.Annotations.fromStack(stack).findError('*', assertions_1.Match.stringLikeRegexp('AwsSolutions-.*'));
|
|
111
|
+
// Test: Frontend construct is properly created and accessible
|
|
112
|
+
test('Frontend construct is created successfully', () => {
|
|
113
|
+
expect(frontend).toBeDefined();
|
|
114
|
+
expect(frontend.node.id).toBe('TestFrontend');
|
|
115
|
+
expect(frontend.bucket).toBeDefined();
|
|
116
|
+
expect(frontend.distribution).toBeDefined();
|
|
117
|
+
expect(frontend.bucketDeployment).toBeDefined();
|
|
118
|
+
});
|
|
119
|
+
// Test: Frontend construct has expected properties
|
|
120
|
+
test('Frontend construct has expected properties', () => {
|
|
121
|
+
expect(frontend.bucket.bucketName).toBeDefined();
|
|
122
|
+
expect(frontend.distribution.distributionId).toBeDefined();
|
|
123
|
+
expect(frontend.domainName).toBe('app.example.com');
|
|
124
|
+
expect(frontend.url()).toBe('https://app.example.com');
|
|
125
|
+
expect(typeof frontend.bucketName()).toBe('string');
|
|
126
|
+
expect(typeof frontend.distributionDomainName()).toBe('string');
|
|
127
|
+
});
|
|
128
|
+
// Test: Template contains expected frontend resources
|
|
129
|
+
test('Template contains expected frontend resources', () => {
|
|
130
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
131
|
+
// Verify S3 bucket exists with security configuration
|
|
132
|
+
template.hasResourceProperties('AWS::S3::Bucket', {
|
|
133
|
+
BucketEncryption: {
|
|
134
|
+
ServerSideEncryptionConfiguration: [
|
|
135
|
+
{
|
|
136
|
+
ServerSideEncryptionByDefault: {
|
|
137
|
+
SSEAlgorithm: 'AES256',
|
|
138
|
+
},
|
|
139
|
+
},
|
|
140
|
+
],
|
|
141
|
+
},
|
|
142
|
+
PublicAccessBlockConfiguration: {
|
|
143
|
+
BlockPublicAcls: true,
|
|
144
|
+
BlockPublicPolicy: true,
|
|
145
|
+
IgnorePublicAcls: true,
|
|
146
|
+
RestrictPublicBuckets: true,
|
|
147
|
+
},
|
|
148
|
+
});
|
|
149
|
+
// Verify CloudFront distribution exists with security configuration
|
|
150
|
+
template.hasResourceProperties('AWS::CloudFront::Distribution', {
|
|
151
|
+
DistributionConfig: {
|
|
152
|
+
Aliases: ['app.example.com'],
|
|
153
|
+
DefaultRootObject: 'index.html',
|
|
154
|
+
DefaultCacheBehavior: {
|
|
155
|
+
ViewerProtocolPolicy: 'redirect-to-https',
|
|
156
|
+
},
|
|
157
|
+
CustomErrorResponses: [
|
|
158
|
+
{
|
|
159
|
+
ErrorCode: 403,
|
|
160
|
+
ResponseCode: 200,
|
|
161
|
+
ResponsePagePath: '/index.html',
|
|
162
|
+
},
|
|
163
|
+
{
|
|
164
|
+
ErrorCode: 404,
|
|
165
|
+
ResponseCode: 200,
|
|
166
|
+
ResponsePagePath: '/index.html',
|
|
167
|
+
},
|
|
168
|
+
],
|
|
169
|
+
ViewerCertificate: {
|
|
170
|
+
AcmCertificateArn: 'arn:aws:acm:us-east-1:123456789012:certificate/12345678-1234-1234-1234-123456789012',
|
|
171
|
+
SslSupportMethod: 'sni-only',
|
|
172
|
+
},
|
|
173
|
+
},
|
|
174
|
+
});
|
|
175
|
+
// Verify CloudFront security headers function exists
|
|
176
|
+
template.hasResourceProperties('AWS::CloudFront::Function', {
|
|
177
|
+
FunctionConfig: {
|
|
178
|
+
Runtime: 'cloudfront-js-1.0',
|
|
179
|
+
},
|
|
180
|
+
});
|
|
181
|
+
// Verify Route53 A record exists for custom domain
|
|
182
|
+
template.hasResourceProperties('AWS::Route53::RecordSet', {
|
|
183
|
+
Type: 'A',
|
|
184
|
+
Name: 'app.example.com.',
|
|
185
|
+
HostedZoneId: 'Z123456789',
|
|
186
|
+
});
|
|
187
|
+
// Verify bucket deployment exists
|
|
188
|
+
template.hasResource('Custom::CDKBucketDeployment', {});
|
|
189
|
+
// Verify auto delete objects custom resource exists
|
|
190
|
+
template.hasResource('Custom::S3AutoDeleteObjects', {});
|
|
191
|
+
});
|
|
192
|
+
// Test: Frontend construct enforces security best practices
|
|
193
|
+
test('Frontend construct enforces security best practices', () => {
|
|
194
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
195
|
+
// Verify HTTPS redirect is enforced
|
|
196
|
+
template.hasResourceProperties('AWS::CloudFront::Distribution', {
|
|
197
|
+
DistributionConfig: {
|
|
198
|
+
DefaultCacheBehavior: {
|
|
199
|
+
ViewerProtocolPolicy: 'redirect-to-https',
|
|
200
|
+
},
|
|
201
|
+
},
|
|
202
|
+
});
|
|
203
|
+
// Verify S3 bucket blocks public access
|
|
204
|
+
template.hasResourceProperties('AWS::S3::Bucket', {
|
|
205
|
+
PublicAccessBlockConfiguration: {
|
|
206
|
+
BlockPublicAcls: true,
|
|
207
|
+
BlockPublicPolicy: true,
|
|
208
|
+
IgnorePublicAcls: true,
|
|
209
|
+
RestrictPublicBuckets: true,
|
|
210
|
+
},
|
|
211
|
+
});
|
|
212
|
+
// Verify S3 bucket has encryption
|
|
213
|
+
template.hasResourceProperties('AWS::S3::Bucket', {
|
|
214
|
+
BucketEncryption: {
|
|
215
|
+
ServerSideEncryptionConfiguration: [
|
|
216
|
+
{
|
|
217
|
+
ServerSideEncryptionByDefault: {
|
|
218
|
+
SSEAlgorithm: 'AES256',
|
|
219
|
+
},
|
|
220
|
+
},
|
|
221
|
+
],
|
|
222
|
+
},
|
|
223
|
+
});
|
|
224
|
+
});
|
|
225
|
+
// Test: Frontend construct supports SPA applications
|
|
226
|
+
test('Frontend construct supports SPA applications', () => {
|
|
227
|
+
const template = assertions_1.Template.fromStack(stack);
|
|
228
|
+
// Verify SPA-friendly error responses
|
|
229
|
+
template.hasResourceProperties('AWS::CloudFront::Distribution', {
|
|
230
|
+
DistributionConfig: {
|
|
231
|
+
CustomErrorResponses: [
|
|
232
|
+
{
|
|
233
|
+
ErrorCode: 403,
|
|
234
|
+
ResponseCode: 200,
|
|
235
|
+
ResponsePagePath: '/index.html',
|
|
236
|
+
},
|
|
237
|
+
{
|
|
238
|
+
ErrorCode: 404,
|
|
239
|
+
ResponseCode: 200,
|
|
240
|
+
ResponsePagePath: '/index.html',
|
|
241
|
+
},
|
|
242
|
+
],
|
|
243
|
+
},
|
|
244
|
+
});
|
|
245
|
+
// Verify default root object is set
|
|
246
|
+
template.hasResourceProperties('AWS::CloudFront::Distribution', {
|
|
247
|
+
DistributionConfig: {
|
|
248
|
+
DefaultRootObject: 'index.html',
|
|
249
|
+
},
|
|
250
|
+
});
|
|
251
|
+
});
|
|
252
|
+
// Test: No unsuppressed warnings
|
|
253
|
+
test('No unsuppressed warnings', () => {
|
|
254
|
+
if (warnings.length > 0) {
|
|
255
|
+
console.log('CDK Nag Warnings:', JSON.stringify(warnings, null, 2));
|
|
256
|
+
}
|
|
257
|
+
expect(warnings).toHaveLength(0);
|
|
258
|
+
});
|
|
259
|
+
// Test: No unsuppressed errors
|
|
260
|
+
test('No unsuppressed errors', () => {
|
|
261
|
+
if (errors.length > 0) {
|
|
262
|
+
console.log('CDK Nag Errors:', JSON.stringify(errors, null, 2));
|
|
263
|
+
}
|
|
264
|
+
expect(errors).toHaveLength(0);
|
|
265
|
+
});
|
|
266
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"frontend-construct-nag.test.js","sourceRoot":"","sources":["../../../use-cases/webapp/tests/frontend-construct-nag.test.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.\n// SPDX-License-Identifier: Apache-2.0\n\nimport * as fs from 'fs';\nimport * as path from 'path';\nimport { App, Stack, Aspects } from 'aws-cdk-lib';\nimport { Template, Annotations, Match } from 'aws-cdk-lib/assertions';\nimport { Certificate } from 'aws-cdk-lib/aws-certificatemanager';\nimport { HostedZone } from 'aws-cdk-lib/aws-route53';\nimport { AwsSolutionsChecks, NagSuppressions } from 'cdk-nag';\nimport { Frontend } from '../frontend-construct';\n\n// Create temporary build directory for tests\nconst testBuildDir = '/tmp/test-frontend-build-nag';\nif (!fs.existsSync(testBuildDir)) {\n  fs.mkdirSync(testBuildDir, { recursive: true });\n}\nfs.writeFileSync(path.join(testBuildDir, 'index.html'), '<!DOCTYPE html><html><head><title>Test</title></head><body><h1>Test App</h1></body></html>');\n\n// Create app and stack\nconst app = new App();\nconst stack = new Stack(app, 'TestStack', {\n  env: {\n    account: '123456789012',\n    region: 'us-east-1',\n  },\n});\n\n// Create SSL certificate for custom domain testing\nconst certificate = Certificate.fromCertificateArn(\n  stack,\n  'Certificate',\n  'arn:aws:acm:us-east-1:123456789012:certificate/12345678-1234-1234-1234-123456789012',\n);\n\n// Create hosted zone for DNS testing\nconst hostedZone = HostedZone.fromHostedZoneAttributes(stack, 'HostedZone', {\n  hostedZoneId: 'Z123456789',\n  zoneName: 'example.com',\n});\n\n// Create the main Frontend construct\nconst frontend = new Frontend(stack, 'TestFrontend', {\n  sourceDirectory: '/tmp/test-frontend-src',\n  buildOutputDirectory: testBuildDir,\n  customDomain: {\n    domainName: 'app.example.com',\n    certificate,\n    hostedZone,\n  },\n  skipBuild: true, // Skip build for testing\n});\n\n// Add CDK Nag suppressions for known acceptable violations\nNagSuppressions.addResourceSuppressions(stack, [\n  {\n    id: 'AwsSolutions-CFR1',\n    reason: 'CloudFront geo restrictions are configured based on application requirements',\n  },\n  {\n    id: 'AwsSolutions-CFR2',\n    reason: 'CloudFront WAF integration is configured based on security requirements',\n  },\n  {\n    id: 'AwsSolutions-CFR3',\n    reason: 'CloudFront access logging is configured based on compliance requirements',\n  },\n  {\n    id: 'AwsSolutions-CFR4',\n    reason: 'CloudFront viewer protocol policy is set to redirect-to-https for security',\n  },\n  {\n    id: 'AwsSolutions-S1',\n    reason: 'S3 bucket access logging is configured based on compliance requirements',\n  },\n  {\n    id: 'AwsSolutions-S2',\n    reason: 'S3 bucket public access is blocked and access is controlled via CloudFront OAC',\n  },\n  {\n    id: 'AwsSolutions-S3',\n    reason: 'S3 bucket SSL requests only policy is enforced via CloudFront HTTPS redirect',\n  },\n  {\n    id: 'AwsSolutions-S10',\n    reason: 'S3 bucket MFA delete is managed through organizational security policies',\n  },\n  {\n    id: 'AwsSolutions-IAM4',\n    reason: 'AWS managed policies are acceptable for standard Lambda execution roles',\n    appliesTo: ['Policy::arn:<AWS::Partition>:iam::aws:policy/service-role/AWSLambdaBasicExecutionRole'],\n  },\n  {\n    id: 'AwsSolutions-IAM5',\n    reason: 'BucketDeployment requires broad S3 permissions to manage deployment assets',\n    appliesTo: [\n      'Action::s3:GetObject*',\n      'Action::s3:GetBucket*',\n      'Action::s3:List*',\n      'Action::s3:DeleteObject*',\n      'Action::s3:Abort*',\n      'Resource::arn:<AWS::Partition>:s3:::cdk-hnb659fds-assets-<AWS::AccountId>-<AWS::Region>/*',\n      'Resource::arn:<AWS::Partition>:s3:::cdk-hnb659fds-assets-123456789012-us-east-1/*',\n      'Resource::<TestFrontendFrontendBucketD37D22DE.Arn>/*',\n      'Resource::*',\n    ],\n  },\n  {\n    id: 'AwsSolutions-L1',\n    reason: 'Lambda runtime versions are managed at the application deployment level',\n  },\n], true);\n\n// Apply CDK Nag checks\nAspects.of(app).add(new AwsSolutionsChecks({ verbose: true }));\n\n// Synthesize the stack\nTemplate.fromStack(stack);\n\n// Check for unsuppressed warnings and errors\nconst warnings = Annotations.fromStack(stack).findWarning('*', Match.stringLikeRegexp('AwsSolutions-.*'));\nconst errors = Annotations.fromStack(stack).findError('*', Match.stringLikeRegexp('AwsSolutions-.*'));\n\n// Test: Frontend construct is properly created and accessible\ntest('Frontend construct is created successfully', () => {\n  expect(frontend).toBeDefined();\n  expect(frontend.node.id).toBe('TestFrontend');\n  expect(frontend.bucket).toBeDefined();\n  expect(frontend.distribution).toBeDefined();\n  expect(frontend.bucketDeployment).toBeDefined();\n});\n\n// Test: Frontend construct has expected properties\ntest('Frontend construct has expected properties', () => {\n  expect(frontend.bucket.bucketName).toBeDefined();\n  expect(frontend.distribution.distributionId).toBeDefined();\n  expect(frontend.domainName).toBe('app.example.com');\n  expect(frontend.url()).toBe('https://app.example.com');\n  expect(typeof frontend.bucketName()).toBe('string');\n  expect(typeof frontend.distributionDomainName()).toBe('string');\n});\n\n// Test: Template contains expected frontend resources\ntest('Template contains expected frontend resources', () => {\n  const template = Template.fromStack(stack);\n\n  // Verify S3 bucket exists with security configuration\n  template.hasResourceProperties('AWS::S3::Bucket', {\n    BucketEncryption: {\n      ServerSideEncryptionConfiguration: [\n        {\n          ServerSideEncryptionByDefault: {\n            SSEAlgorithm: 'AES256',\n          },\n        },\n      ],\n    },\n    PublicAccessBlockConfiguration: {\n      BlockPublicAcls: true,\n      BlockPublicPolicy: true,\n      IgnorePublicAcls: true,\n      RestrictPublicBuckets: true,\n    },\n  });\n\n  // Verify CloudFront distribution exists with security configuration\n  template.hasResourceProperties('AWS::CloudFront::Distribution', {\n    DistributionConfig: {\n      Aliases: ['app.example.com'],\n      DefaultRootObject: 'index.html',\n      DefaultCacheBehavior: {\n        ViewerProtocolPolicy: 'redirect-to-https',\n      },\n      CustomErrorResponses: [\n        {\n          ErrorCode: 403,\n          ResponseCode: 200,\n          ResponsePagePath: '/index.html',\n        },\n        {\n          ErrorCode: 404,\n          ResponseCode: 200,\n          ResponsePagePath: '/index.html',\n        },\n      ],\n      ViewerCertificate: {\n        AcmCertificateArn: 'arn:aws:acm:us-east-1:123456789012:certificate/12345678-1234-1234-1234-123456789012',\n        SslSupportMethod: 'sni-only',\n      },\n    },\n  });\n\n  // Verify CloudFront security headers function exists\n  template.hasResourceProperties('AWS::CloudFront::Function', {\n    FunctionConfig: {\n      Runtime: 'cloudfront-js-1.0',\n    },\n  });\n\n  // Verify Route53 A record exists for custom domain\n  template.hasResourceProperties('AWS::Route53::RecordSet', {\n    Type: 'A',\n    Name: 'app.example.com.',\n    HostedZoneId: 'Z123456789',\n  });\n\n  // Verify bucket deployment exists\n  template.hasResource('Custom::CDKBucketDeployment', {});\n\n  // Verify auto delete objects custom resource exists\n  template.hasResource('Custom::S3AutoDeleteObjects', {});\n});\n\n// Test: Frontend construct enforces security best practices\ntest('Frontend construct enforces security best practices', () => {\n  const template = Template.fromStack(stack);\n\n  // Verify HTTPS redirect is enforced\n  template.hasResourceProperties('AWS::CloudFront::Distribution', {\n    DistributionConfig: {\n      DefaultCacheBehavior: {\n        ViewerProtocolPolicy: 'redirect-to-https',\n      },\n    },\n  });\n\n  // Verify S3 bucket blocks public access\n  template.hasResourceProperties('AWS::S3::Bucket', {\n    PublicAccessBlockConfiguration: {\n      BlockPublicAcls: true,\n      BlockPublicPolicy: true,\n      IgnorePublicAcls: true,\n      RestrictPublicBuckets: true,\n    },\n  });\n\n  // Verify S3 bucket has encryption\n  template.hasResourceProperties('AWS::S3::Bucket', {\n    BucketEncryption: {\n      ServerSideEncryptionConfiguration: [\n        {\n          ServerSideEncryptionByDefault: {\n            SSEAlgorithm: 'AES256',\n          },\n        },\n      ],\n    },\n  });\n});\n\n// Test: Frontend construct supports SPA applications\ntest('Frontend construct supports SPA applications', () => {\n  const template = Template.fromStack(stack);\n\n  // Verify SPA-friendly error responses\n  template.hasResourceProperties('AWS::CloudFront::Distribution', {\n    DistributionConfig: {\n      CustomErrorResponses: [\n        {\n          ErrorCode: 403,\n          ResponseCode: 200,\n          ResponsePagePath: '/index.html',\n        },\n        {\n          ErrorCode: 404,\n          ResponseCode: 200,\n          ResponsePagePath: '/index.html',\n        },\n      ],\n    },\n  });\n\n  // Verify default root object is set\n  template.hasResourceProperties('AWS::CloudFront::Distribution', {\n    DistributionConfig: {\n      DefaultRootObject: 'index.html',\n    },\n  });\n});\n\n// Test: No unsuppressed warnings\ntest('No unsuppressed warnings', () => {\n  if (warnings.length > 0) {\n    console.log('CDK Nag Warnings:', JSON.stringify(warnings, null, 2));\n  }\n  expect(warnings).toHaveLength(0);\n});\n\n// Test: No unsuppressed errors\ntest('No unsuppressed errors', () => {\n  if (errors.length > 0) {\n    console.log('CDK Nag Errors:', JSON.stringify(errors, null, 2));\n  }\n  expect(errors).toHaveLength(0);\n});\n"]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|