@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,432 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ const aws_cdk_lib_1 = require("aws-cdk-lib");
4
+ const assertions_1 = require("aws-cdk-lib/assertions");
5
+ const aws_ec2_1 = require("aws-cdk-lib/aws-ec2");
6
+ const aws_kms_1 = require("aws-cdk-lib/aws-kms");
7
+ const aws_rds_1 = require("aws-cdk-lib/aws-rds");
8
+ const aws_secretsmanager_1 = require("aws-cdk-lib/aws-secretsmanager");
9
+ const cdk_nag_1 = require("cdk-nag");
10
+ const data_loader_1 = require("../data-loader");
11
+ // Create app and stack
12
+ const app = new aws_cdk_lib_1.App();
13
+ const stack = new aws_cdk_lib_1.Stack(app, 'TestStack', {
14
+ env: {
15
+ account: '123456789012',
16
+ region: 'us-east-1',
17
+ },
18
+ });
19
+ // Create KMS key for encryption at rest
20
+ const encryptionKey = new aws_kms_1.Key(stack, 'EncryptionKey', {
21
+ enableKeyRotation: true,
22
+ description: 'KMS key for DataLoader encryption at rest',
23
+ });
24
+ // Create VPC with Flow Logs to fix AwsSolutions-VPC7
25
+ const vpc = new aws_ec2_1.Vpc(stack, 'TestVpc', {
26
+ maxAzs: 2,
27
+ });
28
+ // Add VPC Flow Logs to fix AwsSolutions-VPC7
29
+ new aws_ec2_1.FlowLog(stack, 'VpcFlowLog', {
30
+ resourceType: aws_ec2_1.FlowLogResourceType.fromVpc(vpc),
31
+ trafficType: aws_ec2_1.FlowLogTrafficType.ALL,
32
+ });
33
+ // Create security group for database access
34
+ const dbSecurityGroup = new aws_ec2_1.SecurityGroup(stack, 'DatabaseSecurityGroup', {
35
+ vpc: vpc,
36
+ description: 'Security group for DataLoader database access',
37
+ allowAllOutbound: false,
38
+ });
39
+ // Create database credentials secret
40
+ const dbSecret = new aws_secretsmanager_1.Secret(stack, 'DatabaseSecret', {
41
+ description: 'Database credentials for DataLoader',
42
+ generateSecretString: {
43
+ secretStringTemplate: JSON.stringify({ username: 'postgres' }),
44
+ generateStringKey: 'password',
45
+ excludeCharacters: '"@/\\',
46
+ },
47
+ encryptionKey: encryptionKey,
48
+ });
49
+ // Create Aurora PostgreSQL cluster for testing
50
+ const dbCluster = new aws_rds_1.DatabaseCluster(stack, 'TestDatabaseCluster', {
51
+ engine: aws_rds_1.DatabaseClusterEngine.auroraPostgres({
52
+ version: aws_rds_1.AuroraPostgresEngineVersion.VER_15_4,
53
+ }),
54
+ vpc: vpc,
55
+ credentials: {
56
+ username: 'postgres',
57
+ password: dbSecret.secretValueFromJson('password'),
58
+ },
59
+ defaultDatabaseName: 'testdb',
60
+ securityGroups: [dbSecurityGroup],
61
+ storageEncryptionKey: encryptionKey,
62
+ removalPolicy: aws_cdk_lib_1.RemovalPolicy.DESTROY, // For test environment
63
+ writer: aws_rds_1.ClusterInstance.provisioned('writer', {
64
+ instanceType: aws_ec2_1.InstanceType.of(aws_ec2_1.InstanceClass.T3, aws_ec2_1.InstanceSize.MEDIUM),
65
+ }),
66
+ });
67
+ // Create DataLoader construct
68
+ const dataLoader = new data_loader_1.DataLoader(stack, 'DataLoader', {
69
+ databaseConfig: {
70
+ engine: data_loader_1.DatabaseEngine.POSTGRESQL,
71
+ cluster: dbCluster,
72
+ secret: dbSecret,
73
+ databaseName: 'testdb',
74
+ vpc: vpc,
75
+ securityGroup: dbSecurityGroup,
76
+ },
77
+ fileInputs: [
78
+ {
79
+ filePath: 's3://test-bucket/schema.sql',
80
+ fileType: data_loader_1.FileType.SQL,
81
+ executionOrder: 1,
82
+ },
83
+ {
84
+ filePath: 's3://test-bucket/data.sql',
85
+ fileType: data_loader_1.FileType.SQL,
86
+ executionOrder: 2,
87
+ },
88
+ ],
89
+ removalPolicy: aws_cdk_lib_1.RemovalPolicy.DESTROY, // For test environment
90
+ });
91
+ // Add CDK Nag suppressions for acceptable violations
92
+ // Suppress VPC-related warnings for test environment
93
+ cdk_nag_1.NagSuppressions.addResourceSuppressionsByPath(stack, '/TestStack/TestVpc', [
94
+ {
95
+ id: 'AwsSolutions-VPC7',
96
+ reason: 'VPC Flow Logs are added separately in the test setup',
97
+ },
98
+ ]);
99
+ // Suppress RDS-related warnings for test environment
100
+ cdk_nag_1.NagSuppressions.addResourceSuppressionsByPath(stack, '/TestStack/TestDatabaseCluster', [
101
+ {
102
+ id: 'AwsSolutions-RDS2',
103
+ reason: 'Database encryption is enabled with customer-managed KMS key',
104
+ },
105
+ {
106
+ id: 'AwsSolutions-RDS3',
107
+ reason: 'Multi-AZ disabled for test environment cost optimization',
108
+ },
109
+ {
110
+ id: 'AwsSolutions-RDS6',
111
+ reason: 'IAM database authentication not required for this use case',
112
+ },
113
+ {
114
+ id: 'AwsSolutions-RDS10',
115
+ reason: 'Deletion protection disabled for test environment to allow cleanup',
116
+ },
117
+ {
118
+ id: 'AwsSolutions-RDS11',
119
+ reason: 'Default port acceptable within VPC security boundaries',
120
+ },
121
+ {
122
+ id: 'AwsSolutions-RDS14',
123
+ reason: 'Backtrack not available for PostgreSQL engine',
124
+ },
125
+ {
126
+ id: 'AwsSolutions-RDS16',
127
+ reason: 'Performance Insights disabled for cost optimization in test environment',
128
+ },
129
+ ]);
130
+ // Suppress Secrets Manager warnings
131
+ cdk_nag_1.NagSuppressions.addResourceSuppressionsByPath(stack, '/TestStack/DatabaseSecret', [
132
+ {
133
+ id: 'AwsSolutions-SMG4',
134
+ reason: 'Secret rotation not required for test environment',
135
+ },
136
+ ]);
137
+ // Suppress S3 bucket warnings for DataLoader bucket
138
+ cdk_nag_1.NagSuppressions.addResourceSuppressionsByPath(stack, '/TestStack/DataLoader/DataLoaderBucket', [
139
+ {
140
+ id: 'AwsSolutions-S1',
141
+ reason: 'S3 access logging not required for internal data loading bucket',
142
+ },
143
+ {
144
+ id: 'AwsSolutions-S2',
145
+ reason: 'S3 bucket public read access is blocked by default configuration',
146
+ },
147
+ {
148
+ id: 'AwsSolutions-S3',
149
+ reason: 'SSL-only access is enforced through bucket policy',
150
+ },
151
+ {
152
+ id: 'AwsSolutions-S10',
153
+ reason: 'MFA delete not required for data loading bucket',
154
+ },
155
+ ]);
156
+ // Suppress S3 bucket policy warnings
157
+ cdk_nag_1.NagSuppressions.addResourceSuppressionsByPath(stack, '/TestStack/DataLoader/DataLoaderBucket/Policy', [
158
+ {
159
+ id: 'AwsSolutions-S10',
160
+ reason: 'SSL-only access policy is enforced for data loading bucket security',
161
+ },
162
+ ]);
163
+ // Suppress Lambda function warnings for DataLoader processor
164
+ cdk_nag_1.NagSuppressions.addResourceSuppressionsByPath(stack, '/TestStack/DataLoader/DataLoaderProcessor', [
165
+ {
166
+ id: 'AwsSolutions-L1',
167
+ reason: 'Lambda runtime version is managed at deployment time',
168
+ },
169
+ ]);
170
+ // Suppress security group warnings for DataLoader processor
171
+ cdk_nag_1.NagSuppressions.addResourceSuppressionsByPath(stack, '/TestStack/DataLoader/DataLoaderProcessorSecurityGroup', [
172
+ {
173
+ id: 'AwsSolutions-EC23',
174
+ reason: 'Lambda security group allows all outbound traffic for AWS service access and dependency downloads',
175
+ },
176
+ ]);
177
+ // Suppress IAM role warnings for DataLoader processor
178
+ cdk_nag_1.NagSuppressions.addResourceSuppressionsByPath(stack, '/TestStack/DataLoader/DataLoaderProcessor/ServiceRole', [
179
+ {
180
+ id: 'AwsSolutions-IAM4',
181
+ reason: 'AWSLambdaVPCAccessExecutionRole is required for VPC Lambda functions',
182
+ appliesTo: ['Policy::arn:<AWS::Partition>:iam::aws:policy/service-role/AWSLambdaVPCAccessExecutionRole'],
183
+ },
184
+ {
185
+ id: 'AwsSolutions-IAM5',
186
+ reason: 'Lambda function needs CloudWatch Logs permissions with wildcard for log stream creation',
187
+ appliesTo: ['Resource::arn:<AWS::Partition>:logs:*:*:log-group:/aws/lambda/*:*'],
188
+ },
189
+ ]);
190
+ // Suppress IAM policy warnings for DataLoader processor
191
+ cdk_nag_1.NagSuppressions.addResourceSuppressionsByPath(stack, '/TestStack/DataLoader/DataLoaderProcessor/ServiceRole/DefaultPolicy', [
192
+ {
193
+ id: 'AwsSolutions-IAM5',
194
+ reason: 'Lambda function needs S3 read permissions for data loading files',
195
+ appliesTo: [
196
+ 'Action::s3:GetObject*',
197
+ 'Action::s3:GetBucket*',
198
+ 'Action::s3:List*',
199
+ ],
200
+ },
201
+ {
202
+ id: 'AwsSolutions-IAM5',
203
+ reason: 'Lambda function needs access to all objects in the data loading bucket',
204
+ appliesTo: ['Resource::<DataLoaderDataLoaderBucketF99DADE2.Arn>/*'],
205
+ },
206
+ ]);
207
+ // Suppress Step Functions warnings
208
+ cdk_nag_1.NagSuppressions.addResourceSuppressionsByPath(stack, '/TestStack/DataLoader/DataLoaderStateMachine', [
209
+ {
210
+ id: 'AwsSolutions-SF1',
211
+ reason: 'Step Functions logging configuration is environment-specific',
212
+ },
213
+ {
214
+ id: 'AwsSolutions-SF2',
215
+ reason: 'X-Ray tracing configuration is environment-specific',
216
+ },
217
+ ]);
218
+ // Suppress Step Functions role warnings
219
+ cdk_nag_1.NagSuppressions.addResourceSuppressionsByPath(stack, '/TestStack/DataLoader/DataLoaderStateMachine/Role', [
220
+ {
221
+ id: 'AwsSolutions-IAM5',
222
+ reason: 'Step Functions needs to invoke Lambda functions with specific permissions',
223
+ appliesTo: ['Resource::<DataLoaderDataLoaderProcessor*>'],
224
+ },
225
+ ]);
226
+ // Suppress Step Functions role policy warnings
227
+ cdk_nag_1.NagSuppressions.addResourceSuppressionsByPath(stack, '/TestStack/DataLoader/DataLoaderStateMachine/Role/DefaultPolicy', [
228
+ {
229
+ id: 'AwsSolutions-IAM5',
230
+ reason: 'Step Functions needs to invoke the DataLoader processor Lambda function',
231
+ appliesTo: ['Resource::<DataLoaderDataLoaderProcessor693D75D2.Arn>:*'],
232
+ },
233
+ ]);
234
+ // Suppress custom resource Lambda warnings
235
+ cdk_nag_1.NagSuppressions.addResourceSuppressionsByPath(stack, '/TestStack/DataLoader/StateMachineExecutionTrigger', [
236
+ {
237
+ id: 'AwsSolutions-L1',
238
+ reason: 'Custom resource Lambda runtime version is managed at deployment time',
239
+ },
240
+ ]);
241
+ // Suppress custom resource IAM warnings for the dedicated role
242
+ cdk_nag_1.NagSuppressions.addResourceSuppressionsByPath(stack, '/TestStack/DataLoader/StateMachineExecutionTriggerRole', [
243
+ {
244
+ id: 'AwsSolutions-IAM4',
245
+ reason: 'AWSLambdaBasicExecutionRole is the standard minimal policy for Lambda execution',
246
+ appliesTo: ['Policy::arn:<AWS::Partition>:iam::aws:policy/service-role/AWSLambdaBasicExecutionRole'],
247
+ },
248
+ {
249
+ id: 'AwsSolutions-IAM5',
250
+ reason: 'Custom resource needs Step Functions execution permissions with wildcard for execution ARNs. ' +
251
+ 'Execution ARNs include dynamic execution names that cannot be predetermined.',
252
+ appliesTo: ['Resource::<DataLoaderDataLoaderStateMachine2071C3DC>:*'],
253
+ },
254
+ ]);
255
+ // Suppress custom resource provider warnings
256
+ cdk_nag_1.NagSuppressions.addResourceSuppressionsByPath(stack, '/TestStack/DataLoader/StateMachineExecutionProvider', [
257
+ {
258
+ id: 'AwsSolutions-L1',
259
+ reason: 'Custom resource provider Lambda runtime version is managed at deployment time',
260
+ },
261
+ ]);
262
+ // Suppress custom resource provider IAM warnings
263
+ cdk_nag_1.NagSuppressions.addResourceSuppressionsByPath(stack, '/TestStack/DataLoader/StateMachineExecutionProvider/framework-onEvent/ServiceRole', [
264
+ {
265
+ id: 'AwsSolutions-IAM4',
266
+ reason: 'AWSLambdaBasicExecutionRole is the standard minimal policy for Lambda execution',
267
+ appliesTo: ['Policy::arn:<AWS::Partition>:iam::aws:policy/service-role/AWSLambdaBasicExecutionRole'],
268
+ },
269
+ ]);
270
+ // Suppress custom resource provider policy warnings
271
+ cdk_nag_1.NagSuppressions.addResourceSuppressionsByPath(stack, '/TestStack/DataLoader/StateMachineExecutionProvider/framework-onEvent/ServiceRole/DefaultPolicy', [
272
+ {
273
+ id: 'AwsSolutions-IAM5',
274
+ reason: 'Custom resource needs Step Functions execution permissions',
275
+ appliesTo: ['Resource::<DataLoaderStateMachineExecutionTrigger94199E00.Arn>:*'],
276
+ },
277
+ ]);
278
+ // Suppress bucket deployment warnings if present
279
+ cdk_nag_1.NagSuppressions.addResourceSuppressions(stack, [
280
+ {
281
+ id: 'AwsSolutions-IAM4',
282
+ reason: 'CDK managed bucket deployment resources use AWS managed policies',
283
+ appliesTo: ['Policy::arn:<AWS::Partition>:iam::aws:policy/service-role/AWSLambdaBasicExecutionRole'],
284
+ },
285
+ {
286
+ id: 'AwsSolutions-IAM5',
287
+ reason: 'Bucket deployment requires wildcard permissions for S3 operations',
288
+ appliesTo: ['Resource::*'],
289
+ },
290
+ {
291
+ id: 'AwsSolutions-L1',
292
+ reason: 'CDK managed bucket deployment Lambda runtime is managed by CDK',
293
+ },
294
+ ], true);
295
+ // Apply CDK Nag checks
296
+ aws_cdk_lib_1.Aspects.of(app).add(new cdk_nag_1.AwsSolutionsChecks({ verbose: true }));
297
+ // Synthesize the stack
298
+ assertions_1.Template.fromStack(stack);
299
+ // Check for unsuppressed warnings and errors
300
+ const warnings = assertions_1.Annotations.fromStack(stack).findWarning('*', assertions_1.Match.stringLikeRegexp('AwsSolutions-.*'));
301
+ const errors = assertions_1.Annotations.fromStack(stack).findError('*', assertions_1.Match.stringLikeRegexp('AwsSolutions-.*'));
302
+ // Test: DataLoader construct is properly created
303
+ test('DataLoader construct is created successfully', () => {
304
+ expect(dataLoader).toBeDefined();
305
+ expect(dataLoader.node.id).toBe('DataLoader');
306
+ expect(dataLoader.bucket).toBeDefined();
307
+ expect(dataLoader.stateMachine).toBeDefined();
308
+ expect(dataLoader.processorFunction).toBeDefined();
309
+ expect(dataLoader.customResourceProvider).toBeDefined();
310
+ expect(dataLoader.executionTrigger).toBeDefined();
311
+ });
312
+ // Test: DataLoader has expected properties
313
+ test('DataLoader has expected properties', () => {
314
+ expect(dataLoader.bucket.bucketName).toBeDefined();
315
+ expect(dataLoader.stateMachine.stateMachineArn).toBeDefined();
316
+ expect(dataLoader.processorFunction.functionArn).toBeDefined();
317
+ });
318
+ // Test: Template contains expected DataLoader resources
319
+ test('Template contains expected DataLoader resources', () => {
320
+ const template = assertions_1.Template.fromStack(stack);
321
+ // Verify S3 bucket exists with encryption
322
+ template.hasResourceProperties('AWS::S3::Bucket', {
323
+ BucketEncryption: {
324
+ ServerSideEncryptionConfiguration: [
325
+ {
326
+ ServerSideEncryptionByDefault: {
327
+ SSEAlgorithm: 'AES256',
328
+ },
329
+ },
330
+ ],
331
+ },
332
+ PublicAccessBlockConfiguration: {
333
+ BlockPublicAcls: true,
334
+ BlockPublicPolicy: true,
335
+ IgnorePublicAcls: true,
336
+ RestrictPublicBuckets: true,
337
+ },
338
+ });
339
+ // Verify Lambda function exists with VPC configuration
340
+ template.hasResourceProperties('AWS::Lambda::Function', {
341
+ Runtime: 'python3.13',
342
+ VpcConfig: {
343
+ SecurityGroupIds: assertions_1.Match.anyValue(),
344
+ SubnetIds: assertions_1.Match.anyValue(),
345
+ },
346
+ });
347
+ // Verify Step Functions state machine exists
348
+ template.hasResourceProperties('AWS::StepFunctions::StateMachine', {
349
+ DefinitionString: assertions_1.Match.anyValue(),
350
+ });
351
+ // Verify Aurora cluster exists with encryption
352
+ template.hasResourceProperties('AWS::RDS::DBCluster', {
353
+ Engine: 'aurora-postgresql',
354
+ StorageEncrypted: true,
355
+ });
356
+ // Verify custom resource exists
357
+ template.hasResourceProperties('AWS::CloudFormation::CustomResource', {
358
+ ServiceToken: assertions_1.Match.anyValue(),
359
+ });
360
+ });
361
+ // Test: Security configurations are properly applied
362
+ test('Security configurations are properly applied', () => {
363
+ const template = assertions_1.Template.fromStack(stack);
364
+ // Verify KMS key exists with rotation enabled
365
+ template.hasResourceProperties('AWS::KMS::Key', {
366
+ EnableKeyRotation: true,
367
+ });
368
+ // Verify VPC Flow Logs exist
369
+ template.hasResourceProperties('AWS::EC2::FlowLog', {
370
+ ResourceType: 'VPC',
371
+ TrafficType: 'ALL',
372
+ });
373
+ // Verify Secrets Manager secret exists with KMS encryption
374
+ template.hasResourceProperties('AWS::SecretsManager::Secret', {
375
+ KmsKeyId: assertions_1.Match.anyValue(),
376
+ });
377
+ // Verify security group exists
378
+ template.hasResourceProperties('AWS::EC2::SecurityGroup', {
379
+ GroupDescription: 'Security group for DataLoader database access',
380
+ });
381
+ // Verify Lambda security group exists
382
+ template.hasResourceProperties('AWS::EC2::SecurityGroup', {
383
+ GroupDescription: 'Security group for DataLoader processor Lambda function',
384
+ });
385
+ });
386
+ // Test: IAM permissions are properly configured
387
+ test('IAM permissions are properly configured', () => {
388
+ const template = assertions_1.Template.fromStack(stack);
389
+ // Verify Lambda execution role exists
390
+ template.hasResourceProperties('AWS::IAM::Role', {
391
+ AssumeRolePolicyDocument: {
392
+ Statement: [
393
+ {
394
+ Action: 'sts:AssumeRole',
395
+ Effect: 'Allow',
396
+ Principal: {
397
+ Service: 'lambda.amazonaws.com',
398
+ },
399
+ },
400
+ ],
401
+ },
402
+ });
403
+ // Verify Step Functions role exists
404
+ template.hasResourceProperties('AWS::IAM::Role', {
405
+ AssumeRolePolicyDocument: {
406
+ Statement: [
407
+ {
408
+ Action: 'sts:AssumeRole',
409
+ Effect: 'Allow',
410
+ Principal: {
411
+ Service: 'states.amazonaws.com',
412
+ },
413
+ },
414
+ ],
415
+ },
416
+ });
417
+ });
418
+ // Test: No unsuppressed warnings
419
+ test('No unsuppressed warnings', () => {
420
+ if (warnings.length > 0) {
421
+ console.log('CDK Nag Warnings:', JSON.stringify(warnings, null, 2));
422
+ }
423
+ expect(warnings).toHaveLength(0);
424
+ });
425
+ // Test: No unsuppressed errors
426
+ test('No unsuppressed errors', () => {
427
+ if (errors.length > 0) {
428
+ console.log('CDK Nag Errors:', JSON.stringify(errors, null, 2));
429
+ }
430
+ expect(errors).toHaveLength(0);
431
+ });
432
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"data-loader-nag.test.js","sourceRoot":"","sources":["../../../use-cases/utilities/tests/data-loader-nag.test.ts"],"names":[],"mappings":"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","sourcesContent":["import { App, Stack, Aspects, RemovalPolicy } from 'aws-cdk-lib';\nimport { Template, Annotations, Match } from 'aws-cdk-lib/assertions';\nimport { Vpc, FlowLog, FlowLogResourceType, FlowLogTrafficType, SecurityGroup, InstanceType, InstanceClass, InstanceSize } from 'aws-cdk-lib/aws-ec2';\nimport { Key } from 'aws-cdk-lib/aws-kms';\nimport { DatabaseCluster, DatabaseClusterEngine, AuroraPostgresEngineVersion, ClusterInstance } from 'aws-cdk-lib/aws-rds';\nimport { Secret } from 'aws-cdk-lib/aws-secretsmanager';\nimport { AwsSolutionsChecks, NagSuppressions } from 'cdk-nag';\nimport { DataLoader, DatabaseEngine, FileType } from '../data-loader';\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 KMS key for encryption at rest\nconst encryptionKey = new Key(stack, 'EncryptionKey', {\n  enableKeyRotation: true,\n  description: 'KMS key for DataLoader encryption at rest',\n});\n\n// Create VPC with Flow Logs to fix AwsSolutions-VPC7\nconst vpc = new Vpc(stack, 'TestVpc', {\n  maxAzs: 2,\n});\n\n// Add VPC Flow Logs to fix AwsSolutions-VPC7\nnew FlowLog(stack, 'VpcFlowLog', {\n  resourceType: FlowLogResourceType.fromVpc(vpc),\n  trafficType: FlowLogTrafficType.ALL,\n});\n\n// Create security group for database access\nconst dbSecurityGroup = new SecurityGroup(stack, 'DatabaseSecurityGroup', {\n  vpc: vpc,\n  description: 'Security group for DataLoader database access',\n  allowAllOutbound: false,\n});\n\n// Create database credentials secret\nconst dbSecret = new Secret(stack, 'DatabaseSecret', {\n  description: 'Database credentials for DataLoader',\n  generateSecretString: {\n    secretStringTemplate: JSON.stringify({ username: 'postgres' }),\n    generateStringKey: 'password',\n    excludeCharacters: '\"@/\\\\',\n  },\n  encryptionKey: encryptionKey,\n});\n\n// Create Aurora PostgreSQL cluster for testing\nconst dbCluster = new DatabaseCluster(stack, 'TestDatabaseCluster', {\n  engine: DatabaseClusterEngine.auroraPostgres({\n    version: AuroraPostgresEngineVersion.VER_15_4,\n  }),\n  vpc: vpc,\n  credentials: {\n    username: 'postgres',\n    password: dbSecret.secretValueFromJson('password'),\n  },\n  defaultDatabaseName: 'testdb',\n  securityGroups: [dbSecurityGroup],\n  storageEncryptionKey: encryptionKey,\n  removalPolicy: RemovalPolicy.DESTROY, // For test environment\n  writer: ClusterInstance.provisioned('writer', {\n    instanceType: InstanceType.of(InstanceClass.T3, InstanceSize.MEDIUM),\n  }),\n});\n\n// Create DataLoader construct\nconst dataLoader = new DataLoader(stack, 'DataLoader', {\n  databaseConfig: {\n    engine: DatabaseEngine.POSTGRESQL,\n    cluster: dbCluster,\n    secret: dbSecret,\n    databaseName: 'testdb',\n    vpc: vpc,\n    securityGroup: dbSecurityGroup,\n  },\n  fileInputs: [\n    {\n      filePath: 's3://test-bucket/schema.sql',\n      fileType: FileType.SQL,\n      executionOrder: 1,\n    },\n    {\n      filePath: 's3://test-bucket/data.sql',\n      fileType: FileType.SQL,\n      executionOrder: 2,\n    },\n  ],\n  removalPolicy: RemovalPolicy.DESTROY, // For test environment\n});\n\n// Add CDK Nag suppressions for acceptable violations\n\n// Suppress VPC-related warnings for test environment\nNagSuppressions.addResourceSuppressionsByPath(stack, '/TestStack/TestVpc', [\n  {\n    id: 'AwsSolutions-VPC7',\n    reason: 'VPC Flow Logs are added separately in the test setup',\n  },\n]);\n\n// Suppress RDS-related warnings for test environment\nNagSuppressions.addResourceSuppressionsByPath(stack, '/TestStack/TestDatabaseCluster', [\n  {\n    id: 'AwsSolutions-RDS2',\n    reason: 'Database encryption is enabled with customer-managed KMS key',\n  },\n  {\n    id: 'AwsSolutions-RDS3',\n    reason: 'Multi-AZ disabled for test environment cost optimization',\n  },\n  {\n    id: 'AwsSolutions-RDS6',\n    reason: 'IAM database authentication not required for this use case',\n  },\n  {\n    id: 'AwsSolutions-RDS10',\n    reason: 'Deletion protection disabled for test environment to allow cleanup',\n  },\n  {\n    id: 'AwsSolutions-RDS11',\n    reason: 'Default port acceptable within VPC security boundaries',\n  },\n  {\n    id: 'AwsSolutions-RDS14',\n    reason: 'Backtrack not available for PostgreSQL engine',\n  },\n  {\n    id: 'AwsSolutions-RDS16',\n    reason: 'Performance Insights disabled for cost optimization in test environment',\n  },\n]);\n\n// Suppress Secrets Manager warnings\nNagSuppressions.addResourceSuppressionsByPath(stack, '/TestStack/DatabaseSecret', [\n  {\n    id: 'AwsSolutions-SMG4',\n    reason: 'Secret rotation not required for test environment',\n  },\n]);\n\n// Suppress S3 bucket warnings for DataLoader bucket\nNagSuppressions.addResourceSuppressionsByPath(stack, '/TestStack/DataLoader/DataLoaderBucket', [\n  {\n    id: 'AwsSolutions-S1',\n    reason: 'S3 access logging not required for internal data loading bucket',\n  },\n  {\n    id: 'AwsSolutions-S2',\n    reason: 'S3 bucket public read access is blocked by default configuration',\n  },\n  {\n    id: 'AwsSolutions-S3',\n    reason: 'SSL-only access is enforced through bucket policy',\n  },\n  {\n    id: 'AwsSolutions-S10',\n    reason: 'MFA delete not required for data loading bucket',\n  },\n]);\n\n// Suppress S3 bucket policy warnings\nNagSuppressions.addResourceSuppressionsByPath(stack, '/TestStack/DataLoader/DataLoaderBucket/Policy', [\n  {\n    id: 'AwsSolutions-S10',\n    reason: 'SSL-only access policy is enforced for data loading bucket security',\n  },\n]);\n\n// Suppress Lambda function warnings for DataLoader processor\nNagSuppressions.addResourceSuppressionsByPath(stack, '/TestStack/DataLoader/DataLoaderProcessor', [\n  {\n    id: 'AwsSolutions-L1',\n    reason: 'Lambda runtime version is managed at deployment time',\n  },\n]);\n\n// Suppress security group warnings for DataLoader processor\nNagSuppressions.addResourceSuppressionsByPath(stack, '/TestStack/DataLoader/DataLoaderProcessorSecurityGroup', [\n  {\n    id: 'AwsSolutions-EC23',\n    reason: 'Lambda security group allows all outbound traffic for AWS service access and dependency downloads',\n  },\n]);\n\n// Suppress IAM role warnings for DataLoader processor\nNagSuppressions.addResourceSuppressionsByPath(stack, '/TestStack/DataLoader/DataLoaderProcessor/ServiceRole', [\n  {\n    id: 'AwsSolutions-IAM4',\n    reason: 'AWSLambdaVPCAccessExecutionRole is required for VPC Lambda functions',\n    appliesTo: ['Policy::arn:<AWS::Partition>:iam::aws:policy/service-role/AWSLambdaVPCAccessExecutionRole'],\n  },\n  {\n    id: 'AwsSolutions-IAM5',\n    reason: 'Lambda function needs CloudWatch Logs permissions with wildcard for log stream creation',\n    appliesTo: ['Resource::arn:<AWS::Partition>:logs:*:*:log-group:/aws/lambda/*:*'],\n  },\n]);\n\n// Suppress IAM policy warnings for DataLoader processor\nNagSuppressions.addResourceSuppressionsByPath(stack, '/TestStack/DataLoader/DataLoaderProcessor/ServiceRole/DefaultPolicy', [\n  {\n    id: 'AwsSolutions-IAM5',\n    reason: 'Lambda function needs S3 read permissions for data loading files',\n    appliesTo: [\n      'Action::s3:GetObject*',\n      'Action::s3:GetBucket*',\n      'Action::s3:List*',\n    ],\n  },\n  {\n    id: 'AwsSolutions-IAM5',\n    reason: 'Lambda function needs access to all objects in the data loading bucket',\n    appliesTo: ['Resource::<DataLoaderDataLoaderBucketF99DADE2.Arn>/*'],\n  },\n]);\n\n// Suppress Step Functions warnings\nNagSuppressions.addResourceSuppressionsByPath(stack, '/TestStack/DataLoader/DataLoaderStateMachine', [\n  {\n    id: 'AwsSolutions-SF1',\n    reason: 'Step Functions logging configuration is environment-specific',\n  },\n  {\n    id: 'AwsSolutions-SF2',\n    reason: 'X-Ray tracing configuration is environment-specific',\n  },\n]);\n\n// Suppress Step Functions role warnings\nNagSuppressions.addResourceSuppressionsByPath(stack, '/TestStack/DataLoader/DataLoaderStateMachine/Role', [\n  {\n    id: 'AwsSolutions-IAM5',\n    reason: 'Step Functions needs to invoke Lambda functions with specific permissions',\n    appliesTo: ['Resource::<DataLoaderDataLoaderProcessor*>'],\n  },\n]);\n\n// Suppress Step Functions role policy warnings\nNagSuppressions.addResourceSuppressionsByPath(stack, '/TestStack/DataLoader/DataLoaderStateMachine/Role/DefaultPolicy', [\n  {\n    id: 'AwsSolutions-IAM5',\n    reason: 'Step Functions needs to invoke the DataLoader processor Lambda function',\n    appliesTo: ['Resource::<DataLoaderDataLoaderProcessor693D75D2.Arn>:*'],\n  },\n]);\n\n// Suppress custom resource Lambda warnings\nNagSuppressions.addResourceSuppressionsByPath(stack, '/TestStack/DataLoader/StateMachineExecutionTrigger', [\n  {\n    id: 'AwsSolutions-L1',\n    reason: 'Custom resource Lambda runtime version is managed at deployment time',\n  },\n]);\n\n// Suppress custom resource IAM warnings for the dedicated role\nNagSuppressions.addResourceSuppressionsByPath(stack, '/TestStack/DataLoader/StateMachineExecutionTriggerRole', [\n  {\n    id: 'AwsSolutions-IAM4',\n    reason: 'AWSLambdaBasicExecutionRole is the standard minimal policy for Lambda execution',\n    appliesTo: ['Policy::arn:<AWS::Partition>:iam::aws:policy/service-role/AWSLambdaBasicExecutionRole'],\n  },\n  {\n    id: 'AwsSolutions-IAM5',\n    reason: 'Custom resource needs Step Functions execution permissions with wildcard for execution ARNs. ' +\n            'Execution ARNs include dynamic execution names that cannot be predetermined.',\n    appliesTo: ['Resource::<DataLoaderDataLoaderStateMachine2071C3DC>:*'],\n  },\n]);\n\n// Suppress custom resource provider warnings\nNagSuppressions.addResourceSuppressionsByPath(stack, '/TestStack/DataLoader/StateMachineExecutionProvider', [\n  {\n    id: 'AwsSolutions-L1',\n    reason: 'Custom resource provider Lambda runtime version is managed at deployment time',\n  },\n]);\n\n// Suppress custom resource provider IAM warnings\nNagSuppressions.addResourceSuppressionsByPath(stack, '/TestStack/DataLoader/StateMachineExecutionProvider/framework-onEvent/ServiceRole', [\n  {\n    id: 'AwsSolutions-IAM4',\n    reason: 'AWSLambdaBasicExecutionRole is the standard minimal policy for Lambda execution',\n    appliesTo: ['Policy::arn:<AWS::Partition>:iam::aws:policy/service-role/AWSLambdaBasicExecutionRole'],\n  },\n]);\n\n// Suppress custom resource provider policy warnings\nNagSuppressions.addResourceSuppressionsByPath(stack, '/TestStack/DataLoader/StateMachineExecutionProvider/framework-onEvent/ServiceRole/DefaultPolicy', [\n  {\n    id: 'AwsSolutions-IAM5',\n    reason: 'Custom resource needs Step Functions execution permissions',\n    appliesTo: ['Resource::<DataLoaderStateMachineExecutionTrigger94199E00.Arn>:*'],\n  },\n]);\n\n// Suppress bucket deployment warnings if present\nNagSuppressions.addResourceSuppressions(stack, [\n  {\n    id: 'AwsSolutions-IAM4',\n    reason: 'CDK managed bucket deployment resources use AWS managed policies',\n    appliesTo: ['Policy::arn:<AWS::Partition>:iam::aws:policy/service-role/AWSLambdaBasicExecutionRole'],\n  },\n  {\n    id: 'AwsSolutions-IAM5',\n    reason: 'Bucket deployment requires wildcard permissions for S3 operations',\n    appliesTo: ['Resource::*'],\n  },\n  {\n    id: 'AwsSolutions-L1',\n    reason: 'CDK managed bucket deployment Lambda runtime is managed by CDK',\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: DataLoader construct is properly created\ntest('DataLoader construct is created successfully', () => {\n  expect(dataLoader).toBeDefined();\n  expect(dataLoader.node.id).toBe('DataLoader');\n  expect(dataLoader.bucket).toBeDefined();\n  expect(dataLoader.stateMachine).toBeDefined();\n  expect(dataLoader.processorFunction).toBeDefined();\n  expect(dataLoader.customResourceProvider).toBeDefined();\n  expect(dataLoader.executionTrigger).toBeDefined();\n});\n\n// Test: DataLoader has expected properties\ntest('DataLoader has expected properties', () => {\n  expect(dataLoader.bucket.bucketName).toBeDefined();\n  expect(dataLoader.stateMachine.stateMachineArn).toBeDefined();\n  expect(dataLoader.processorFunction.functionArn).toBeDefined();\n});\n\n// Test: Template contains expected DataLoader resources\ntest('Template contains expected DataLoader resources', () => {\n  const template = Template.fromStack(stack);\n\n  // Verify S3 bucket exists with encryption\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 Lambda function exists with VPC configuration\n  template.hasResourceProperties('AWS::Lambda::Function', {\n    Runtime: 'python3.13',\n    VpcConfig: {\n      SecurityGroupIds: Match.anyValue(),\n      SubnetIds: Match.anyValue(),\n    },\n  });\n\n  // Verify Step Functions state machine exists\n  template.hasResourceProperties('AWS::StepFunctions::StateMachine', {\n    DefinitionString: Match.anyValue(),\n  });\n\n  // Verify Aurora cluster exists with encryption\n  template.hasResourceProperties('AWS::RDS::DBCluster', {\n    Engine: 'aurora-postgresql',\n    StorageEncrypted: true,\n  });\n\n  // Verify custom resource exists\n  template.hasResourceProperties('AWS::CloudFormation::CustomResource', {\n    ServiceToken: Match.anyValue(),\n  });\n});\n\n// Test: Security configurations are properly applied\ntest('Security configurations are properly applied', () => {\n  const template = Template.fromStack(stack);\n\n  // Verify KMS key exists with rotation enabled\n  template.hasResourceProperties('AWS::KMS::Key', {\n    EnableKeyRotation: true,\n  });\n\n  // Verify VPC Flow Logs exist\n  template.hasResourceProperties('AWS::EC2::FlowLog', {\n    ResourceType: 'VPC',\n    TrafficType: 'ALL',\n  });\n\n  // Verify Secrets Manager secret exists with KMS encryption\n  template.hasResourceProperties('AWS::SecretsManager::Secret', {\n    KmsKeyId: Match.anyValue(),\n  });\n\n  // Verify security group exists\n  template.hasResourceProperties('AWS::EC2::SecurityGroup', {\n    GroupDescription: 'Security group for DataLoader database access',\n  });\n\n  // Verify Lambda security group exists\n  template.hasResourceProperties('AWS::EC2::SecurityGroup', {\n    GroupDescription: 'Security group for DataLoader processor Lambda function',\n  });\n});\n\n// Test: IAM permissions are properly configured\ntest('IAM permissions are properly configured', () => {\n  const template = Template.fromStack(stack);\n\n  // Verify Lambda execution role exists\n  template.hasResourceProperties('AWS::IAM::Role', {\n    AssumeRolePolicyDocument: {\n      Statement: [\n        {\n          Action: 'sts:AssumeRole',\n          Effect: 'Allow',\n          Principal: {\n            Service: 'lambda.amazonaws.com',\n          },\n        },\n      ],\n    },\n  });\n\n  // Verify Step Functions role exists\n  template.hasResourceProperties('AWS::IAM::Role', {\n    AssumeRolePolicyDocument: {\n      Statement: [\n        {\n          Action: 'sts:AssumeRole',\n          Effect: 'Allow',\n          Principal: {\n            Service: 'states.amazonaws.com',\n          },\n        },\n      ],\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 {};