@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,385 @@
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_cloudfront_1 = require("aws-cdk-lib/aws-cloudfront");
11
+ const aws_route53_1 = require("aws-cdk-lib/aws-route53");
12
+ const aws_s3_1 = require("aws-cdk-lib/aws-s3");
13
+ const aws_s3_deployment_1 = require("aws-cdk-lib/aws-s3-deployment");
14
+ const frontend_construct_1 = require("../frontend-construct");
15
+ // Mock execSync to avoid actual build execution during tests
16
+ jest.mock('child_process', () => ({
17
+ execSync: jest.fn(),
18
+ }));
19
+ describe('@webapp Frontend', () => {
20
+ let app;
21
+ let stack;
22
+ let template;
23
+ let testBuildDir;
24
+ let testSrcDir;
25
+ beforeAll(() => {
26
+ // Create a temporary build directory for tests
27
+ testBuildDir = '/tmp/test-frontend-build';
28
+ if (!fs.existsSync(testBuildDir)) {
29
+ fs.mkdirSync(testBuildDir, { recursive: true });
30
+ }
31
+ fs.writeFileSync(path.join(testBuildDir, 'index.html'), '<!DOCTYPE html><html><head><title>Test</title></head><body><h1>Test App</h1></body></html>');
32
+ // Create a temporary source directory for tests
33
+ testSrcDir = '/tmp/test-frontend-src';
34
+ if (!fs.existsSync(testSrcDir)) {
35
+ fs.mkdirSync(testSrcDir, { recursive: true });
36
+ }
37
+ // Create default build directory inside source directory
38
+ const defaultBuildDir = path.join(testSrcDir, 'build');
39
+ if (!fs.existsSync(defaultBuildDir)) {
40
+ fs.mkdirSync(defaultBuildDir, { recursive: true });
41
+ }
42
+ fs.writeFileSync(path.join(defaultBuildDir, 'index.html'), '<!DOCTYPE html><html><head><title>Default Test</title></head><body><h1>Default Test App</h1></body></html>');
43
+ });
44
+ beforeEach(() => {
45
+ app = new aws_cdk_lib_1.App();
46
+ stack = new aws_cdk_lib_1.Stack(app, 'TestStack');
47
+ });
48
+ describe('@webapp Basic functionality', () => {
49
+ test('@webapp creates frontend construct with minimal configuration', () => {
50
+ const frontend = new frontend_construct_1.Frontend(stack, 'Frontend', {
51
+ sourceDirectory: '/tmp/test-frontend-src',
52
+ buildOutputDirectory: testBuildDir,
53
+ skipBuild: true, // Skip build for testing
54
+ });
55
+ template = assertions_1.Template.fromStack(stack);
56
+ // Verify S3 bucket is created
57
+ template.hasResourceProperties('AWS::S3::Bucket', {
58
+ BucketEncryption: {
59
+ ServerSideEncryptionConfiguration: [
60
+ {
61
+ ServerSideEncryptionByDefault: {
62
+ SSEAlgorithm: 'AES256',
63
+ },
64
+ },
65
+ ],
66
+ },
67
+ PublicAccessBlockConfiguration: {
68
+ BlockPublicAcls: true,
69
+ BlockPublicPolicy: true,
70
+ IgnorePublicAcls: true,
71
+ RestrictPublicBuckets: true,
72
+ },
73
+ });
74
+ // Verify CloudFront distribution is created
75
+ template.hasResourceProperties('AWS::CloudFront::Distribution', {
76
+ DistributionConfig: {
77
+ DefaultRootObject: 'index.html',
78
+ CustomErrorResponses: [
79
+ {
80
+ ErrorCode: 403,
81
+ ResponseCode: 200,
82
+ ResponsePagePath: '/index.html',
83
+ },
84
+ {
85
+ ErrorCode: 404,
86
+ ResponseCode: 200,
87
+ ResponsePagePath: '/index.html',
88
+ },
89
+ ],
90
+ },
91
+ });
92
+ // Verify security headers function is created
93
+ template.hasResourceProperties('AWS::CloudFront::Function', {
94
+ FunctionConfig: {
95
+ Runtime: 'cloudfront-js-1.0',
96
+ },
97
+ });
98
+ // Verify bucket deployment is created
99
+ template.hasResource('Custom::CDKBucketDeployment', {});
100
+ // Test public methods
101
+ expect(frontend.bucket).toBeInstanceOf(aws_s3_1.Bucket);
102
+ expect(frontend.distribution).toBeInstanceOf(aws_cloudfront_1.Distribution);
103
+ expect(frontend.bucketDeployment).toBeInstanceOf(aws_s3_deployment_1.BucketDeployment);
104
+ expect(typeof frontend.distributionDomainName()).toBe('string');
105
+ expect(typeof frontend.bucketName()).toBe('string');
106
+ expect(frontend.url()).toContain('https://');
107
+ });
108
+ test('@webapp creates frontend construct with custom build command', () => {
109
+ new frontend_construct_1.Frontend(stack, 'Frontend', {
110
+ sourceDirectory: '/tmp/test-frontend-src',
111
+ buildOutputDirectory: testBuildDir,
112
+ buildCommand: 'yarn build',
113
+ skipBuild: true,
114
+ });
115
+ template = assertions_1.Template.fromStack(stack);
116
+ // Should still create the same resources
117
+ template.resourceCountIs('AWS::S3::Bucket', 1);
118
+ template.resourceCountIs('AWS::CloudFront::Distribution', 1);
119
+ });
120
+ test('@webapp creates frontend construct with custom error responses', () => {
121
+ const customErrorResponses = [
122
+ {
123
+ httpStatus: 500,
124
+ responseHttpStatus: 200,
125
+ responsePagePath: '/error.html',
126
+ },
127
+ ];
128
+ new frontend_construct_1.Frontend(stack, 'Frontend', {
129
+ sourceDirectory: '/tmp/test-frontend-src',
130
+ buildOutputDirectory: testBuildDir,
131
+ errorResponses: customErrorResponses,
132
+ skipBuild: true,
133
+ });
134
+ template = assertions_1.Template.fromStack(stack);
135
+ template.hasResourceProperties('AWS::CloudFront::Distribution', {
136
+ DistributionConfig: {
137
+ CustomErrorResponses: [
138
+ {
139
+ ErrorCode: 500,
140
+ ResponseCode: 200,
141
+ ResponsePagePath: '/error.html',
142
+ },
143
+ ],
144
+ },
145
+ });
146
+ });
147
+ });
148
+ describe('@webapp Custom domain functionality', () => {
149
+ test('@webapp creates frontend construct with custom domain', () => {
150
+ const certificate = aws_certificatemanager_1.Certificate.fromCertificateArn(stack, 'Certificate', 'arn:aws:acm:us-east-1:123456789012:certificate/12345678-1234-1234-1234-123456789012');
151
+ const frontend = new frontend_construct_1.Frontend(stack, 'Frontend', {
152
+ sourceDirectory: '/tmp/test-frontend-src',
153
+ buildOutputDirectory: testBuildDir,
154
+ customDomain: {
155
+ domainName: 'app.example.com',
156
+ certificate,
157
+ },
158
+ skipBuild: true,
159
+ });
160
+ template = assertions_1.Template.fromStack(stack);
161
+ // Verify CloudFront distribution has custom domain
162
+ template.hasResourceProperties('AWS::CloudFront::Distribution', {
163
+ DistributionConfig: {
164
+ Aliases: ['app.example.com'],
165
+ ViewerCertificate: {
166
+ AcmCertificateArn: 'arn:aws:acm:us-east-1:123456789012:certificate/12345678-1234-1234-1234-123456789012',
167
+ SslSupportMethod: 'sni-only',
168
+ },
169
+ },
170
+ });
171
+ expect(frontend.domainName).toBe('app.example.com');
172
+ expect(frontend.url()).toBe('https://app.example.com');
173
+ });
174
+ test('@webapp creates frontend construct with custom domain and hosted zone', () => {
175
+ const certificate = aws_certificatemanager_1.Certificate.fromCertificateArn(stack, 'Certificate', 'arn:aws:acm:us-east-1:123456789012:certificate/12345678-1234-1234-1234-123456789012');
176
+ const hostedZone = aws_route53_1.HostedZone.fromHostedZoneAttributes(stack, 'HostedZone', {
177
+ hostedZoneId: 'Z123456789',
178
+ zoneName: 'example.com',
179
+ });
180
+ new frontend_construct_1.Frontend(stack, 'Frontend', {
181
+ sourceDirectory: '/tmp/test-frontend-src',
182
+ buildOutputDirectory: testBuildDir,
183
+ customDomain: {
184
+ domainName: 'app.example.com',
185
+ certificate,
186
+ hostedZone,
187
+ },
188
+ skipBuild: true,
189
+ });
190
+ template = assertions_1.Template.fromStack(stack);
191
+ // Verify Route53 A record is created
192
+ template.hasResourceProperties('AWS::Route53::RecordSet', {
193
+ Type: 'A',
194
+ Name: 'app.example.com.',
195
+ HostedZoneId: 'Z123456789',
196
+ });
197
+ });
198
+ });
199
+ describe('@webapp Validation', () => {
200
+ test('@webapp throws error when sourceDirectory is missing', () => {
201
+ expect(() => {
202
+ new frontend_construct_1.Frontend(stack, 'Frontend', {
203
+ sourceDirectory: '',
204
+ buildOutputDirectory: testBuildDir,
205
+ skipBuild: true,
206
+ });
207
+ }).toThrow('sourceDirectory is required');
208
+ });
209
+ test('@webapp uses default buildOutputDirectory when not provided', () => {
210
+ // Use the existing test build directory
211
+ const frontend = new frontend_construct_1.Frontend(stack, 'Frontend', {
212
+ sourceDirectory: testSrcDir,
213
+ // buildOutputDirectory not provided - should use default './build/'
214
+ skipBuild: true,
215
+ });
216
+ expect(frontend).toBeDefined();
217
+ // The construct should be created successfully with default buildOutputDirectory
218
+ });
219
+ test('@webapp applies custom removal policy to resources', () => {
220
+ const retainStack = new aws_cdk_lib_1.Stack(app, 'RetainTestStack');
221
+ const frontend = new frontend_construct_1.Frontend(retainStack, 'Frontend', {
222
+ sourceDirectory: testSrcDir,
223
+ buildOutputDirectory: testBuildDir,
224
+ skipBuild: true,
225
+ removalPolicy: aws_cdk_lib_1.RemovalPolicy.RETAIN,
226
+ });
227
+ expect(frontend).toBeDefined();
228
+ // Check that the bucket has the correct removal policy
229
+ const retainTemplate = assertions_1.Template.fromStack(retainStack);
230
+ retainTemplate.hasResource('AWS::S3::Bucket', {
231
+ DeletionPolicy: 'Retain',
232
+ UpdateReplacePolicy: 'Retain',
233
+ });
234
+ });
235
+ test('@webapp uses default DESTROY removal policy when not specified', () => {
236
+ const destroyStack = new aws_cdk_lib_1.Stack(app, 'DestroyTestStack');
237
+ const frontend = new frontend_construct_1.Frontend(destroyStack, 'Frontend', {
238
+ sourceDirectory: testSrcDir,
239
+ buildOutputDirectory: testBuildDir,
240
+ skipBuild: true,
241
+ // removalPolicy not specified - should default to DESTROY
242
+ });
243
+ expect(frontend).toBeDefined();
244
+ // Check that the bucket has the default DESTROY removal policy
245
+ const destroyTemplate = assertions_1.Template.fromStack(destroyStack);
246
+ destroyTemplate.hasResource('AWS::S3::Bucket', {
247
+ DeletionPolicy: 'Delete',
248
+ UpdateReplacePolicy: 'Delete',
249
+ });
250
+ });
251
+ test('@webapp throws error when domainName is provided without certificate', () => {
252
+ expect(() => {
253
+ new frontend_construct_1.Frontend(stack, 'Frontend', {
254
+ sourceDirectory: '/tmp/test-frontend-src',
255
+ buildOutputDirectory: testBuildDir,
256
+ customDomain: {
257
+ domainName: 'app.example.com',
258
+ }, // Type assertion to bypass TypeScript validation for testing
259
+ skipBuild: true,
260
+ });
261
+ }).toThrow('certificate is required when domainName is provided');
262
+ });
263
+ });
264
+ describe('@webapp Security features', () => {
265
+ test('@webapp creates S3 bucket with security best practices', () => {
266
+ new frontend_construct_1.Frontend(stack, 'Frontend', {
267
+ sourceDirectory: '/tmp/test-frontend-src',
268
+ buildOutputDirectory: testBuildDir,
269
+ skipBuild: true,
270
+ });
271
+ template = assertions_1.Template.fromStack(stack);
272
+ // Verify S3 bucket security settings
273
+ template.hasResourceProperties('AWS::S3::Bucket', {
274
+ BucketEncryption: {
275
+ ServerSideEncryptionConfiguration: [
276
+ {
277
+ ServerSideEncryptionByDefault: {
278
+ SSEAlgorithm: 'AES256',
279
+ },
280
+ },
281
+ ],
282
+ },
283
+ PublicAccessBlockConfiguration: {
284
+ BlockPublicAcls: true,
285
+ BlockPublicPolicy: true,
286
+ IgnorePublicAcls: true,
287
+ RestrictPublicBuckets: true,
288
+ },
289
+ });
290
+ });
291
+ test('@webapp creates CloudFront distribution with security headers', () => {
292
+ new frontend_construct_1.Frontend(stack, 'Frontend', {
293
+ sourceDirectory: '/tmp/test-frontend-src',
294
+ buildOutputDirectory: testBuildDir,
295
+ skipBuild: true,
296
+ });
297
+ template = assertions_1.Template.fromStack(stack);
298
+ // Verify security headers function exists
299
+ template.hasResourceProperties('AWS::CloudFront::Function', {
300
+ FunctionConfig: {
301
+ Runtime: 'cloudfront-js-1.0',
302
+ },
303
+ });
304
+ // Verify HTTPS redirect is enabled
305
+ template.hasResourceProperties('AWS::CloudFront::Distribution', {
306
+ DistributionConfig: {
307
+ DefaultCacheBehavior: {
308
+ ViewerProtocolPolicy: 'redirect-to-https',
309
+ },
310
+ },
311
+ });
312
+ });
313
+ test('@webapp enables auto delete objects on S3 bucket', () => {
314
+ new frontend_construct_1.Frontend(stack, 'Frontend', {
315
+ sourceDirectory: '/tmp/test-frontend-src',
316
+ buildOutputDirectory: testBuildDir,
317
+ skipBuild: true,
318
+ });
319
+ template = assertions_1.Template.fromStack(stack);
320
+ // Verify auto delete objects custom resource is created
321
+ template.hasResource('Custom::S3AutoDeleteObjects', {});
322
+ });
323
+ });
324
+ describe('@webapp Default values', () => {
325
+ test('@webapp uses default SPA error responses', () => {
326
+ new frontend_construct_1.Frontend(stack, 'Frontend', {
327
+ sourceDirectory: '/tmp/test-frontend-src',
328
+ buildOutputDirectory: testBuildDir,
329
+ skipBuild: true,
330
+ });
331
+ template = assertions_1.Template.fromStack(stack);
332
+ template.hasResourceProperties('AWS::CloudFront::Distribution', {
333
+ DistributionConfig: {
334
+ CustomErrorResponses: [
335
+ {
336
+ ErrorCode: 403,
337
+ ResponseCode: 200,
338
+ ResponsePagePath: '/index.html',
339
+ },
340
+ {
341
+ ErrorCode: 404,
342
+ ResponseCode: 200,
343
+ ResponsePagePath: '/index.html',
344
+ },
345
+ ],
346
+ },
347
+ });
348
+ });
349
+ test('@webapp exports default SPA error responses', () => {
350
+ expect(frontend_construct_1.DEFAULT_SPA_ERROR_RESPONSES).toEqual([
351
+ {
352
+ httpStatus: 403,
353
+ responseHttpStatus: 200,
354
+ responsePagePath: '/index.html',
355
+ },
356
+ {
357
+ httpStatus: 404,
358
+ responseHttpStatus: 200,
359
+ responsePagePath: '/index.html',
360
+ },
361
+ ]);
362
+ });
363
+ });
364
+ describe('@webapp Additional distribution properties', () => {
365
+ test('@webapp accepts additional CloudFront distribution properties', () => {
366
+ new frontend_construct_1.Frontend(stack, 'Frontend', {
367
+ sourceDirectory: '/tmp/test-frontend-src',
368
+ buildOutputDirectory: testBuildDir,
369
+ distributionProps: {
370
+ comment: 'Custom frontend distribution',
371
+ enabled: true,
372
+ },
373
+ skipBuild: true,
374
+ });
375
+ template = assertions_1.Template.fromStack(stack);
376
+ template.hasResourceProperties('AWS::CloudFront::Distribution', {
377
+ DistributionConfig: {
378
+ Comment: 'Custom frontend distribution',
379
+ Enabled: true,
380
+ },
381
+ });
382
+ });
383
+ });
384
+ });
385
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"frontend-construct.test.js","sourceRoot":"","sources":["../../../use-cases/webapp/tests/frontend-construct.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, RemovalPolicy, Stack } from 'aws-cdk-lib';\nimport { Template } from 'aws-cdk-lib/assertions';\nimport { Certificate } from 'aws-cdk-lib/aws-certificatemanager';\nimport { Distribution } from 'aws-cdk-lib/aws-cloudfront';\nimport { HostedZone } from 'aws-cdk-lib/aws-route53';\nimport { Bucket } from 'aws-cdk-lib/aws-s3';\nimport { BucketDeployment } from 'aws-cdk-lib/aws-s3-deployment';\nimport { Frontend, DEFAULT_SPA_ERROR_RESPONSES } from '../frontend-construct';\n\n// Mock execSync to avoid actual build execution during tests\njest.mock('child_process', () => ({\n  execSync: jest.fn(),\n}));\n\ndescribe('@webapp Frontend', () => {\n  let app: App;\n  let stack: Stack;\n  let template: Template;\n  let testBuildDir: string;\n  let testSrcDir: string;\n\n  beforeAll(() => {\n    // Create a temporary build directory for tests\n    testBuildDir = '/tmp/test-frontend-build';\n    if (!fs.existsSync(testBuildDir)) {\n      fs.mkdirSync(testBuildDir, { recursive: true });\n    }\n    fs.writeFileSync(path.join(testBuildDir, 'index.html'), '<!DOCTYPE html><html><head><title>Test</title></head><body><h1>Test App</h1></body></html>');\n\n    // Create a temporary source directory for tests\n    testSrcDir = '/tmp/test-frontend-src';\n    if (!fs.existsSync(testSrcDir)) {\n      fs.mkdirSync(testSrcDir, { recursive: true });\n    }\n\n    // Create default build directory inside source directory\n    const defaultBuildDir = path.join(testSrcDir, 'build');\n    if (!fs.existsSync(defaultBuildDir)) {\n      fs.mkdirSync(defaultBuildDir, { recursive: true });\n    }\n    fs.writeFileSync(path.join(defaultBuildDir, 'index.html'), '<!DOCTYPE html><html><head><title>Default Test</title></head><body><h1>Default Test App</h1></body></html>');\n  });\n\n  beforeEach(() => {\n    app = new App();\n    stack = new Stack(app, 'TestStack');\n  });\n\n  describe('@webapp Basic functionality', () => {\n    test('@webapp creates frontend construct with minimal configuration', () => {\n      const frontend = new Frontend(stack, 'Frontend', {\n        sourceDirectory: '/tmp/test-frontend-src',\n        buildOutputDirectory: testBuildDir,\n        skipBuild: true, // Skip build for testing\n      });\n\n      template = Template.fromStack(stack);\n\n      // Verify S3 bucket is created\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 is created\n      template.hasResourceProperties('AWS::CloudFront::Distribution', {\n        DistributionConfig: {\n          DefaultRootObject: 'index.html',\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 security headers function is created\n      template.hasResourceProperties('AWS::CloudFront::Function', {\n        FunctionConfig: {\n          Runtime: 'cloudfront-js-1.0',\n        },\n      });\n\n      // Verify bucket deployment is created\n      template.hasResource('Custom::CDKBucketDeployment', {});\n\n      // Test public methods\n      expect(frontend.bucket).toBeInstanceOf(Bucket);\n      expect(frontend.distribution).toBeInstanceOf(Distribution);\n      expect(frontend.bucketDeployment).toBeInstanceOf(BucketDeployment);\n      expect(typeof frontend.distributionDomainName()).toBe('string');\n      expect(typeof frontend.bucketName()).toBe('string');\n      expect(frontend.url()).toContain('https://');\n    });\n\n    test('@webapp creates frontend construct with custom build command', () => {\n      new Frontend(stack, 'Frontend', {\n        sourceDirectory: '/tmp/test-frontend-src',\n        buildOutputDirectory: testBuildDir,\n        buildCommand: 'yarn build',\n        skipBuild: true,\n      });\n\n      template = Template.fromStack(stack);\n\n      // Should still create the same resources\n      template.resourceCountIs('AWS::S3::Bucket', 1);\n      template.resourceCountIs('AWS::CloudFront::Distribution', 1);\n    });\n\n    test('@webapp creates frontend construct with custom error responses', () => {\n      const customErrorResponses = [\n        {\n          httpStatus: 500,\n          responseHttpStatus: 200,\n          responsePagePath: '/error.html',\n        },\n      ];\n\n      new Frontend(stack, 'Frontend', {\n        sourceDirectory: '/tmp/test-frontend-src',\n        buildOutputDirectory: testBuildDir,\n        errorResponses: customErrorResponses,\n        skipBuild: true,\n      });\n\n      template = Template.fromStack(stack);\n\n      template.hasResourceProperties('AWS::CloudFront::Distribution', {\n        DistributionConfig: {\n          CustomErrorResponses: [\n            {\n              ErrorCode: 500,\n              ResponseCode: 200,\n              ResponsePagePath: '/error.html',\n            },\n          ],\n        },\n      });\n    });\n  });\n\n  describe('@webapp Custom domain functionality', () => {\n    test('@webapp creates frontend construct with custom domain', () => {\n      const certificate = Certificate.fromCertificateArn(\n        stack,\n        'Certificate',\n        'arn:aws:acm:us-east-1:123456789012:certificate/12345678-1234-1234-1234-123456789012',\n      );\n\n      const frontend = new Frontend(stack, 'Frontend', {\n        sourceDirectory: '/tmp/test-frontend-src',\n        buildOutputDirectory: testBuildDir,\n        customDomain: {\n          domainName: 'app.example.com',\n          certificate,\n        },\n        skipBuild: true,\n      });\n\n      template = Template.fromStack(stack);\n\n      // Verify CloudFront distribution has custom domain\n      template.hasResourceProperties('AWS::CloudFront::Distribution', {\n        DistributionConfig: {\n          Aliases: ['app.example.com'],\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      expect(frontend.domainName).toBe('app.example.com');\n      expect(frontend.url()).toBe('https://app.example.com');\n    });\n\n    test('@webapp creates frontend construct with custom domain and hosted zone', () => {\n      const certificate = Certificate.fromCertificateArn(\n        stack,\n        'Certificate',\n        'arn:aws:acm:us-east-1:123456789012:certificate/12345678-1234-1234-1234-123456789012',\n      );\n\n      const hostedZone = HostedZone.fromHostedZoneAttributes(stack, 'HostedZone', {\n        hostedZoneId: 'Z123456789',\n        zoneName: 'example.com',\n      });\n\n      new Frontend(stack, 'Frontend', {\n        sourceDirectory: '/tmp/test-frontend-src',\n        buildOutputDirectory: testBuildDir,\n        customDomain: {\n          domainName: 'app.example.com',\n          certificate,\n          hostedZone,\n        },\n        skipBuild: true,\n      });\n\n      template = Template.fromStack(stack);\n\n      // Verify Route53 A record is created\n      template.hasResourceProperties('AWS::Route53::RecordSet', {\n        Type: 'A',\n        Name: 'app.example.com.',\n        HostedZoneId: 'Z123456789',\n      });\n    });\n  });\n\n  describe('@webapp Validation', () => {\n    test('@webapp throws error when sourceDirectory is missing', () => {\n      expect(() => {\n        new Frontend(stack, 'Frontend', {\n          sourceDirectory: '',\n          buildOutputDirectory: testBuildDir,\n          skipBuild: true,\n        });\n      }).toThrow('sourceDirectory is required');\n    });\n\n    test('@webapp uses default buildOutputDirectory when not provided', () => {\n      // Use the existing test build directory\n      const frontend = new Frontend(stack, 'Frontend', {\n        sourceDirectory: testSrcDir,\n        // buildOutputDirectory not provided - should use default './build/'\n        skipBuild: true,\n      });\n\n      expect(frontend).toBeDefined();\n      // The construct should be created successfully with default buildOutputDirectory\n    });\n\n    test('@webapp applies custom removal policy to resources', () => {\n      const retainStack = new Stack(app, 'RetainTestStack');\n      const frontend = new Frontend(retainStack, 'Frontend', {\n        sourceDirectory: testSrcDir,\n        buildOutputDirectory: testBuildDir,\n        skipBuild: true,\n        removalPolicy: RemovalPolicy.RETAIN,\n      });\n\n      expect(frontend).toBeDefined();\n\n      // Check that the bucket has the correct removal policy\n      const retainTemplate = Template.fromStack(retainStack);\n      retainTemplate.hasResource('AWS::S3::Bucket', {\n        DeletionPolicy: 'Retain',\n        UpdateReplacePolicy: 'Retain',\n      });\n    });\n\n    test('@webapp uses default DESTROY removal policy when not specified', () => {\n      const destroyStack = new Stack(app, 'DestroyTestStack');\n      const frontend = new Frontend(destroyStack, 'Frontend', {\n        sourceDirectory: testSrcDir,\n        buildOutputDirectory: testBuildDir,\n        skipBuild: true,\n        // removalPolicy not specified - should default to DESTROY\n      });\n\n      expect(frontend).toBeDefined();\n\n      // Check that the bucket has the default DESTROY removal policy\n      const destroyTemplate = Template.fromStack(destroyStack);\n      destroyTemplate.hasResource('AWS::S3::Bucket', {\n        DeletionPolicy: 'Delete',\n        UpdateReplacePolicy: 'Delete',\n      });\n    });\n\n    test('@webapp throws error when domainName is provided without certificate', () => {\n      expect(() => {\n        new Frontend(stack, 'Frontend', {\n          sourceDirectory: '/tmp/test-frontend-src',\n          buildOutputDirectory: testBuildDir,\n          customDomain: {\n            domainName: 'app.example.com',\n          } as any, // Type assertion to bypass TypeScript validation for testing\n          skipBuild: true,\n        });\n      }).toThrow('certificate is required when domainName is provided');\n    });\n  });\n\n  describe('@webapp Security features', () => {\n    test('@webapp creates S3 bucket with security best practices', () => {\n      new Frontend(stack, 'Frontend', {\n        sourceDirectory: '/tmp/test-frontend-src',\n        buildOutputDirectory: testBuildDir,\n        skipBuild: true,\n      });\n\n      template = Template.fromStack(stack);\n\n      // Verify S3 bucket security settings\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\n    test('@webapp creates CloudFront distribution with security headers', () => {\n      new Frontend(stack, 'Frontend', {\n        sourceDirectory: '/tmp/test-frontend-src',\n        buildOutputDirectory: testBuildDir,\n        skipBuild: true,\n      });\n\n      template = Template.fromStack(stack);\n\n      // Verify security headers function exists\n      template.hasResourceProperties('AWS::CloudFront::Function', {\n        FunctionConfig: {\n          Runtime: 'cloudfront-js-1.0',\n        },\n      });\n\n      // Verify HTTPS redirect is enabled\n      template.hasResourceProperties('AWS::CloudFront::Distribution', {\n        DistributionConfig: {\n          DefaultCacheBehavior: {\n            ViewerProtocolPolicy: 'redirect-to-https',\n          },\n        },\n      });\n    });\n\n    test('@webapp enables auto delete objects on S3 bucket', () => {\n      new Frontend(stack, 'Frontend', {\n        sourceDirectory: '/tmp/test-frontend-src',\n        buildOutputDirectory: testBuildDir,\n        skipBuild: true,\n      });\n\n      template = Template.fromStack(stack);\n\n      // Verify auto delete objects custom resource is created\n      template.hasResource('Custom::S3AutoDeleteObjects', {});\n    });\n  });\n\n  describe('@webapp Default values', () => {\n    test('@webapp uses default SPA error responses', () => {\n      new Frontend(stack, 'Frontend', {\n        sourceDirectory: '/tmp/test-frontend-src',\n        buildOutputDirectory: testBuildDir,\n        skipBuild: true,\n      });\n\n      template = Template.fromStack(stack);\n\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\n    test('@webapp exports default SPA error responses', () => {\n      expect(DEFAULT_SPA_ERROR_RESPONSES).toEqual([\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\n  describe('@webapp Additional distribution properties', () => {\n    test('@webapp accepts additional CloudFront distribution properties', () => {\n      new Frontend(stack, 'Frontend', {\n        sourceDirectory: '/tmp/test-frontend-src',\n        buildOutputDirectory: testBuildDir,\n        distributionProps: {\n          comment: 'Custom frontend distribution',\n          enabled: true,\n        },\n        skipBuild: true,\n      });\n\n      template = Template.fromStack(stack);\n\n      template.hasResourceProperties('AWS::CloudFront::Distribution', {\n        DistributionConfig: {\n          Comment: 'Custom frontend distribution',\n          Enabled: true,\n        },\n      });\n    });\n  });\n});\n"]}
package/package.json ADDED
@@ -0,0 +1,183 @@
1
+ {
2
+ "name": "@cdklabs/cdk-appmod-catalog-blueprints",
3
+ "description": "Serverless infrastructure components organized by business use cases",
4
+ "repository": {
5
+ "type": "git",
6
+ "url": "git@github.com:cdklabs/cdk-appmod-catalog-blueprints.git"
7
+ },
8
+ "scripts": {
9
+ "build": "npx projen build",
10
+ "build:fast": "npx projen build:fast",
11
+ "bump": "npx projen bump",
12
+ "clobber": "npx projen clobber",
13
+ "compat": "npx projen compat",
14
+ "compile": "npx projen compile",
15
+ "default": "npx projen default",
16
+ "docgen": "npx projen docgen",
17
+ "eject": "npx projen eject",
18
+ "eslint": "npx projen eslint",
19
+ "integ": "npx projen integ",
20
+ "integ:update": "npx projen integ:update",
21
+ "package": "npx projen package",
22
+ "package-all": "npx projen package-all",
23
+ "package:dotnet": "npx projen package:dotnet",
24
+ "package:go": "npx projen package:go",
25
+ "package:java": "npx projen package:java",
26
+ "package:js": "npx projen package:js",
27
+ "package:python": "npx projen package:python",
28
+ "post-compile": "npx projen post-compile",
29
+ "post-upgrade": "npx projen post-upgrade",
30
+ "pre-compile": "npx projen pre-compile",
31
+ "release": "npx projen release",
32
+ "rosetta:extract": "npx projen rosetta:extract",
33
+ "test": "npx projen test",
34
+ "test:cdk-nag:all": "npx projen test:cdk-nag:all",
35
+ "test:cdk-nag:document-processing": "npx projen test:cdk-nag:document-processing",
36
+ "test:cdk-nag:webapp": "npx projen test:cdk-nag:webapp",
37
+ "test:document-processing": "npx projen test:document-processing",
38
+ "test:document-processing:unit": "npx projen test:document-processing:unit",
39
+ "test:security": "npx projen test:security",
40
+ "test:watch": "npx projen test:watch",
41
+ "test:webapp": "npx projen test:webapp",
42
+ "test:webapp:core": "npx projen test:webapp:core",
43
+ "test:webapp:frontend": "npx projen test:webapp:frontend",
44
+ "test:webapp:quickstart": "npx projen test:webapp:quickstart",
45
+ "test:webapp:unit": "npx projen test:webapp:unit",
46
+ "unbump": "npx projen unbump",
47
+ "upgrade": "npx projen upgrade",
48
+ "upgrade-cdklabs-projen-project-types": "npx projen upgrade-cdklabs-projen-project-types",
49
+ "upgrade-dev-deps": "npx projen upgrade-dev-deps",
50
+ "watch": "npx projen watch",
51
+ "projen": "npx projen"
52
+ },
53
+ "author": {
54
+ "name": "Amazon Web Services",
55
+ "email": "aws-cdk-dev@amazon.com",
56
+ "organization": true
57
+ },
58
+ "devDependencies": {
59
+ "@aws-cdk/aws-lambda-python-alpha": "2.192.0-alpha.0",
60
+ "@aws-cdk/integ-runner": "latest",
61
+ "@aws-cdk/integ-tests-alpha": "latest",
62
+ "@stylistic/eslint-plugin": "^2",
63
+ "@types/jest": "^29.5.14",
64
+ "@types/node": "^18",
65
+ "@typescript-eslint/eslint-plugin": "^8",
66
+ "@typescript-eslint/parser": "^8",
67
+ "aws-cdk-lib": "2.216.0",
68
+ "cdk-nag": "^2.37.35",
69
+ "cdklabs-projen-project-types": "^0.3.1",
70
+ "commit-and-tag-version": "^12",
71
+ "constructs": "10.0.5",
72
+ "eslint": "^9",
73
+ "eslint-import-resolver-typescript": "^4.2.5",
74
+ "eslint-plugin-import": "^2.31.0",
75
+ "jest": "^29.7.0",
76
+ "jest-junit": "^16",
77
+ "jsii": "~5.9.5",
78
+ "jsii-diff": "^1.110.0",
79
+ "jsii-docgen": "^10.5.0",
80
+ "jsii-pacmak": "^1.110.0",
81
+ "jsii-rosetta": "~5.8.0",
82
+ "projen": "^0.95.2",
83
+ "ts-jest": "^29.3.0",
84
+ "ts-node": "^10.9.2",
85
+ "typescript": "^5.8.2"
86
+ },
87
+ "peerDependencies": {
88
+ "@aws-cdk/aws-lambda-python-alpha": "2.192.0-alpha.0",
89
+ "aws-cdk-lib": "^2.216.0",
90
+ "constructs": "^10.0.5"
91
+ },
92
+ "keywords": [
93
+ "cdk"
94
+ ],
95
+ "engines": {
96
+ "node": ">= 18.12.0"
97
+ },
98
+ "main": "lib/index.js",
99
+ "license": "Apache-2.0",
100
+ "publishConfig": {
101
+ "access": "public"
102
+ },
103
+ "version": "1.0.0",
104
+ "jest": {
105
+ "coverageProvider": "v8",
106
+ "testMatch": [
107
+ "<rootDir>/@(use-cases|test)/**/*(*.)@(spec|test).ts?(x)",
108
+ "<rootDir>/@(use-cases|test)/**/__tests__/**/*.ts?(x)",
109
+ "<rootDir>/@(projenrc)/**/*(*.)@(spec|test).ts?(x)",
110
+ "<rootDir>/@(projenrc)/**/__tests__/**/*.ts?(x)"
111
+ ],
112
+ "clearMocks": true,
113
+ "collectCoverage": true,
114
+ "coverageReporters": [
115
+ "json",
116
+ "lcov",
117
+ "clover",
118
+ "cobertura",
119
+ "text"
120
+ ],
121
+ "coverageDirectory": "coverage",
122
+ "coveragePathIgnorePatterns": [
123
+ "/node_modules/"
124
+ ],
125
+ "testPathIgnorePatterns": [
126
+ "/node_modules/"
127
+ ],
128
+ "watchPathIgnorePatterns": [
129
+ "/node_modules/"
130
+ ],
131
+ "reporters": [
132
+ "default",
133
+ [
134
+ "jest-junit",
135
+ {
136
+ "outputDirectory": "test-reports"
137
+ }
138
+ ]
139
+ ],
140
+ "transform": {
141
+ "^.+\\.[t]sx?$": [
142
+ "ts-jest",
143
+ {
144
+ "tsconfig": "tsconfig.dev.json"
145
+ }
146
+ ]
147
+ }
148
+ },
149
+ "types": "lib/index.d.ts",
150
+ "stability": "experimental",
151
+ "jsii": {
152
+ "outdir": "dist",
153
+ "targets": {
154
+ "java": {
155
+ "package": "io.github.cdklabs.appmod.catalog.blueprints",
156
+ "maven": {
157
+ "groupId": "io.github.cdklabs",
158
+ "artifactId": "appmod-catalog-blueprints"
159
+ }
160
+ },
161
+ "python": {
162
+ "distName": "appmod-catalog-blueprints",
163
+ "module": "appmod_catalog_blueprints"
164
+ },
165
+ "dotnet": {
166
+ "namespace": "CdklabsAppmodCatalogBlueprints",
167
+ "packageId": "CdklabsAppmodCatalogBlueprints"
168
+ },
169
+ "go": {
170
+ "moduleName": "github.com/cdklabs/appmod-catalog-blueprints-go"
171
+ }
172
+ },
173
+ "tsc": {
174
+ "outDir": "lib",
175
+ "rootDir": "use-cases"
176
+ }
177
+ },
178
+ "files": [
179
+ "lib/",
180
+ ".jsii"
181
+ ],
182
+ "//": "~~ Generated by projen. To modify, edit .projenrc.ts and run \"npx projen\"."
183
+ }