@cdklabs/cdk-appmod-catalog-blueprints 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (105) hide show
  1. package/.jsii +8644 -0
  2. package/LICENSE +202 -0
  3. package/README.md +212 -0
  4. package/lib/document-processing/agentic-document-processing.d.ts +16 -0
  5. package/lib/document-processing/agentic-document-processing.js +90 -0
  6. package/lib/document-processing/base-document-processing.d.ts +189 -0
  7. package/lib/document-processing/base-document-processing.js +509 -0
  8. package/lib/document-processing/bedrock-document-processing.d.ts +167 -0
  9. package/lib/document-processing/bedrock-document-processing.js +297 -0
  10. package/lib/document-processing/index.d.ts +3 -0
  11. package/lib/document-processing/index.js +20 -0
  12. package/lib/document-processing/resources/default-bedrock-invoke/index.py +63 -0
  13. package/lib/document-processing/resources/default-bedrock-invoke/requirements.txt +4 -0
  14. package/lib/document-processing/resources/default-doc-retrieval-lambda/index.mjs +92 -0
  15. package/lib/document-processing/resources/default-doc-retrieval-lambda/package.json +10 -0
  16. package/lib/document-processing/resources/default-error-handler/index.js +46 -0
  17. package/lib/document-processing/resources/default-error-handler/package.json +4 -0
  18. package/lib/document-processing/resources/default-image-processor/classifier.mjs +665 -0
  19. package/lib/document-processing/resources/default-image-processor/extractors.mjs +465 -0
  20. package/lib/document-processing/resources/default-image-processor/index.mjs +143 -0
  21. package/lib/document-processing/resources/default-image-processor/package-lock.json +12 -0
  22. package/lib/document-processing/resources/default-image-processor/package.json +4 -0
  23. package/lib/document-processing/resources/default-image-validator/index.mjs +76 -0
  24. package/lib/document-processing/resources/default-image-validator/package-lock.json +154 -0
  25. package/lib/document-processing/resources/default-image-validator/package.json +7 -0
  26. package/lib/document-processing/resources/default-pdf-processor/index.js +46 -0
  27. package/lib/document-processing/resources/default-pdf-validator/index.js +36 -0
  28. package/lib/document-processing/resources/default-sqs-consumer/index.py +111 -0
  29. package/lib/document-processing/resources/default-sqs-consumer/requirements.txt +4 -0
  30. package/lib/document-processing/resources/default-sqs-consumer/sample_payload.json +20 -0
  31. package/lib/document-processing/resources/default-sqs-consumer/sample_payload_multi.json +24 -0
  32. package/lib/document-processing/resources/default-strands-agent/index.py +111 -0
  33. package/lib/document-processing/resources/default-strands-agent/requirements.txt +6 -0
  34. package/lib/document-processing/tests/agentic-document-processing-nag.test.d.ts +1 -0
  35. package/lib/document-processing/tests/agentic-document-processing-nag.test.js +107 -0
  36. package/lib/document-processing/tests/agentic-document-processing.test.d.ts +1 -0
  37. package/lib/document-processing/tests/agentic-document-processing.test.js +125 -0
  38. package/lib/document-processing/tests/bedrock-document-processing-nag.test.d.ts +1 -0
  39. package/lib/document-processing/tests/bedrock-document-processing-nag.test.js +101 -0
  40. package/lib/document-processing/tests/bedrock-document-processing.test.d.ts +1 -0
  41. package/lib/document-processing/tests/bedrock-document-processing.test.js +79 -0
  42. package/lib/framework/custom-resource/default-runtimes.d.ts +21 -0
  43. package/lib/framework/custom-resource/default-runtimes.js +34 -0
  44. package/lib/framework/custom-resource/index.d.ts +1 -0
  45. package/lib/framework/custom-resource/index.js +18 -0
  46. package/lib/framework/foundation/access-log.d.ts +69 -0
  47. package/lib/framework/foundation/access-log.js +121 -0
  48. package/lib/framework/foundation/eventbridge-broker.d.ts +18 -0
  49. package/lib/framework/foundation/eventbridge-broker.js +42 -0
  50. package/lib/framework/foundation/index.d.ts +3 -0
  51. package/lib/framework/foundation/index.js +20 -0
  52. package/lib/framework/foundation/network.d.ts +19 -0
  53. package/lib/framework/foundation/network.js +83 -0
  54. package/lib/framework/index.d.ts +2 -0
  55. package/lib/framework/index.js +19 -0
  56. package/lib/framework/quickstart/base-quickstart.d.ts +30 -0
  57. package/lib/framework/quickstart/base-quickstart.js +30 -0
  58. package/lib/index.d.ts +4 -0
  59. package/lib/index.js +21 -0
  60. package/lib/tsconfig.tsbuildinfo +1 -0
  61. package/lib/utilities/cdk-nag-config.d.ts +42 -0
  62. package/lib/utilities/cdk-nag-config.js +194 -0
  63. package/lib/utilities/data-loader-lambda/index.py +282 -0
  64. package/lib/utilities/data-loader-lambda/requirements.txt +3 -0
  65. package/lib/utilities/data-loader.d.ts +173 -0
  66. package/lib/utilities/data-loader.js +447 -0
  67. package/lib/utilities/index.d.ts +3 -0
  68. package/lib/utilities/index.js +20 -0
  69. package/lib/utilities/lambda-iam-utils.d.ts +145 -0
  70. package/lib/utilities/lambda-iam-utils.js +235 -0
  71. package/lib/utilities/lambda_layers/data-masking/layer-construct.d.ts +42 -0
  72. package/lib/utilities/lambda_layers/data-masking/layer-construct.js +53 -0
  73. package/lib/utilities/lambda_layers/data-masking/layer-construct.ts +88 -0
  74. package/lib/utilities/observability/bedrock-observability.d.ts +18 -0
  75. package/lib/utilities/observability/bedrock-observability.js +131 -0
  76. package/lib/utilities/observability/cloudfront-distribution-observability-property-injector.d.ts +6 -0
  77. package/lib/utilities/observability/cloudfront-distribution-observability-property-injector.js +22 -0
  78. package/lib/utilities/observability/index.d.ts +6 -0
  79. package/lib/utilities/observability/index.js +25 -0
  80. package/lib/utilities/observability/lambda-observability-property-injector.d.ts +8 -0
  81. package/lib/utilities/observability/lambda-observability-property-injector.js +43 -0
  82. package/lib/utilities/observability/log-group-data-protection-props.d.ts +19 -0
  83. package/lib/utilities/observability/log-group-data-protection-props.js +5 -0
  84. package/lib/utilities/observability/observability.d.ts +83 -0
  85. package/lib/utilities/observability/observability.js +278 -0
  86. package/lib/utilities/observability/observable.d.ts +32 -0
  87. package/lib/utilities/observability/observable.js +3 -0
  88. package/lib/utilities/observability/powertools-config.d.ts +3 -0
  89. package/lib/utilities/observability/powertools-config.js +25 -0
  90. package/lib/utilities/observability/resources/bedrock-manage-logging-configuration/index.py +27 -0
  91. package/lib/utilities/observability/state-machine-observability-property-injector.d.ts +8 -0
  92. package/lib/utilities/observability/state-machine-observability-property-injector.js +49 -0
  93. package/lib/utilities/tests/data-loader-nag.test.d.ts +1 -0
  94. package/lib/utilities/tests/data-loader-nag.test.js +432 -0
  95. package/lib/utilities/tests/data-loader.test.d.ts +1 -0
  96. package/lib/utilities/tests/data-loader.test.js +284 -0
  97. package/lib/webapp/frontend-construct.d.ts +136 -0
  98. package/lib/webapp/frontend-construct.js +253 -0
  99. package/lib/webapp/index.d.ts +1 -0
  100. package/lib/webapp/index.js +18 -0
  101. package/lib/webapp/tests/frontend-construct-nag.test.d.ts +1 -0
  102. package/lib/webapp/tests/frontend-construct-nag.test.js +266 -0
  103. package/lib/webapp/tests/frontend-construct.test.d.ts +1 -0
  104. package/lib/webapp/tests/frontend-construct.test.js +385 -0
  105. package/package.json +183 -0
