@cdklabs/cdk-appmod-catalog-blueprints 1.5.0 → 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.
Files changed (92) hide show
  1. package/.jsii +2537 -204
  2. package/lib/document-processing/adapter/adapter.d.ts +4 -2
  3. package/lib/document-processing/adapter/adapter.js +1 -1
  4. package/lib/document-processing/adapter/queued-s3-adapter.d.ts +9 -2
  5. package/lib/document-processing/adapter/queued-s3-adapter.js +29 -15
  6. package/lib/document-processing/agentic-document-processing.d.ts +4 -0
  7. package/lib/document-processing/agentic-document-processing.js +20 -10
  8. package/lib/document-processing/base-document-processing.d.ts +54 -2
  9. package/lib/document-processing/base-document-processing.js +136 -82
  10. package/lib/document-processing/bedrock-document-processing.d.ts +202 -2
  11. package/lib/document-processing/bedrock-document-processing.js +717 -77
  12. package/lib/document-processing/chunking-config.d.ts +614 -0
  13. package/lib/document-processing/chunking-config.js +5 -0
  14. package/lib/document-processing/default-document-processing-config.js +1 -1
  15. package/lib/document-processing/index.d.ts +1 -0
  16. package/lib/document-processing/index.js +2 -1
  17. package/lib/document-processing/resources/aggregation/handler.py +567 -0
  18. package/lib/document-processing/resources/aggregation/requirements.txt +7 -0
  19. package/lib/document-processing/resources/aggregation/test_handler.py +362 -0
  20. package/lib/document-processing/resources/cleanup/handler.py +276 -0
  21. package/lib/document-processing/resources/cleanup/requirements.txt +5 -0
  22. package/lib/document-processing/resources/cleanup/test_handler.py +436 -0
  23. package/lib/document-processing/resources/default-bedrock-invoke/index.py +85 -3
  24. package/lib/document-processing/resources/default-bedrock-invoke/test_index.py +622 -0
  25. package/lib/document-processing/resources/pdf-chunking/README.md +313 -0
  26. package/lib/document-processing/resources/pdf-chunking/chunking_strategies.py +460 -0
  27. package/lib/document-processing/resources/pdf-chunking/error_handling.py +491 -0
  28. package/lib/document-processing/resources/pdf-chunking/handler.py +958 -0
  29. package/lib/document-processing/resources/pdf-chunking/metrics.py +435 -0
  30. package/lib/document-processing/resources/pdf-chunking/requirements.txt +3 -0
  31. package/lib/document-processing/resources/pdf-chunking/strategy_selection.py +420 -0
  32. package/lib/document-processing/resources/pdf-chunking/structured_logging.py +457 -0
  33. package/lib/document-processing/resources/pdf-chunking/test_chunking_strategies.py +353 -0
  34. package/lib/document-processing/resources/pdf-chunking/test_error_handling.py +487 -0
  35. package/lib/document-processing/resources/pdf-chunking/test_handler.py +609 -0
  36. package/lib/document-processing/resources/pdf-chunking/test_integration.py +694 -0
  37. package/lib/document-processing/resources/pdf-chunking/test_metrics.py +532 -0
  38. package/lib/document-processing/resources/pdf-chunking/test_strategy_selection.py +471 -0
  39. package/lib/document-processing/resources/pdf-chunking/test_structured_logging.py +449 -0
  40. package/lib/document-processing/resources/pdf-chunking/test_token_estimation.py +374 -0
  41. package/lib/document-processing/resources/pdf-chunking/token_estimation.py +189 -0
  42. package/lib/document-processing/tests/agentic-document-processing-nag.test.js +4 -3
  43. package/lib/document-processing/tests/agentic-document-processing.test.js +488 -4
  44. package/lib/document-processing/tests/base-document-processing-nag.test.js +9 -2
  45. package/lib/document-processing/tests/base-document-processing-schema.test.d.ts +1 -0
  46. package/lib/document-processing/tests/base-document-processing-schema.test.js +337 -0
  47. package/lib/document-processing/tests/base-document-processing.test.js +114 -8
  48. package/lib/document-processing/tests/bedrock-document-processing-chunking-nag.test.d.ts +1 -0
  49. package/lib/document-processing/tests/bedrock-document-processing-chunking-nag.test.js +382 -0
  50. package/lib/document-processing/tests/bedrock-document-processing-nag.test.js +4 -3
  51. package/lib/document-processing/tests/bedrock-document-processing-security.test.d.ts +1 -0
  52. package/lib/document-processing/tests/bedrock-document-processing-security.test.js +389 -0
  53. package/lib/document-processing/tests/bedrock-document-processing.test.js +808 -8
  54. package/lib/document-processing/tests/chunking-config.test.d.ts +1 -0
  55. package/lib/document-processing/tests/chunking-config.test.js +238 -0
  56. package/lib/document-processing/tests/queued-s3-adapter-nag.test.js +9 -2
  57. package/lib/document-processing/tests/queued-s3-adapter.test.js +17 -6
  58. package/lib/framework/agents/base-agent.js +1 -1
  59. package/lib/framework/agents/batch-agent.js +1 -1
  60. package/lib/framework/agents/default-agent-config.js +1 -1
  61. package/lib/framework/bedrock/bedrock.js +1 -1
  62. package/lib/framework/custom-resource/default-runtimes.js +1 -1
  63. package/lib/framework/foundation/access-log.js +1 -1
  64. package/lib/framework/foundation/eventbridge-broker.js +1 -1
  65. package/lib/framework/foundation/network.js +1 -1
  66. package/lib/framework/tests/access-log.test.js +5 -2
  67. package/lib/framework/tests/batch-agent.test.js +5 -2
  68. package/lib/framework/tests/bedrock.test.js +5 -2
  69. package/lib/framework/tests/eventbridge-broker.test.js +5 -2
  70. package/lib/framework/tests/framework-nag.test.js +16 -8
  71. package/lib/framework/tests/network.test.js +9 -4
  72. package/lib/tsconfig.tsbuildinfo +1 -1
  73. package/lib/utilities/data-loader.js +1 -1
  74. package/lib/utilities/lambda-iam-utils.js +1 -1
  75. package/lib/utilities/observability/cloudfront-distribution-observability-property-injector.js +1 -1
  76. package/lib/utilities/observability/default-observability-config.js +1 -1
  77. package/lib/utilities/observability/lambda-observability-property-injector.js +1 -1
  78. package/lib/utilities/observability/log-group-data-protection-utils.js +1 -1
  79. package/lib/utilities/observability/powertools-config.d.ts +10 -1
  80. package/lib/utilities/observability/powertools-config.js +19 -3
  81. package/lib/utilities/observability/state-machine-observability-property-injector.js +1 -1
  82. package/lib/utilities/test-utils.d.ts +43 -0
  83. package/lib/utilities/test-utils.js +56 -0
  84. package/lib/utilities/tests/data-loader-nag.test.js +3 -2
  85. package/lib/utilities/tests/data-loader.test.js +3 -2
  86. package/lib/webapp/frontend-construct.js +1 -1
  87. package/lib/webapp/tests/frontend-construct-nag.test.js +3 -2
  88. package/lib/webapp/tests/frontend-construct.test.js +3 -2
  89. package/package.json +6 -5
  90. package/lib/document-processing/resources/default-error-handler/index.js +0 -46
  91. package/lib/document-processing/resources/default-pdf-processor/index.js +0 -46
  92. 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"]}