@cdklabs/cdk-appmod-catalog-blueprints 1.2.2 → 1.4.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 (69) hide show
  1. package/.jsii +804 -173
  2. package/README.md +76 -98
  3. package/lib/document-processing/adapter/queued-s3-adapter.js +1 -1
  4. package/lib/document-processing/agentic-document-processing.d.ts +5 -28
  5. package/lib/document-processing/agentic-document-processing.js +8 -63
  6. package/lib/document-processing/base-document-processing.js +4 -20
  7. package/lib/document-processing/bedrock-document-processing.d.ts +4 -32
  8. package/lib/document-processing/bedrock-document-processing.js +10 -37
  9. package/lib/document-processing/default-document-processing-config.js +1 -1
  10. package/lib/document-processing/tests/agentic-document-processing-nag.test.js +12 -11
  11. package/lib/document-processing/tests/agentic-document-processing.test.js +136 -67
  12. package/lib/document-processing/tests/base-document-processing-nag.test.d.ts +1 -0
  13. package/lib/document-processing/tests/base-document-processing-nag.test.js +161 -0
  14. package/lib/document-processing/tests/base-document-processing.test.d.ts +1 -0
  15. package/lib/document-processing/tests/base-document-processing.test.js +499 -0
  16. package/lib/document-processing/tests/bedrock-document-processing-nag.test.js +3 -2
  17. package/lib/document-processing/tests/bedrock-document-processing.test.js +221 -40
  18. package/lib/document-processing/tests/queued-s3-adapter-nag.test.d.ts +1 -0
  19. package/lib/document-processing/tests/queued-s3-adapter-nag.test.js +122 -0
  20. package/lib/document-processing/tests/queued-s3-adapter.test.d.ts +1 -0
  21. package/lib/document-processing/tests/queued-s3-adapter.test.js +276 -0
  22. package/lib/framework/agents/base-agent.d.ts +90 -0
  23. package/lib/framework/agents/base-agent.js +55 -0
  24. package/lib/framework/agents/batch-agent.d.ts +11 -0
  25. package/lib/framework/agents/batch-agent.js +64 -0
  26. package/lib/framework/agents/default-agent-config.d.ts +3 -0
  27. package/lib/framework/agents/default-agent-config.js +12 -0
  28. package/lib/framework/agents/index.d.ts +3 -0
  29. package/lib/framework/agents/index.js +20 -0
  30. package/lib/framework/agents/resources/default-strands-agent/batch.py +99 -0
  31. package/lib/framework/agents/resources/default-strands-agent/models.py +7 -0
  32. package/lib/framework/agents/resources/default-strands-agent/requirements.txt +7 -0
  33. package/lib/framework/agents/resources/default-strands-agent/utils.py +36 -0
  34. package/lib/framework/bedrock/bedrock.d.ts +38 -0
  35. package/lib/framework/bedrock/bedrock.js +54 -0
  36. package/lib/framework/bedrock/index.d.ts +1 -0
  37. package/lib/framework/bedrock/index.js +18 -0
  38. package/lib/framework/custom-resource/default-runtimes.js +1 -1
  39. package/lib/framework/foundation/access-log.js +1 -1
  40. package/lib/framework/foundation/eventbridge-broker.js +1 -1
  41. package/lib/framework/foundation/network.js +1 -1
  42. package/lib/framework/index.d.ts +2 -0
  43. package/lib/framework/index.js +3 -1
  44. package/lib/framework/tests/access-log.test.d.ts +1 -0
  45. package/lib/framework/tests/access-log.test.js +146 -0
  46. package/lib/framework/tests/batch-agent.test.d.ts +1 -0
  47. package/lib/framework/tests/batch-agent.test.js +164 -0
  48. package/lib/framework/tests/bedrock.test.d.ts +1 -0
  49. package/lib/framework/tests/bedrock.test.js +68 -0
  50. package/lib/framework/tests/eventbridge-broker.test.d.ts +1 -0
  51. package/lib/framework/tests/eventbridge-broker.test.js +73 -0
  52. package/lib/framework/tests/framework-nag.test.d.ts +1 -0
  53. package/lib/framework/tests/framework-nag.test.js +155 -0
  54. package/lib/framework/tests/network.test.d.ts +1 -0
  55. package/lib/framework/tests/network.test.js +120 -0
  56. package/lib/tsconfig.tsbuildinfo +1 -1
  57. package/lib/utilities/data-loader.js +1 -1
  58. package/lib/utilities/lambda-iam-utils.js +4 -3
  59. package/lib/utilities/observability/cloudfront-distribution-observability-property-injector.js +1 -1
  60. package/lib/utilities/observability/default-observability-config.js +1 -1
  61. package/lib/utilities/observability/index.d.ts +1 -0
  62. package/lib/utilities/observability/index.js +2 -1
  63. package/lib/utilities/observability/lambda-observability-property-injector.js +1 -1
  64. package/lib/utilities/observability/log-group-data-protection-utils.d.ts +6 -0
  65. package/lib/utilities/observability/log-group-data-protection-utils.js +37 -0
  66. package/lib/utilities/observability/powertools-config.js +1 -1
  67. package/lib/utilities/observability/state-machine-observability-property-injector.js +1 -1
  68. package/lib/webapp/frontend-construct.js +1 -1
  69. package/package.json +8 -8