@@ -0,0 +1,509 @@
1
+ "use strict";
2
+ var _a;
3
+ Object.defineProperty(exports, "__esModule", { value: true });
4
+ exports.BaseDocumentProcessing = exports.DocumentProcessingPrefix = void 0;
5
+ const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
6
+ // Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
7
+ // SPDX-License-Identifier: Apache-2.0
8
+ const path = require("node:path");
9
+ const aws_lambda_python_alpha_1 = require("@aws-cdk/aws-lambda-python-alpha");
10
+ const aws_cdk_lib_1 = require("aws-cdk-lib");
11
+ const aws_dynamodb_1 = require("aws-cdk-lib/aws-dynamodb");
12
+ const aws_ec2_1 = require("aws-cdk-lib/aws-ec2");
13
+ const aws_iam_1 = require("aws-cdk-lib/aws-iam");
14
+ const aws_kms_1 = require("aws-cdk-lib/aws-kms");
15
+ const aws_lambda_event_sources_1 = require("aws-cdk-lib/aws-lambda-event-sources");
16
+ const aws_s3_1 = require("aws-cdk-lib/aws-s3");
17
+ const aws_s3_notifications_1 = require("aws-cdk-lib/aws-s3-notifications");
18
+ const aws_sqs_1 = require("aws-cdk-lib/aws-sqs");
19
+ const aws_stepfunctions_1 = require("aws-cdk-lib/aws-stepfunctions");
20
+ const aws_stepfunctions_tasks_1 = require("aws-cdk-lib/aws-stepfunctions-tasks");
21
+ const constructs_1 = require("constructs");
22
+ const framework_1 = require("../framework");
23
+ const utilities_1 = require("../utilities");
24
+ const lambda_observability_property_injector_1 = require("../utilities/observability/lambda-observability-property-injector");
25
+ const powertools_config_1 = require("../utilities/observability/powertools-config");
26
+ const state_machine_observability_property_injector_1 = require("../utilities/observability/state-machine-observability-property-injector");
27
+ /**
28
+ * S3 prefix constants for organizing documents throughout the processing lifecycle.
29
+ *
30
+ * Documents flow through these prefixes based on processing outcomes:
31
+ * - Upload → raw/ (triggers processing)
32
+ * - Success → processed/ (workflow completed successfully)
33
+ * - Failure → failed/ (workflow encountered errors)
34
+ */
35
+ var DocumentProcessingPrefix;
36
+ (function (DocumentProcessingPrefix) {
37
+ /** Prefix for newly uploaded documents awaiting processing */
38
+ DocumentProcessingPrefix["RAW"] = "raw/";
39
+ /** Prefix for documents that failed processing */
40
+ DocumentProcessingPrefix["FAILED"] = "failed/";
41
+ /** Prefix for successfully processed documents */
42
+ DocumentProcessingPrefix["PROCESSED"] = "processed/";
43
+ })(DocumentProcessingPrefix || (exports.DocumentProcessingPrefix = DocumentProcessingPrefix = {}));
44
+ /**
45
+ * Abstract base class for serverless document processing workflows.
46
+ *
47
+ * Provides a complete document processing pipeline with:
48
+ * - **S3 Storage**: Organized with prefixes (raw/, processed/, failed/) for document lifecycle management
49
+ * - **SQS Queue**: Reliable message processing with configurable visibility timeout and dead letter queue
50
+ * - **DynamoDB Table**: Workflow metadata tracking with DocumentId as partition key
51
+ * - **Step Functions**: Orchestrated workflow with automatic file movement based on processing outcome
52
+ * - **Auto-triggering**: S3 event notifications automatically start processing when files are uploaded to raw/ prefix
53
+ * - **Error Handling**: Failed documents are moved to failed/ prefix with error details stored in DynamoDB
54
+ * - **EventBridge Integration**: Optional custom event publishing for workflow state changes
55
+ *
56
+ * ## Architecture Flow
57
+ * S3 Upload (raw/) → SQS → Lambda Consumer → Step Functions → Processing Steps → S3 (processed/failed/)
58
+ *
59
+ * ## Implementation Requirements
60
+ * Subclasses must implement four abstract methods to define the processing workflow:
61
+ * - `classificationStep()`: Document type classification
62
+ * - `extractionStep()`: Data extraction from documents
63
+ * - `enrichmentStep()`: Optional data enrichment (return undefined to skip)
64
+ * - `postProcessingStep()`: Optional post-processing (return undefined to skip)
65
+ */
66
+ class BaseDocumentProcessing extends constructs_1.Construct {
67
+ /**
68
+ * Creates a new BaseDocumentProcessing construct.
69
+ *
70
+ * Initializes the complete document processing infrastructure including S3 bucket,
71
+ * SQS queue, DynamoDB table, and sets up S3 event notifications to trigger processing.
72
+ *
73
+ * @param scope - The scope in which to define this construct
74
+ * @param id - The scoped construct ID. Must be unique within the scope.
75
+ * @param props - Configuration properties for the document processing pipeline
76
+ */
77
+ constructor(scope, id, props) {
78
+ super(scope, id);
79
+ this.props = props;
80
+ if (props.network) {
81
+ props.network.createServiceEndpoint('vpce-sqs', aws_ec2_1.InterfaceVpcEndpointAwsService.SQS);
82
+ props.network.createServiceEndpoint('vpce-s3', aws_ec2_1.InterfaceVpcEndpointAwsService.S3);
83
+ props.network.createServiceEndpoint('vpce-sfn', aws_ec2_1.InterfaceVpcEndpointAwsService.STEP_FUNCTIONS);
84
+ props.network.createServiceEndpoint('vpce-eb', aws_ec2_1.InterfaceVpcEndpointAwsService.EVENTBRIDGE);
85
+ if (props.enableObservability) {
86
+ props.network.createServiceEndpoint('vpce-logs', aws_ec2_1.InterfaceVpcEndpointAwsService.CLOUDWATCH_LOGS);
87
+ props.network.createServiceEndpoint('vpce-metrics', aws_ec2_1.InterfaceVpcEndpointAwsService.CLOUDWATCH_MONITORING);
88
+ }
89
+ }
90
+ this.encryptionKey = props.encryptionKey || new aws_kms_1.Key(this, 'IDPEncryptionKey', {
91
+ enableKeyRotation: true,
92
+ removalPolicy: props.removalPolicy || aws_cdk_lib_1.RemovalPolicy.DESTROY,
93
+ });
94
+ const bucketName = `documentprocessingbucket-${aws_cdk_lib_1.Names.uniqueResourceName(this, {
95
+ maxLength: 60 - 'documentprocessingbucket-'.length,
96
+ })}`.toLowerCase();
97
+ const bucketArn = `arn:aws:s3:::${bucketName}`;
98
+ this.encryptionKey.grantEncryptDecrypt(new aws_iam_1.ServicePrincipal('s3.amazonaws.com', {
99
+ conditions: {
100
+ ArnEquals: {
101
+ 'kms:EncryptionContext:aws:s3:arn': bucketArn,
102
+ },
103
+ },
104
+ }));
105
+ this.bucket = props.bucket || new aws_s3_1.Bucket(this, 'DocumentProcessingBucket', {
106
+ bucketName,
107
+ autoDeleteObjects: (props.removalPolicy && props.removalPolicy === aws_cdk_lib_1.RemovalPolicy.DESTROY) || !props.removalPolicy ? true : false,
108
+ removalPolicy: props.removalPolicy || aws_cdk_lib_1.RemovalPolicy.DESTROY,
109
+ encryption: aws_s3_1.BucketEncryption.KMS,
110
+ enforceSSL: true,
111
+ bucketKeyEnabled: true,
112
+ });
113
+ this.bucketEncryptionKey = this.bucket.encryptionKey;
114
+ const tempLogGroupDataProtection = props.logGroupDataProtection || {
115
+ logGroupEncryptionKey: new aws_kms_1.Key(this, 'LogGroupEncryptionKey', {
116
+ enableKeyRotation: true,
117
+ removalPolicy: props.removalPolicy || aws_cdk_lib_1.RemovalPolicy.DESTROY,
118
+ }),
119
+ };
120
+ if (!tempLogGroupDataProtection.logGroupEncryptionKey) {
121
+ this.logGroupDataProtection = {
122
+ dataProtectionIdentifiers: tempLogGroupDataProtection.dataProtectionIdentifiers,
123
+ logGroupEncryptionKey: new aws_kms_1.Key(this, 'LogGroupEncryptionKey', {
124
+ enableKeyRotation: true,
125
+ removalPolicy: props.removalPolicy || aws_cdk_lib_1.RemovalPolicy.DESTROY,
126
+ }),
127
+ };
128
+ }
129
+ else {
130
+ this.logGroupDataProtection = tempLogGroupDataProtection;
131
+ }
132
+ this.deadLetterQueue = new aws_sqs_1.Queue(this, 'DocumentProcessingDLQ', {
133
+ visibilityTimeout: props.queueVisibilityTimeout || aws_cdk_lib_1.Duration.seconds(300),
134
+ removalPolicy: props.removalPolicy || aws_cdk_lib_1.RemovalPolicy.DESTROY,
135
+ enforceSSL: true,
136
+ encryption: aws_sqs_1.QueueEncryption.KMS,
137
+ encryptionMasterKey: this.encryptionKey,
138
+ });
139
+ this.queue = new aws_sqs_1.Queue(this, 'DocumentProcessingQueue', {
140
+ visibilityTimeout: props.queueVisibilityTimeout || aws_cdk_lib_1.Duration.seconds(300),
141
+ removalPolicy: props.removalPolicy || aws_cdk_lib_1.RemovalPolicy.DESTROY,
142
+ enforceSSL: true,
143
+ deadLetterQueue: {
144
+ maxReceiveCount: props.dlqMaxReceiveCount || 5,
145
+ queue: this.deadLetterQueue,
146
+ },
147
+ encryption: aws_sqs_1.QueueEncryption.KMS,
148
+ encryptionMasterKey: this.encryptionKey,
149
+ });
150
+ this.documentProcessingTable = props.documentProcessingTable || new aws_dynamodb_1.Table(this, 'DocumentProcessingTable', {
151
+ partitionKey: {
152
+ name: 'DocumentId',
153
+ type: aws_dynamodb_1.AttributeType.STRING,
154
+ },
155
+ billingMode: aws_dynamodb_1.BillingMode.PAY_PER_REQUEST,
156
+ removalPolicy: props.removalPolicy || aws_cdk_lib_1.RemovalPolicy.DESTROY,
157
+ pointInTimeRecoverySpecification: {
158
+ pointInTimeRecoveryEnabled: true,
159
+ },
160
+ encryption: aws_dynamodb_1.TableEncryption.CUSTOMER_MANAGED,
161
+ encryptionKey: this.encryptionKey,
162
+ });
163
+ if (props.enableObservability) {
164
+ aws_cdk_lib_1.PropertyInjectors.of(this).add(new state_machine_observability_property_injector_1.StateMachineObservabilityPropertyInjector(this.logGroupDataProtection), new lambda_observability_property_injector_1.LambdaObservabilityPropertyInjector(this.logGroupDataProtection));
165
+ }
166
+ this.metricNamespace = props.metricNamespace || 'appmod-catalog';
167
+ this.metricServiceName = props.metricServiceName || 'document-processing';
168
+ }
169
+ handleStateMachineCreation(stateMachineId) {
170
+ const classificationStep = this.classificationStep();
171
+ const processingStep = this.processingStep();
172
+ const enrichmentStep = this.enrichmentStep();
173
+ const postProcessingStep = this.postProcessingStep();
174
+ const initMetadataEntry = new aws_stepfunctions_tasks_1.DynamoPutItem(this, 'InitMetadataEntry', {
175
+ table: this.documentProcessingTable,
176
+ item: {
177
+ DocumentId: aws_stepfunctions_tasks_1.DynamoAttributeValue.fromString(aws_stepfunctions_1.JsonPath.stringAt('$.documentId')),
178
+ Bucket: aws_stepfunctions_tasks_1.DynamoAttributeValue.fromString(aws_stepfunctions_1.JsonPath.stringAt('$.bucket')),
179
+ Key: aws_stepfunctions_tasks_1.DynamoAttributeValue.fromString(aws_stepfunctions_1.JsonPath.stringAt('$.key')),
180
+ WorkflowStatus: aws_stepfunctions_tasks_1.DynamoAttributeValue.fromString('pending'),
181
+ StateMachineExecId: aws_stepfunctions_tasks_1.DynamoAttributeValue.fromString(aws_stepfunctions_1.JsonPath.stringAt('$$.Execution.Id')),
182
+ },
183
+ resultPath: aws_stepfunctions_1.JsonPath.DISCARD,
184
+ });
185
+ // File movement operations
186
+ const moveToFailed = this.createMoveToFailedChain();
187
+ const moveToProcessed = this.createMoveToProcessedChain();
188
+ const processingChain = processingStep
189
+ .addCatch(new aws_stepfunctions_tasks_1.DynamoUpdateItem(this, 'ProcessingFailDDBUpdate', {
190
+ table: this.documentProcessingTable,
191
+ key: {
192
+ DocumentId: aws_stepfunctions_tasks_1.DynamoAttributeValue.fromString(aws_stepfunctions_1.JsonPath.stringAt('$.documentId')),
193
+ },
194
+ updateExpression: 'SET WorkflowStatus = :newStatus',
195
+ expressionAttributeValues: {
196
+ ':newStatus': aws_stepfunctions_tasks_1.DynamoAttributeValue.fromString('processing-failure'),
197
+ },
198
+ resultPath: aws_stepfunctions_1.JsonPath.DISCARD,
199
+ }).next(moveToFailed), {
200
+ resultPath: aws_stepfunctions_1.JsonPath.DISCARD,
201
+ })
202
+ .next(new aws_stepfunctions_tasks_1.DynamoUpdateItem(this, 'ProcessingSuccessUpdate', {
203
+ table: this.documentProcessingTable,
204
+ key: {
205
+ DocumentId: aws_stepfunctions_tasks_1.DynamoAttributeValue.fromString(aws_stepfunctions_1.JsonPath.stringAt('$.documentId')),
206
+ },
207
+ updateExpression: 'SET WorkflowStatus = :newStatus, ProcessingResult = :processingResult',
208
+ expressionAttributeValues: {
209
+ ':newStatus': aws_stepfunctions_tasks_1.DynamoAttributeValue.fromString('processing-complete'),
210
+ ':processingResult': aws_stepfunctions_tasks_1.DynamoAttributeValue.fromString(aws_stepfunctions_1.JsonPath.jsonToString(aws_stepfunctions_1.JsonPath.objectAt('$.processingResult'))),
211
+ },
212
+ resultPath: aws_stepfunctions_1.JsonPath.DISCARD,
213
+ }));
214
+ // Build the complete chain including optional steps
215
+ if (enrichmentStep) {
216
+ const enrichmentChain = enrichmentStep
217
+ .addCatch(new aws_stepfunctions_tasks_1.DynamoUpdateItem(this, 'EnrichmentFailDDBUpdate', {
218
+ table: this.documentProcessingTable,
219
+ key: {
220
+ DocumentId: aws_stepfunctions_tasks_1.DynamoAttributeValue.fromString(aws_stepfunctions_1.JsonPath.stringAt('$.documentId')),
221
+ },
222
+ updateExpression: 'SET WorkflowStatus = :newStatus',
223
+ expressionAttributeValues: {
224
+ ':newStatus': aws_stepfunctions_tasks_1.DynamoAttributeValue.fromString('enrichment-failure'),
225
+ },
226
+ resultPath: aws_stepfunctions_1.JsonPath.DISCARD,
227
+ }).next(moveToFailed), {
228
+ resultPath: aws_stepfunctions_1.JsonPath.DISCARD,
229
+ })
230
+ .next(new aws_stepfunctions_tasks_1.DynamoUpdateItem(this, 'EnrichmentSuccessUpdate', {
231
+ table: this.documentProcessingTable,
232
+ key: {
233
+ DocumentId: aws_stepfunctions_tasks_1.DynamoAttributeValue.fromString(aws_stepfunctions_1.JsonPath.stringAt('$.documentId')),
234
+ },
235
+ updateExpression: 'SET WorkflowStatus = :newStatus, EnrichmentResult = :enrichmentResult',
236
+ expressionAttributeValues: {
237
+ ':newStatus': postProcessingStep ? aws_stepfunctions_tasks_1.DynamoAttributeValue.fromString('enrichment-complete') : aws_stepfunctions_tasks_1.DynamoAttributeValue.fromString('complete'),
238
+ ':enrichmentResult': aws_stepfunctions_tasks_1.DynamoAttributeValue.fromString(aws_stepfunctions_1.JsonPath.jsonToString(aws_stepfunctions_1.JsonPath.objectAt('$.enrichedResult'))),
239
+ },
240
+ resultPath: aws_stepfunctions_1.JsonPath.DISCARD,
241
+ }));
242
+ processingChain.next(enrichmentChain);
243
+ if (postProcessingStep) {
244
+ const postProcessingChain = postProcessingStep
245
+ .addCatch(new aws_stepfunctions_tasks_1.DynamoUpdateItem(this, 'PostProcessingFailDDBUpdate', {
246
+ table: this.documentProcessingTable,
247
+ key: {
248
+ DocumentId: aws_stepfunctions_tasks_1.DynamoAttributeValue.fromString(aws_stepfunctions_1.JsonPath.stringAt('$.documentId')),
249
+ },
250
+ updateExpression: 'SET WorkflowStatus = :newStatus',
251
+ expressionAttributeValues: {
252
+ ':newStatus': aws_stepfunctions_tasks_1.DynamoAttributeValue.fromString('post-processing-failure'),
253
+ },
254
+ resultPath: aws_stepfunctions_1.JsonPath.DISCARD,
255
+ }).next(moveToFailed), {
256
+ resultPath: aws_stepfunctions_1.JsonPath.DISCARD,
257
+ })
258
+ .next(new aws_stepfunctions_tasks_1.DynamoUpdateItem(this, 'PostProcessingSuccessUpdate', {
259
+ table: this.documentProcessingTable,
260
+ key: {
261
+ DocumentId: aws_stepfunctions_tasks_1.DynamoAttributeValue.fromString(aws_stepfunctions_1.JsonPath.stringAt('$.documentId')),
262
+ },
263
+ updateExpression: 'SET WorkflowStatus = :newStatus, PostProcessingResult = :postProcessingResult',
264
+ expressionAttributeValues: {
265
+ ':newStatus': aws_stepfunctions_tasks_1.DynamoAttributeValue.fromString('complete'),
266
+ ':postProcessingResult': aws_stepfunctions_tasks_1.DynamoAttributeValue.fromString(aws_stepfunctions_1.JsonPath.jsonToString(aws_stepfunctions_1.JsonPath.objectAt('$.postProcessedResult'))),
267
+ },
268
+ resultPath: aws_stepfunctions_1.JsonPath.DISCARD,
269
+ }).next(moveToProcessed));
270
+ enrichmentChain.next(postProcessingChain);
271
+ }
272
+ else {
273
+ enrichmentChain.next(moveToProcessed);
274
+ }
275
+ }
276
+ else if (postProcessingStep) {
277
+ const postProcessingChain = postProcessingStep
278
+ .addCatch(new aws_stepfunctions_tasks_1.DynamoUpdateItem(this, 'PostProcessingFailDDBUpdate', {
279
+ table: this.documentProcessingTable,
280
+ key: {
281
+ DocumentId: aws_stepfunctions_tasks_1.DynamoAttributeValue.fromString(aws_stepfunctions_1.JsonPath.stringAt('$.documentId')),
282
+ },
283
+ updateExpression: 'SET WorkflowStatus = :newStatus',
284
+ expressionAttributeValues: {
285
+ ':newStatus': aws_stepfunctions_tasks_1.DynamoAttributeValue.fromString('post-processing-failure'),
286
+ },
287
+ resultPath: aws_stepfunctions_1.JsonPath.DISCARD,
288
+ }).next(moveToFailed), {
289
+ resultPath: aws_stepfunctions_1.JsonPath.DISCARD,
290
+ })
291
+ .next(new aws_stepfunctions_tasks_1.DynamoUpdateItem(this, 'PostProcessingSuccessUpdate', {
292
+ table: this.documentProcessingTable,
293
+ key: {
294
+ DocumentId: aws_stepfunctions_tasks_1.DynamoAttributeValue.fromString(aws_stepfunctions_1.JsonPath.stringAt('$.documentId')),
295
+ },
296
+ updateExpression: 'SET WorkflowStatus = :newStatus, PostProcessingResult = :postProcessingResult',
297
+ expressionAttributeValues: {
298
+ ':newStatus': aws_stepfunctions_tasks_1.DynamoAttributeValue.fromString('complete'),
299
+ ':postProcessingResult': aws_stepfunctions_tasks_1.DynamoAttributeValue.fromString(aws_stepfunctions_1.JsonPath.jsonToString(aws_stepfunctions_1.JsonPath.objectAt('$.postProcessedResult'))),
300
+ },
301
+ resultPath: aws_stepfunctions_1.JsonPath.DISCARD,
302
+ }).next(moveToProcessed));
303
+ processingChain.next(postProcessingChain);
304
+ }
305
+ else {
306
+ // No optional steps - mark as complete after extraction
307
+ processingChain.next(new aws_stepfunctions_tasks_1.DynamoUpdateItem(this, 'WorkflowCompleteUpdate', {
308
+ table: this.documentProcessingTable,
309
+ key: {
310
+ DocumentId: aws_stepfunctions_tasks_1.DynamoAttributeValue.fromString(aws_stepfunctions_1.JsonPath.stringAt('$.documentId')),
311
+ },
312
+ updateExpression: 'SET WorkflowStatus = :newStatus',
313
+ expressionAttributeValues: {
314
+ ':newStatus': aws_stepfunctions_tasks_1.DynamoAttributeValue.fromString('complete'),
315
+ },
316
+ resultPath: aws_stepfunctions_1.JsonPath.DISCARD,
317
+ }).next(moveToProcessed));
318
+ }
319
+ const workflowDefinition = initMetadataEntry.next(classificationStep
320
+ .addCatch(new aws_stepfunctions_tasks_1.DynamoUpdateItem(this, 'ClassificationFailDDBUpdate', {
321
+ table: this.documentProcessingTable,
322
+ key: {
323
+ DocumentId: aws_stepfunctions_tasks_1.DynamoAttributeValue.fromString(aws_stepfunctions_1.JsonPath.stringAt('$.documentId')),
324
+ },
325
+ updateExpression: 'SET WorkflowStatus = :newStatus',
326
+ expressionAttributeValues: {
327
+ ':newStatus': aws_stepfunctions_tasks_1.DynamoAttributeValue.fromString('classification-failure'),
328
+ },
329
+ resultPath: aws_stepfunctions_1.JsonPath.DISCARD,
330
+ }).next(moveToFailed), {
331
+ resultPath: aws_stepfunctions_1.JsonPath.DISCARD,
332
+ })
333
+ .next(new aws_stepfunctions_tasks_1.DynamoUpdateItem(this, 'ClassificationSuccessUpdate', {
334
+ table: this.documentProcessingTable,
335
+ key: {
336
+ DocumentId: aws_stepfunctions_tasks_1.DynamoAttributeValue.fromString(aws_stepfunctions_1.JsonPath.stringAt('$.documentId')),
337
+ },
338
+ updateExpression: 'SET WorkflowStatus = :newStatus, ClassificationResult = :classificationResult',
339
+ expressionAttributeValues: {
340
+ ':newStatus': aws_stepfunctions_tasks_1.DynamoAttributeValue.fromString('classification-complete'),
341
+ ':classificationResult': aws_stepfunctions_tasks_1.DynamoAttributeValue.fromString(aws_stepfunctions_1.JsonPath.jsonToString(aws_stepfunctions_1.JsonPath.objectAt('$.classificationResult'))),
342
+ },
343
+ resultPath: aws_stepfunctions_1.JsonPath.DISCARD,
344
+ }))).next(processingChain);
345
+ const role = this.createStateMachineRole();
346
+ this.encryptionKey.grantEncryptDecrypt(role);
347
+ if (this.bucketEncryptionKey) {
348
+ this.bucketEncryptionKey.grantEncryptDecrypt(role);
349
+ }
350
+ const stateMachine = new aws_stepfunctions_1.StateMachine(this, stateMachineId, {
351
+ definitionBody: aws_stepfunctions_1.DefinitionBody.fromChainable(workflowDefinition),
352
+ timeout: this.props.workflowTimeout || aws_cdk_lib_1.Duration.minutes(15),
353
+ role,
354
+ encryptionConfiguration: new aws_stepfunctions_1.CustomerManagedEncryptionConfiguration(this.encryptionKey),
355
+ });
356
+ this.handleWorkflowTrigger(stateMachine);
357
+ return stateMachine;
358
+ }
359
+ handleWorkflowTrigger(stateMachine) {
360
+ this.bucket.addEventNotification(aws_s3_1.EventType.OBJECT_CREATED, new aws_s3_notifications_1.SqsDestination(this.queue), {
361
+ prefix: DocumentProcessingPrefix.RAW,
362
+ });
363
+ this.createSQSConsumerLambda(stateMachine);
364
+ }
365
+ createSQSConsumerLambda(stateMachine) {
366
+ const { region, account } = utilities_1.LambdaIamUtils.getStackInfo(this);
367
+ // Create logs permissions and get unique function name
368
+ const logsPermissions = utilities_1.LambdaIamUtils.createLogsPermissions({
369
+ scope: this,
370
+ functionName: 'SQSConsumer',
371
+ region,
372
+ account,
373
+ });
374
+ // Create policy statements for SQS consumer Lambda
375
+ const policyStatements = [
376
+ ...logsPermissions.policyStatements,
377
+ new aws_iam_1.PolicyStatement({
378
+ effect: aws_iam_1.Effect.ALLOW,
379
+ actions: ['states:StartExecution'],
380
+ resources: [stateMachine.stateMachineArn],
381
+ }),
382
+ ];
383
+ if (this.props.network) {
384
+ policyStatements.push(utilities_1.LambdaIamUtils.generateLambdaVPCPermissions());
385
+ }
386
+ // Create IAM role for SQS consumer Lambda
387
+ const sqsConsumerRole = new aws_iam_1.Role(this, 'SQSConsumerRole', {
388
+ assumedBy: new aws_iam_1.ServicePrincipal('lambda.amazonaws.com'),
389
+ inlinePolicies: {
390
+ SQSConsumerExecutionPolicy: new aws_iam_1.PolicyDocument({
391
+ statements: policyStatements,
392
+ }),
393
+ },
394
+ });
395
+ this.encryptionKey.grantEncryptDecrypt(sqsConsumerRole);
396
+ // Create SQS consumer Lambda function
397
+ const sqsConsumerLambda = new aws_lambda_python_alpha_1.PythonFunction(this, 'SQSConsumer', {
398
+ functionName: logsPermissions.uniqueFunctionName,
399
+ runtime: framework_1.DefaultRuntimes.PYTHON,
400
+ role: sqsConsumerRole,
401
+ entry: path.join(__dirname, '/resources/default-sqs-consumer'),
402
+ environment: {
403
+ STATE_MACHINE_ARN: stateMachine.stateMachineArn,
404
+ ...powertools_config_1.PowertoolsConfig.generateDefaultLambdaConfig(this.props.enableObservability, this.metricNamespace, this.metricServiceName),
405
+ },
406
+ timeout: aws_cdk_lib_1.Duration.minutes(5),
407
+ description: 'Consumes SQS messages and triggers Step Functions executions for document processing',
408
+ environmentEncryption: this.encryptionKey,
409
+ vpc: this.props.network ? this.props.network.vpc : undefined,
410
+ vpcSubnets: this.props.network ? this.props.network.applicationSubnetSelection() : undefined,
411
+ });
412
+ // Add SQS event source to Lambda
413
+ sqsConsumerLambda.addEventSource(new aws_lambda_event_sources_1.SqsEventSource(this.queue, {
414
+ batchSize: 10,
415
+ maxBatchingWindow: aws_cdk_lib_1.Duration.seconds(5),
416
+ reportBatchItemFailures: true,
417
+ }));
418
+ return sqsConsumerLambda;
419
+ }
420
+ createStateMachineRole() {
421
+ return new aws_iam_1.Role(this, 'StateMachineRole', {
422
+ assumedBy: new aws_iam_1.ServicePrincipal('states.amazonaws.com'),
423
+ inlinePolicies: {
424
+ StateMachineExecutionPolicy: new aws_iam_1.PolicyDocument({
425
+ statements: [
426
+ new aws_iam_1.PolicyStatement({
427
+ effect: aws_iam_1.Effect.ALLOW,
428
+ actions: ['s3:GetObject', 's3:CopyObject', 's3:DeleteObject', 's3:PutObject'],
429
+ resources: [`${this.bucket.bucketArn}/*`],
430
+ }),
431
+ new aws_iam_1.PolicyStatement({
432
+ effect: aws_iam_1.Effect.ALLOW,
433
+ actions: ['dynamodb:PutItem', 'dynamodb:UpdateItem'],
434
+ resources: [this.documentProcessingTable.tableArn],
435
+ }),
436
+ ],
437
+ }),
438
+ },
439
+ });
440
+ }
441
+ createMoveToFailedChain() {
442
+ const failedChain = new aws_stepfunctions_tasks_1.CallAwsService(this, 'CopyToFailed', {
443
+ service: 's3',
444
+ action: 'copyObject',
445
+ parameters: {
446
+ Bucket: aws_stepfunctions_1.JsonPath.stringAt('$.bucket'),
447
+ CopySource: aws_stepfunctions_1.JsonPath.format('{}/{}', aws_stepfunctions_1.JsonPath.stringAt('$.bucket'), aws_stepfunctions_1.JsonPath.stringAt('$.key')),
448
+ Key: aws_stepfunctions_1.JsonPath.format('failed/{}', aws_stepfunctions_1.JsonPath.stringAt('$.filename')),
449
+ },
450
+ iamResources: [`${this.bucket.bucketArn}/*`],
451
+ resultPath: aws_stepfunctions_1.JsonPath.DISCARD,
452
+ }).next(new aws_stepfunctions_tasks_1.CallAwsService(this, 'DeleteFromRaw', {
453
+ service: 's3',
454
+ action: 'deleteObject',
455
+ parameters: {
456
+ Bucket: aws_stepfunctions_1.JsonPath.stringAt('$.bucket'),
457
+ Key: aws_stepfunctions_1.JsonPath.stringAt('$.key'),
458
+ },
459
+ iamResources: [`${this.bucket.bucketArn}/*`],
460
+ resultPath: aws_stepfunctions_1.JsonPath.DISCARD,
461
+ }));
462
+ if (this.props.eventbridgeBroker) {
463
+ failedChain.next(this.props.eventbridgeBroker.sendViaSfnChain('document-processing-failed', {
464
+ documentId: aws_stepfunctions_1.JsonPath.stringAt('$.documentId'),
465
+ bucket: aws_stepfunctions_1.JsonPath.stringAt('$.bucket'),
466
+ filename: aws_stepfunctions_1.JsonPath.stringAt('$.filename'),
467
+ }));
468
+ }
469
+ return failedChain;
470
+ }
471
+ createMoveToProcessedChain() {
472
+ const processedChain = new aws_stepfunctions_tasks_1.CallAwsService(this, 'CopyToProcessed', {
473
+ service: 's3',
474
+ action: 'copyObject',
475
+ parameters: {
476
+ Bucket: aws_stepfunctions_1.JsonPath.stringAt('$.bucket'),
477
+ CopySource: aws_stepfunctions_1.JsonPath.format('{}/{}', aws_stepfunctions_1.JsonPath.stringAt('$.bucket'), aws_stepfunctions_1.JsonPath.stringAt('$.key')),
478
+ Key: aws_stepfunctions_1.JsonPath.format('processed/{}', aws_stepfunctions_1.JsonPath.stringAt('$.filename')),
479
+ },
480
+ iamResources: [`${this.bucket.bucketArn}/*`],
481
+ resultPath: aws_stepfunctions_1.JsonPath.DISCARD,
482
+ }).next(new aws_stepfunctions_tasks_1.CallAwsService(this, 'DeleteFromRawSuccess', {
483
+ service: 's3',
484
+ action: 'deleteObject',
485
+ parameters: {
486
+ Bucket: aws_stepfunctions_1.JsonPath.stringAt('$.bucket'),
487
+ Key: aws_stepfunctions_1.JsonPath.stringAt('$.key'),
488
+ },
489
+ iamResources: [`${this.bucket.bucketArn}/*`],
490
+ resultPath: aws_stepfunctions_1.JsonPath.DISCARD,
491
+ }));
492
+ if (this.props.eventbridgeBroker) {
493
+ processedChain.next(this.props.eventbridgeBroker.sendViaSfnChain('document-processed-successful', {
494
+ documentId: aws_stepfunctions_1.JsonPath.stringAt('$.documentId'),
495
+ bucket: aws_stepfunctions_1.JsonPath.stringAt('$.bucket'),
496
+ filename: aws_stepfunctions_1.JsonPath.stringAt('$.filename'),
497
+ classification: aws_stepfunctions_1.JsonPath.stringAt('$.classificationResult.documentClassification'),
498
+ }));
499
+ }
500
+ return processedChain;
501
+ }
502
+ metrics() {
503
+ return [];
504
+ }
505
+ }
506
+ exports.BaseDocumentProcessing = BaseDocumentProcessing;
507
+ _a = JSII_RTTI_SYMBOL_1;
508
+ BaseDocumentProcessing[_a] = { fqn: "@cdklabs/cdk-appmod-catalog-blueprints.BaseDocumentProcessing", version: "1.0.0" };
509
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"base-document-processing.js","sourceRoot":"","sources":["../../use-cases/document-processing/base-document-processing.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.\n// SPDX-License-Identifier: Apache-2.0\n\nimport * as path from 'node:path';\nimport { PythonFunction } from '@aws-cdk/aws-lambda-python-alpha';\nimport { Duration, Names, PropertyInjectors, RemovalPolicy } from 'aws-cdk-lib';\nimport { IMetric } from 'aws-cdk-lib/aws-cloudwatch';\nimport { AttributeType, BillingMode, Table, TableEncryption } from 'aws-cdk-lib/aws-dynamodb';\nimport { InterfaceVpcEndpointAwsService } from 'aws-cdk-lib/aws-ec2';\nimport { Effect, PolicyDocument, PolicyStatement, Role, ServicePrincipal } from 'aws-cdk-lib/aws-iam';\nimport { IKey, Key } from 'aws-cdk-lib/aws-kms';\nimport { Function } from 'aws-cdk-lib/aws-lambda';\nimport { SqsEventSource } from 'aws-cdk-lib/aws-lambda-event-sources';\nimport { Bucket, BucketEncryption, EventType } from 'aws-cdk-lib/aws-s3';\nimport { SqsDestination } from 'aws-cdk-lib/aws-s3-notifications';\nimport { Queue, QueueEncryption } from 'aws-cdk-lib/aws-sqs';\nimport { CustomerManagedEncryptionConfiguration, DefinitionBody, JsonPath, StateMachine } from 'aws-cdk-lib/aws-stepfunctions';\nimport { BedrockInvokeModel, DynamoAttributeValue, DynamoPutItem, DynamoUpdateItem, LambdaInvoke, CallAwsService, StepFunctionsStartExecution } from 'aws-cdk-lib/aws-stepfunctions-tasks';\nimport { Construct } from 'constructs';\nimport { DefaultRuntimes, Network } from '../framework';\nimport { EventbridgeBroker } from '../framework/foundation/eventbridge-broker';\nimport { LambdaIamUtils, LogGroupDataProtectionProps } from '../utilities';\nimport { LambdaObservabilityPropertyInjector } from '../utilities/observability/lambda-observability-property-injector';\nimport { IObservable, ObservableProps } from '../utilities/observability/observable';\nimport { PowertoolsConfig } from '../utilities/observability/powertools-config';\nimport { StateMachineObservabilityPropertyInjector } from '../utilities/observability/state-machine-observability-property-injector';\n\n/**\n * Configuration properties for BaseDocumentProcessing construct.\n */\nexport interface BaseDocumentProcessingProps extends ObservableProps {\n  /**\n   * S3 bucket for document storage with organized prefixes (raw/, processed/, failed/).\n   * If not provided, a new bucket will be created with auto-delete enabled based on removalPolicy.\n   */\n  readonly bucket?: Bucket;\n  /**\n   * DynamoDB table for storing document processing metadata and workflow state.\n   * If not provided, a new table will be created with DocumentId as partition key.\n   */\n  readonly documentProcessingTable?: Table;\n  /**\n   * SQS queue visibility timeout for processing messages.\n   * Should be longer than expected processing time to prevent duplicate processing.\n   * @default Duration.seconds(300)\n   */\n  readonly queueVisibilityTimeout?: Duration;\n\n  /**\n   * The number of times a message can be unsuccessfully dequeued before being moved to the dead-letter queue.\n   *\n   * @default 5\n   */\n  readonly dlqMaxReceiveCount?: number;\n\n  /**\n   * Maximum execution time for the Step Functions workflow.\n   * @default Duration.minutes(30)\n   */\n  readonly workflowTimeout?: Duration;\n  /**\n   * Removal policy for created resources (bucket, table, queue).\n   * @default RemovalPolicy.DESTROY\n   */\n  readonly removalPolicy?: RemovalPolicy;\n  /**\n   * Optional EventBridge broker for publishing custom events during processing.\n   * If not provided, no custom events will be sent out.\n   */\n  readonly eventbridgeBroker?: EventbridgeBroker;\n\n  /**\n   * Enable logging and tracing for all supporting resource\n   * @default false\n   */\n  readonly enableObservability?: boolean;\n\n  /**\n   * Resources that can run inside a VPC will follow the provided network configuration\n   * @default resources will run outside of a VPC\n   */\n  readonly network?: Network;\n\n  /**\n   * KMS key to be used.\n   * @default A new key would be created\n   */\n  readonly encryptionKey?: Key;\n}\n\n/**\n * S3 prefix constants for organizing documents throughout the processing lifecycle.\n *\n * Documents flow through these prefixes based on processing outcomes:\n * - Upload → raw/ (triggers processing)\n * - Success → processed/ (workflow completed successfully)\n * - Failure → failed/ (workflow encountered errors)\n */\nexport enum DocumentProcessingPrefix {\n  /** Prefix for newly uploaded documents awaiting processing */\n  RAW = 'raw/',\n  /** Prefix for documents that failed processing */\n  FAILED = 'failed/',\n  /** Prefix for successfully processed documents */\n  PROCESSED = 'processed/',\n}\n\n/**\n * Union type for Step Functions tasks that can be used in document processing workflows.\n * Supports Bedrock model invocation, Lambda function invocation, and nested Step Functions execution.\n */\nexport type DocumentProcessingStepType = BedrockInvokeModel | LambdaInvoke | StepFunctionsStartExecution;\n\n/**\n * Abstract base class for serverless document processing workflows.\n *\n * Provides a complete document processing pipeline with:\n * - **S3 Storage**: Organized with prefixes (raw/, processed/, failed/) for document lifecycle management\n * - **SQS Queue**: Reliable message processing with configurable visibility timeout and dead letter queue\n * - **DynamoDB Table**: Workflow metadata tracking with DocumentId as partition key\n * - **Step Functions**: Orchestrated workflow with automatic file movement based on processing outcome\n * - **Auto-triggering**: S3 event notifications automatically start processing when files are uploaded to raw/ prefix\n * - **Error Handling**: Failed documents are moved to failed/ prefix with error details stored in DynamoDB\n * - **EventBridge Integration**: Optional custom event publishing for workflow state changes\n *\n * ## Architecture Flow\n * S3 Upload (raw/) → SQS → Lambda Consumer → Step Functions → Processing Steps → S3 (processed/failed/)\n *\n * ## Implementation Requirements\n * Subclasses must implement four abstract methods to define the processing workflow:\n * - `classificationStep()`: Document type classification\n * - `extractionStep()`: Data extraction from documents\n * - `enrichmentStep()`: Optional data enrichment (return undefined to skip)\n * - `postProcessingStep()`: Optional post-processing (return undefined to skip)\n */\nexport abstract class BaseDocumentProcessing extends Construct implements IObservable {\n  /** Business metric service name. This is part of the initial service dimension */\n  readonly metricServiceName: string;\n  /** Business metric namespace. */\n  readonly metricNamespace: string;\n  /** log group data protection configuration */\n  readonly logGroupDataProtection: LogGroupDataProtectionProps;\n  /** S3 bucket for document storage with organized prefixes (raw/, processed/, failed/) */\n  readonly bucket: Bucket;\n  /** SQS queue for reliable message processing with dead letter queue support */\n  readonly queue: Queue;\n  /** DynamoDB table for storing document processing metadata and workflow state */\n  readonly documentProcessingTable: Table;\n  /** Configuration properties for the document processing pipeline */\n  private readonly props: BaseDocumentProcessingProps;\n  /** Dead letter queue  */\n  readonly deadLetterQueue: Queue;\n  /** KMS key */\n  readonly encryptionKey: Key;\n  /** Encryption key used by the DocumentProcessingBucket */\n  readonly bucketEncryptionKey?: IKey;\n\n  /**\n   * Creates a new BaseDocumentProcessing construct.\n   *\n   * Initializes the complete document processing infrastructure including S3 bucket,\n   * SQS queue, DynamoDB table, and sets up S3 event notifications to trigger processing.\n   *\n   * @param scope - The scope in which to define this construct\n   * @param id - The scoped construct ID. Must be unique within the scope.\n   * @param props - Configuration properties for the document processing pipeline\n   */\n  constructor(scope: Construct, id: string, props: BaseDocumentProcessingProps) {\n    super(scope, id);\n    this.props = props;\n    if (props.network) {\n      props.network.createServiceEndpoint('vpce-sqs', InterfaceVpcEndpointAwsService.SQS);\n      props.network.createServiceEndpoint('vpce-s3', InterfaceVpcEndpointAwsService.S3);\n      props.network.createServiceEndpoint('vpce-sfn', InterfaceVpcEndpointAwsService.STEP_FUNCTIONS);\n      props.network.createServiceEndpoint('vpce-eb', InterfaceVpcEndpointAwsService.EVENTBRIDGE);\n      if (props.enableObservability) {\n        props.network.createServiceEndpoint('vpce-logs', InterfaceVpcEndpointAwsService.CLOUDWATCH_LOGS);\n        props.network.createServiceEndpoint('vpce-metrics', InterfaceVpcEndpointAwsService.CLOUDWATCH_MONITORING);\n      }\n    }\n\n    this.encryptionKey = props.encryptionKey || new Key(this, 'IDPEncryptionKey', {\n      enableKeyRotation: true,\n      removalPolicy: props.removalPolicy || RemovalPolicy.DESTROY,\n    });\n\n    const bucketName = `documentprocessingbucket-${Names.uniqueResourceName(this, {\n      maxLength: 60 - 'documentprocessingbucket-'.length,\n    })}`.toLowerCase();\n\n    const bucketArn = `arn:aws:s3:::${bucketName}`;\n\n    this.encryptionKey.grantEncryptDecrypt(new ServicePrincipal('s3.amazonaws.com', {\n      conditions: {\n        ArnEquals: {\n          'kms:EncryptionContext:aws:s3:arn': bucketArn,\n        },\n      },\n    }));\n\n    this.bucket = props.bucket || new Bucket(this, 'DocumentProcessingBucket', {\n      bucketName,\n      autoDeleteObjects: (props.removalPolicy && props.removalPolicy === RemovalPolicy.DESTROY) || !props.removalPolicy ? true : false,\n      removalPolicy: props.removalPolicy || RemovalPolicy.DESTROY,\n      encryption: BucketEncryption.KMS,\n      enforceSSL: true,\n      bucketKeyEnabled: true,\n    });\n\n\n    this.bucketEncryptionKey = this.bucket.encryptionKey;\n\n    const tempLogGroupDataProtection = props.logGroupDataProtection || {\n      logGroupEncryptionKey: new Key(this, 'LogGroupEncryptionKey', {\n        enableKeyRotation: true,\n        removalPolicy: props.removalPolicy || RemovalPolicy.DESTROY,\n      }),\n    };\n\n    if (!tempLogGroupDataProtection.logGroupEncryptionKey) {\n      this.logGroupDataProtection = {\n        dataProtectionIdentifiers: tempLogGroupDataProtection.dataProtectionIdentifiers,\n        logGroupEncryptionKey: new Key(this, 'LogGroupEncryptionKey', {\n          enableKeyRotation: true,\n          removalPolicy: props.removalPolicy || RemovalPolicy.DESTROY,\n        }),\n      };\n    } else {\n      this.logGroupDataProtection = tempLogGroupDataProtection;\n    }\n\n    this.deadLetterQueue = new Queue(this, 'DocumentProcessingDLQ', {\n      visibilityTimeout: props.queueVisibilityTimeout || Duration.seconds(300),\n      removalPolicy: props.removalPolicy || RemovalPolicy.DESTROY,\n      enforceSSL: true,\n      encryption: QueueEncryption.KMS,\n      encryptionMasterKey: this.encryptionKey,\n    });\n\n    this.queue = new Queue(this, 'DocumentProcessingQueue', {\n      visibilityTimeout: props.queueVisibilityTimeout || Duration.seconds(300),\n      removalPolicy: props.removalPolicy || RemovalPolicy.DESTROY,\n      enforceSSL: true,\n      deadLetterQueue: {\n        maxReceiveCount: props.dlqMaxReceiveCount || 5,\n        queue: this.deadLetterQueue,\n      },\n      encryption: QueueEncryption.KMS,\n      encryptionMasterKey: this.encryptionKey,\n    });\n\n    this.documentProcessingTable = props.documentProcessingTable || new Table(this, 'DocumentProcessingTable', {\n      partitionKey: {\n        name: 'DocumentId',\n        type: AttributeType.STRING,\n      },\n      billingMode: BillingMode.PAY_PER_REQUEST,\n      removalPolicy: props.removalPolicy || RemovalPolicy.DESTROY,\n      pointInTimeRecoverySpecification: {\n        pointInTimeRecoveryEnabled: true,\n      },\n      encryption: TableEncryption.CUSTOMER_MANAGED,\n      encryptionKey: this.encryptionKey,\n    });\n\n    if (props.enableObservability) {\n      PropertyInjectors.of(this).add(\n        new StateMachineObservabilityPropertyInjector(this.logGroupDataProtection),\n        new LambdaObservabilityPropertyInjector(this.logGroupDataProtection),\n      );\n    }\n\n    this.metricNamespace = props.metricNamespace || 'appmod-catalog';\n    this.metricServiceName = props.metricServiceName || 'document-processing';\n  }\n\n\n  protected handleStateMachineCreation(stateMachineId: string) {\n    const classificationStep = this.classificationStep();\n    const processingStep = this.processingStep();\n    const enrichmentStep = this.enrichmentStep();\n    const postProcessingStep = this.postProcessingStep();\n\n    const initMetadataEntry = new DynamoPutItem(this, 'InitMetadataEntry', {\n      table: this.documentProcessingTable,\n      item: {\n        DocumentId: DynamoAttributeValue.fromString(JsonPath.stringAt('$.documentId')),\n        Bucket: DynamoAttributeValue.fromString(JsonPath.stringAt('$.bucket')),\n        Key: DynamoAttributeValue.fromString(JsonPath.stringAt('$.key')),\n        WorkflowStatus: DynamoAttributeValue.fromString('pending'),\n        StateMachineExecId: DynamoAttributeValue.fromString(JsonPath.stringAt('$$.Execution.Id')),\n      },\n      resultPath: JsonPath.DISCARD,\n    });\n\n    // File movement operations\n    const moveToFailed = this.createMoveToFailedChain();\n    const moveToProcessed = this.createMoveToProcessedChain();\n\n    const processingChain = processingStep\n      .addCatch(new DynamoUpdateItem(this, 'ProcessingFailDDBUpdate', {\n        table: this.documentProcessingTable,\n        key: {\n          DocumentId: DynamoAttributeValue.fromString(JsonPath.stringAt('$.documentId')),\n        },\n        updateExpression: 'SET WorkflowStatus = :newStatus',\n        expressionAttributeValues: {\n          ':newStatus': DynamoAttributeValue.fromString('processing-failure'),\n        },\n        resultPath: JsonPath.DISCARD,\n      }).next(moveToFailed), {\n        resultPath: JsonPath.DISCARD,\n      })\n      .next(\n        new DynamoUpdateItem(this, 'ProcessingSuccessUpdate', {\n          table: this.documentProcessingTable,\n          key: {\n            DocumentId: DynamoAttributeValue.fromString(JsonPath.stringAt('$.documentId')),\n          },\n          updateExpression: 'SET WorkflowStatus = :newStatus, ProcessingResult = :processingResult',\n          expressionAttributeValues: {\n            ':newStatus': DynamoAttributeValue.fromString('processing-complete'),\n            ':processingResult': DynamoAttributeValue.fromString(JsonPath.jsonToString(JsonPath.objectAt('$.processingResult'))),\n          },\n          resultPath: JsonPath.DISCARD,\n        }),\n      );\n\n    // Build the complete chain including optional steps\n    if (enrichmentStep) {\n      const enrichmentChain = enrichmentStep\n        .addCatch(new DynamoUpdateItem(this, 'EnrichmentFailDDBUpdate', {\n          table: this.documentProcessingTable,\n          key: {\n            DocumentId: DynamoAttributeValue.fromString(JsonPath.stringAt('$.documentId')),\n          },\n          updateExpression: 'SET WorkflowStatus = :newStatus',\n          expressionAttributeValues: {\n            ':newStatus': DynamoAttributeValue.fromString('enrichment-failure'),\n          },\n          resultPath: JsonPath.DISCARD,\n        }).next(moveToFailed), {\n          resultPath: JsonPath.DISCARD,\n        })\n        .next(\n          new DynamoUpdateItem(this, 'EnrichmentSuccessUpdate', {\n            table: this.documentProcessingTable,\n            key: {\n              DocumentId: DynamoAttributeValue.fromString(JsonPath.stringAt('$.documentId')),\n            },\n            updateExpression: 'SET WorkflowStatus = :newStatus, EnrichmentResult = :enrichmentResult',\n            expressionAttributeValues: {\n              ':newStatus': postProcessingStep ? DynamoAttributeValue.fromString('enrichment-complete') : DynamoAttributeValue.fromString('complete'),\n              ':enrichmentResult': DynamoAttributeValue.fromString(JsonPath.jsonToString(JsonPath.objectAt('$.enrichedResult'))),\n            },\n            resultPath: JsonPath.DISCARD,\n          }),\n        );\n\n      processingChain.next(enrichmentChain);\n\n      if (postProcessingStep) {\n        const postProcessingChain = postProcessingStep\n          .addCatch(new DynamoUpdateItem(this, 'PostProcessingFailDDBUpdate', {\n            table: this.documentProcessingTable,\n            key: {\n              DocumentId: DynamoAttributeValue.fromString(JsonPath.stringAt('$.documentId')),\n            },\n            updateExpression: 'SET WorkflowStatus = :newStatus',\n            expressionAttributeValues: {\n              ':newStatus': DynamoAttributeValue.fromString('post-processing-failure'),\n            },\n            resultPath: JsonPath.DISCARD,\n          }).next(moveToFailed), {\n            resultPath: JsonPath.DISCARD,\n          })\n          .next(\n            new DynamoUpdateItem(this, 'PostProcessingSuccessUpdate', {\n              table: this.documentProcessingTable,\n              key: {\n                DocumentId: DynamoAttributeValue.fromString(JsonPath.stringAt('$.documentId')),\n              },\n              updateExpression: 'SET WorkflowStatus = :newStatus, PostProcessingResult = :postProcessingResult',\n              expressionAttributeValues: {\n                ':newStatus': DynamoAttributeValue.fromString('complete'),\n                ':postProcessingResult': DynamoAttributeValue.fromString(JsonPath.jsonToString(JsonPath.objectAt('$.postProcessedResult'))),\n              },\n              resultPath: JsonPath.DISCARD,\n            }).next(moveToProcessed),\n          );\n        enrichmentChain.next(postProcessingChain);\n      } else {\n        enrichmentChain.next(moveToProcessed);\n      }\n    } else if (postProcessingStep) {\n      const postProcessingChain = postProcessingStep\n        .addCatch(new DynamoUpdateItem(this, 'PostProcessingFailDDBUpdate', {\n          table: this.documentProcessingTable,\n          key: {\n            DocumentId: DynamoAttributeValue.fromString(JsonPath.stringAt('$.documentId')),\n          },\n          updateExpression: 'SET WorkflowStatus = :newStatus',\n          expressionAttributeValues: {\n            ':newStatus': DynamoAttributeValue.fromString('post-processing-failure'),\n          },\n          resultPath: JsonPath.DISCARD,\n        }).next(moveToFailed), {\n          resultPath: JsonPath.DISCARD,\n        })\n        .next(\n          new DynamoUpdateItem(this, 'PostProcessingSuccessUpdate', {\n            table: this.documentProcessingTable,\n            key: {\n              DocumentId: DynamoAttributeValue.fromString(JsonPath.stringAt('$.documentId')),\n            },\n            updateExpression: 'SET WorkflowStatus = :newStatus, PostProcessingResult = :postProcessingResult',\n            expressionAttributeValues: {\n              ':newStatus': DynamoAttributeValue.fromString('complete'),\n              ':postProcessingResult': DynamoAttributeValue.fromString(JsonPath.jsonToString(JsonPath.objectAt('$.postProcessedResult'))),\n            },\n            resultPath: JsonPath.DISCARD,\n          }).next(moveToProcessed),\n        );\n      processingChain.next(postProcessingChain);\n    } else {\n      // No optional steps - mark as complete after extraction\n      processingChain.next(\n        new DynamoUpdateItem(this, 'WorkflowCompleteUpdate', {\n          table: this.documentProcessingTable,\n          key: {\n            DocumentId: DynamoAttributeValue.fromString(JsonPath.stringAt('$.documentId')),\n          },\n          updateExpression: 'SET WorkflowStatus = :newStatus',\n          expressionAttributeValues: {\n            ':newStatus': DynamoAttributeValue.fromString('complete'),\n          },\n          resultPath: JsonPath.DISCARD,\n        }).next(moveToProcessed),\n      );\n    }\n\n    const workflowDefinition = initMetadataEntry.next(\n      classificationStep\n        .addCatch(new DynamoUpdateItem(this, 'ClassificationFailDDBUpdate', {\n          table: this.documentProcessingTable,\n          key: {\n            DocumentId: DynamoAttributeValue.fromString(JsonPath.stringAt('$.documentId')),\n          },\n          updateExpression: 'SET WorkflowStatus = :newStatus',\n          expressionAttributeValues: {\n            ':newStatus': DynamoAttributeValue.fromString('classification-failure'),\n          },\n          resultPath: JsonPath.DISCARD,\n        }).next(moveToFailed), {\n          resultPath: JsonPath.DISCARD,\n        })\n        .next(\n          new DynamoUpdateItem(this, 'ClassificationSuccessUpdate', {\n            table: this.documentProcessingTable,\n            key: {\n              DocumentId: DynamoAttributeValue.fromString(JsonPath.stringAt('$.documentId')),\n            },\n            updateExpression: 'SET WorkflowStatus = :newStatus, ClassificationResult = :classificationResult',\n            expressionAttributeValues: {\n              ':newStatus': DynamoAttributeValue.fromString('classification-complete'),\n              ':classificationResult': DynamoAttributeValue.fromString(JsonPath.jsonToString(JsonPath.objectAt('$.classificationResult'))),\n            },\n            resultPath: JsonPath.DISCARD,\n          }),\n        ),\n    ).next(processingChain);\n\n    const role = this.createStateMachineRole();\n    this.encryptionKey.grantEncryptDecrypt(role);\n    if (this.bucketEncryptionKey) {\n      this.bucketEncryptionKey.grantEncryptDecrypt(role);\n    }\n    const stateMachine = new StateMachine(this, stateMachineId, {\n      definitionBody: DefinitionBody.fromChainable(workflowDefinition),\n      timeout: this.props.workflowTimeout || Duration.minutes(15),\n      role,\n      encryptionConfiguration: new CustomerManagedEncryptionConfiguration(this.encryptionKey),\n    });\n\n    this.handleWorkflowTrigger(stateMachine);\n\n    return stateMachine;\n  }\n\n  protected handleWorkflowTrigger(stateMachine: StateMachine) {\n    this.bucket.addEventNotification(EventType.OBJECT_CREATED, new SqsDestination(this.queue), {\n      prefix: DocumentProcessingPrefix.RAW,\n    });\n    this.createSQSConsumerLambda(stateMachine);\n  }\n\n  private createSQSConsumerLambda(stateMachine: StateMachine): Function {\n    const { region, account } = LambdaIamUtils.getStackInfo(this);\n    // Create logs permissions and get unique function name\n    const logsPermissions = LambdaIamUtils.createLogsPermissions({\n      scope: this,\n      functionName: 'SQSConsumer',\n      region,\n      account,\n    });\n\n    // Create policy statements for SQS consumer Lambda\n    const policyStatements = [\n      ...logsPermissions.policyStatements,\n      new PolicyStatement({\n        effect: Effect.ALLOW,\n        actions: ['states:StartExecution'],\n        resources: [stateMachine.stateMachineArn],\n      }),\n    ];\n\n    if (this.props.network) {\n      policyStatements.push(LambdaIamUtils.generateLambdaVPCPermissions());\n    }\n\n    // Create IAM role for SQS consumer Lambda\n    const sqsConsumerRole = new Role(this, 'SQSConsumerRole', {\n      assumedBy: new ServicePrincipal('lambda.amazonaws.com'),\n      inlinePolicies: {\n        SQSConsumerExecutionPolicy: new PolicyDocument({\n          statements: policyStatements,\n        }),\n      },\n    });\n\n    this.encryptionKey.grantEncryptDecrypt(sqsConsumerRole);\n\n    // Create SQS consumer Lambda function\n    const sqsConsumerLambda = new PythonFunction(this, 'SQSConsumer', {\n      functionName: logsPermissions.uniqueFunctionName,\n      runtime: DefaultRuntimes.PYTHON,\n      role: sqsConsumerRole,\n      entry: path.join(__dirname, '/resources/default-sqs-consumer'),\n      environment: {\n        STATE_MACHINE_ARN: stateMachine.stateMachineArn,\n        ...PowertoolsConfig.generateDefaultLambdaConfig(this.props.enableObservability, this.metricNamespace, this.metricServiceName),\n      },\n      timeout: Duration.minutes(5),\n      description: 'Consumes SQS messages and triggers Step Functions executions for document processing',\n      environmentEncryption: this.encryptionKey,\n      vpc: this.props.network ? this.props.network.vpc : undefined,\n      vpcSubnets: this.props.network ? this.props.network.applicationSubnetSelection() : undefined,\n    });\n\n    // Add SQS event source to Lambda\n    sqsConsumerLambda.addEventSource(\n      new SqsEventSource(this.queue, {\n        batchSize: 10,\n        maxBatchingWindow: Duration.seconds(5),\n        reportBatchItemFailures: true,\n      }),\n    );\n\n    return sqsConsumerLambda;\n  }\n\n  private createStateMachineRole(): Role {\n    return new Role(this, 'StateMachineRole', {\n      assumedBy: new ServicePrincipal('states.amazonaws.com'),\n      inlinePolicies: {\n        StateMachineExecutionPolicy: new PolicyDocument({\n          statements: [\n            new PolicyStatement({\n              effect: Effect.ALLOW,\n              actions: ['s3:GetObject', 's3:CopyObject', 's3:DeleteObject', 's3:PutObject'],\n              resources: [`${this.bucket.bucketArn}/*`],\n            }),\n            new PolicyStatement({\n              effect: Effect.ALLOW,\n              actions: ['dynamodb:PutItem', 'dynamodb:UpdateItem'],\n              resources: [this.documentProcessingTable.tableArn],\n            }),\n          ],\n        }),\n      },\n    });\n  }\n\n  private createMoveToFailedChain() {\n    const failedChain = new CallAwsService(this, 'CopyToFailed', {\n      service: 's3',\n      action: 'copyObject',\n      parameters: {\n        Bucket: JsonPath.stringAt('$.bucket'),\n        CopySource: JsonPath.format('{}/{}', JsonPath.stringAt('$.bucket'), JsonPath.stringAt('$.key')),\n        Key: JsonPath.format('failed/{}', JsonPath.stringAt('$.filename')),\n      },\n      iamResources: [`${this.bucket.bucketArn}/*`],\n      resultPath: JsonPath.DISCARD,\n    }).next(\n      new CallAwsService(this, 'DeleteFromRaw', {\n        service: 's3',\n        action: 'deleteObject',\n        parameters: {\n          Bucket: JsonPath.stringAt('$.bucket'),\n          Key: JsonPath.stringAt('$.key'),\n        },\n        iamResources: [`${this.bucket.bucketArn}/*`],\n        resultPath: JsonPath.DISCARD,\n      }),\n    );\n\n    if (this.props.eventbridgeBroker) {\n      failedChain.next(\n        this.props.eventbridgeBroker.sendViaSfnChain(\n          'document-processing-failed',\n          {\n            documentId: JsonPath.stringAt('$.documentId'),\n            bucket: JsonPath.stringAt('$.bucket'),\n            filename: JsonPath.stringAt('$.filename'),\n          },\n        ),\n      );\n    }\n\n    return failedChain;\n  }\n\n  private createMoveToProcessedChain() {\n    const processedChain = new CallAwsService(this, 'CopyToProcessed', {\n      service: 's3',\n      action: 'copyObject',\n      parameters: {\n        Bucket: JsonPath.stringAt('$.bucket'),\n        CopySource: JsonPath.format('{}/{}', JsonPath.stringAt('$.bucket'), JsonPath.stringAt('$.key')),\n        Key: JsonPath.format('processed/{}', JsonPath.stringAt('$.filename')),\n      },\n      iamResources: [`${this.bucket.bucketArn}/*`],\n      resultPath: JsonPath.DISCARD,\n    }).next(\n      new CallAwsService(this, 'DeleteFromRawSuccess', {\n        service: 's3',\n        action: 'deleteObject',\n        parameters: {\n          Bucket: JsonPath.stringAt('$.bucket'),\n          Key: JsonPath.stringAt('$.key'),\n        },\n        iamResources: [`${this.bucket.bucketArn}/*`],\n        resultPath: JsonPath.DISCARD,\n      }),\n    );\n\n    if (this.props.eventbridgeBroker) {\n      processedChain.next(\n        this.props.eventbridgeBroker.sendViaSfnChain(\n          'document-processed-successful',\n          {\n            documentId: JsonPath.stringAt('$.documentId'),\n            bucket: JsonPath.stringAt('$.bucket'),\n            filename: JsonPath.stringAt('$.filename'),\n            classification: JsonPath.stringAt('$.classificationResult.documentClassification'),\n          },\n        ),\n      );\n    }\n\n    return processedChain;\n  }\n\n  public metrics(): IMetric[] {\n    return [];\n  }\n\n  /**\n   * Defines the document classification step of the workflow.\n   *\n   * **CRITICAL**: Must set `outputPath` to preserve workflow state for subsequent steps.\n   * The classification result should be available at `$.classificationResult` for DynamoDB storage.\n   *\n   * @returns Step Functions task for document classification\n   */\n  protected abstract classificationStep(): DocumentProcessingStepType;\n\n  /**\n   * Defines the document processing step of the workflow.\n   *\n   * **CRITICAL**: Must set `outputPath` to preserve workflow state for subsequent steps.\n   * The extraction result should be available at `$.processingResult` for DynamoDB storage.\n   *\n   * @returns Step Functions task for document extraction\n   */\n  protected abstract processingStep(): DocumentProcessingStepType;\n\n  /**\n   * Defines the optional document enrichment step of the workflow.\n   *\n   * **CRITICAL**: If implemented, must set `outputPath` to preserve workflow state.\n   * The enrichment result should be available at `$.enrichedResult` for DynamoDB storage.\n   *\n   * @returns Step Functions task for document enrichment, or undefined to skip this step\n   */\n  protected abstract enrichmentStep(): DocumentProcessingStepType | undefined;\n\n  /**\n   * Defines the optional post-processing step of the workflow.\n   *\n   * **CRITICAL**: If implemented, must set `outputPath` to preserve workflow state.\n   * The post-processing result should be available at `$.postProcessedResult` for DynamoDB storage.\n   *\n   * @returns Step Functions task for post-processing, or undefined to skip this step\n   */\n  protected abstract postProcessingStep(): DocumentProcessingStepType | undefined;\n}\n"]}