@cdklabs/cdk-appmod-catalog-blueprints 1.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.jsii +8644 -0
- package/LICENSE +202 -0
- package/README.md +212 -0
- package/lib/document-processing/agentic-document-processing.d.ts +16 -0
- package/lib/document-processing/agentic-document-processing.js +90 -0
- package/lib/document-processing/base-document-processing.d.ts +189 -0
- package/lib/document-processing/base-document-processing.js +509 -0
- package/lib/document-processing/bedrock-document-processing.d.ts +167 -0
- package/lib/document-processing/bedrock-document-processing.js +297 -0
- package/lib/document-processing/index.d.ts +3 -0
- package/lib/document-processing/index.js +20 -0
- package/lib/document-processing/resources/default-bedrock-invoke/index.py +63 -0
- package/lib/document-processing/resources/default-bedrock-invoke/requirements.txt +4 -0
- package/lib/document-processing/resources/default-doc-retrieval-lambda/index.mjs +92 -0
- package/lib/document-processing/resources/default-doc-retrieval-lambda/package.json +10 -0
- package/lib/document-processing/resources/default-error-handler/index.js +46 -0
- package/lib/document-processing/resources/default-error-handler/package.json +4 -0
- package/lib/document-processing/resources/default-image-processor/classifier.mjs +665 -0
- package/lib/document-processing/resources/default-image-processor/extractors.mjs +465 -0
- package/lib/document-processing/resources/default-image-processor/index.mjs +143 -0
- package/lib/document-processing/resources/default-image-processor/package-lock.json +12 -0
- package/lib/document-processing/resources/default-image-processor/package.json +4 -0
- package/lib/document-processing/resources/default-image-validator/index.mjs +76 -0
- package/lib/document-processing/resources/default-image-validator/package-lock.json +154 -0
- package/lib/document-processing/resources/default-image-validator/package.json +7 -0
- package/lib/document-processing/resources/default-pdf-processor/index.js +46 -0
- package/lib/document-processing/resources/default-pdf-validator/index.js +36 -0
- package/lib/document-processing/resources/default-sqs-consumer/index.py +111 -0
- package/lib/document-processing/resources/default-sqs-consumer/requirements.txt +4 -0
- package/lib/document-processing/resources/default-sqs-consumer/sample_payload.json +20 -0
- package/lib/document-processing/resources/default-sqs-consumer/sample_payload_multi.json +24 -0
- package/lib/document-processing/resources/default-strands-agent/index.py +111 -0
- package/lib/document-processing/resources/default-strands-agent/requirements.txt +6 -0
- package/lib/document-processing/tests/agentic-document-processing-nag.test.d.ts +1 -0
- package/lib/document-processing/tests/agentic-document-processing-nag.test.js +107 -0
- package/lib/document-processing/tests/agentic-document-processing.test.d.ts +1 -0
- package/lib/document-processing/tests/agentic-document-processing.test.js +125 -0
- package/lib/document-processing/tests/bedrock-document-processing-nag.test.d.ts +1 -0
- package/lib/document-processing/tests/bedrock-document-processing-nag.test.js +101 -0
- package/lib/document-processing/tests/bedrock-document-processing.test.d.ts +1 -0
- package/lib/document-processing/tests/bedrock-document-processing.test.js +79 -0
- package/lib/framework/custom-resource/default-runtimes.d.ts +21 -0
- package/lib/framework/custom-resource/default-runtimes.js +34 -0
- package/lib/framework/custom-resource/index.d.ts +1 -0
- package/lib/framework/custom-resource/index.js +18 -0
- package/lib/framework/foundation/access-log.d.ts +69 -0
- package/lib/framework/foundation/access-log.js +121 -0
- package/lib/framework/foundation/eventbridge-broker.d.ts +18 -0
- package/lib/framework/foundation/eventbridge-broker.js +42 -0
- package/lib/framework/foundation/index.d.ts +3 -0
- package/lib/framework/foundation/index.js +20 -0
- package/lib/framework/foundation/network.d.ts +19 -0
- package/lib/framework/foundation/network.js +83 -0
- package/lib/framework/index.d.ts +2 -0
- package/lib/framework/index.js +19 -0
- package/lib/framework/quickstart/base-quickstart.d.ts +30 -0
- package/lib/framework/quickstart/base-quickstart.js +30 -0
- package/lib/index.d.ts +4 -0
- package/lib/index.js +21 -0
- package/lib/tsconfig.tsbuildinfo +1 -0
- package/lib/utilities/cdk-nag-config.d.ts +42 -0
- package/lib/utilities/cdk-nag-config.js +194 -0
- package/lib/utilities/data-loader-lambda/index.py +282 -0
- package/lib/utilities/data-loader-lambda/requirements.txt +3 -0
- package/lib/utilities/data-loader.d.ts +173 -0
- package/lib/utilities/data-loader.js +447 -0
- package/lib/utilities/index.d.ts +3 -0
- package/lib/utilities/index.js +20 -0
- package/lib/utilities/lambda-iam-utils.d.ts +145 -0
- package/lib/utilities/lambda-iam-utils.js +235 -0
- package/lib/utilities/lambda_layers/data-masking/layer-construct.d.ts +42 -0
- package/lib/utilities/lambda_layers/data-masking/layer-construct.js +53 -0
- package/lib/utilities/lambda_layers/data-masking/layer-construct.ts +88 -0
- package/lib/utilities/observability/bedrock-observability.d.ts +18 -0
- package/lib/utilities/observability/bedrock-observability.js +131 -0
- package/lib/utilities/observability/cloudfront-distribution-observability-property-injector.d.ts +6 -0
- package/lib/utilities/observability/cloudfront-distribution-observability-property-injector.js +22 -0
- package/lib/utilities/observability/index.d.ts +6 -0
- package/lib/utilities/observability/index.js +25 -0
- package/lib/utilities/observability/lambda-observability-property-injector.d.ts +8 -0
- package/lib/utilities/observability/lambda-observability-property-injector.js +43 -0
- package/lib/utilities/observability/log-group-data-protection-props.d.ts +19 -0
- package/lib/utilities/observability/log-group-data-protection-props.js +5 -0
- package/lib/utilities/observability/observability.d.ts +83 -0
- package/lib/utilities/observability/observability.js +278 -0
- package/lib/utilities/observability/observable.d.ts +32 -0
- package/lib/utilities/observability/observable.js +3 -0
- package/lib/utilities/observability/powertools-config.d.ts +3 -0
- package/lib/utilities/observability/powertools-config.js +25 -0
- package/lib/utilities/observability/resources/bedrock-manage-logging-configuration/index.py +27 -0
- package/lib/utilities/observability/state-machine-observability-property-injector.d.ts +8 -0
- package/lib/utilities/observability/state-machine-observability-property-injector.js +49 -0
- package/lib/utilities/tests/data-loader-nag.test.d.ts +1 -0
- package/lib/utilities/tests/data-loader-nag.test.js +432 -0
- package/lib/utilities/tests/data-loader.test.d.ts +1 -0
- package/lib/utilities/tests/data-loader.test.js +284 -0
- package/lib/webapp/frontend-construct.d.ts +136 -0
- package/lib/webapp/frontend-construct.js +253 -0
- package/lib/webapp/index.d.ts +1 -0
- package/lib/webapp/index.js +18 -0
- package/lib/webapp/tests/frontend-construct-nag.test.d.ts +1 -0
- package/lib/webapp/tests/frontend-construct-nag.test.js +266 -0
- package/lib/webapp/tests/frontend-construct.test.d.ts +1 -0
- package/lib/webapp/tests/frontend-construct.test.js +385 -0
- package/package.json +183 -0
|
@@ -0,0 +1,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 {};
|