@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.
Files changed (105) hide show
  1. package/.jsii +8644 -0
  2. package/LICENSE +202 -0
  3. package/README.md +212 -0
  4. package/lib/document-processing/agentic-document-processing.d.ts +16 -0
  5. package/lib/document-processing/agentic-document-processing.js +90 -0
  6. package/lib/document-processing/base-document-processing.d.ts +189 -0
  7. package/lib/document-processing/base-document-processing.js +509 -0
  8. package/lib/document-processing/bedrock-document-processing.d.ts +167 -0
  9. package/lib/document-processing/bedrock-document-processing.js +297 -0
  10. package/lib/document-processing/index.d.ts +3 -0
  11. package/lib/document-processing/index.js +20 -0
  12. package/lib/document-processing/resources/default-bedrock-invoke/index.py +63 -0
  13. package/lib/document-processing/resources/default-bedrock-invoke/requirements.txt +4 -0
  14. package/lib/document-processing/resources/default-doc-retrieval-lambda/index.mjs +92 -0
  15. package/lib/document-processing/resources/default-doc-retrieval-lambda/package.json +10 -0
  16. package/lib/document-processing/resources/default-error-handler/index.js +46 -0
  17. package/lib/document-processing/resources/default-error-handler/package.json +4 -0
  18. package/lib/document-processing/resources/default-image-processor/classifier.mjs +665 -0
  19. package/lib/document-processing/resources/default-image-processor/extractors.mjs +465 -0
  20. package/lib/document-processing/resources/default-image-processor/index.mjs +143 -0
  21. package/lib/document-processing/resources/default-image-processor/package-lock.json +12 -0
  22. package/lib/document-processing/resources/default-image-processor/package.json +4 -0
  23. package/lib/document-processing/resources/default-image-validator/index.mjs +76 -0
  24. package/lib/document-processing/resources/default-image-validator/package-lock.json +154 -0
  25. package/lib/document-processing/resources/default-image-validator/package.json +7 -0
  26. package/lib/document-processing/resources/default-pdf-processor/index.js +46 -0
  27. package/lib/document-processing/resources/default-pdf-validator/index.js +36 -0
  28. package/lib/document-processing/resources/default-sqs-consumer/index.py +111 -0
  29. package/lib/document-processing/resources/default-sqs-consumer/requirements.txt +4 -0
  30. package/lib/document-processing/resources/default-sqs-consumer/sample_payload.json +20 -0
  31. package/lib/document-processing/resources/default-sqs-consumer/sample_payload_multi.json +24 -0
  32. package/lib/document-processing/resources/default-strands-agent/index.py +111 -0
  33. package/lib/document-processing/resources/default-strands-agent/requirements.txt +6 -0
  34. package/lib/document-processing/tests/agentic-document-processing-nag.test.d.ts +1 -0
  35. package/lib/document-processing/tests/agentic-document-processing-nag.test.js +107 -0
  36. package/lib/document-processing/tests/agentic-document-processing.test.d.ts +1 -0
  37. package/lib/document-processing/tests/agentic-document-processing.test.js +125 -0
  38. package/lib/document-processing/tests/bedrock-document-processing-nag.test.d.ts +1 -0
  39. package/lib/document-processing/tests/bedrock-document-processing-nag.test.js +101 -0
  40. package/lib/document-processing/tests/bedrock-document-processing.test.d.ts +1 -0
  41. package/lib/document-processing/tests/bedrock-document-processing.test.js +79 -0
  42. package/lib/framework/custom-resource/default-runtimes.d.ts +21 -0
  43. package/lib/framework/custom-resource/default-runtimes.js +34 -0
  44. package/lib/framework/custom-resource/index.d.ts +1 -0
  45. package/lib/framework/custom-resource/index.js +18 -0
  46. package/lib/framework/foundation/access-log.d.ts +69 -0
  47. package/lib/framework/foundation/access-log.js +121 -0
  48. package/lib/framework/foundation/eventbridge-broker.d.ts +18 -0
  49. package/lib/framework/foundation/eventbridge-broker.js +42 -0
  50. package/lib/framework/foundation/index.d.ts +3 -0
  51. package/lib/framework/foundation/index.js +20 -0
  52. package/lib/framework/foundation/network.d.ts +19 -0
  53. package/lib/framework/foundation/network.js +83 -0
  54. package/lib/framework/index.d.ts +2 -0
  55. package/lib/framework/index.js +19 -0
  56. package/lib/framework/quickstart/base-quickstart.d.ts +30 -0
  57. package/lib/framework/quickstart/base-quickstart.js +30 -0
  58. package/lib/index.d.ts +4 -0
  59. package/lib/index.js +21 -0
  60. package/lib/tsconfig.tsbuildinfo +1 -0
  61. package/lib/utilities/cdk-nag-config.d.ts +42 -0
  62. package/lib/utilities/cdk-nag-config.js +194 -0
  63. package/lib/utilities/data-loader-lambda/index.py +282 -0
  64. package/lib/utilities/data-loader-lambda/requirements.txt +3 -0
  65. package/lib/utilities/data-loader.d.ts +173 -0
  66. package/lib/utilities/data-loader.js +447 -0
  67. package/lib/utilities/index.d.ts +3 -0
  68. package/lib/utilities/index.js +20 -0
  69. package/lib/utilities/lambda-iam-utils.d.ts +145 -0
  70. package/lib/utilities/lambda-iam-utils.js +235 -0
  71. package/lib/utilities/lambda_layers/data-masking/layer-construct.d.ts +42 -0
  72. package/lib/utilities/lambda_layers/data-masking/layer-construct.js +53 -0
  73. package/lib/utilities/lambda_layers/data-masking/layer-construct.ts +88 -0
  74. package/lib/utilities/observability/bedrock-observability.d.ts +18 -0
  75. package/lib/utilities/observability/bedrock-observability.js +131 -0
  76. package/lib/utilities/observability/cloudfront-distribution-observability-property-injector.d.ts +6 -0
  77. package/lib/utilities/observability/cloudfront-distribution-observability-property-injector.js +22 -0
  78. package/lib/utilities/observability/index.d.ts +6 -0
  79. package/lib/utilities/observability/index.js +25 -0
  80. package/lib/utilities/observability/lambda-observability-property-injector.d.ts +8 -0
  81. package/lib/utilities/observability/lambda-observability-property-injector.js +43 -0
  82. package/lib/utilities/observability/log-group-data-protection-props.d.ts +19 -0
  83. package/lib/utilities/observability/log-group-data-protection-props.js +5 -0
  84. package/lib/utilities/observability/observability.d.ts +83 -0
  85. package/lib/utilities/observability/observability.js +278 -0
  86. package/lib/utilities/observability/observable.d.ts +32 -0
  87. package/lib/utilities/observability/observable.js +3 -0
  88. package/lib/utilities/observability/powertools-config.d.ts +3 -0
  89. package/lib/utilities/observability/powertools-config.js +25 -0
  90. package/lib/utilities/observability/resources/bedrock-manage-logging-configuration/index.py +27 -0
  91. package/lib/utilities/observability/state-machine-observability-property-injector.d.ts +8 -0
  92. package/lib/utilities/observability/state-machine-observability-property-injector.js +49 -0
  93. package/lib/utilities/tests/data-loader-nag.test.d.ts +1 -0
  94. package/lib/utilities/tests/data-loader-nag.test.js +432 -0
  95. package/lib/utilities/tests/data-loader.test.d.ts +1 -0
  96. package/lib/utilities/tests/data-loader.test.js +284 -0
  97. package/lib/webapp/frontend-construct.d.ts +136 -0
  98. package/lib/webapp/frontend-construct.js +253 -0
  99. package/lib/webapp/index.d.ts +1 -0
  100. package/lib/webapp/index.js +18 -0
  101. package/lib/webapp/tests/frontend-construct-nag.test.d.ts +1 -0
  102. package/lib/webapp/tests/frontend-construct-nag.test.js +266 -0
  103. package/lib/webapp/tests/frontend-construct.test.d.ts +1 -0
  104. package/lib/webapp/tests/frontend-construct.test.js +385 -0
  105. 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 {};