@cdklabs/cdk-appmod-catalog-blueprints 1.4.1 → 1.6.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 +2579 -194
- package/lib/document-processing/adapter/adapter.d.ts +4 -2
- package/lib/document-processing/adapter/adapter.js +1 -1
- package/lib/document-processing/adapter/queued-s3-adapter.d.ts +9 -2
- package/lib/document-processing/adapter/queued-s3-adapter.js +29 -15
- package/lib/document-processing/agentic-document-processing.d.ts +4 -0
- package/lib/document-processing/agentic-document-processing.js +20 -10
- package/lib/document-processing/base-document-processing.d.ts +54 -2
- package/lib/document-processing/base-document-processing.js +136 -82
- package/lib/document-processing/bedrock-document-processing.d.ts +202 -2
- package/lib/document-processing/bedrock-document-processing.js +717 -77
- package/lib/document-processing/chunking-config.d.ts +614 -0
- package/lib/document-processing/chunking-config.js +5 -0
- package/lib/document-processing/default-document-processing-config.js +1 -1
- package/lib/document-processing/index.d.ts +1 -0
- package/lib/document-processing/index.js +2 -1
- package/lib/document-processing/resources/aggregation/handler.py +567 -0
- package/lib/document-processing/resources/aggregation/requirements.txt +7 -0
- package/lib/document-processing/resources/aggregation/test_handler.py +362 -0
- package/lib/document-processing/resources/cleanup/handler.py +276 -0
- package/lib/document-processing/resources/cleanup/requirements.txt +5 -0
- package/lib/document-processing/resources/cleanup/test_handler.py +436 -0
- package/lib/document-processing/resources/default-bedrock-invoke/index.py +85 -3
- package/lib/document-processing/resources/default-bedrock-invoke/test_index.py +622 -0
- package/lib/document-processing/resources/pdf-chunking/README.md +313 -0
- package/lib/document-processing/resources/pdf-chunking/chunking_strategies.py +460 -0
- package/lib/document-processing/resources/pdf-chunking/error_handling.py +491 -0
- package/lib/document-processing/resources/pdf-chunking/handler.py +958 -0
- package/lib/document-processing/resources/pdf-chunking/metrics.py +435 -0
- package/lib/document-processing/resources/pdf-chunking/requirements.txt +3 -0
- package/lib/document-processing/resources/pdf-chunking/strategy_selection.py +420 -0
- package/lib/document-processing/resources/pdf-chunking/structured_logging.py +457 -0
- package/lib/document-processing/resources/pdf-chunking/test_chunking_strategies.py +353 -0
- package/lib/document-processing/resources/pdf-chunking/test_error_handling.py +487 -0
- package/lib/document-processing/resources/pdf-chunking/test_handler.py +609 -0
- package/lib/document-processing/resources/pdf-chunking/test_integration.py +694 -0
- package/lib/document-processing/resources/pdf-chunking/test_metrics.py +532 -0
- package/lib/document-processing/resources/pdf-chunking/test_strategy_selection.py +471 -0
- package/lib/document-processing/resources/pdf-chunking/test_structured_logging.py +449 -0
- package/lib/document-processing/resources/pdf-chunking/test_token_estimation.py +374 -0
- package/lib/document-processing/resources/pdf-chunking/token_estimation.py +189 -0
- package/lib/document-processing/tests/agentic-document-processing-nag.test.js +4 -3
- package/lib/document-processing/tests/agentic-document-processing.test.js +488 -4
- package/lib/document-processing/tests/base-document-processing-nag.test.js +9 -2
- package/lib/document-processing/tests/base-document-processing-schema.test.d.ts +1 -0
- package/lib/document-processing/tests/base-document-processing-schema.test.js +337 -0
- package/lib/document-processing/tests/base-document-processing.test.js +114 -8
- package/lib/document-processing/tests/bedrock-document-processing-chunking-nag.test.d.ts +1 -0
- package/lib/document-processing/tests/bedrock-document-processing-chunking-nag.test.js +382 -0
- package/lib/document-processing/tests/bedrock-document-processing-nag.test.js +4 -3
- package/lib/document-processing/tests/bedrock-document-processing-security.test.d.ts +1 -0
- package/lib/document-processing/tests/bedrock-document-processing-security.test.js +389 -0
- package/lib/document-processing/tests/bedrock-document-processing.test.js +808 -8
- package/lib/document-processing/tests/chunking-config.test.d.ts +1 -0
- package/lib/document-processing/tests/chunking-config.test.js +238 -0
- package/lib/document-processing/tests/queued-s3-adapter-nag.test.js +9 -2
- package/lib/document-processing/tests/queued-s3-adapter.test.js +17 -6
- package/lib/framework/agents/base-agent.js +1 -1
- package/lib/framework/agents/batch-agent.js +1 -1
- package/lib/framework/agents/default-agent-config.js +1 -1
- package/lib/framework/bedrock/bedrock.js +1 -1
- package/lib/framework/custom-resource/default-runtimes.js +1 -1
- package/lib/framework/foundation/access-log.js +1 -1
- package/lib/framework/foundation/eventbridge-broker.js +1 -1
- package/lib/framework/foundation/network.d.ts +4 -2
- package/lib/framework/foundation/network.js +52 -41
- package/lib/framework/tests/access-log.test.js +5 -2
- package/lib/framework/tests/batch-agent.test.js +5 -2
- package/lib/framework/tests/bedrock.test.js +5 -2
- package/lib/framework/tests/eventbridge-broker.test.js +5 -2
- package/lib/framework/tests/framework-nag.test.js +26 -7
- package/lib/framework/tests/network.test.js +30 -2
- package/lib/tsconfig.tsbuildinfo +1 -1
- package/lib/utilities/data-loader.js +1 -1
- package/lib/utilities/lambda-iam-utils.js +1 -1
- package/lib/utilities/observability/cloudfront-distribution-observability-property-injector.js +1 -1
- package/lib/utilities/observability/default-observability-config.js +1 -1
- package/lib/utilities/observability/lambda-observability-property-injector.js +1 -1
- package/lib/utilities/observability/log-group-data-protection-utils.js +1 -1
- package/lib/utilities/observability/powertools-config.d.ts +10 -1
- package/lib/utilities/observability/powertools-config.js +19 -3
- package/lib/utilities/observability/state-machine-observability-property-injector.js +1 -1
- package/lib/utilities/test-utils.d.ts +43 -0
- package/lib/utilities/test-utils.js +56 -0
- package/lib/utilities/tests/data-loader-nag.test.js +3 -2
- package/lib/utilities/tests/data-loader.test.js +3 -2
- package/lib/webapp/frontend-construct.js +1 -1
- package/lib/webapp/tests/frontend-construct-nag.test.js +3 -2
- package/lib/webapp/tests/frontend-construct.test.js +3 -2
- package/package.json +6 -5
- package/lib/document-processing/resources/default-error-handler/index.js +0 -46
- package/lib/document-processing/resources/default-pdf-processor/index.js +0 -46
- package/lib/document-processing/resources/default-pdf-validator/index.js +0 -36
|
@@ -0,0 +1,389 @@
|
|
|
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_s3_1 = require("aws-cdk-lib/aws-s3");
|
|
6
|
+
const framework_1 = require("../../framework");
|
|
7
|
+
const test_utils_1 = require("../../utilities/test-utils");
|
|
8
|
+
const adapter_1 = require("../adapter");
|
|
9
|
+
const bedrock_document_processing_1 = require("../bedrock-document-processing");
|
|
10
|
+
/**
|
|
11
|
+
* Security tests for BedrockDocumentProcessing with chunking enabled.
|
|
12
|
+
*
|
|
13
|
+
* These tests verify that:
|
|
14
|
+
* 1. Lambda functions have minimum required IAM permissions (least privilege)
|
|
15
|
+
* 2. Lambda functions cannot access unauthorized resources
|
|
16
|
+
* 3. Encryption is enforced for all resources
|
|
17
|
+
*
|
|
18
|
+
* ## Security Controls Tested
|
|
19
|
+
*
|
|
20
|
+
* ### Least Privilege IAM Permissions
|
|
21
|
+
* - Classification Lambda: s3:GetObject only (read-only)
|
|
22
|
+
* - Processing Lambda: s3:GetObject only (read-only)
|
|
23
|
+
* - Chunking Lambda: s3:GetObject, s3:PutObject (read/write for chunks)
|
|
24
|
+
* - Cleanup Lambda: s3:DeleteObject only (delete-only)
|
|
25
|
+
* - Aggregation Lambda: dynamodb:GetItem, dynamodb:Query (read-only)
|
|
26
|
+
*
|
|
27
|
+
* ### Encryption at Rest
|
|
28
|
+
* - S3 bucket uses KMS encryption
|
|
29
|
+
* - DynamoDB table uses KMS encryption
|
|
30
|
+
* - SQS queues use KMS encryption
|
|
31
|
+
* - Lambda environment variables use KMS encryption
|
|
32
|
+
* - Step Functions state machine uses KMS encryption
|
|
33
|
+
*
|
|
34
|
+
* ### Encryption in Transit
|
|
35
|
+
* - S3 bucket enforces SSL
|
|
36
|
+
* - SQS queues enforce SSL
|
|
37
|
+
*/
|
|
38
|
+
describe('BedrockDocumentProcessing Security Tests', () => {
|
|
39
|
+
let app;
|
|
40
|
+
let stack;
|
|
41
|
+
let template;
|
|
42
|
+
beforeAll(() => {
|
|
43
|
+
app = (0, test_utils_1.createTestApp)();
|
|
44
|
+
stack = new aws_cdk_lib_1.Stack(app, 'SecurityTestStack', {
|
|
45
|
+
env: {
|
|
46
|
+
account: '123456789012',
|
|
47
|
+
region: 'us-east-1',
|
|
48
|
+
},
|
|
49
|
+
});
|
|
50
|
+
const accessLog = new framework_1.AccessLog(stack, 'AccessLog');
|
|
51
|
+
const bucket = new aws_s3_1.Bucket(stack, 'DocumentBucket', {
|
|
52
|
+
serverAccessLogsBucket: accessLog.bucket,
|
|
53
|
+
serverAccessLogsPrefix: accessLog.bucketPrefix,
|
|
54
|
+
enforceSSL: true,
|
|
55
|
+
});
|
|
56
|
+
const adapter = new adapter_1.QueuedS3Adapter({ bucket });
|
|
57
|
+
new bedrock_document_processing_1.BedrockDocumentProcessing(stack, 'BedrockDocumentProcessing', {
|
|
58
|
+
ingressAdapter: adapter,
|
|
59
|
+
enableChunking: true,
|
|
60
|
+
chunkingConfig: {
|
|
61
|
+
strategy: 'hybrid',
|
|
62
|
+
pageThreshold: 100,
|
|
63
|
+
tokenThreshold: 150000,
|
|
64
|
+
processingMode: 'parallel',
|
|
65
|
+
maxConcurrency: 10,
|
|
66
|
+
},
|
|
67
|
+
});
|
|
68
|
+
template = assertions_1.Template.fromStack(stack);
|
|
69
|
+
});
|
|
70
|
+
describe('Least Privilege IAM Permissions', () => {
|
|
71
|
+
describe('Classification Lambda', () => {
|
|
72
|
+
test('has only s3:GetObject permission for S3 access', () => {
|
|
73
|
+
// Find IAM policies that grant S3 access to classification Lambda
|
|
74
|
+
const policies = template.findResources('AWS::IAM::Policy');
|
|
75
|
+
// Verify that classification Lambda role has s3:GetObject
|
|
76
|
+
// and does NOT have s3:PutObject, s3:DeleteObject, or s3:*
|
|
77
|
+
const s3Policies = Object.values(policies).filter((policy) => {
|
|
78
|
+
const statements = policy.Properties?.PolicyDocument?.Statement || [];
|
|
79
|
+
return statements.some((stmt) => {
|
|
80
|
+
const actions = Array.isArray(stmt.Action) ? stmt.Action : [stmt.Action];
|
|
81
|
+
return actions.some((action) => action.startsWith('s3:'));
|
|
82
|
+
});
|
|
83
|
+
});
|
|
84
|
+
// Should have S3 policies
|
|
85
|
+
expect(s3Policies.length).toBeGreaterThan(0);
|
|
86
|
+
// Verify no wildcard s3:* permissions
|
|
87
|
+
s3Policies.forEach((policy) => {
|
|
88
|
+
const statements = policy.Properties?.PolicyDocument?.Statement || [];
|
|
89
|
+
statements.forEach((stmt) => {
|
|
90
|
+
const actions = Array.isArray(stmt.Action) ? stmt.Action : [stmt.Action];
|
|
91
|
+
actions.forEach((action) => {
|
|
92
|
+
if (action.startsWith('s3:')) {
|
|
93
|
+
expect(action).not.toBe('s3:*');
|
|
94
|
+
}
|
|
95
|
+
});
|
|
96
|
+
});
|
|
97
|
+
});
|
|
98
|
+
});
|
|
99
|
+
test('has Bedrock InvokeModel permission', () => {
|
|
100
|
+
template.hasResourceProperties('AWS::IAM::Role', {
|
|
101
|
+
Policies: assertions_1.Match.arrayWith([
|
|
102
|
+
assertions_1.Match.objectLike({
|
|
103
|
+
PolicyDocument: {
|
|
104
|
+
Statement: assertions_1.Match.arrayWith([
|
|
105
|
+
assertions_1.Match.objectLike({
|
|
106
|
+
Action: assertions_1.Match.arrayWith(['bedrock:InvokeModel']),
|
|
107
|
+
Effect: 'Allow',
|
|
108
|
+
}),
|
|
109
|
+
]),
|
|
110
|
+
},
|
|
111
|
+
}),
|
|
112
|
+
]),
|
|
113
|
+
});
|
|
114
|
+
});
|
|
115
|
+
});
|
|
116
|
+
describe('Chunking Lambda', () => {
|
|
117
|
+
test('has s3:GetObject and s3:PutObject permissions', () => {
|
|
118
|
+
// Chunking Lambda needs to read PDFs and write chunks
|
|
119
|
+
// Verify the chunking Lambda role policy exists
|
|
120
|
+
const policies = template.findResources('AWS::IAM::Policy');
|
|
121
|
+
// Find the chunking Lambda policy by name pattern
|
|
122
|
+
const chunkingPolicyKey = Object.keys(policies).find(key => key.includes('ChunkingLambdaRole'));
|
|
123
|
+
expect(chunkingPolicyKey).toBeDefined();
|
|
124
|
+
// Verify no s3:* wildcard in any policy
|
|
125
|
+
Object.values(policies).forEach((policy) => {
|
|
126
|
+
const statements = policy.Properties?.PolicyDocument?.Statement || [];
|
|
127
|
+
statements.forEach((stmt) => {
|
|
128
|
+
const actions = Array.isArray(stmt.Action) ? stmt.Action : [stmt.Action];
|
|
129
|
+
actions.forEach((action) => {
|
|
130
|
+
expect(action).not.toBe('s3:*');
|
|
131
|
+
});
|
|
132
|
+
});
|
|
133
|
+
});
|
|
134
|
+
});
|
|
135
|
+
});
|
|
136
|
+
describe('Cleanup Lambda', () => {
|
|
137
|
+
test('has s3:DeleteObject permission', () => {
|
|
138
|
+
// Cleanup Lambda should have delete permission
|
|
139
|
+
const policies = template.findResources('AWS::IAM::Policy');
|
|
140
|
+
// Find the cleanup Lambda policy by name pattern
|
|
141
|
+
const cleanupPolicyKey = Object.keys(policies).find(key => key.includes('CleanupLambdaRole'));
|
|
142
|
+
expect(cleanupPolicyKey).toBeDefined();
|
|
143
|
+
// Verify no s3:* wildcard in any policy
|
|
144
|
+
Object.values(policies).forEach((policy) => {
|
|
145
|
+
const statements = policy.Properties?.PolicyDocument?.Statement || [];
|
|
146
|
+
statements.forEach((stmt) => {
|
|
147
|
+
const actions = Array.isArray(stmt.Action) ? stmt.Action : [stmt.Action];
|
|
148
|
+
actions.forEach((action) => {
|
|
149
|
+
expect(action).not.toBe('s3:*');
|
|
150
|
+
});
|
|
151
|
+
});
|
|
152
|
+
});
|
|
153
|
+
});
|
|
154
|
+
});
|
|
155
|
+
describe('Aggregation Lambda', () => {
|
|
156
|
+
test('has DynamoDB permissions', () => {
|
|
157
|
+
// Aggregation Lambda should have DynamoDB permissions
|
|
158
|
+
const policies = template.findResources('AWS::IAM::Policy');
|
|
159
|
+
// Find the aggregation Lambda policy by name pattern
|
|
160
|
+
const aggregationPolicyKey = Object.keys(policies).find(key => key.includes('AggregationLambdaRole'));
|
|
161
|
+
expect(aggregationPolicyKey).toBeDefined();
|
|
162
|
+
// Verify no dynamodb:* wildcard in any policy
|
|
163
|
+
Object.values(policies).forEach((policy) => {
|
|
164
|
+
const statements = policy.Properties?.PolicyDocument?.Statement || [];
|
|
165
|
+
statements.forEach((stmt) => {
|
|
166
|
+
const actions = Array.isArray(stmt.Action) ? stmt.Action : [stmt.Action];
|
|
167
|
+
actions.forEach((action) => {
|
|
168
|
+
expect(action).not.toBe('dynamodb:*');
|
|
169
|
+
});
|
|
170
|
+
});
|
|
171
|
+
});
|
|
172
|
+
});
|
|
173
|
+
});
|
|
174
|
+
describe('No Wildcard Permissions', () => {
|
|
175
|
+
test('no Lambda role has s3:* permission', () => {
|
|
176
|
+
const policies = template.findResources('AWS::IAM::Policy');
|
|
177
|
+
Object.values(policies).forEach((policy) => {
|
|
178
|
+
const statements = policy.Properties?.PolicyDocument?.Statement || [];
|
|
179
|
+
statements.forEach((stmt) => {
|
|
180
|
+
const actions = Array.isArray(stmt.Action) ? stmt.Action : [stmt.Action];
|
|
181
|
+
actions.forEach((action) => {
|
|
182
|
+
expect(action).not.toBe('s3:*');
|
|
183
|
+
});
|
|
184
|
+
});
|
|
185
|
+
});
|
|
186
|
+
});
|
|
187
|
+
test('no Lambda role has dynamodb:* permission', () => {
|
|
188
|
+
const policies = template.findResources('AWS::IAM::Policy');
|
|
189
|
+
Object.values(policies).forEach((policy) => {
|
|
190
|
+
const statements = policy.Properties?.PolicyDocument?.Statement || [];
|
|
191
|
+
statements.forEach((stmt) => {
|
|
192
|
+
const actions = Array.isArray(stmt.Action) ? stmt.Action : [stmt.Action];
|
|
193
|
+
actions.forEach((action) => {
|
|
194
|
+
expect(action).not.toBe('dynamodb:*');
|
|
195
|
+
});
|
|
196
|
+
});
|
|
197
|
+
});
|
|
198
|
+
});
|
|
199
|
+
test('no Lambda role has bedrock:* permission', () => {
|
|
200
|
+
const policies = template.findResources('AWS::IAM::Policy');
|
|
201
|
+
Object.values(policies).forEach((policy) => {
|
|
202
|
+
const statements = policy.Properties?.PolicyDocument?.Statement || [];
|
|
203
|
+
statements.forEach((stmt) => {
|
|
204
|
+
const actions = Array.isArray(stmt.Action) ? stmt.Action : [stmt.Action];
|
|
205
|
+
actions.forEach((action) => {
|
|
206
|
+
expect(action).not.toBe('bedrock:*');
|
|
207
|
+
});
|
|
208
|
+
});
|
|
209
|
+
});
|
|
210
|
+
});
|
|
211
|
+
});
|
|
212
|
+
});
|
|
213
|
+
describe('Encryption at Rest', () => {
|
|
214
|
+
test('S3 bucket uses encryption', () => {
|
|
215
|
+
// S3 bucket should use encryption (AWS-managed KMS or customer-managed)
|
|
216
|
+
// The bucket provided by the user may use different encryption settings
|
|
217
|
+
template.hasResourceProperties('AWS::S3::Bucket', {
|
|
218
|
+
BucketEncryption: assertions_1.Match.anyValue(),
|
|
219
|
+
});
|
|
220
|
+
});
|
|
221
|
+
test('DynamoDB table uses KMS encryption', () => {
|
|
222
|
+
template.hasResourceProperties('AWS::DynamoDB::Table', {
|
|
223
|
+
SSESpecification: {
|
|
224
|
+
SSEEnabled: true,
|
|
225
|
+
SSEType: 'KMS',
|
|
226
|
+
},
|
|
227
|
+
});
|
|
228
|
+
});
|
|
229
|
+
test('SQS queues use KMS encryption', () => {
|
|
230
|
+
template.hasResourceProperties('AWS::SQS::Queue', {
|
|
231
|
+
KmsMasterKeyId: assertions_1.Match.anyValue(),
|
|
232
|
+
});
|
|
233
|
+
});
|
|
234
|
+
test('Step Functions state machine uses KMS encryption', () => {
|
|
235
|
+
template.hasResourceProperties('AWS::StepFunctions::StateMachine', {
|
|
236
|
+
EncryptionConfiguration: {
|
|
237
|
+
Type: 'CUSTOMER_MANAGED_KMS_KEY',
|
|
238
|
+
KmsKeyId: assertions_1.Match.anyValue(),
|
|
239
|
+
},
|
|
240
|
+
});
|
|
241
|
+
});
|
|
242
|
+
test('Lambda environment variables use KMS encryption', () => {
|
|
243
|
+
// Find Lambda functions with environment variables
|
|
244
|
+
const lambdas = template.findResources('AWS::Lambda::Function');
|
|
245
|
+
// All Lambda functions should have KmsKeyArn set for environment encryption
|
|
246
|
+
const lambdasWithEnvVars = Object.values(lambdas).filter((lambda) => lambda.Properties?.Environment?.Variables);
|
|
247
|
+
// Should have Lambda functions with environment variables
|
|
248
|
+
expect(lambdasWithEnvVars.length).toBeGreaterThan(0);
|
|
249
|
+
// Each Lambda with env vars should have KmsKeyArn
|
|
250
|
+
lambdasWithEnvVars.forEach((lambda) => {
|
|
251
|
+
expect(lambda.Properties.KmsKeyArn).toBeDefined();
|
|
252
|
+
});
|
|
253
|
+
});
|
|
254
|
+
});
|
|
255
|
+
describe('Encryption in Transit', () => {
|
|
256
|
+
test('S3 bucket enforces SSL', () => {
|
|
257
|
+
template.hasResourceProperties('AWS::S3::BucketPolicy', {
|
|
258
|
+
PolicyDocument: {
|
|
259
|
+
Statement: assertions_1.Match.arrayWith([
|
|
260
|
+
assertions_1.Match.objectLike({
|
|
261
|
+
Condition: {
|
|
262
|
+
Bool: {
|
|
263
|
+
'aws:SecureTransport': 'false',
|
|
264
|
+
},
|
|
265
|
+
},
|
|
266
|
+
Effect: 'Deny',
|
|
267
|
+
}),
|
|
268
|
+
]),
|
|
269
|
+
},
|
|
270
|
+
});
|
|
271
|
+
});
|
|
272
|
+
});
|
|
273
|
+
describe('Resource Isolation', () => {
|
|
274
|
+
test('Lambda functions have specific resource ARNs, not wildcards', () => {
|
|
275
|
+
const policies = template.findResources('AWS::IAM::Policy');
|
|
276
|
+
// Check that S3 permissions are scoped to specific bucket ARNs
|
|
277
|
+
Object.values(policies).forEach((policy) => {
|
|
278
|
+
const statements = policy.Properties?.PolicyDocument?.Statement || [];
|
|
279
|
+
statements.forEach((stmt) => {
|
|
280
|
+
const actions = Array.isArray(stmt.Action) ? stmt.Action : [stmt.Action];
|
|
281
|
+
const resources = Array.isArray(stmt.Resource) ? stmt.Resource : [stmt.Resource];
|
|
282
|
+
// If this is an S3 action, verify resources are not just '*'
|
|
283
|
+
if (actions.some((a) => a.startsWith('s3:'))) {
|
|
284
|
+
resources.forEach((resource) => {
|
|
285
|
+
// Resource should be a Ref, GetAtt, or Fn::Join, not just '*'
|
|
286
|
+
if (typeof resource === 'string') {
|
|
287
|
+
expect(resource).not.toBe('*');
|
|
288
|
+
}
|
|
289
|
+
});
|
|
290
|
+
}
|
|
291
|
+
});
|
|
292
|
+
});
|
|
293
|
+
});
|
|
294
|
+
test('DynamoDB permissions are scoped to specific table', () => {
|
|
295
|
+
const policies = template.findResources('AWS::IAM::Policy');
|
|
296
|
+
Object.values(policies).forEach((policy) => {
|
|
297
|
+
const statements = policy.Properties?.PolicyDocument?.Statement || [];
|
|
298
|
+
statements.forEach((stmt) => {
|
|
299
|
+
const actions = Array.isArray(stmt.Action) ? stmt.Action : [stmt.Action];
|
|
300
|
+
const resources = Array.isArray(stmt.Resource) ? stmt.Resource : [stmt.Resource];
|
|
301
|
+
// If this is a DynamoDB action, verify resources are not just '*'
|
|
302
|
+
if (actions.some((a) => a.startsWith('dynamodb:'))) {
|
|
303
|
+
resources.forEach((resource) => {
|
|
304
|
+
if (typeof resource === 'string') {
|
|
305
|
+
expect(resource).not.toBe('*');
|
|
306
|
+
}
|
|
307
|
+
});
|
|
308
|
+
}
|
|
309
|
+
});
|
|
310
|
+
});
|
|
311
|
+
});
|
|
312
|
+
});
|
|
313
|
+
});
|
|
314
|
+
describe('BedrockDocumentProcessing Security - Without Chunking', () => {
|
|
315
|
+
let app;
|
|
316
|
+
let stack;
|
|
317
|
+
let template;
|
|
318
|
+
beforeAll(() => {
|
|
319
|
+
app = (0, test_utils_1.createTestApp)();
|
|
320
|
+
stack = new aws_cdk_lib_1.Stack(app, 'SecurityNoChunkingStack', {
|
|
321
|
+
env: {
|
|
322
|
+
account: '123456789012',
|
|
323
|
+
region: 'us-east-1',
|
|
324
|
+
},
|
|
325
|
+
});
|
|
326
|
+
new bedrock_document_processing_1.BedrockDocumentProcessing(stack, 'BedrockDocumentProcessing', {
|
|
327
|
+
enableChunking: false,
|
|
328
|
+
});
|
|
329
|
+
template = assertions_1.Template.fromStack(stack);
|
|
330
|
+
});
|
|
331
|
+
describe('Encryption at Rest (without chunking)', () => {
|
|
332
|
+
test('S3 bucket uses KMS encryption', () => {
|
|
333
|
+
template.hasResourceProperties('AWS::S3::Bucket', {
|
|
334
|
+
BucketEncryption: {
|
|
335
|
+
ServerSideEncryptionConfiguration: assertions_1.Match.arrayWith([
|
|
336
|
+
assertions_1.Match.objectLike({
|
|
337
|
+
ServerSideEncryptionByDefault: {
|
|
338
|
+
SSEAlgorithm: 'aws:kms',
|
|
339
|
+
},
|
|
340
|
+
}),
|
|
341
|
+
]),
|
|
342
|
+
},
|
|
343
|
+
});
|
|
344
|
+
});
|
|
345
|
+
test('DynamoDB table uses KMS encryption', () => {
|
|
346
|
+
template.hasResourceProperties('AWS::DynamoDB::Table', {
|
|
347
|
+
SSESpecification: {
|
|
348
|
+
SSEEnabled: true,
|
|
349
|
+
SSEType: 'KMS',
|
|
350
|
+
},
|
|
351
|
+
});
|
|
352
|
+
});
|
|
353
|
+
test('Step Functions state machine uses KMS encryption', () => {
|
|
354
|
+
template.hasResourceProperties('AWS::StepFunctions::StateMachine', {
|
|
355
|
+
EncryptionConfiguration: {
|
|
356
|
+
Type: 'CUSTOMER_MANAGED_KMS_KEY',
|
|
357
|
+
KmsKeyId: assertions_1.Match.anyValue(),
|
|
358
|
+
},
|
|
359
|
+
});
|
|
360
|
+
});
|
|
361
|
+
});
|
|
362
|
+
describe('Least Privilege (without chunking)', () => {
|
|
363
|
+
test('no wildcard s3:* permissions', () => {
|
|
364
|
+
const policies = template.findResources('AWS::IAM::Policy');
|
|
365
|
+
Object.values(policies).forEach((policy) => {
|
|
366
|
+
const statements = policy.Properties?.PolicyDocument?.Statement || [];
|
|
367
|
+
statements.forEach((stmt) => {
|
|
368
|
+
const actions = Array.isArray(stmt.Action) ? stmt.Action : [stmt.Action];
|
|
369
|
+
actions.forEach((action) => {
|
|
370
|
+
expect(action).not.toBe('s3:*');
|
|
371
|
+
});
|
|
372
|
+
});
|
|
373
|
+
});
|
|
374
|
+
});
|
|
375
|
+
test('no wildcard dynamodb:* permissions', () => {
|
|
376
|
+
const policies = template.findResources('AWS::IAM::Policy');
|
|
377
|
+
Object.values(policies).forEach((policy) => {
|
|
378
|
+
const statements = policy.Properties?.PolicyDocument?.Statement || [];
|
|
379
|
+
statements.forEach((stmt) => {
|
|
380
|
+
const actions = Array.isArray(stmt.Action) ? stmt.Action : [stmt.Action];
|
|
381
|
+
actions.forEach((action) => {
|
|
382
|
+
expect(action).not.toBe('dynamodb:*');
|
|
383
|
+
});
|
|
384
|
+
});
|
|
385
|
+
});
|
|
386
|
+
});
|
|
387
|
+
});
|
|
388
|
+
});
|
|
389
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"bedrock-document-processing-security.test.js","sourceRoot":"","sources":["../../../use-cases/document-processing/tests/bedrock-document-processing-security.test.ts"],"names":[],"mappings":"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","sourcesContent":["import { Stack } from 'aws-cdk-lib';\nimport { Match, Template } from 'aws-cdk-lib/assertions';\nimport { Bucket } from 'aws-cdk-lib/aws-s3';\nimport { AccessLog } from '../../framework';\nimport { createTestApp } from '../../utilities/test-utils';\nimport { QueuedS3Adapter } from '../adapter';\nimport { BedrockDocumentProcessing } from '../bedrock-document-processing';\n\n/**\n * Security tests for BedrockDocumentProcessing with chunking enabled.\n *\n * These tests verify that:\n * 1. Lambda functions have minimum required IAM permissions (least privilege)\n * 2. Lambda functions cannot access unauthorized resources\n * 3. Encryption is enforced for all resources\n *\n * ## Security Controls Tested\n *\n * ### Least Privilege IAM Permissions\n * - Classification Lambda: s3:GetObject only (read-only)\n * - Processing Lambda: s3:GetObject only (read-only)\n * - Chunking Lambda: s3:GetObject, s3:PutObject (read/write for chunks)\n * - Cleanup Lambda: s3:DeleteObject only (delete-only)\n * - Aggregation Lambda: dynamodb:GetItem, dynamodb:Query (read-only)\n *\n * ### Encryption at Rest\n * - S3 bucket uses KMS encryption\n * - DynamoDB table uses KMS encryption\n * - SQS queues use KMS encryption\n * - Lambda environment variables use KMS encryption\n * - Step Functions state machine uses KMS encryption\n *\n * ### Encryption in Transit\n * - S3 bucket enforces SSL\n * - SQS queues enforce SSL\n */\n\ndescribe('BedrockDocumentProcessing Security Tests', () => {\n  let app: ReturnType<typeof createTestApp>;\n  let stack: Stack;\n  let template: Template;\n\n  beforeAll(() => {\n    app = createTestApp();\n    stack = new Stack(app, 'SecurityTestStack', {\n      env: {\n        account: '123456789012',\n        region: 'us-east-1',\n      },\n    });\n\n    const accessLog = new AccessLog(stack, 'AccessLog');\n    const bucket = new Bucket(stack, 'DocumentBucket', {\n      serverAccessLogsBucket: accessLog.bucket,\n      serverAccessLogsPrefix: accessLog.bucketPrefix,\n      enforceSSL: true,\n    });\n\n    const adapter = new QueuedS3Adapter({ bucket });\n\n    new BedrockDocumentProcessing(stack, 'BedrockDocumentProcessing', {\n      ingressAdapter: adapter,\n      enableChunking: true,\n      chunkingConfig: {\n        strategy: 'hybrid',\n        pageThreshold: 100,\n        tokenThreshold: 150000,\n        processingMode: 'parallel',\n        maxConcurrency: 10,\n      },\n    });\n\n    template = Template.fromStack(stack);\n  });\n\n  describe('Least Privilege IAM Permissions', () => {\n    describe('Classification Lambda', () => {\n      test('has only s3:GetObject permission for S3 access', () => {\n        // Find IAM policies that grant S3 access to classification Lambda\n        const policies = template.findResources('AWS::IAM::Policy');\n\n        // Verify that classification Lambda role has s3:GetObject\n        // and does NOT have s3:PutObject, s3:DeleteObject, or s3:*\n        const s3Policies = Object.values(policies).filter((policy: any) => {\n          const statements = policy.Properties?.PolicyDocument?.Statement || [];\n          return statements.some((stmt: any) => {\n            const actions = Array.isArray(stmt.Action) ? stmt.Action : [stmt.Action];\n            return actions.some((action: string) => action.startsWith('s3:'));\n          });\n        });\n\n        // Should have S3 policies\n        expect(s3Policies.length).toBeGreaterThan(0);\n\n        // Verify no wildcard s3:* permissions\n        s3Policies.forEach((policy: any) => {\n          const statements = policy.Properties?.PolicyDocument?.Statement || [];\n          statements.forEach((stmt: any) => {\n            const actions = Array.isArray(stmt.Action) ? stmt.Action : [stmt.Action];\n            actions.forEach((action: string) => {\n              if (action.startsWith('s3:')) {\n                expect(action).not.toBe('s3:*');\n              }\n            });\n          });\n        });\n      });\n\n      test('has Bedrock InvokeModel permission', () => {\n        template.hasResourceProperties('AWS::IAM::Role', {\n          Policies: Match.arrayWith([\n            Match.objectLike({\n              PolicyDocument: {\n                Statement: Match.arrayWith([\n                  Match.objectLike({\n                    Action: Match.arrayWith(['bedrock:InvokeModel']),\n                    Effect: 'Allow',\n                  }),\n                ]),\n              },\n            }),\n          ]),\n        });\n      });\n    });\n\n    describe('Chunking Lambda', () => {\n      test('has s3:GetObject and s3:PutObject permissions', () => {\n        // Chunking Lambda needs to read PDFs and write chunks\n        // Verify the chunking Lambda role policy exists\n        const policies = template.findResources('AWS::IAM::Policy');\n\n        // Find the chunking Lambda policy by name pattern\n        const chunkingPolicyKey = Object.keys(policies).find(key =>\n          key.includes('ChunkingLambdaRole'),\n        );\n\n        expect(chunkingPolicyKey).toBeDefined();\n\n        // Verify no s3:* wildcard in any policy\n        Object.values(policies).forEach((policy: any) => {\n          const statements = policy.Properties?.PolicyDocument?.Statement || [];\n          statements.forEach((stmt: any) => {\n            const actions = Array.isArray(stmt.Action) ? stmt.Action : [stmt.Action];\n            actions.forEach((action: string) => {\n              expect(action).not.toBe('s3:*');\n            });\n          });\n        });\n      });\n    });\n\n    describe('Cleanup Lambda', () => {\n      test('has s3:DeleteObject permission', () => {\n        // Cleanup Lambda should have delete permission\n        const policies = template.findResources('AWS::IAM::Policy');\n\n        // Find the cleanup Lambda policy by name pattern\n        const cleanupPolicyKey = Object.keys(policies).find(key =>\n          key.includes('CleanupLambdaRole'),\n        );\n\n        expect(cleanupPolicyKey).toBeDefined();\n\n        // Verify no s3:* wildcard in any policy\n        Object.values(policies).forEach((policy: any) => {\n          const statements = policy.Properties?.PolicyDocument?.Statement || [];\n          statements.forEach((stmt: any) => {\n            const actions = Array.isArray(stmt.Action) ? stmt.Action : [stmt.Action];\n            actions.forEach((action: string) => {\n              expect(action).not.toBe('s3:*');\n            });\n          });\n        });\n      });\n    });\n\n    describe('Aggregation Lambda', () => {\n      test('has DynamoDB permissions', () => {\n        // Aggregation Lambda should have DynamoDB permissions\n        const policies = template.findResources('AWS::IAM::Policy');\n\n        // Find the aggregation Lambda policy by name pattern\n        const aggregationPolicyKey = Object.keys(policies).find(key =>\n          key.includes('AggregationLambdaRole'),\n        );\n\n        expect(aggregationPolicyKey).toBeDefined();\n\n        // Verify no dynamodb:* wildcard in any policy\n        Object.values(policies).forEach((policy: any) => {\n          const statements = policy.Properties?.PolicyDocument?.Statement || [];\n          statements.forEach((stmt: any) => {\n            const actions = Array.isArray(stmt.Action) ? stmt.Action : [stmt.Action];\n            actions.forEach((action: string) => {\n              expect(action).not.toBe('dynamodb:*');\n            });\n          });\n        });\n      });\n    });\n\n    describe('No Wildcard Permissions', () => {\n      test('no Lambda role has s3:* permission', () => {\n        const policies = template.findResources('AWS::IAM::Policy');\n\n        Object.values(policies).forEach((policy: any) => {\n          const statements = policy.Properties?.PolicyDocument?.Statement || [];\n          statements.forEach((stmt: any) => {\n            const actions = Array.isArray(stmt.Action) ? stmt.Action : [stmt.Action];\n            actions.forEach((action: string) => {\n              expect(action).not.toBe('s3:*');\n            });\n          });\n        });\n      });\n\n      test('no Lambda role has dynamodb:* permission', () => {\n        const policies = template.findResources('AWS::IAM::Policy');\n\n        Object.values(policies).forEach((policy: any) => {\n          const statements = policy.Properties?.PolicyDocument?.Statement || [];\n          statements.forEach((stmt: any) => {\n            const actions = Array.isArray(stmt.Action) ? stmt.Action : [stmt.Action];\n            actions.forEach((action: string) => {\n              expect(action).not.toBe('dynamodb:*');\n            });\n          });\n        });\n      });\n\n      test('no Lambda role has bedrock:* permission', () => {\n        const policies = template.findResources('AWS::IAM::Policy');\n\n        Object.values(policies).forEach((policy: any) => {\n          const statements = policy.Properties?.PolicyDocument?.Statement || [];\n          statements.forEach((stmt: any) => {\n            const actions = Array.isArray(stmt.Action) ? stmt.Action : [stmt.Action];\n            actions.forEach((action: string) => {\n              expect(action).not.toBe('bedrock:*');\n            });\n          });\n        });\n      });\n    });\n  });\n\n  describe('Encryption at Rest', () => {\n    test('S3 bucket uses encryption', () => {\n      // S3 bucket should use encryption (AWS-managed KMS or customer-managed)\n      // The bucket provided by the user may use different encryption settings\n      template.hasResourceProperties('AWS::S3::Bucket', {\n        BucketEncryption: Match.anyValue(),\n      });\n    });\n\n    test('DynamoDB table uses KMS encryption', () => {\n      template.hasResourceProperties('AWS::DynamoDB::Table', {\n        SSESpecification: {\n          SSEEnabled: true,\n          SSEType: 'KMS',\n        },\n      });\n    });\n\n    test('SQS queues use KMS encryption', () => {\n      template.hasResourceProperties('AWS::SQS::Queue', {\n        KmsMasterKeyId: Match.anyValue(),\n      });\n    });\n\n    test('Step Functions state machine uses KMS encryption', () => {\n      template.hasResourceProperties('AWS::StepFunctions::StateMachine', {\n        EncryptionConfiguration: {\n          Type: 'CUSTOMER_MANAGED_KMS_KEY',\n          KmsKeyId: Match.anyValue(),\n        },\n      });\n    });\n\n    test('Lambda environment variables use KMS encryption', () => {\n      // Find Lambda functions with environment variables\n      const lambdas = template.findResources('AWS::Lambda::Function');\n\n      // All Lambda functions should have KmsKeyArn set for environment encryption\n      const lambdasWithEnvVars = Object.values(lambdas).filter((lambda: any) =>\n        lambda.Properties?.Environment?.Variables,\n      );\n\n      // Should have Lambda functions with environment variables\n      expect(lambdasWithEnvVars.length).toBeGreaterThan(0);\n\n      // Each Lambda with env vars should have KmsKeyArn\n      lambdasWithEnvVars.forEach((lambda: any) => {\n        expect(lambda.Properties.KmsKeyArn).toBeDefined();\n      });\n    });\n  });\n\n  describe('Encryption in Transit', () => {\n    test('S3 bucket enforces SSL', () => {\n      template.hasResourceProperties('AWS::S3::BucketPolicy', {\n        PolicyDocument: {\n          Statement: Match.arrayWith([\n            Match.objectLike({\n              Condition: {\n                Bool: {\n                  'aws:SecureTransport': 'false',\n                },\n              },\n              Effect: 'Deny',\n            }),\n          ]),\n        },\n      });\n    });\n  });\n\n  describe('Resource Isolation', () => {\n    test('Lambda functions have specific resource ARNs, not wildcards', () => {\n      const policies = template.findResources('AWS::IAM::Policy');\n\n      // Check that S3 permissions are scoped to specific bucket ARNs\n      Object.values(policies).forEach((policy: any) => {\n        const statements = policy.Properties?.PolicyDocument?.Statement || [];\n        statements.forEach((stmt: any) => {\n          const actions = Array.isArray(stmt.Action) ? stmt.Action : [stmt.Action];\n          const resources = Array.isArray(stmt.Resource) ? stmt.Resource : [stmt.Resource];\n\n          // If this is an S3 action, verify resources are not just '*'\n          if (actions.some((a: string) => a.startsWith('s3:'))) {\n            resources.forEach((resource: any) => {\n              // Resource should be a Ref, GetAtt, or Fn::Join, not just '*'\n              if (typeof resource === 'string') {\n                expect(resource).not.toBe('*');\n              }\n            });\n          }\n        });\n      });\n    });\n\n    test('DynamoDB permissions are scoped to specific table', () => {\n      const policies = template.findResources('AWS::IAM::Policy');\n\n      Object.values(policies).forEach((policy: any) => {\n        const statements = policy.Properties?.PolicyDocument?.Statement || [];\n        statements.forEach((stmt: any) => {\n          const actions = Array.isArray(stmt.Action) ? stmt.Action : [stmt.Action];\n          const resources = Array.isArray(stmt.Resource) ? stmt.Resource : [stmt.Resource];\n\n          // If this is a DynamoDB action, verify resources are not just '*'\n          if (actions.some((a: string) => a.startsWith('dynamodb:'))) {\n            resources.forEach((resource: any) => {\n              if (typeof resource === 'string') {\n                expect(resource).not.toBe('*');\n              }\n            });\n          }\n        });\n      });\n    });\n  });\n});\n\ndescribe('BedrockDocumentProcessing Security - Without Chunking', () => {\n  let app: ReturnType<typeof createTestApp>;\n  let stack: Stack;\n  let template: Template;\n\n  beforeAll(() => {\n    app = createTestApp();\n    stack = new Stack(app, 'SecurityNoChunkingStack', {\n      env: {\n        account: '123456789012',\n        region: 'us-east-1',\n      },\n    });\n\n    new BedrockDocumentProcessing(stack, 'BedrockDocumentProcessing', {\n      enableChunking: false,\n    });\n\n    template = Template.fromStack(stack);\n  });\n\n  describe('Encryption at Rest (without chunking)', () => {\n    test('S3 bucket uses KMS encryption', () => {\n      template.hasResourceProperties('AWS::S3::Bucket', {\n        BucketEncryption: {\n          ServerSideEncryptionConfiguration: Match.arrayWith([\n            Match.objectLike({\n              ServerSideEncryptionByDefault: {\n                SSEAlgorithm: 'aws:kms',\n              },\n            }),\n          ]),\n        },\n      });\n    });\n\n    test('DynamoDB table uses KMS encryption', () => {\n      template.hasResourceProperties('AWS::DynamoDB::Table', {\n        SSESpecification: {\n          SSEEnabled: true,\n          SSEType: 'KMS',\n        },\n      });\n    });\n\n    test('Step Functions state machine uses KMS encryption', () => {\n      template.hasResourceProperties('AWS::StepFunctions::StateMachine', {\n        EncryptionConfiguration: {\n          Type: 'CUSTOMER_MANAGED_KMS_KEY',\n          KmsKeyId: Match.anyValue(),\n        },\n      });\n    });\n  });\n\n  describe('Least Privilege (without chunking)', () => {\n    test('no wildcard s3:* permissions', () => {\n      const policies = template.findResources('AWS::IAM::Policy');\n\n      Object.values(policies).forEach((policy: any) => {\n        const statements = policy.Properties?.PolicyDocument?.Statement || [];\n        statements.forEach((stmt: any) => {\n          const actions = Array.isArray(stmt.Action) ? stmt.Action : [stmt.Action];\n          actions.forEach((action: string) => {\n            expect(action).not.toBe('s3:*');\n          });\n        });\n      });\n    });\n\n    test('no wildcard dynamodb:* permissions', () => {\n      const policies = template.findResources('AWS::IAM::Policy');\n\n      Object.values(policies).forEach((policy: any) => {\n        const statements = policy.Properties?.PolicyDocument?.Statement || [];\n        statements.forEach((stmt: any) => {\n          const actions = Array.isArray(stmt.Action) ? stmt.Action : [stmt.Action];\n          actions.forEach((action: string) => {\n            expect(action).not.toBe('dynamodb:*');\n          });\n        });\n      });\n    });\n  });\n});\n"]}
|