@@ -0,0 +1,499 @@
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_dynamodb_1 = require("aws-cdk-lib/aws-dynamodb");
6
+ const aws_kms_1 = require("aws-cdk-lib/aws-kms");
7
+ const aws_lambda_1 = require("aws-cdk-lib/aws-lambda");
8
+ const aws_s3_1 = require("aws-cdk-lib/aws-s3");
9
+ const aws_stepfunctions_tasks_1 = require("aws-cdk-lib/aws-stepfunctions-tasks");
10
+ const framework_1 = require("../../framework");
11
+ const eventbridge_broker_1 = require("../../framework/foundation/eventbridge-broker");
12
+ const adapter_1 = require("../adapter");
13
+ const base_document_processing_1 = require("../base-document-processing");
14
+ // Concrete test implementation of BaseDocumentProcessing
15
+ class TestDocumentProcessing extends base_document_processing_1.BaseDocumentProcessing {
16
+ constructor(scope, id, props) {
17
+ super(scope, id, props);
18
+ this.classificationFn = new aws_lambda_1.Function(this, 'ClassificationFn', {
19
+ runtime: aws_lambda_1.Runtime.NODEJS_20_X,
20
+ handler: 'index.handler',
21
+ code: aws_lambda_1.Code.fromInline('exports.handler = async () => ({ documentClassification: "TEST" });'),
22
+ });
23
+ this.processingFn = new aws_lambda_1.Function(this, 'ProcessingFn', {
24
+ runtime: aws_lambda_1.Runtime.NODEJS_20_X,
25
+ handler: 'index.handler',
26
+ code: aws_lambda_1.Code.fromInline('exports.handler = async () => ({ result: {} });'),
27
+ });
28
+ }
29
+ classificationStep() {
30
+ return new aws_stepfunctions_tasks_1.LambdaInvoke(this, 'MockClassification', {
31
+ lambdaFunction: this.classificationFn,
32
+ resultPath: '$.classificationResult',
33
+ });
34
+ }
35
+ processingStep() {
36
+ return new aws_stepfunctions_tasks_1.LambdaInvoke(this, 'MockProcessing', {
37
+ lambdaFunction: this.processingFn,
38
+ resultPath: '$.processingResult',
39
+ });
40
+ }
41
+ enrichmentStep() {
42
+ return undefined;
43
+ }
44
+ postProcessingStep() {
45
+ return undefined;
46
+ }
47
+ createStateMachine() {
48
+ return this.handleStateMachineCreation('test-state-machine');
49
+ }
50
+ }
51
+ // Test implementation with enrichment step
52
+ class TestDocumentProcessingWithEnrichment extends base_document_processing_1.BaseDocumentProcessing {
53
+ constructor(scope, id, props) {
54
+ super(scope, id, props);
55
+ this.classificationFn = new aws_lambda_1.Function(this, 'ClassificationFn', {
56
+ runtime: aws_lambda_1.Runtime.NODEJS_20_X,
57
+ handler: 'index.handler',
58
+ code: aws_lambda_1.Code.fromInline('exports.handler = async () => ({ documentClassification: "TEST" });'),
59
+ });
60
+ this.processingFn = new aws_lambda_1.Function(this, 'ProcessingFn', {
61
+ runtime: aws_lambda_1.Runtime.NODEJS_20_X,
62
+ handler: 'index.handler',
63
+ code: aws_lambda_1.Code.fromInline('exports.handler = async () => ({ result: {} });'),
64
+ });
65
+ this.enrichmentFn = new aws_lambda_1.Function(this, 'EnrichmentFn', {
66
+ runtime: aws_lambda_1.Runtime.NODEJS_20_X,
67
+ handler: 'index.handler',
68
+ code: aws_lambda_1.Code.fromInline('exports.handler = async () => ({ enriched: true });'),
69
+ });
70
+ }
71
+ classificationStep() {
72
+ return new aws_stepfunctions_tasks_1.LambdaInvoke(this, 'MockClassification', {
73
+ lambdaFunction: this.classificationFn,
74
+ resultPath: '$.classificationResult',
75
+ });
76
+ }
77
+ processingStep() {
78
+ return new aws_stepfunctions_tasks_1.LambdaInvoke(this, 'MockProcessing', {
79
+ lambdaFunction: this.processingFn,
80
+ resultPath: '$.processingResult',
81
+ });
82
+ }
83
+ enrichmentStep() {
84
+ return new aws_stepfunctions_tasks_1.LambdaInvoke(this, 'MockEnrichment', {
85
+ lambdaFunction: this.enrichmentFn,
86
+ resultPath: '$.enrichedResult',
87
+ });
88
+ }
89
+ postProcessingStep() {
90
+ return undefined;
91
+ }
92
+ createStateMachine() {
93
+ return this.handleStateMachineCreation('test-state-machine-enrichment');
94
+ }
95
+ }
96
+ // Test implementation with post-processing step
97
+ class TestDocumentProcessingWithPostProcessing extends base_document_processing_1.BaseDocumentProcessing {
98
+ constructor(scope, id, props) {
99
+ super(scope, id, props);
100
+ this.classificationFn = new aws_lambda_1.Function(this, 'ClassificationFn', {
101
+ runtime: aws_lambda_1.Runtime.NODEJS_20_X,
102
+ handler: 'index.handler',
103
+ code: aws_lambda_1.Code.fromInline('exports.handler = async () => ({ documentClassification: "TEST" });'),
104
+ });
105
+ this.processingFn = new aws_lambda_1.Function(this, 'ProcessingFn', {
106
+ runtime: aws_lambda_1.Runtime.NODEJS_20_X,
107
+ handler: 'index.handler',
108
+ code: aws_lambda_1.Code.fromInline('exports.handler = async () => ({ result: {} });'),
109
+ });
110
+ this.postProcessingFn = new aws_lambda_1.Function(this, 'PostProcessingFn', {
111
+ runtime: aws_lambda_1.Runtime.NODEJS_20_X,
112
+ handler: 'index.handler',
113
+ code: aws_lambda_1.Code.fromInline('exports.handler = async () => ({ processed: true });'),
114
+ });
115
+ }
116
+ classificationStep() {
117
+ return new aws_stepfunctions_tasks_1.LambdaInvoke(this, 'MockClassification', {
118
+ lambdaFunction: this.classificationFn,
119
+ resultPath: '$.classificationResult',
120
+ });
121
+ }
122
+ processingStep() {
123
+ return new aws_stepfunctions_tasks_1.LambdaInvoke(this, 'MockProcessing', {
124
+ lambdaFunction: this.processingFn,
125
+ resultPath: '$.processingResult',
126
+ });
127
+ }
128
+ enrichmentStep() {
129
+ return undefined;
130
+ }
131
+ postProcessingStep() {
132
+ return new aws_stepfunctions_tasks_1.LambdaInvoke(this, 'MockPostProcessing', {
133
+ lambdaFunction: this.postProcessingFn,
134
+ resultPath: '$.postProcessedResult',
135
+ });
136
+ }
137
+ createStateMachine() {
138
+ return this.handleStateMachineCreation('test-state-machine-post-processing');
139
+ }
140
+ }
141
+ describe('BaseDocumentProcessing', () => {
142
+ let minimalStack;
143
+ let customStack;
144
+ let enrichmentStack;
145
+ let postProcessingStack;
146
+ let eventBridgeStack;
147
+ let minimalTemplate;
148
+ let customTemplate;
149
+ let enrichmentTemplate;
150
+ let postProcessingTemplate;
151
+ let eventBridgeTemplate;
152
+ beforeAll(() => {
153
+ // Minimal configuration
154
+ minimalStack = new aws_cdk_lib_1.Stack();
155
+ const minimalConstruct = new TestDocumentProcessing(minimalStack, 'MinimalTest', {});
156
+ minimalConstruct.createStateMachine();
157
+ // Custom configuration with custom table, bucket, and timeout
158
+ customStack = new aws_cdk_lib_1.Stack();
159
+ const customTable = new aws_dynamodb_1.Table(customStack, 'CustomTable', {
160
+ partitionKey: { name: 'DocumentId', type: aws_dynamodb_1.AttributeType.STRING },
161
+ });
162
+ const customKey = new aws_kms_1.Key(customStack, 'CustomKey', { enableKeyRotation: true });
163
+ const accessLog = new framework_1.AccessLog(customStack, 'AccessLog');
164
+ const customBucket = new aws_s3_1.Bucket(customStack, 'CustomBucket', {
165
+ serverAccessLogsBucket: accessLog.bucket,
166
+ serverAccessLogsPrefix: accessLog.bucketPrefix,
167
+ enforceSSL: true,
168
+ });
169
+ const customAdapter = new adapter_1.QueuedS3Adapter({ bucket: customBucket });
170
+ const customConstruct = new TestDocumentProcessing(customStack, 'CustomTest', {
171
+ documentProcessingTable: customTable,
172
+ ingressAdapter: customAdapter,
173
+ workflowTimeout: aws_cdk_lib_1.Duration.minutes(30),
174
+ removalPolicy: aws_cdk_lib_1.RemovalPolicy.RETAIN,
175
+ encryptionKey: customKey,
176
+ });
177
+ customConstruct.createStateMachine();
178
+ // With enrichment step
179
+ enrichmentStack = new aws_cdk_lib_1.Stack();
180
+ const enrichmentConstruct = new TestDocumentProcessingWithEnrichment(enrichmentStack, 'EnrichmentTest', {});
181
+ enrichmentConstruct.createStateMachine();
182
+ // With post-processing step
183
+ postProcessingStack = new aws_cdk_lib_1.Stack();
184
+ const postProcessingConstruct = new TestDocumentProcessingWithPostProcessing(postProcessingStack, 'PostProcessingTest', {});
185
+ postProcessingConstruct.createStateMachine();
186
+ // With EventBridge broker
187
+ eventBridgeStack = new aws_cdk_lib_1.Stack();
188
+ const broker = new eventbridge_broker_1.EventbridgeBroker(eventBridgeStack, 'TestBroker', {
189
+ name: 'test-broker',
190
+ eventSource: 'test-source',
191
+ });
192
+ const eventBridgeConstruct = new TestDocumentProcessing(eventBridgeStack, 'EventBridgeTest', {
193
+ eventbridgeBroker: broker,
194
+ });
195
+ eventBridgeConstruct.createStateMachine();
196
+ // Generate templates once
197
+ minimalTemplate = assertions_1.Template.fromStack(minimalStack);
198
+ customTemplate = assertions_1.Template.fromStack(customStack);
199
+ enrichmentTemplate = assertions_1.Template.fromStack(enrichmentStack);
200
+ postProcessingTemplate = assertions_1.Template.fromStack(postProcessingStack);
201
+ eventBridgeTemplate = assertions_1.Template.fromStack(eventBridgeStack);
202
+ });
203
+ describe('Basic functionality', () => {
204
+ test('creates construct with minimal configuration', () => {
205
+ expect(minimalTemplate).toBeDefined();
206
+ });
207
+ test('creates DynamoDB table with correct configuration', () => {
208
+ minimalTemplate.hasResourceProperties('AWS::DynamoDB::Table', {
209
+ KeySchema: [{
210
+ AttributeName: 'DocumentId',
211
+ KeyType: 'HASH',
212
+ }],
213
+ BillingMode: 'PAY_PER_REQUEST',
214
+ PointInTimeRecoverySpecification: {
215
+ PointInTimeRecoveryEnabled: true,
216
+ },
217
+ SSESpecification: {
218
+ SSEEnabled: true,
219
+ SSEType: 'KMS',
220
+ },
221
+ });
222
+ });
223
+ test('creates KMS encryption key with rotation enabled', () => {
224
+ minimalTemplate.hasResourceProperties('AWS::KMS::Key', {
225
+ EnableKeyRotation: true,
226
+ });
227
+ });
228
+ test('creates S3 bucket for document storage', () => {
229
+ minimalTemplate.hasResourceProperties('AWS::S3::Bucket', {
230
+ BucketEncryption: {
231
+ ServerSideEncryptionConfiguration: [{
232
+ ServerSideEncryptionByDefault: {
233
+ SSEAlgorithm: 'aws:kms',
234
+ },
235
+ }],
236
+ },
237
+ });
238
+ });
239
+ test('creates SQS queue with DLQ', () => {
240
+ minimalTemplate.hasResourceProperties('AWS::SQS::Queue', {
241
+ RedrivePolicy: assertions_1.Match.objectLike({
242
+ maxReceiveCount: 5,
243
+ }),
244
+ });
245
+ });
246
+ test('creates Step Functions state machine', () => {
247
+ minimalTemplate.resourceCountIs('AWS::StepFunctions::StateMachine', 1);
248
+ });
249
+ });
250
+ describe('State machine configuration', () => {
251
+ test('creates state machine with timeout', () => {
252
+ minimalTemplate.resourceCountIs('AWS::StepFunctions::StateMachine', 1);
253
+ });
254
+ test('creates state machine with custom timeout', () => {
255
+ customTemplate.resourceCountIs('AWS::StepFunctions::StateMachine', 1);
256
+ });
257
+ test('creates state machine with encryption', () => {
258
+ minimalTemplate.hasResourceProperties('AWS::StepFunctions::StateMachine', {
259
+ EncryptionConfiguration: {
260
+ Type: 'CUSTOMER_MANAGED_KMS_KEY',
261
+ },
262
+ });
263
+ });
264
+ test('creates state machine role with DynamoDB permissions', () => {
265
+ minimalTemplate.hasResourceProperties('AWS::IAM::Role', {
266
+ AssumeRolePolicyDocument: {
267
+ Statement: [{
268
+ Action: 'sts:AssumeRole',
269
+ Effect: 'Allow',
270
+ Principal: {
271
+ Service: 'states.amazonaws.com',
272
+ },
273
+ }],
274
+ },
275
+ });
276
+ minimalTemplate.hasResourceProperties('AWS::IAM::Policy', {
277
+ PolicyDocument: {
278
+ Statement: assertions_1.Match.arrayWith([
279
+ assertions_1.Match.objectLike({
280
+ Effect: 'Allow',
281
+ Resource: assertions_1.Match.anyValue(),
282
+ }),
283
+ ]),
284
+ },
285
+ });
286
+ });
287
+ });
288
+ describe('Error handling chains', () => {
289
+ test('includes DynamoDB update for classification failure', () => {
290
+ minimalTemplate.hasResourceProperties('AWS::StepFunctions::StateMachine', {
291
+ DefinitionString: assertions_1.Match.objectLike({
292
+ 'Fn::Join': assertions_1.Match.arrayWith(['']),
293
+ }),
294
+ });
295
+ });
296
+ test('includes DynamoDB update for processing failure', () => {
297
+ minimalTemplate.hasResourceProperties('AWS::StepFunctions::StateMachine', {
298
+ DefinitionString: assertions_1.Match.objectLike({
299
+ 'Fn::Join': assertions_1.Match.arrayWith(['']),
300
+ }),
301
+ });
302
+ });
303
+ test('includes workflow status updates for success', () => {
304
+ minimalTemplate.hasResourceProperties('AWS::StepFunctions::StateMachine', {
305
+ DefinitionString: assertions_1.Match.objectLike({
306
+ 'Fn::Join': assertions_1.Match.arrayWith(['']),
307
+ }),
308
+ });
309
+ });
310
+ });
311
+ describe('Optional workflow steps', () => {
312
+ test('includes enrichment step when provided', () => {
313
+ enrichmentTemplate.hasResourceProperties('AWS::StepFunctions::StateMachine', {
314
+ DefinitionString: assertions_1.Match.objectLike({
315
+ 'Fn::Join': assertions_1.Match.arrayWith(['']),
316
+ }),
317
+ });
318
+ });
319
+ test('includes post-processing step when provided', () => {
320
+ postProcessingTemplate.hasResourceProperties('AWS::StepFunctions::StateMachine', {
321
+ DefinitionString: assertions_1.Match.objectLike({
322
+ 'Fn::Join': assertions_1.Match.arrayWith(['']),
323
+ }),
324
+ });
325
+ });
326
+ test('marks workflow complete after processing when no optional steps', () => {
327
+ minimalTemplate.hasResourceProperties('AWS::StepFunctions::StateMachine', {
328
+ DefinitionString: assertions_1.Match.objectLike({
329
+ 'Fn::Join': assertions_1.Match.arrayWith(['']),
330
+ }),
331
+ });
332
+ });
333
+ });
334
+ describe('Custom configuration', () => {
335
+ test('uses provided DynamoDB table', () => {
336
+ customTemplate.resourceCountIs('AWS::DynamoDB::Table', 1);
337
+ customTemplate.hasResourceProperties('AWS::DynamoDB::Table', {
338
+ KeySchema: [{
339
+ AttributeName: 'DocumentId',
340
+ KeyType: 'HASH',
341
+ }],
342
+ });
343
+ });
344
+ test('uses provided encryption key', () => {
345
+ customTemplate.resourceCountIs('AWS::KMS::Key', 2); // Custom key + adapter key
346
+ });
347
+ test('uses provided S3 bucket via adapter', () => {
348
+ customTemplate.hasResourceProperties('AWS::S3::Bucket', {
349
+ BucketEncryption: assertions_1.Match.objectLike({
350
+ ServerSideEncryptionConfiguration: assertions_1.Match.anyValue(),
351
+ }),
352
+ });
353
+ });
354
+ test('applies custom removal policy', () => {
355
+ customTemplate.hasResource('AWS::DynamoDB::Table', {
356
+ DeletionPolicy: 'Retain',
357
+ UpdateReplacePolicy: 'Retain',
358
+ });
359
+ });
360
+ });
361
+ describe('EventBridge integration', () => {
362
+ test('includes EventBridge event bus when broker provided', () => {
363
+ eventBridgeTemplate.hasResourceProperties('AWS::Events::EventBus', {
364
+ Name: 'test-broker',
365
+ });
366
+ });
367
+ test('includes EventBridge put events in state machine', () => {
368
+ eventBridgeTemplate.hasResourceProperties('AWS::StepFunctions::StateMachine', {
369
+ DefinitionString: assertions_1.Match.objectLike({
370
+ 'Fn::Join': assertions_1.Match.arrayWith(['']),
371
+ }),
372
+ });
373
+ });
374
+ });
375
+ describe('Security', () => {
376
+ test('encrypts DynamoDB table with KMS', () => {
377
+ minimalTemplate.hasResourceProperties('AWS::DynamoDB::Table', {
378
+ SSESpecification: {
379
+ SSEEnabled: true,
380
+ SSEType: 'KMS',
381
+ },
382
+ });
383
+ });
384
+ test('encrypts S3 bucket with KMS', () => {
385
+ minimalTemplate.hasResourceProperties('AWS::S3::Bucket', {
386
+ BucketEncryption: {
387
+ ServerSideEncryptionConfiguration: [{
388
+ ServerSideEncryptionByDefault: {
389
+ SSEAlgorithm: 'aws:kms',
390
+ },
391
+ }],
392
+ },
393
+ });
394
+ });
395
+ test('encrypts SQS queue with KMS', () => {
396
+ minimalTemplate.hasResourceProperties('AWS::SQS::Queue', {
397
+ KmsMasterKeyId: assertions_1.Match.anyValue(),
398
+ });
399
+ });
400
+ test('encrypts state machine with customer managed key', () => {
401
+ minimalTemplate.hasResourceProperties('AWS::StepFunctions::StateMachine', {
402
+ EncryptionConfiguration: {
403
+ Type: 'CUSTOMER_MANAGED_KMS_KEY',
404
+ KmsKeyId: assertions_1.Match.anyValue(),
405
+ },
406
+ });
407
+ });
408
+ test('enforces SSL on S3 bucket', () => {
409
+ minimalTemplate.hasResourceProperties('AWS::S3::BucketPolicy', {
410
+ PolicyDocument: {
411
+ Statement: assertions_1.Match.arrayWith([
412
+ assertions_1.Match.objectLike({
413
+ Effect: 'Deny',
414
+ Condition: {
415
+ Bool: {
416
+ 'aws:SecureTransport': 'false',
417
+ },
418
+ },
419
+ }),
420
+ ]),
421
+ },
422
+ });
423
+ });
424
+ test('enforces SSL on SQS queue', () => {
425
+ minimalTemplate.hasResourceProperties('AWS::SQS::QueuePolicy', {
426
+ PolicyDocument: {
427
+ Statement: assertions_1.Match.arrayWith([
428
+ assertions_1.Match.objectLike({
429
+ Effect: 'Deny',
430
+ Condition: {
431
+ Bool: {
432
+ 'aws:SecureTransport': 'false',
433
+ },
434
+ },
435
+ }),
436
+ ]),
437
+ },
438
+ });
439
+ });
440
+ });
441
+ describe('Adapter integration', () => {
442
+ test('creates S3 event notification to SQS', () => {
443
+ minimalTemplate.hasResourceProperties('AWS::Lambda::Function', {
444
+ Handler: 'index.handler',
445
+ Environment: {
446
+ Variables: {
447
+ STATE_MACHINE_ARN: assertions_1.Match.anyValue(),
448
+ RAW_PREFIX: 'raw/',
449
+ },
450
+ },
451
+ });
452
+ });
453
+ test('creates SQS consumer Lambda with event source', () => {
454
+ minimalTemplate.hasResourceProperties('AWS::Lambda::EventSourceMapping', {
455
+ BatchSize: 10,
456
+ EventSourceArn: assertions_1.Match.anyValue(),
457
+ FunctionName: assertions_1.Match.anyValue(),
458
+ });
459
+ });
460
+ test('creates dead letter queue for failed messages', () => {
461
+ minimalTemplate.resourceCountIs('AWS::SQS::Queue', 2);
462
+ });
463
+ });
464
+ describe('Observability', () => {
465
+ test('creates construct without observability by default', () => {
466
+ const stack = new aws_cdk_lib_1.Stack();
467
+ const construct = new TestDocumentProcessing(stack, 'NoObservability', {
468
+ enableObservability: false,
469
+ });
470
+ construct.createStateMachine();
471
+ const template = assertions_1.Template.fromStack(stack);
472
+ template.hasResourceProperties('AWS::StepFunctions::StateMachine', {
473
+ LoggingConfiguration: assertions_1.Match.absent(),
474
+ });
475
+ });
476
+ test('enables observability when configured', () => {
477
+ const stack = new aws_cdk_lib_1.Stack();
478
+ const construct = new TestDocumentProcessing(stack, 'WithObservability', {
479
+ enableObservability: true,
480
+ });
481
+ construct.createStateMachine();
482
+ const template = assertions_1.Template.fromStack(stack);
483
+ template.hasResourceProperties('AWS::StepFunctions::StateMachine', {
484
+ LoggingConfiguration: assertions_1.Match.objectLike({
485
+ Level: 'ALL',
486
+ }),
487
+ });
488
+ });
489
+ });
490
+ describe('Resource counts', () => {
491
+ test('creates expected number of resources', () => {
492
+ minimalTemplate.resourceCountIs('AWS::DynamoDB::Table', 1);
493
+ minimalTemplate.resourceCountIs('AWS::S3::Bucket', 1);
494
+ minimalTemplate.resourceCountIs('AWS::SQS::Queue', 2);
495
+ minimalTemplate.resourceCountIs('AWS::StepFunctions::StateMachine', 1);
496
+ });
497
+ });
498
+ });
499
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"base-document-processing.test.js","sourceRoot":"","sources":["../../../use-cases/document-processing/tests/base-document-processing.test.ts"],"names":[],"mappings":"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","sourcesContent":["import { Duration, RemovalPolicy, Stack } from 'aws-cdk-lib';\nimport { Template, Match } from 'aws-cdk-lib/assertions';\nimport { AttributeType, Table } from 'aws-cdk-lib/aws-dynamodb';\nimport { Key } from 'aws-cdk-lib/aws-kms';\nimport { Code, Function, Runtime } from 'aws-cdk-lib/aws-lambda';\nimport { Bucket } from 'aws-cdk-lib/aws-s3';\nimport { LambdaInvoke } from 'aws-cdk-lib/aws-stepfunctions-tasks';\nimport { AccessLog } from '../../framework';\nimport { EventbridgeBroker } from '../../framework/foundation/eventbridge-broker';\nimport { QueuedS3Adapter } from '../adapter';\nimport { BaseDocumentProcessing, DocumentProcessingStepType } from '../base-document-processing';\n\n// Concrete test implementation of BaseDocumentProcessing\nclass TestDocumentProcessing extends BaseDocumentProcessing {\n  private classificationFn: Function;\n  private processingFn: Function;\n\n  constructor(scope: any, id: string, props: any) {\n    super(scope, id, props);\n\n    this.classificationFn = new Function(this, 'ClassificationFn', {\n      runtime: Runtime.NODEJS_20_X,\n      handler: 'index.handler',\n      code: Code.fromInline('exports.handler = async () => ({ documentClassification: \"TEST\" });'),\n    });\n\n    this.processingFn = new Function(this, 'ProcessingFn', {\n      runtime: Runtime.NODEJS_20_X,\n      handler: 'index.handler',\n      code: Code.fromInline('exports.handler = async () => ({ result: {} });'),\n    });\n  }\n\n  protected classificationStep(): DocumentProcessingStepType {\n    return new LambdaInvoke(this, 'MockClassification', {\n      lambdaFunction: this.classificationFn,\n      resultPath: '$.classificationResult',\n    });\n  }\n\n  protected processingStep(): DocumentProcessingStepType {\n    return new LambdaInvoke(this, 'MockProcessing', {\n      lambdaFunction: this.processingFn,\n      resultPath: '$.processingResult',\n    });\n  }\n\n  protected enrichmentStep(): DocumentProcessingStepType | undefined {\n    return undefined;\n  }\n\n  protected postProcessingStep(): DocumentProcessingStepType | undefined {\n    return undefined;\n  }\n\n  public createStateMachine() {\n    return this.handleStateMachineCreation('test-state-machine');\n  }\n}\n\n// Test implementation with enrichment step\nclass TestDocumentProcessingWithEnrichment extends BaseDocumentProcessing {\n  private classificationFn: Function;\n  private processingFn: Function;\n  private enrichmentFn: Function;\n\n  constructor(scope: any, id: string, props: any) {\n    super(scope, id, props);\n\n    this.classificationFn = new Function(this, 'ClassificationFn', {\n      runtime: Runtime.NODEJS_20_X,\n      handler: 'index.handler',\n      code: Code.fromInline('exports.handler = async () => ({ documentClassification: \"TEST\" });'),\n    });\n\n    this.processingFn = new Function(this, 'ProcessingFn', {\n      runtime: Runtime.NODEJS_20_X,\n      handler: 'index.handler',\n      code: Code.fromInline('exports.handler = async () => ({ result: {} });'),\n    });\n\n    this.enrichmentFn = new Function(this, 'EnrichmentFn', {\n      runtime: Runtime.NODEJS_20_X,\n      handler: 'index.handler',\n      code: Code.fromInline('exports.handler = async () => ({ enriched: true });'),\n    });\n  }\n\n  protected classificationStep(): DocumentProcessingStepType {\n    return new LambdaInvoke(this, 'MockClassification', {\n      lambdaFunction: this.classificationFn,\n      resultPath: '$.classificationResult',\n    });\n  }\n\n  protected processingStep(): DocumentProcessingStepType {\n    return new LambdaInvoke(this, 'MockProcessing', {\n      lambdaFunction: this.processingFn,\n      resultPath: '$.processingResult',\n    });\n  }\n\n  protected enrichmentStep(): DocumentProcessingStepType | undefined {\n    return new LambdaInvoke(this, 'MockEnrichment', {\n      lambdaFunction: this.enrichmentFn,\n      resultPath: '$.enrichedResult',\n    });\n  }\n\n  protected postProcessingStep(): DocumentProcessingStepType | undefined {\n    return undefined;\n  }\n\n  public createStateMachine() {\n    return this.handleStateMachineCreation('test-state-machine-enrichment');\n  }\n}\n\n// Test implementation with post-processing step\nclass TestDocumentProcessingWithPostProcessing extends BaseDocumentProcessing {\n  private classificationFn: Function;\n  private processingFn: Function;\n  private postProcessingFn: Function;\n\n  constructor(scope: any, id: string, props: any) {\n    super(scope, id, props);\n\n    this.classificationFn = new Function(this, 'ClassificationFn', {\n      runtime: Runtime.NODEJS_20_X,\n      handler: 'index.handler',\n      code: Code.fromInline('exports.handler = async () => ({ documentClassification: \"TEST\" });'),\n    });\n\n    this.processingFn = new Function(this, 'ProcessingFn', {\n      runtime: Runtime.NODEJS_20_X,\n      handler: 'index.handler',\n      code: Code.fromInline('exports.handler = async () => ({ result: {} });'),\n    });\n\n    this.postProcessingFn = new Function(this, 'PostProcessingFn', {\n      runtime: Runtime.NODEJS_20_X,\n      handler: 'index.handler',\n      code: Code.fromInline('exports.handler = async () => ({ processed: true });'),\n    });\n  }\n\n  protected classificationStep(): DocumentProcessingStepType {\n    return new LambdaInvoke(this, 'MockClassification', {\n      lambdaFunction: this.classificationFn,\n      resultPath: '$.classificationResult',\n    });\n  }\n\n  protected processingStep(): DocumentProcessingStepType {\n    return new LambdaInvoke(this, 'MockProcessing', {\n      lambdaFunction: this.processingFn,\n      resultPath: '$.processingResult',\n    });\n  }\n\n  protected enrichmentStep(): DocumentProcessingStepType | undefined {\n    return undefined;\n  }\n\n  protected postProcessingStep(): DocumentProcessingStepType | undefined {\n    return new LambdaInvoke(this, 'MockPostProcessing', {\n      lambdaFunction: this.postProcessingFn,\n      resultPath: '$.postProcessedResult',\n    });\n  }\n\n  public createStateMachine() {\n    return this.handleStateMachineCreation('test-state-machine-post-processing');\n  }\n}\n\ndescribe('BaseDocumentProcessing', () => {\n  let minimalStack: Stack;\n  let customStack: Stack;\n  let enrichmentStack: Stack;\n  let postProcessingStack: Stack;\n  let eventBridgeStack: Stack;\n  let minimalTemplate: Template;\n  let customTemplate: Template;\n  let enrichmentTemplate: Template;\n  let postProcessingTemplate: Template;\n  let eventBridgeTemplate: Template;\n\n  beforeAll(() => {\n    // Minimal configuration\n    minimalStack = new Stack();\n    const minimalConstruct = new TestDocumentProcessing(minimalStack, 'MinimalTest', {});\n    minimalConstruct.createStateMachine();\n\n    // Custom configuration with custom table, bucket, and timeout\n    customStack = new Stack();\n    const customTable = new Table(customStack, 'CustomTable', {\n      partitionKey: { name: 'DocumentId', type: AttributeType.STRING },\n    });\n    const customKey = new Key(customStack, 'CustomKey', { enableKeyRotation: true });\n    const accessLog = new AccessLog(customStack, 'AccessLog');\n    const customBucket = new Bucket(customStack, 'CustomBucket', {\n      serverAccessLogsBucket: accessLog.bucket,\n      serverAccessLogsPrefix: accessLog.bucketPrefix,\n      enforceSSL: true,\n    });\n    const customAdapter = new QueuedS3Adapter({ bucket: customBucket });\n    const customConstruct = new TestDocumentProcessing(customStack, 'CustomTest', {\n      documentProcessingTable: customTable,\n      ingressAdapter: customAdapter,\n      workflowTimeout: Duration.minutes(30),\n      removalPolicy: RemovalPolicy.RETAIN,\n      encryptionKey: customKey,\n    });\n    customConstruct.createStateMachine();\n\n    // With enrichment step\n    enrichmentStack = new Stack();\n    const enrichmentConstruct = new TestDocumentProcessingWithEnrichment(enrichmentStack, 'EnrichmentTest', {});\n    enrichmentConstruct.createStateMachine();\n\n    // With post-processing step\n    postProcessingStack = new Stack();\n    const postProcessingConstruct = new TestDocumentProcessingWithPostProcessing(postProcessingStack, 'PostProcessingTest', {});\n    postProcessingConstruct.createStateMachine();\n\n    // With EventBridge broker\n    eventBridgeStack = new Stack();\n    const broker = new EventbridgeBroker(eventBridgeStack, 'TestBroker', {\n      name: 'test-broker',\n      eventSource: 'test-source',\n    });\n    const eventBridgeConstruct = new TestDocumentProcessing(eventBridgeStack, 'EventBridgeTest', {\n      eventbridgeBroker: broker,\n    });\n    eventBridgeConstruct.createStateMachine();\n\n    // Generate templates once\n    minimalTemplate = Template.fromStack(minimalStack);\n    customTemplate = Template.fromStack(customStack);\n    enrichmentTemplate = Template.fromStack(enrichmentStack);\n    postProcessingTemplate = Template.fromStack(postProcessingStack);\n    eventBridgeTemplate = Template.fromStack(eventBridgeStack);\n  });\n\n  describe('Basic functionality', () => {\n    test('creates construct with minimal configuration', () => {\n      expect(minimalTemplate).toBeDefined();\n    });\n\n    test('creates DynamoDB table with correct configuration', () => {\n      minimalTemplate.hasResourceProperties('AWS::DynamoDB::Table', {\n        KeySchema: [{\n          AttributeName: 'DocumentId',\n          KeyType: 'HASH',\n        }],\n        BillingMode: 'PAY_PER_REQUEST',\n        PointInTimeRecoverySpecification: {\n          PointInTimeRecoveryEnabled: true,\n        },\n        SSESpecification: {\n          SSEEnabled: true,\n          SSEType: 'KMS',\n        },\n      });\n    });\n\n    test('creates KMS encryption key with rotation enabled', () => {\n      minimalTemplate.hasResourceProperties('AWS::KMS::Key', {\n        EnableKeyRotation: true,\n      });\n    });\n\n    test('creates S3 bucket for document storage', () => {\n      minimalTemplate.hasResourceProperties('AWS::S3::Bucket', {\n        BucketEncryption: {\n          ServerSideEncryptionConfiguration: [{\n            ServerSideEncryptionByDefault: {\n              SSEAlgorithm: 'aws:kms',\n            },\n          }],\n        },\n      });\n    });\n\n    test('creates SQS queue with DLQ', () => {\n      minimalTemplate.hasResourceProperties('AWS::SQS::Queue', {\n        RedrivePolicy: Match.objectLike({\n          maxReceiveCount: 5,\n        }),\n      });\n    });\n\n    test('creates Step Functions state machine', () => {\n      minimalTemplate.resourceCountIs('AWS::StepFunctions::StateMachine', 1);\n    });\n  });\n\n  describe('State machine configuration', () => {\n    test('creates state machine with timeout', () => {\n      minimalTemplate.resourceCountIs('AWS::StepFunctions::StateMachine', 1);\n    });\n\n    test('creates state machine with custom timeout', () => {\n      customTemplate.resourceCountIs('AWS::StepFunctions::StateMachine', 1);\n    });\n\n    test('creates state machine with encryption', () => {\n      minimalTemplate.hasResourceProperties('AWS::StepFunctions::StateMachine', {\n        EncryptionConfiguration: {\n          Type: 'CUSTOMER_MANAGED_KMS_KEY',\n        },\n      });\n    });\n\n    test('creates state machine role with DynamoDB permissions', () => {\n      minimalTemplate.hasResourceProperties('AWS::IAM::Role', {\n        AssumeRolePolicyDocument: {\n          Statement: [{\n            Action: 'sts:AssumeRole',\n            Effect: 'Allow',\n            Principal: {\n              Service: 'states.amazonaws.com',\n            },\n          }],\n        },\n      });\n\n      minimalTemplate.hasResourceProperties('AWS::IAM::Policy', {\n        PolicyDocument: {\n          Statement: Match.arrayWith([\n            Match.objectLike({\n              Effect: 'Allow',\n              Resource: Match.anyValue(),\n            }),\n          ]),\n        },\n      });\n    });\n  });\n\n  describe('Error handling chains', () => {\n    test('includes DynamoDB update for classification failure', () => {\n      minimalTemplate.hasResourceProperties('AWS::StepFunctions::StateMachine', {\n        DefinitionString: Match.objectLike({\n          'Fn::Join': Match.arrayWith(['']),\n        }),\n      });\n    });\n\n    test('includes DynamoDB update for processing failure', () => {\n      minimalTemplate.hasResourceProperties('AWS::StepFunctions::StateMachine', {\n        DefinitionString: Match.objectLike({\n          'Fn::Join': Match.arrayWith(['']),\n        }),\n      });\n    });\n\n    test('includes workflow status updates for success', () => {\n      minimalTemplate.hasResourceProperties('AWS::StepFunctions::StateMachine', {\n        DefinitionString: Match.objectLike({\n          'Fn::Join': Match.arrayWith(['']),\n        }),\n      });\n    });\n  });\n\n  describe('Optional workflow steps', () => {\n    test('includes enrichment step when provided', () => {\n      enrichmentTemplate.hasResourceProperties('AWS::StepFunctions::StateMachine', {\n        DefinitionString: Match.objectLike({\n          'Fn::Join': Match.arrayWith(['']),\n        }),\n      });\n    });\n\n    test('includes post-processing step when provided', () => {\n      postProcessingTemplate.hasResourceProperties('AWS::StepFunctions::StateMachine', {\n        DefinitionString: Match.objectLike({\n          'Fn::Join': Match.arrayWith(['']),\n        }),\n      });\n    });\n\n    test('marks workflow complete after processing when no optional steps', () => {\n      minimalTemplate.hasResourceProperties('AWS::StepFunctions::StateMachine', {\n        DefinitionString: Match.objectLike({\n          'Fn::Join': Match.arrayWith(['']),\n        }),\n      });\n    });\n  });\n\n  describe('Custom configuration', () => {\n    test('uses provided DynamoDB table', () => {\n      customTemplate.resourceCountIs('AWS::DynamoDB::Table', 1);\n      customTemplate.hasResourceProperties('AWS::DynamoDB::Table', {\n        KeySchema: [{\n          AttributeName: 'DocumentId',\n          KeyType: 'HASH',\n        }],\n      });\n    });\n\n    test('uses provided encryption key', () => {\n      customTemplate.resourceCountIs('AWS::KMS::Key', 2); // Custom key + adapter key\n    });\n\n    test('uses provided S3 bucket via adapter', () => {\n      customTemplate.hasResourceProperties('AWS::S3::Bucket', {\n        BucketEncryption: Match.objectLike({\n          ServerSideEncryptionConfiguration: Match.anyValue(),\n        }),\n      });\n    });\n\n    test('applies custom removal policy', () => {\n      customTemplate.hasResource('AWS::DynamoDB::Table', {\n        DeletionPolicy: 'Retain',\n        UpdateReplacePolicy: 'Retain',\n      });\n    });\n  });\n\n  describe('EventBridge integration', () => {\n    test('includes EventBridge event bus when broker provided', () => {\n      eventBridgeTemplate.hasResourceProperties('AWS::Events::EventBus', {\n        Name: 'test-broker',\n      });\n    });\n\n    test('includes EventBridge put events in state machine', () => {\n      eventBridgeTemplate.hasResourceProperties('AWS::StepFunctions::StateMachine', {\n        DefinitionString: Match.objectLike({\n          'Fn::Join': Match.arrayWith(['']),\n        }),\n      });\n    });\n  });\n\n  describe('Security', () => {\n    test('encrypts DynamoDB table with KMS', () => {\n      minimalTemplate.hasResourceProperties('AWS::DynamoDB::Table', {\n        SSESpecification: {\n          SSEEnabled: true,\n          SSEType: 'KMS',\n        },\n      });\n    });\n\n    test('encrypts S3 bucket with KMS', () => {\n      minimalTemplate.hasResourceProperties('AWS::S3::Bucket', {\n        BucketEncryption: {\n          ServerSideEncryptionConfiguration: [{\n            ServerSideEncryptionByDefault: {\n              SSEAlgorithm: 'aws:kms',\n            },\n          }],\n        },\n      });\n    });\n\n    test('encrypts SQS queue with KMS', () => {\n      minimalTemplate.hasResourceProperties('AWS::SQS::Queue', {\n        KmsMasterKeyId: Match.anyValue(),\n      });\n    });\n\n    test('encrypts state machine with customer managed key', () => {\n      minimalTemplate.hasResourceProperties('AWS::StepFunctions::StateMachine', {\n        EncryptionConfiguration: {\n          Type: 'CUSTOMER_MANAGED_KMS_KEY',\n          KmsKeyId: Match.anyValue(),\n        },\n      });\n    });\n\n    test('enforces SSL on S3 bucket', () => {\n      minimalTemplate.hasResourceProperties('AWS::S3::BucketPolicy', {\n        PolicyDocument: {\n          Statement: Match.arrayWith([\n            Match.objectLike({\n              Effect: 'Deny',\n              Condition: {\n                Bool: {\n                  'aws:SecureTransport': 'false',\n                },\n              },\n            }),\n          ]),\n        },\n      });\n    });\n\n    test('enforces SSL on SQS queue', () => {\n      minimalTemplate.hasResourceProperties('AWS::SQS::QueuePolicy', {\n        PolicyDocument: {\n          Statement: Match.arrayWith([\n            Match.objectLike({\n              Effect: 'Deny',\n              Condition: {\n                Bool: {\n                  'aws:SecureTransport': 'false',\n                },\n              },\n            }),\n          ]),\n        },\n      });\n    });\n  });\n\n  describe('Adapter integration', () => {\n    test('creates S3 event notification to SQS', () => {\n      minimalTemplate.hasResourceProperties('AWS::Lambda::Function', {\n        Handler: 'index.handler',\n        Environment: {\n          Variables: {\n            STATE_MACHINE_ARN: Match.anyValue(),\n            RAW_PREFIX: 'raw/',\n          },\n        },\n      });\n    });\n\n    test('creates SQS consumer Lambda with event source', () => {\n      minimalTemplate.hasResourceProperties('AWS::Lambda::EventSourceMapping', {\n        BatchSize: 10,\n        EventSourceArn: Match.anyValue(),\n        FunctionName: Match.anyValue(),\n      });\n    });\n\n    test('creates dead letter queue for failed messages', () => {\n      minimalTemplate.resourceCountIs('AWS::SQS::Queue', 2);\n    });\n  });\n\n  describe('Observability', () => {\n    test('creates construct without observability by default', () => {\n      const stack = new Stack();\n      const construct = new TestDocumentProcessing(stack, 'NoObservability', {\n        enableObservability: false,\n      });\n      construct.createStateMachine();\n      const template = Template.fromStack(stack);\n\n      template.hasResourceProperties('AWS::StepFunctions::StateMachine', {\n        LoggingConfiguration: Match.absent(),\n      });\n    });\n\n    test('enables observability when configured', () => {\n      const stack = new Stack();\n      const construct = new TestDocumentProcessing(stack, 'WithObservability', {\n        enableObservability: true,\n      });\n      construct.createStateMachine();\n      const template = Template.fromStack(stack);\n\n      template.hasResourceProperties('AWS::StepFunctions::StateMachine', {\n        LoggingConfiguration: Match.objectLike({\n          Level: 'ALL',\n        }),\n      });\n    });\n  });\n\n  describe('Resource counts', () => {\n    test('creates expected number of resources', () => {\n      minimalTemplate.resourceCountIs('AWS::DynamoDB::Table', 1);\n      minimalTemplate.resourceCountIs('AWS::S3::Bucket', 1);\n      minimalTemplate.resourceCountIs('AWS::SQS::Queue', 2);\n      minimalTemplate.resourceCountIs('AWS::StepFunctions::StateMachine', 1);\n    });\n  });\n});\n"]}
@@ -34,7 +34,8 @@ const adapter = new adapter_1.QueuedS3Adapter({
34
34
  // Create the main BedrockDocumentProcessing construct
35
35
  new bedrock_document_processing_1.BedrockDocumentProcessing(stack, 'BedrockDocumentProcessing', {
36
36
  ingressAdapter: adapter,
37
- useCrossRegionInference: true,
37
+ classificationBedrockModel: { useCrossRegionInference: true },
38
+ processingBedrockModel: { useCrossRegionInference: true },
38
39
  eventbridgeBroker: broker,
39
40
  enableObservability: true,
40
41
  });
@@ -102,4 +103,4 @@ test('No unsuppressed errors', () => {
102
103
  }
103
104
  expect(errors).toHaveLength(0);
104
105
  });
105
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"bedrock-document-processing-nag.test.js","sourceRoot":"","sources":["../../../use-cases/document-processing/tests/bedrock-document-processing-nag.test.ts"],"names":[],"mappings":"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","sourcesContent":["import { App, Stack, Aspects } from 'aws-cdk-lib';\nimport { Annotations, Match } from 'aws-cdk-lib/assertions';\nimport { Bucket } from 'aws-cdk-lib/aws-s3';\nimport { AwsSolutionsChecks, NagSuppressions } from 'cdk-nag';\nimport { AccessLog } from '../../framework';\nimport { EventbridgeBroker } from '../../framework/foundation/eventbridge-broker';\nimport { QueuedS3Adapter } from '../adapter';\nimport { BedrockDocumentProcessing } from '../bedrock-document-processing';\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 EventBridge broker\nconst broker = new EventbridgeBroker(stack, 'IDPBroker', {\n  name: 'idp-broker',\n  eventSource: 'intelligent-document-processing',\n});\n\nconst accessLog = new AccessLog(stack, 'AccessLog');\n\n// Create S3 bucket for agentic tools\nconst bucket = new Bucket(stack, 'BedrockDocumentProcessingBucket', {\n  serverAccessLogsBucket: accessLog.bucket,\n  serverAccessLogsPrefix: accessLog.bucketPrefix,\n  enforceSSL: true,\n});\n\nconst adapter = new QueuedS3Adapter({\n  bucket,\n});\n\n// Create the main BedrockDocumentProcessing construct\nnew BedrockDocumentProcessing(stack, 'BedrockDocumentProcessing', {\n  ingressAdapter: adapter,\n  useCrossRegionInference: true,\n  eventbridgeBroker: broker,\n  enableObservability: true,\n});\n\n// Suppress CDK-managed BucketNotificationsHandler AWS managed policy\nNagSuppressions.addResourceSuppressionsByPath(\n  stack,\n  '/TestStack/BucketNotificationsHandler050a0587b7544547bf325f094a3db834/Role',\n  [\n    {\n      id: 'AwsSolutions-IAM4',\n      reason: 'CDK-managed BucketNotificationsHandler requires AWSLambdaBasicExecutionRole for S3 event processing',\n      appliesTo: ['Policy::arn:<AWS::Partition>:iam::aws:policy/service-role/AWSLambdaBasicExecutionRole'],\n    },\n  ],\n);\n\n// Suppress S3 bucket wildcard permissions for Lambda roles\nNagSuppressions.addResourceSuppressions(\n  stack,\n  [\n    {\n      id: 'AwsSolutions-IAM5',\n      reason: 'Lambda functions require wildcard access to S3 bucket objects for document processing',\n      appliesTo: ['Resource::<BedrockDocumentProcessingBucketA06CA1E2.Arn>/*'],\n    },\n  ],\n  true,\n);\n\n// Suppress Bedrock foundation model wildcard permissions\nNagSuppressions.addResourceSuppressions(\n  stack,\n  [\n    {\n      id: 'AwsSolutions-IAM5',\n      reason: 'Cross-region inference requires wildcard region access to Bedrock foundation models',\n      appliesTo: ['Resource::arn:aws:bedrock:*::foundation-model/anthropic.claude-3-7-sonnet-20250219-v1:0'],\n    },\n  ],\n  true,\n);\n\n// Suppress SQSConsumerRole wildcard permissions for Lambda log streams\nNagSuppressions.addResourceSuppressions(\n  stack,\n  [\n    {\n      id: 'AwsSolutions-IAM5',\n      reason: 'Lambda log stream ARN is only known at runtime, wildcard required for CloudWatch Logs access',\n    },\n  ],\n  true,\n);\n\n// Suppress StateMachineRole wildcard permissions for Lambda invocation\nNagSuppressions.addResourceSuppressionsByPath(\n  stack,\n  '/TestStack/BedrockDocumentProcessing/StateMachineRole/DefaultPolicy',\n  [\n    {\n      id: 'AwsSolutions-IAM5',\n      reason: 'Step Functions requires wildcard permissions to invoke Lambda functions with version-specific ARNs',\n    },\n  ],\n);\n\n// Suppress Lambda log group wildcard permissions\nNagSuppressions.addResourceSuppressions(\n  stack,\n  [\n    {\n      id: 'AwsSolutions-IAM5',\n      reason: 'Lambda log stream names are generated at runtime, wildcard required for CloudWatch Logs access',\n    },\n  ],\n  true,\n);\n\n// Apply CDK Nag checks\nAspects.of(app).add(new AwsSolutionsChecks({ verbose: true }));\n\n// Synthesize the stack and 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: 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"]}
106
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"bedrock-document-processing-nag.test.js","sourceRoot":"","sources":["../../../use-cases/document-processing/tests/bedrock-document-processing-nag.test.ts"],"names":[],"mappings":"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","sourcesContent":["import { App, Stack, Aspects } from 'aws-cdk-lib';\nimport { Annotations, Match } from 'aws-cdk-lib/assertions';\nimport { Bucket } from 'aws-cdk-lib/aws-s3';\nimport { AwsSolutionsChecks, NagSuppressions } from 'cdk-nag';\nimport { AccessLog } from '../../framework';\nimport { EventbridgeBroker } from '../../framework/foundation/eventbridge-broker';\nimport { QueuedS3Adapter } from '../adapter';\nimport { BedrockDocumentProcessing } from '../bedrock-document-processing';\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 EventBridge broker\nconst broker = new EventbridgeBroker(stack, 'IDPBroker', {\n  name: 'idp-broker',\n  eventSource: 'intelligent-document-processing',\n});\n\nconst accessLog = new AccessLog(stack, 'AccessLog');\n\n// Create S3 bucket for agentic tools\nconst bucket = new Bucket(stack, 'BedrockDocumentProcessingBucket', {\n  serverAccessLogsBucket: accessLog.bucket,\n  serverAccessLogsPrefix: accessLog.bucketPrefix,\n  enforceSSL: true,\n});\n\nconst adapter = new QueuedS3Adapter({\n  bucket,\n});\n\n// Create the main BedrockDocumentProcessing construct\nnew BedrockDocumentProcessing(stack, 'BedrockDocumentProcessing', {\n  ingressAdapter: adapter,\n  classificationBedrockModel: { useCrossRegionInference: true },\n  processingBedrockModel: { useCrossRegionInference: true },\n  eventbridgeBroker: broker,\n  enableObservability: true,\n});\n\n// Suppress CDK-managed BucketNotificationsHandler AWS managed policy\nNagSuppressions.addResourceSuppressionsByPath(\n  stack,\n  '/TestStack/BucketNotificationsHandler050a0587b7544547bf325f094a3db834/Role',\n  [\n    {\n      id: 'AwsSolutions-IAM4',\n      reason: 'CDK-managed BucketNotificationsHandler requires AWSLambdaBasicExecutionRole for S3 event processing',\n      appliesTo: ['Policy::arn:<AWS::Partition>:iam::aws:policy/service-role/AWSLambdaBasicExecutionRole'],\n    },\n  ],\n);\n\n// Suppress S3 bucket wildcard permissions for Lambda roles\nNagSuppressions.addResourceSuppressions(\n  stack,\n  [\n    {\n      id: 'AwsSolutions-IAM5',\n      reason: 'Lambda functions require wildcard access to S3 bucket objects for document processing',\n      appliesTo: ['Resource::<BedrockDocumentProcessingBucketA06CA1E2.Arn>/*'],\n    },\n  ],\n  true,\n);\n\n// Suppress Bedrock foundation model wildcard permissions\nNagSuppressions.addResourceSuppressions(\n  stack,\n  [\n    {\n      id: 'AwsSolutions-IAM5',\n      reason: 'Cross-region inference requires wildcard region access to Bedrock foundation models',\n      appliesTo: ['Resource::arn:aws:bedrock:*::foundation-model/anthropic.claude-3-7-sonnet-20250219-v1:0'],\n    },\n  ],\n  true,\n);\n\n// Suppress SQSConsumerRole wildcard permissions for Lambda log streams\nNagSuppressions.addResourceSuppressions(\n  stack,\n  [\n    {\n      id: 'AwsSolutions-IAM5',\n      reason: 'Lambda log stream ARN is only known at runtime, wildcard required for CloudWatch Logs access',\n    },\n  ],\n  true,\n);\n\n// Suppress StateMachineRole wildcard permissions for Lambda invocation\nNagSuppressions.addResourceSuppressionsByPath(\n  stack,\n  '/TestStack/BedrockDocumentProcessing/StateMachineRole/DefaultPolicy',\n  [\n    {\n      id: 'AwsSolutions-IAM5',\n      reason: 'Step Functions requires wildcard permissions to invoke Lambda functions with version-specific ARNs',\n    },\n  ],\n);\n\n// Suppress Lambda log group wildcard permissions\nNagSuppressions.addResourceSuppressions(\n  stack,\n  [\n    {\n      id: 'AwsSolutions-IAM5',\n      reason: 'Lambda log stream names are generated at runtime, wildcard required for CloudWatch Logs access',\n    },\n  ],\n  true,\n);\n\n// Apply CDK Nag checks\nAspects.of(app).add(new AwsSolutionsChecks({ verbose: true }));\n\n// Synthesize the stack and 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: 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"]}