@cdklabs/cdk-appmod-catalog-blueprints 1.3.0 → 1.4.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.jsii +4 -4
- package/README.md +76 -98
- package/lib/document-processing/adapter/queued-s3-adapter.js +1 -1
- package/lib/document-processing/agentic-document-processing.js +1 -1
- package/lib/document-processing/base-document-processing.js +1 -1
- package/lib/document-processing/bedrock-document-processing.js +1 -1
- package/lib/document-processing/default-document-processing-config.js +1 -1
- package/lib/document-processing/tests/agentic-document-processing.test.js +104 -60
- package/lib/document-processing/tests/base-document-processing-nag.test.d.ts +1 -0
- package/lib/document-processing/tests/base-document-processing-nag.test.js +161 -0
- package/lib/document-processing/tests/base-document-processing.test.d.ts +1 -0
- package/lib/document-processing/tests/base-document-processing.test.js +499 -0
- package/lib/document-processing/tests/bedrock-document-processing.test.js +212 -36
- package/lib/document-processing/tests/queued-s3-adapter-nag.test.d.ts +1 -0
- package/lib/document-processing/tests/queued-s3-adapter-nag.test.js +122 -0
- package/lib/document-processing/tests/queued-s3-adapter.test.d.ts +1 -0
- package/lib/document-processing/tests/queued-s3-adapter.test.js +276 -0
- package/lib/framework/agents/base-agent.js +1 -1
- package/lib/framework/agents/batch-agent.js +1 -1
- package/lib/framework/agents/default-agent-config.js +1 -1
- package/lib/framework/bedrock/bedrock.js +1 -1
- package/lib/framework/custom-resource/default-runtimes.js +1 -1
- package/lib/framework/foundation/access-log.js +1 -1
- package/lib/framework/foundation/eventbridge-broker.js +1 -1
- package/lib/framework/foundation/network.js +1 -1
- package/lib/framework/tests/access-log.test.d.ts +1 -0
- package/lib/framework/tests/access-log.test.js +146 -0
- package/lib/framework/tests/batch-agent.test.d.ts +1 -0
- package/lib/framework/tests/batch-agent.test.js +164 -0
- package/lib/framework/tests/bedrock.test.d.ts +1 -0
- package/lib/framework/tests/bedrock.test.js +68 -0
- package/lib/framework/tests/eventbridge-broker.test.d.ts +1 -0
- package/lib/framework/tests/eventbridge-broker.test.js +73 -0
- package/lib/framework/tests/framework-nag.test.d.ts +1 -0
- package/lib/framework/tests/framework-nag.test.js +155 -0
- package/lib/framework/tests/network.test.d.ts +1 -0
- package/lib/framework/tests/network.test.js +120 -0
- package/lib/tsconfig.tsbuildinfo +1 -1
- package/lib/utilities/data-loader.js +1 -1
- package/lib/utilities/lambda-iam-utils.js +1 -1
- package/lib/utilities/observability/cloudfront-distribution-observability-property-injector.js +1 -1
- package/lib/utilities/observability/default-observability-config.js +1 -1
- package/lib/utilities/observability/lambda-observability-property-injector.js +1 -1
- package/lib/utilities/observability/log-group-data-protection-utils.js +1 -1
- package/lib/utilities/observability/powertools-config.js +1 -1
- package/lib/utilities/observability/state-machine-observability-property-injector.js +1 -1
- package/lib/webapp/frontend-construct.js +1 -1
- 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"]}
|