@cdklabs/cdk-appmod-catalog-blueprints 1.0.1 → 1.2.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.jsii +701 -204
- package/README.md +100 -160
- package/lib/document-processing/adapter/adapter.d.ts +47 -0
- package/lib/document-processing/adapter/adapter.js +5 -0
- package/lib/document-processing/adapter/index.d.ts +2 -0
- package/lib/document-processing/adapter/index.js +19 -0
- package/lib/document-processing/adapter/queued-s3-adapter.d.ts +66 -0
- package/lib/document-processing/adapter/queued-s3-adapter.js +230 -0
- package/lib/document-processing/agentic-document-processing.d.ts +22 -0
- package/lib/document-processing/agentic-document-processing.js +11 -14
- package/lib/document-processing/base-document-processing.d.ts +8 -44
- package/lib/document-processing/base-document-processing.js +23 -190
- package/lib/document-processing/bedrock-document-processing.js +3 -13
- package/lib/document-processing/default-document-processing-config.d.ts +3 -0
- package/lib/document-processing/default-document-processing-config.js +14 -0
- package/lib/document-processing/index.d.ts +2 -0
- package/lib/document-processing/index.js +3 -1
- package/lib/document-processing/resources/default-bedrock-invoke/index.py +36 -24
- package/lib/document-processing/resources/default-sqs-consumer/index.py +10 -5
- package/lib/document-processing/resources/default-strands-agent/index.py +8 -5
- package/lib/document-processing/tests/agentic-document-processing-nag.test.js +6 -2
- package/lib/document-processing/tests/agentic-document-processing.test.js +5 -19
- package/lib/document-processing/tests/bedrock-document-processing-nag.test.js +6 -2
- 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/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.d.ts +9 -0
- package/lib/utilities/observability/default-observability-config.js +20 -0
- package/lib/utilities/observability/index.d.ts +1 -0
- package/lib/utilities/observability/index.js +2 -1
- package/lib/utilities/observability/lambda-observability-property-injector.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 +9 -9
|
@@ -0,0 +1,230 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var _a;
|
|
3
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
4
|
+
exports.QueuedS3Adapter = 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_ec2_1 = require("aws-cdk-lib/aws-ec2");
|
|
12
|
+
const aws_iam_1 = require("aws-cdk-lib/aws-iam");
|
|
13
|
+
const aws_kms_1 = require("aws-cdk-lib/aws-kms");
|
|
14
|
+
const aws_lambda_event_sources_1 = require("aws-cdk-lib/aws-lambda-event-sources");
|
|
15
|
+
const aws_s3_1 = require("aws-cdk-lib/aws-s3");
|
|
16
|
+
const aws_s3_notifications_1 = require("aws-cdk-lib/aws-s3-notifications");
|
|
17
|
+
const aws_sqs_1 = require("aws-cdk-lib/aws-sqs");
|
|
18
|
+
const aws_stepfunctions_1 = require("aws-cdk-lib/aws-stepfunctions");
|
|
19
|
+
const aws_stepfunctions_tasks_1 = require("aws-cdk-lib/aws-stepfunctions-tasks");
|
|
20
|
+
const framework_1 = require("../../framework");
|
|
21
|
+
const utilities_1 = require("../../utilities");
|
|
22
|
+
const default_document_processing_config_1 = require("../default-document-processing-config");
|
|
23
|
+
/**
|
|
24
|
+
* This adapter allows the intelligent document processing workflow
|
|
25
|
+
* to be triggered by files that are uploaded into a S3 Bucket.
|
|
26
|
+
*/
|
|
27
|
+
class QueuedS3Adapter {
|
|
28
|
+
constructor(adapterProps = {}) {
|
|
29
|
+
this.adapterProps = adapterProps;
|
|
30
|
+
this.resources = {};
|
|
31
|
+
this.prefixes = {
|
|
32
|
+
raw: this.adapterProps.rawPrefix || 'raw/',
|
|
33
|
+
processed: this.adapterProps.processedPrefix || 'processed/',
|
|
34
|
+
failed: this.adapterProps.failedPrefix || 'failed/',
|
|
35
|
+
};
|
|
36
|
+
}
|
|
37
|
+
init(scope, props) {
|
|
38
|
+
if (props.network) {
|
|
39
|
+
props.network.createServiceEndpoint('vpce-sqs', aws_ec2_1.InterfaceVpcEndpointAwsService.SQS);
|
|
40
|
+
props.network.createServiceEndpoint('vpce-s3', aws_ec2_1.InterfaceVpcEndpointAwsService.S3);
|
|
41
|
+
}
|
|
42
|
+
const encryptionKey = props.encryptionKey || new aws_kms_1.Key(scope, 'QueuedS3AdapterEncryptionKey', {
|
|
43
|
+
enableKeyRotation: true,
|
|
44
|
+
removalPolicy: props.removalPolicy || aws_cdk_lib_1.RemovalPolicy.DESTROY,
|
|
45
|
+
});
|
|
46
|
+
this.resources.encryptionKey = encryptionKey;
|
|
47
|
+
const bucket = this.adapterProps.bucket || new aws_s3_1.Bucket(scope, 'DocumentProcessingBucket', {
|
|
48
|
+
autoDeleteObjects: (props.removalPolicy && props.removalPolicy === aws_cdk_lib_1.RemovalPolicy.DESTROY) || !props.removalPolicy ? true : false,
|
|
49
|
+
removalPolicy: props.removalPolicy || aws_cdk_lib_1.RemovalPolicy.DESTROY,
|
|
50
|
+
encryption: aws_s3_1.BucketEncryption.KMS,
|
|
51
|
+
enforceSSL: true,
|
|
52
|
+
bucketKeyEnabled: true,
|
|
53
|
+
});
|
|
54
|
+
this.resources.bucket = bucket;
|
|
55
|
+
const deadLetterQueue = new aws_sqs_1.Queue(scope, 'DocumentProcessingDLQ', {
|
|
56
|
+
visibilityTimeout: this.adapterProps.queueVisibilityTimeout || aws_cdk_lib_1.Duration.seconds(300),
|
|
57
|
+
removalPolicy: props.removalPolicy || aws_cdk_lib_1.RemovalPolicy.DESTROY,
|
|
58
|
+
enforceSSL: true,
|
|
59
|
+
encryption: aws_sqs_1.QueueEncryption.KMS,
|
|
60
|
+
encryptionMasterKey: encryptionKey,
|
|
61
|
+
});
|
|
62
|
+
const queue = new aws_sqs_1.Queue(scope, 'DocumentProcessingQueue', {
|
|
63
|
+
visibilityTimeout: this.adapterProps.queueVisibilityTimeout || aws_cdk_lib_1.Duration.seconds(300),
|
|
64
|
+
removalPolicy: props.removalPolicy || aws_cdk_lib_1.RemovalPolicy.DESTROY,
|
|
65
|
+
enforceSSL: true,
|
|
66
|
+
deadLetterQueue: {
|
|
67
|
+
maxReceiveCount: this.adapterProps.dlqMaxReceiveCount || 5,
|
|
68
|
+
queue: deadLetterQueue,
|
|
69
|
+
},
|
|
70
|
+
encryption: aws_sqs_1.QueueEncryption.KMS,
|
|
71
|
+
encryptionMasterKey: encryptionKey,
|
|
72
|
+
});
|
|
73
|
+
this.resources.deadLetterQueue = deadLetterQueue;
|
|
74
|
+
this.resources.queue = queue;
|
|
75
|
+
}
|
|
76
|
+
createIngressTrigger(scope, stateMachine, props) {
|
|
77
|
+
const bucket = this.resources.bucket;
|
|
78
|
+
const queue = this.resources.queue;
|
|
79
|
+
const encryptionKey = this.resources.encryptionKey;
|
|
80
|
+
bucket.addEventNotification(aws_s3_1.EventType.OBJECT_CREATED, new aws_s3_notifications_1.SqsDestination(queue), {
|
|
81
|
+
prefix: this.prefixes.raw,
|
|
82
|
+
});
|
|
83
|
+
const sqsConsumerLambdaFn = this.createSQSConsumerLambda(scope, stateMachine, props, encryptionKey, queue);
|
|
84
|
+
this.resources.sqsConsumerLambdaFunction = sqsConsumerLambdaFn;
|
|
85
|
+
return this.resources;
|
|
86
|
+
}
|
|
87
|
+
createSQSConsumerLambda(scope, stateMachine, props, encryptionKey, queue) {
|
|
88
|
+
const metricNamespace = props.metricNamespace || utilities_1.DefaultObservabilityConfig.DEFAULT_METRIC_NAMESPACE;
|
|
89
|
+
const metricServiceName = props.metricServiceName || default_document_processing_config_1.DefaultDocumentProcessingConfig.DEFAULT_OBSERVABILITY_METRIC_SVC_NAME;
|
|
90
|
+
const { region, account } = utilities_1.LambdaIamUtils.getStackInfo(scope);
|
|
91
|
+
// Create logs permissions and get unique function name
|
|
92
|
+
const logsPermissions = utilities_1.LambdaIamUtils.createLogsPermissions({
|
|
93
|
+
scope,
|
|
94
|
+
functionName: 'SQSConsumer',
|
|
95
|
+
region,
|
|
96
|
+
account,
|
|
97
|
+
enableObservability: props.enableObservability,
|
|
98
|
+
});
|
|
99
|
+
// Create policy statements for SQS consumer Lambda
|
|
100
|
+
const policyStatements = [
|
|
101
|
+
...logsPermissions.policyStatements,
|
|
102
|
+
new aws_iam_1.PolicyStatement({
|
|
103
|
+
effect: aws_iam_1.Effect.ALLOW,
|
|
104
|
+
actions: ['states:StartExecution'],
|
|
105
|
+
resources: [stateMachine.stateMachineArn],
|
|
106
|
+
}),
|
|
107
|
+
];
|
|
108
|
+
if (props.network) {
|
|
109
|
+
policyStatements.push(utilities_1.LambdaIamUtils.generateLambdaVPCPermissions());
|
|
110
|
+
}
|
|
111
|
+
// Create IAM role for SQS consumer Lambda
|
|
112
|
+
const sqsConsumerRole = new aws_iam_1.Role(scope, 'SQSConsumerRole', {
|
|
113
|
+
assumedBy: new aws_iam_1.ServicePrincipal('lambda.amazonaws.com'),
|
|
114
|
+
inlinePolicies: {
|
|
115
|
+
SQSConsumerExecutionPolicy: new aws_iam_1.PolicyDocument({
|
|
116
|
+
statements: policyStatements,
|
|
117
|
+
}),
|
|
118
|
+
},
|
|
119
|
+
});
|
|
120
|
+
encryptionKey.grantEncryptDecrypt(sqsConsumerRole);
|
|
121
|
+
// Create SQS consumer Lambda function
|
|
122
|
+
const sqsConsumerLambda = new aws_lambda_python_alpha_1.PythonFunction(scope, 'SQSConsumer', {
|
|
123
|
+
functionName: logsPermissions.uniqueFunctionName,
|
|
124
|
+
runtime: framework_1.DefaultRuntimes.PYTHON,
|
|
125
|
+
role: sqsConsumerRole,
|
|
126
|
+
entry: path.join(__dirname, '/../resources/default-sqs-consumer'),
|
|
127
|
+
environment: {
|
|
128
|
+
STATE_MACHINE_ARN: stateMachine.stateMachineArn,
|
|
129
|
+
RAW_PREFIX: this.prefixes.raw,
|
|
130
|
+
...utilities_1.PowertoolsConfig.generateDefaultLambdaConfig(props.enableObservability, metricNamespace, metricServiceName),
|
|
131
|
+
},
|
|
132
|
+
timeout: aws_cdk_lib_1.Duration.minutes(5),
|
|
133
|
+
description: 'Consumes SQS messages and triggers Step Functions executions for document processing',
|
|
134
|
+
environmentEncryption: encryptionKey,
|
|
135
|
+
vpc: props.network ? props.network.vpc : undefined,
|
|
136
|
+
vpcSubnets: props.network ? props.network.applicationSubnetSelection() : undefined,
|
|
137
|
+
});
|
|
138
|
+
// Add SQS event source to Lambda
|
|
139
|
+
sqsConsumerLambda.addEventSource(new aws_lambda_event_sources_1.SqsEventSource(queue, {
|
|
140
|
+
batchSize: 10,
|
|
141
|
+
maxBatchingWindow: aws_cdk_lib_1.Duration.seconds(5),
|
|
142
|
+
reportBatchItemFailures: true,
|
|
143
|
+
}));
|
|
144
|
+
return sqsConsumerLambda;
|
|
145
|
+
}
|
|
146
|
+
generateAdapterIAMPolicies(additionalIAMActions, narrowActions) {
|
|
147
|
+
const bucket = this.resources.bucket;
|
|
148
|
+
const normalizedIAMActions = additionalIAMActions || [];
|
|
149
|
+
const statements = [];
|
|
150
|
+
if (!narrowActions) {
|
|
151
|
+
statements.push(new aws_iam_1.PolicyStatement({
|
|
152
|
+
effect: aws_iam_1.Effect.ALLOW,
|
|
153
|
+
actions: ['s3:GetObject', 's3:CopyObject', 's3:DeleteObject', 's3:PutObject', ...normalizedIAMActions],
|
|
154
|
+
resources: [`${bucket.bucketArn}/*`],
|
|
155
|
+
}));
|
|
156
|
+
if (bucket.encryptionKey) {
|
|
157
|
+
statements.push(new aws_iam_1.PolicyStatement({
|
|
158
|
+
effect: aws_iam_1.Effect.ALLOW,
|
|
159
|
+
actions: [
|
|
160
|
+
'kms:Encrypt',
|
|
161
|
+
'kms:ReEncrypt*',
|
|
162
|
+
'kms:GenerateDataKey*',
|
|
163
|
+
'kms:Decrypt',
|
|
164
|
+
],
|
|
165
|
+
resources: [bucket.encryptionKey.keyArn],
|
|
166
|
+
}));
|
|
167
|
+
}
|
|
168
|
+
}
|
|
169
|
+
else {
|
|
170
|
+
statements.push(new aws_iam_1.PolicyStatement({
|
|
171
|
+
effect: aws_iam_1.Effect.ALLOW,
|
|
172
|
+
actions: normalizedIAMActions,
|
|
173
|
+
resources: [`${bucket.bucketArn}/*`],
|
|
174
|
+
}));
|
|
175
|
+
}
|
|
176
|
+
return statements;
|
|
177
|
+
}
|
|
178
|
+
createFailedChain(scope) {
|
|
179
|
+
const bucket = this.resources.bucket;
|
|
180
|
+
const failedChain = new aws_stepfunctions_tasks_1.CallAwsService(scope, 'CopyToFailed', {
|
|
181
|
+
service: 's3',
|
|
182
|
+
action: 'copyObject',
|
|
183
|
+
parameters: {
|
|
184
|
+
Bucket: aws_stepfunctions_1.JsonPath.stringAt('$.content.bucket'),
|
|
185
|
+
CopySource: aws_stepfunctions_1.JsonPath.format('{}/{}', aws_stepfunctions_1.JsonPath.stringAt('$.content.bucket'), aws_stepfunctions_1.JsonPath.stringAt('$.content.key')),
|
|
186
|
+
Key: aws_stepfunctions_1.JsonPath.format(`${this.prefixes.failed}/{}`, aws_stepfunctions_1.JsonPath.stringAt('$.content.filename')),
|
|
187
|
+
},
|
|
188
|
+
iamResources: [`${bucket.bucketArn}/*`],
|
|
189
|
+
resultPath: aws_stepfunctions_1.JsonPath.DISCARD,
|
|
190
|
+
}).next(new aws_stepfunctions_tasks_1.CallAwsService(scope, 'DeleteFromRaw', {
|
|
191
|
+
service: 's3',
|
|
192
|
+
action: 'deleteObject',
|
|
193
|
+
parameters: {
|
|
194
|
+
Bucket: aws_stepfunctions_1.JsonPath.stringAt('$.content.bucket'),
|
|
195
|
+
Key: aws_stepfunctions_1.JsonPath.stringAt('$.content.key'),
|
|
196
|
+
},
|
|
197
|
+
iamResources: [`${bucket.bucketArn}/*`],
|
|
198
|
+
resultPath: aws_stepfunctions_1.JsonPath.DISCARD,
|
|
199
|
+
}));
|
|
200
|
+
return failedChain;
|
|
201
|
+
}
|
|
202
|
+
createSuccessChain(scope) {
|
|
203
|
+
const bucket = this.resources.bucket;
|
|
204
|
+
const chain = new aws_stepfunctions_tasks_1.CallAwsService(scope, 'CopyToProcessed', {
|
|
205
|
+
service: 's3',
|
|
206
|
+
action: 'copyObject',
|
|
207
|
+
parameters: {
|
|
208
|
+
Bucket: aws_stepfunctions_1.JsonPath.stringAt('$.content.bucket'),
|
|
209
|
+
CopySource: aws_stepfunctions_1.JsonPath.format('{}/{}', aws_stepfunctions_1.JsonPath.stringAt('$.content.bucket'), aws_stepfunctions_1.JsonPath.stringAt('$.content.key')),
|
|
210
|
+
Key: aws_stepfunctions_1.JsonPath.format(`${this.prefixes.processed}/{}`, aws_stepfunctions_1.JsonPath.stringAt('$.content.filename')),
|
|
211
|
+
},
|
|
212
|
+
iamResources: [`${bucket.bucketArn}/*`],
|
|
213
|
+
resultPath: aws_stepfunctions_1.JsonPath.DISCARD,
|
|
214
|
+
}).next(new aws_stepfunctions_tasks_1.CallAwsService(scope, 'DeleteFromRawSuccess', {
|
|
215
|
+
service: 's3',
|
|
216
|
+
action: 'deleteObject',
|
|
217
|
+
parameters: {
|
|
218
|
+
Bucket: aws_stepfunctions_1.JsonPath.stringAt('$.content.bucket'),
|
|
219
|
+
Key: aws_stepfunctions_1.JsonPath.stringAt('$.content.key'),
|
|
220
|
+
},
|
|
221
|
+
iamResources: [`${bucket.bucketArn}/*`],
|
|
222
|
+
resultPath: aws_stepfunctions_1.JsonPath.DISCARD,
|
|
223
|
+
}));
|
|
224
|
+
return chain;
|
|
225
|
+
}
|
|
226
|
+
}
|
|
227
|
+
exports.QueuedS3Adapter = QueuedS3Adapter;
|
|
228
|
+
_a = JSII_RTTI_SYMBOL_1;
|
|
229
|
+
QueuedS3Adapter[_a] = { fqn: "@cdklabs/cdk-appmod-catalog-blueprints.QueuedS3Adapter", version: "1.2.0" };
|
|
230
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"queued-s3-adapter.js","sourceRoot":"","sources":["../../../use-cases/document-processing/adapter/queued-s3-adapter.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, RemovalPolicy } from 'aws-cdk-lib';\nimport { InterfaceVpcEndpointAwsService } from 'aws-cdk-lib/aws-ec2';\nimport { Effect, PolicyDocument, PolicyStatement, Role, ServicePrincipal } from 'aws-cdk-lib/aws-iam';\nimport { 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 { Chain, JsonPath, StateMachine } from 'aws-cdk-lib/aws-stepfunctions';\nimport { CallAwsService } from 'aws-cdk-lib/aws-stepfunctions-tasks';\nimport { Construct } from 'constructs';\nimport { BaseDocumentProcessingProps } from '../base-document-processing';\nimport { IAdapter } from './adapter';\nimport { DefaultRuntimes } from '../../framework';\nimport { DefaultObservabilityConfig, LambdaIamUtils, PowertoolsConfig } from '../../utilities';\nimport { DefaultDocumentProcessingConfig } from '../default-document-processing-config';\n\n/**\n * Struct for S3 Prefixes\n */\ninterface S3Prefixes {\n  readonly raw: string;\n  readonly processed: string;\n  readonly failed: string;\n}\n\n/**\n * Props for the Queued S3 Adapter\n */\nexport interface QueuedS3AdapterProps {\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   * @default create a new bucket\n   */\n  readonly bucket?: Bucket;\n\n  /**\n   * S3 prefix where the raw files would be stored.\n   * This serves as the trigger point for processing\n   *\n   * @default \"raw/\"\n   */\n  readonly rawPrefix?: string;\n\n  /**\n   * S3 prefix where the processed files would be stored.\n   *\n   * @default \"processed/\"\n   */\n  readonly processedPrefix?: string;\n\n  /**\n   * S3 prefix where the files that failed processing would be stored.\n   *\n   * @default \"failed/\"\n   */\n  readonly failedPrefix?: string;\n\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/**\n * This adapter allows the intelligent document processing workflow\n * to be triggered by files that are uploaded into a S3 Bucket.\n */\nexport class QueuedS3Adapter implements IAdapter {\n  private readonly adapterProps: QueuedS3AdapterProps;\n  private readonly resources: Record<string, any>;\n  private readonly prefixes: S3Prefixes;\n\n  constructor(adapterProps: QueuedS3AdapterProps = {}) {\n    this.adapterProps = adapterProps;\n    this.resources = {};\n    this.prefixes = {\n      raw: this.adapterProps.rawPrefix || 'raw/',\n      processed: this.adapterProps.processedPrefix || 'processed/',\n      failed: this.adapterProps.failedPrefix || 'failed/',\n    };\n  }\n\n  init(scope: Construct, props: BaseDocumentProcessingProps): void {\n    if (props.network) {\n      props.network.createServiceEndpoint('vpce-sqs', InterfaceVpcEndpointAwsService.SQS);\n      props.network.createServiceEndpoint('vpce-s3', InterfaceVpcEndpointAwsService.S3);\n    }\n\n    const encryptionKey = props.encryptionKey || new Key(scope, 'QueuedS3AdapterEncryptionKey', {\n      enableKeyRotation: true,\n      removalPolicy: props.removalPolicy || RemovalPolicy.DESTROY,\n    });\n\n    this.resources.encryptionKey = encryptionKey;\n\n    const bucket = this.adapterProps.bucket || new Bucket(scope, 'DocumentProcessingBucket', {\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    this.resources.bucket = bucket;\n\n    const deadLetterQueue = new Queue(scope, 'DocumentProcessingDLQ', {\n      visibilityTimeout: this.adapterProps.queueVisibilityTimeout || Duration.seconds(300),\n      removalPolicy: props.removalPolicy || RemovalPolicy.DESTROY,\n      enforceSSL: true,\n      encryption: QueueEncryption.KMS,\n      encryptionMasterKey: encryptionKey,\n    });\n\n    const queue = new Queue(scope, 'DocumentProcessingQueue', {\n      visibilityTimeout: this.adapterProps.queueVisibilityTimeout || Duration.seconds(300),\n      removalPolicy: props.removalPolicy || RemovalPolicy.DESTROY,\n      enforceSSL: true,\n      deadLetterQueue: {\n        maxReceiveCount: this.adapterProps.dlqMaxReceiveCount || 5,\n        queue: deadLetterQueue,\n      },\n      encryption: QueueEncryption.KMS,\n      encryptionMasterKey: encryptionKey,\n    });\n\n    this.resources.deadLetterQueue = deadLetterQueue;\n    this.resources.queue = queue;\n  }\n\n  createIngressTrigger(scope: Construct, stateMachine: StateMachine, props: BaseDocumentProcessingProps): Record<string, any> {\n    const bucket: Bucket = this.resources.bucket;\n    const queue: Queue = this.resources.queue;\n    const encryptionKey: Key = this.resources.encryptionKey;\n\n    bucket.addEventNotification(EventType.OBJECT_CREATED, new SqsDestination(queue), {\n      prefix: this.prefixes.raw,\n    });\n\n    const sqsConsumerLambdaFn = this.createSQSConsumerLambda(scope, stateMachine, props, encryptionKey, queue);\n\n    this.resources.sqsConsumerLambdaFunction = sqsConsumerLambdaFn;\n\n    return this.resources;\n  }\n\n  private createSQSConsumerLambda(scope: Construct, stateMachine: StateMachine\n    , props: BaseDocumentProcessingProps, encryptionKey: Key, queue: Queue): Function {\n    const metricNamespace = props.metricNamespace || DefaultObservabilityConfig.DEFAULT_METRIC_NAMESPACE;\n    const metricServiceName = props.metricServiceName || DefaultDocumentProcessingConfig.DEFAULT_OBSERVABILITY_METRIC_SVC_NAME;\n\n    const { region, account } = LambdaIamUtils.getStackInfo(scope);\n    // Create logs permissions and get unique function name\n    const logsPermissions = LambdaIamUtils.createLogsPermissions({\n      scope,\n      functionName: 'SQSConsumer',\n      region,\n      account,\n      enableObservability: props.enableObservability,\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 (props.network) {\n      policyStatements.push(LambdaIamUtils.generateLambdaVPCPermissions());\n    }\n\n    // Create IAM role for SQS consumer Lambda\n    const sqsConsumerRole = new Role(scope, 'SQSConsumerRole', {\n      assumedBy: new ServicePrincipal('lambda.amazonaws.com'),\n      inlinePolicies: {\n        SQSConsumerExecutionPolicy: new PolicyDocument({\n          statements: policyStatements,\n        }),\n      },\n    });\n\n    encryptionKey.grantEncryptDecrypt(sqsConsumerRole);\n\n    // Create SQS consumer Lambda function\n    const sqsConsumerLambda = new PythonFunction(scope, '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        RAW_PREFIX: this.prefixes.raw,\n        ...PowertoolsConfig.generateDefaultLambdaConfig(props.enableObservability, metricNamespace, metricServiceName),\n      },\n      timeout: Duration.minutes(5),\n      description: 'Consumes SQS messages and triggers Step Functions executions for document processing',\n      environmentEncryption: encryptionKey,\n      vpc: props.network ? props.network.vpc : undefined,\n      vpcSubnets: props.network ? props.network.applicationSubnetSelection() : undefined,\n    });\n\n    // Add SQS event source to Lambda\n    sqsConsumerLambda.addEventSource(\n      new SqsEventSource(queue, {\n        batchSize: 10,\n        maxBatchingWindow: Duration.seconds(5),\n        reportBatchItemFailures: true,\n      }),\n    );\n\n    return sqsConsumerLambda;\n  }\n\n  generateAdapterIAMPolicies(additionalIAMActions?: string[], narrowActions?: boolean): PolicyStatement[] {\n    const bucket: Bucket = this.resources.bucket;\n\n    const normalizedIAMActions = additionalIAMActions || [];\n\n    const statements = [];\n    if (!narrowActions) {\n      statements.push(new PolicyStatement({\n        effect: Effect.ALLOW,\n        actions: ['s3:GetObject', 's3:CopyObject', 's3:DeleteObject', 's3:PutObject', ...normalizedIAMActions],\n        resources: [`${bucket.bucketArn}/*`],\n      }));\n\n      if (bucket.encryptionKey) {\n        statements.push(new PolicyStatement({\n          effect: Effect.ALLOW,\n          actions: [\n            'kms:Encrypt',\n            'kms:ReEncrypt*',\n            'kms:GenerateDataKey*',\n            'kms:Decrypt',\n          ],\n          resources: [bucket.encryptionKey.keyArn],\n        }));\n      }\n\n    } else {\n      statements.push(new PolicyStatement({\n        effect: Effect.ALLOW,\n        actions: normalizedIAMActions,\n        resources: [`${bucket.bucketArn}/*`],\n      }));\n    }\n\n    return statements;\n  }\n\n  createFailedChain(scope: Construct): Chain {\n    const bucket: Bucket = this.resources.bucket;\n\n    const failedChain = new CallAwsService(scope, 'CopyToFailed', {\n      service: 's3',\n      action: 'copyObject',\n      parameters: {\n        Bucket: JsonPath.stringAt('$.content.bucket'),\n        CopySource: JsonPath.format('{}/{}', JsonPath.stringAt('$.content.bucket'), JsonPath.stringAt('$.content.key')),\n        Key: JsonPath.format(`${this.prefixes.failed}/{}`, JsonPath.stringAt('$.content.filename')),\n      },\n      iamResources: [`${bucket.bucketArn}/*`],\n      resultPath: JsonPath.DISCARD,\n    }).next(\n      new CallAwsService(scope, 'DeleteFromRaw', {\n        service: 's3',\n        action: 'deleteObject',\n        parameters: {\n          Bucket: JsonPath.stringAt('$.content.bucket'),\n          Key: JsonPath.stringAt('$.content.key'),\n        },\n        iamResources: [`${bucket.bucketArn}/*`],\n        resultPath: JsonPath.DISCARD,\n      }),\n    );\n\n    return failedChain;\n  }\n\n  createSuccessChain(scope: Construct): Chain {\n    const bucket: Bucket = this.resources.bucket;\n\n    const chain = new CallAwsService(scope, 'CopyToProcessed', {\n      service: 's3',\n      action: 'copyObject',\n      parameters: {\n        Bucket: JsonPath.stringAt('$.content.bucket'),\n        CopySource: JsonPath.format('{}/{}', JsonPath.stringAt('$.content.bucket'), JsonPath.stringAt('$.content.key')),\n        Key: JsonPath.format(`${this.prefixes.processed}/{}`, JsonPath.stringAt('$.content.filename')),\n      },\n      iamResources: [`${bucket.bucketArn}/*`],\n      resultPath: JsonPath.DISCARD,\n    }).next(\n      new CallAwsService(scope, 'DeleteFromRawSuccess', {\n        service: 's3',\n        action: 'deleteObject',\n        parameters: {\n          Bucket: JsonPath.stringAt('$.content.bucket'),\n          Key: JsonPath.stringAt('$.content.key'),\n        },\n        iamResources: [`${bucket.bucketArn}/*`],\n        resultPath: JsonPath.DISCARD,\n      }),\n    );\n\n    return chain;\n  }\n}"]}
|
|
@@ -1,10 +1,32 @@
|
|
|
1
1
|
import { LayerVersion } from 'aws-cdk-lib/aws-lambda';
|
|
2
|
+
import { Bucket } from 'aws-cdk-lib/aws-s3';
|
|
2
3
|
import { Construct } from 'constructs';
|
|
3
4
|
import { DocumentProcessingStepType } from './base-document-processing';
|
|
4
5
|
import { BedrockDocumentProcessing, BedrockDocumentProcessingProps } from './bedrock-document-processing';
|
|
5
6
|
export interface AgentProps {
|
|
7
|
+
/**
|
|
8
|
+
* Bucket where the tools are located in
|
|
9
|
+
* Primarily use to grant read permission to the
|
|
10
|
+
* processing agent to access the tools.
|
|
11
|
+
*
|
|
12
|
+
* @default No extra IAM permissions would be automatically
|
|
13
|
+
* assigned to the processing agent.
|
|
14
|
+
*/
|
|
15
|
+
readonly toolsBucket?: Bucket;
|
|
16
|
+
/**
|
|
17
|
+
* System prompt for the agent
|
|
18
|
+
*/
|
|
6
19
|
readonly agentSystemPrompt?: string;
|
|
20
|
+
/**
|
|
21
|
+
* S3 path where the tools are located.
|
|
22
|
+
* The agent would dynamically load the tools
|
|
23
|
+
*/
|
|
7
24
|
readonly toolsLocation?: string[];
|
|
25
|
+
/**
|
|
26
|
+
* If there are python dependencies that are needed by
|
|
27
|
+
* the provided tools, provide the Lambda Layers with the
|
|
28
|
+
* dependencies.
|
|
29
|
+
*/
|
|
8
30
|
readonly lambdaLayers?: LayerVersion[];
|
|
9
31
|
}
|
|
10
32
|
export interface AgenticDocumentProcessingProps extends BedrockDocumentProcessingProps {
|
|
@@ -5,7 +5,6 @@ exports.AgenticDocumentProcessing = void 0;
|
|
|
5
5
|
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
|
6
6
|
const aws_lambda_python_alpha_1 = require("@aws-cdk/aws-lambda-python-alpha");
|
|
7
7
|
const aws_cdk_lib_1 = require("aws-cdk-lib");
|
|
8
|
-
const aws_iam_1 = require("aws-cdk-lib/aws-iam");
|
|
9
8
|
const aws_lambda_1 = require("aws-cdk-lib/aws-lambda");
|
|
10
9
|
const aws_stepfunctions_tasks_1 = require("aws-cdk-lib/aws-stepfunctions-tasks");
|
|
11
10
|
const bedrock_document_processing_1 = require("./bedrock-document-processing");
|
|
@@ -21,23 +20,21 @@ class AgenticDocumentProcessing extends bedrock_document_processing_1.BedrockDoc
|
|
|
21
20
|
const fmModel = this.bedrockDocumentProcessingProps.processingModelId || bedrock_document_processing_1.BedrockDocumentProcessing.DEFAULT_PROCESSING_MODEL_ID;
|
|
22
21
|
const adjustedModelId = this.bedrockDocumentProcessingProps.useCrossRegionInference ? `${this.crossRegionInferencePrefix}.${fmModel.modelId}` : fmModel.modelId;
|
|
23
22
|
const role = this.generateLambdaRoleForBedrock(fmModel, 'ProcessingAgentLambdaRole');
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
's3:ListBucket',
|
|
28
|
-
],
|
|
29
|
-
resources: [
|
|
30
|
-
this.bucket.bucketArn,
|
|
31
|
-
],
|
|
32
|
-
}));
|
|
23
|
+
this.ingressAdapter.generateAdapterIAMPolicies(['s3:ListBucket']).forEach((statement) => {
|
|
24
|
+
role.addToPrincipalPolicy(statement);
|
|
25
|
+
});
|
|
33
26
|
const environmentVariables = {
|
|
34
27
|
MODEL_ID: adjustedModelId,
|
|
35
28
|
INVOKE_TYPE: 'agent',
|
|
36
29
|
...powertools_config_1.PowertoolsConfig.generateDefaultLambdaConfig(this.bedrockDocumentProcessingProps.enableObservability, this.metricNamespace, this.metricServiceName),
|
|
37
30
|
};
|
|
38
31
|
this.encryptionKey.grantEncryptDecrypt(role);
|
|
39
|
-
|
|
40
|
-
|
|
32
|
+
const toolsBucket = agentProps.processingAgentParameters?.toolsBucket;
|
|
33
|
+
if (toolsBucket) {
|
|
34
|
+
toolsBucket.grantRead(role);
|
|
35
|
+
if (toolsBucket.encryptionKey) {
|
|
36
|
+
toolsBucket.encryptionKey.grantDecrypt(role);
|
|
37
|
+
}
|
|
41
38
|
}
|
|
42
39
|
if (agentProps.processingAgentParameters?.toolsLocation) {
|
|
43
40
|
environmentVariables.TOOLS_CONFIG = JSON.stringify(agentProps.processingAgentParameters?.toolsLocation);
|
|
@@ -87,5 +84,5 @@ class AgenticDocumentProcessing extends bedrock_document_processing_1.BedrockDoc
|
|
|
87
84
|
}
|
|
88
85
|
exports.AgenticDocumentProcessing = AgenticDocumentProcessing;
|
|
89
86
|
_a = JSII_RTTI_SYMBOL_1;
|
|
90
|
-
AgenticDocumentProcessing[_a] = { fqn: "@cdklabs/cdk-appmod-catalog-blueprints.AgenticDocumentProcessing", version: "1.0
|
|
91
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"agentic-document-processing.js","sourceRoot":"","sources":["../../use-cases/document-processing/agentic-document-processing.ts"],"names":[],"mappings":"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","sourcesContent":["import { PythonFunction } from '@aws-cdk/aws-lambda-python-alpha';\nimport { Duration, Stack } from 'aws-cdk-lib';\nimport { Effect, PolicyStatement } from 'aws-cdk-lib/aws-iam';\nimport { Architecture, LayerVersion } from 'aws-cdk-lib/aws-lambda';\nimport { LambdaInvoke } from 'aws-cdk-lib/aws-stepfunctions-tasks';\nimport { Construct } from 'constructs';\nimport { DocumentProcessingStepType } from './base-document-processing';\nimport { BedrockDocumentProcessing, BedrockDocumentProcessingProps } from './bedrock-document-processing';\nimport { DefaultRuntimes } from '../framework';\nimport { LambdaIamUtils } from '../utilities';\nimport { PowertoolsConfig } from '../utilities/observability/powertools-config';\n\nexport interface AgentProps {\n  readonly agentSystemPrompt?: string;\n  readonly toolsLocation?: string[];\n  readonly lambdaLayers?: LayerVersion[];\n}\n\nexport interface AgenticDocumentProcessingProps extends BedrockDocumentProcessingProps {\n  readonly processingAgentParameters?: AgentProps;\n}\n\nexport class AgenticDocumentProcessing extends BedrockDocumentProcessing {\n  constructor(scope: Construct, id: string, props: AgenticDocumentProcessingProps) {\n    super(scope, id, props);\n  }\n\n  protected processingStep(): DocumentProcessingStepType {\n    const agentProps = this.bedrockDocumentProcessingProps as AgenticDocumentProcessingProps;\n    const fmModel = this.bedrockDocumentProcessingProps.processingModelId || BedrockDocumentProcessing.DEFAULT_PROCESSING_MODEL_ID;\n    const adjustedModelId = this.bedrockDocumentProcessingProps.useCrossRegionInference ? `${this.crossRegionInferencePrefix}.${fmModel.modelId}` : fmModel.modelId;\n    const role = this.generateLambdaRoleForBedrock(fmModel, 'ProcessingAgentLambdaRole');\n    role.addToPrincipalPolicy(new PolicyStatement({\n      effect: Effect.ALLOW,\n      actions: [\n        's3:ListBucket',\n      ],\n      resources: [\n        this.bucket.bucketArn,\n      ],\n    }));\n    const environmentVariables:Record<string, string> = {\n      MODEL_ID: adjustedModelId,\n      INVOKE_TYPE: 'agent',\n      ...PowertoolsConfig.generateDefaultLambdaConfig(\n        this.bedrockDocumentProcessingProps.enableObservability,\n        this.metricNamespace,\n        this.metricServiceName,\n      ),\n    };\n\n    this.encryptionKey.grantEncryptDecrypt(role);\n    if (this.bucketEncryptionKey) {\n      this.bucketEncryptionKey.grantEncryptDecrypt(role);\n    }\n\n    if (agentProps.processingAgentParameters?.toolsLocation) {\n      environmentVariables.TOOLS_CONFIG = JSON.stringify(agentProps.processingAgentParameters?.toolsLocation);\n    }\n\n    if (agentProps.processingAgentParameters?.agentSystemPrompt) {\n      environmentVariables.SYSTEM_PROMPT = agentProps.processingAgentParameters?.agentSystemPrompt;\n    }\n\n    if (agentProps.processingPrompt) {\n      environmentVariables.PROMPT = agentProps.processingPrompt;\n    }\n\n    const { region, account } = Stack.of(this);\n    const generatedLogPermissions = LambdaIamUtils.createLogsPermissions({\n      account,\n      functionName: 'agentic-idp-processing',\n      region,\n      scope: this,\n      enableObservability: this.bedrockDocumentProcessingProps.enableObservability,\n    });\n\n    const agenticFunction = new PythonFunction(this, 'ProcessingAgentFunction', {\n      functionName: generatedLogPermissions.uniqueFunctionName,\n      architecture: Architecture.X86_64,\n      entry: `${__dirname}/resources/default-strands-agent`,\n      runtime: DefaultRuntimes.PYTHON,\n      layers: agentProps.processingAgentParameters?.lambdaLayers,\n      environment: environmentVariables,\n      role,\n      timeout: this.bedrockDocumentProcessingProps.stepTimeouts || Duration.minutes(5),\n      memorySize: 1024,\n      environmentEncryption: this.encryptionKey,\n      vpc: this.bedrockDocumentProcessingProps.network ? this.bedrockDocumentProcessingProps.network.vpc : undefined,\n      vpcSubnets: this.bedrockDocumentProcessingProps.network ? this.bedrockDocumentProcessingProps.network.applicationSubnetSelection() : undefined,\n    });\n\n    for (const statement of generatedLogPermissions.policyStatements) {\n      agenticFunction.role?.addToPrincipalPolicy(statement);\n    }\n\n    if (this.bedrockDocumentProcessingProps.network) {\n      agenticFunction.role?.addToPrincipalPolicy(LambdaIamUtils.generateLambdaVPCPermissions());\n    }\n\n    return new LambdaInvoke(this, 'ProcessingStep', {\n      lambdaFunction: agenticFunction,\n      resultPath: '$.processingResult',\n      resultSelector: {\n        'result.$': '$.Payload.result',\n      },\n    });\n  }\n}"]}
|
|
87
|
+
AgenticDocumentProcessing[_a] = { fqn: "@cdklabs/cdk-appmod-catalog-blueprints.AgenticDocumentProcessing", version: "1.2.0" };
|
|
88
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"agentic-document-processing.js","sourceRoot":"","sources":["../../use-cases/document-processing/agentic-document-processing.ts"],"names":[],"mappings":"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","sourcesContent":["import { PythonFunction } from '@aws-cdk/aws-lambda-python-alpha';\nimport { Duration, Stack } from 'aws-cdk-lib';\nimport { Architecture, LayerVersion } from 'aws-cdk-lib/aws-lambda';\nimport { Bucket } from 'aws-cdk-lib/aws-s3';\nimport { LambdaInvoke } from 'aws-cdk-lib/aws-stepfunctions-tasks';\nimport { Construct } from 'constructs';\nimport { DocumentProcessingStepType } from './base-document-processing';\nimport { BedrockDocumentProcessing, BedrockDocumentProcessingProps } from './bedrock-document-processing';\nimport { DefaultRuntimes } from '../framework';\nimport { LambdaIamUtils } from '../utilities';\nimport { PowertoolsConfig } from '../utilities/observability/powertools-config';\n\nexport interface AgentProps {\n  /**\n   * Bucket where the tools are located in\n   * Primarily use to grant read permission to the\n   * processing agent to access the tools.\n   *\n   * @default No extra IAM permissions would be automatically\n   * assigned to the processing agent.\n   */\n  readonly toolsBucket?: Bucket;\n\n  /**\n   * System prompt for the agent\n   */\n  readonly agentSystemPrompt?: string;\n\n  /**\n   * S3 path where the tools are located.\n   * The agent would dynamically load the tools\n   */\n  readonly toolsLocation?: string[];\n\n  /**\n   * If there are python dependencies that are needed by\n   * the provided tools, provide the Lambda Layers with the\n   * dependencies.\n   */\n  readonly lambdaLayers?: LayerVersion[];\n}\n\nexport interface AgenticDocumentProcessingProps extends BedrockDocumentProcessingProps {\n  readonly processingAgentParameters?: AgentProps;\n}\n\nexport class AgenticDocumentProcessing extends BedrockDocumentProcessing {\n  constructor(scope: Construct, id: string, props: AgenticDocumentProcessingProps) {\n    super(scope, id, props);\n  }\n\n  protected processingStep(): DocumentProcessingStepType {\n    const agentProps = this.bedrockDocumentProcessingProps as AgenticDocumentProcessingProps;\n    const fmModel = this.bedrockDocumentProcessingProps.processingModelId || BedrockDocumentProcessing.DEFAULT_PROCESSING_MODEL_ID;\n    const adjustedModelId = this.bedrockDocumentProcessingProps.useCrossRegionInference ? `${this.crossRegionInferencePrefix}.${fmModel.modelId}` : fmModel.modelId;\n    const role = this.generateLambdaRoleForBedrock(fmModel, 'ProcessingAgentLambdaRole');\n    this.ingressAdapter.generateAdapterIAMPolicies(['s3:ListBucket']).forEach((statement) => {\n      role.addToPrincipalPolicy(statement);\n    });\n    const environmentVariables:Record<string, string> = {\n      MODEL_ID: adjustedModelId,\n      INVOKE_TYPE: 'agent',\n      ...PowertoolsConfig.generateDefaultLambdaConfig(\n        this.bedrockDocumentProcessingProps.enableObservability,\n        this.metricNamespace,\n        this.metricServiceName,\n      ),\n    };\n\n    this.encryptionKey.grantEncryptDecrypt(role);\n\n    const toolsBucket = agentProps.processingAgentParameters?.toolsBucket;\n\n    if (toolsBucket) {\n      toolsBucket.grantRead(role);\n\n      if (toolsBucket.encryptionKey) {\n        toolsBucket.encryptionKey.grantDecrypt(role);\n      }\n    }\n\n    if (agentProps.processingAgentParameters?.toolsLocation) {\n      environmentVariables.TOOLS_CONFIG = JSON.stringify(agentProps.processingAgentParameters?.toolsLocation);\n    }\n\n    if (agentProps.processingAgentParameters?.agentSystemPrompt) {\n      environmentVariables.SYSTEM_PROMPT = agentProps.processingAgentParameters?.agentSystemPrompt;\n    }\n\n    if (agentProps.processingPrompt) {\n      environmentVariables.PROMPT = agentProps.processingPrompt;\n    }\n\n    const { region, account } = Stack.of(this);\n    const generatedLogPermissions = LambdaIamUtils.createLogsPermissions({\n      account,\n      functionName: 'agentic-idp-processing',\n      region,\n      scope: this,\n      enableObservability: this.bedrockDocumentProcessingProps.enableObservability,\n    });\n\n    const agenticFunction = new PythonFunction(this, 'ProcessingAgentFunction', {\n      functionName: generatedLogPermissions.uniqueFunctionName,\n      architecture: Architecture.X86_64,\n      entry: `${__dirname}/resources/default-strands-agent`,\n      runtime: DefaultRuntimes.PYTHON,\n      layers: agentProps.processingAgentParameters?.lambdaLayers,\n      environment: environmentVariables,\n      role,\n      timeout: this.bedrockDocumentProcessingProps.stepTimeouts || Duration.minutes(5),\n      memorySize: 1024,\n      environmentEncryption: this.encryptionKey,\n      vpc: this.bedrockDocumentProcessingProps.network ? this.bedrockDocumentProcessingProps.network.vpc : undefined,\n      vpcSubnets: this.bedrockDocumentProcessingProps.network ? this.bedrockDocumentProcessingProps.network.applicationSubnetSelection() : undefined,\n    });\n\n    for (const statement of generatedLogPermissions.policyStatements) {\n      agenticFunction.role?.addToPrincipalPolicy(statement);\n    }\n\n    if (this.bedrockDocumentProcessingProps.network) {\n      agenticFunction.role?.addToPrincipalPolicy(LambdaIamUtils.generateLambdaVPCPermissions());\n    }\n\n    return new LambdaInvoke(this, 'ProcessingStep', {\n      lambdaFunction: agenticFunction,\n      resultPath: '$.processingResult',\n      resultSelector: {\n        'result.$': '$.Payload.result',\n      },\n    });\n  }\n}"]}
|
|
@@ -1,12 +1,11 @@
|
|
|
1
1
|
import { Duration, RemovalPolicy } from 'aws-cdk-lib';
|
|
2
2
|
import { IMetric } from 'aws-cdk-lib/aws-cloudwatch';
|
|
3
3
|
import { Table } from 'aws-cdk-lib/aws-dynamodb';
|
|
4
|
-
import {
|
|
5
|
-
import { Bucket } from 'aws-cdk-lib/aws-s3';
|
|
6
|
-
import { Queue } from 'aws-cdk-lib/aws-sqs';
|
|
4
|
+
import { Key } from 'aws-cdk-lib/aws-kms';
|
|
7
5
|
import { StateMachine } from 'aws-cdk-lib/aws-stepfunctions';
|
|
8
6
|
import { BedrockInvokeModel, LambdaInvoke, StepFunctionsStartExecution } from 'aws-cdk-lib/aws-stepfunctions-tasks';
|
|
9
7
|
import { Construct } from 'constructs';
|
|
8
|
+
import { IAdapter } from './adapter';
|
|
10
9
|
import { Network } from '../framework';
|
|
11
10
|
import { EventbridgeBroker } from '../framework/foundation/eventbridge-broker';
|
|
12
11
|
import { LogGroupDataProtectionProps } from '../utilities';
|
|
@@ -16,27 +15,16 @@ import { IObservable, ObservableProps } from '../utilities/observability/observa
|
|
|
16
15
|
*/
|
|
17
16
|
export interface BaseDocumentProcessingProps extends ObservableProps {
|
|
18
17
|
/**
|
|
19
|
-
*
|
|
20
|
-
*
|
|
18
|
+
* Adapter that defines how the document processing workflow is triggered
|
|
19
|
+
*
|
|
20
|
+
* @default QueuedS3Adapter
|
|
21
21
|
*/
|
|
22
|
-
readonly
|
|
22
|
+
readonly ingressAdapter?: IAdapter;
|
|
23
23
|
/**
|
|
24
24
|
* DynamoDB table for storing document processing metadata and workflow state.
|
|
25
25
|
* If not provided, a new table will be created with DocumentId as partition key.
|
|
26
26
|
*/
|
|
27
27
|
readonly documentProcessingTable?: Table;
|
|
28
|
-
/**
|
|
29
|
-
* SQS queue visibility timeout for processing messages.
|
|
30
|
-
* Should be longer than expected processing time to prevent duplicate processing.
|
|
31
|
-
* @default Duration.seconds(300)
|
|
32
|
-
*/
|
|
33
|
-
readonly queueVisibilityTimeout?: Duration;
|
|
34
|
-
/**
|
|
35
|
-
* The number of times a message can be unsuccessfully dequeued before being moved to the dead-letter queue.
|
|
36
|
-
*
|
|
37
|
-
* @default 5
|
|
38
|
-
*/
|
|
39
|
-
readonly dlqMaxReceiveCount?: number;
|
|
40
28
|
/**
|
|
41
29
|
* Maximum execution time for the Step Functions workflow.
|
|
42
30
|
* @default Duration.minutes(30)
|
|
@@ -68,22 +56,6 @@ export interface BaseDocumentProcessingProps extends ObservableProps {
|
|
|
68
56
|
*/
|
|
69
57
|
readonly encryptionKey?: Key;
|
|
70
58
|
}
|
|
71
|
-
/**
|
|
72
|
-
* S3 prefix constants for organizing documents throughout the processing lifecycle.
|
|
73
|
-
*
|
|
74
|
-
* Documents flow through these prefixes based on processing outcomes:
|
|
75
|
-
* - Upload → raw/ (triggers processing)
|
|
76
|
-
* - Success → processed/ (workflow completed successfully)
|
|
77
|
-
* - Failure → failed/ (workflow encountered errors)
|
|
78
|
-
*/
|
|
79
|
-
export declare enum DocumentProcessingPrefix {
|
|
80
|
-
/** Prefix for newly uploaded documents awaiting processing */
|
|
81
|
-
RAW = "raw/",
|
|
82
|
-
/** Prefix for documents that failed processing */
|
|
83
|
-
FAILED = "failed/",
|
|
84
|
-
/** Prefix for successfully processed documents */
|
|
85
|
-
PROCESSED = "processed/"
|
|
86
|
-
}
|
|
87
59
|
/**
|
|
88
60
|
* Union type for Step Functions tasks that can be used in document processing workflows.
|
|
89
61
|
* Supports Bedrock model invocation, Lambda function invocation, and nested Step Functions execution.
|
|
@@ -118,20 +90,14 @@ export declare abstract class BaseDocumentProcessing extends Construct implement
|
|
|
118
90
|
readonly metricNamespace: string;
|
|
119
91
|
/** log group data protection configuration */
|
|
120
92
|
readonly logGroupDataProtection: LogGroupDataProtectionProps;
|
|
121
|
-
/** S3 bucket for document storage with organized prefixes (raw/, processed/, failed/) */
|
|
122
|
-
readonly bucket: Bucket;
|
|
123
|
-
/** SQS queue for reliable message processing with dead letter queue support */
|
|
124
|
-
readonly queue: Queue;
|
|
125
93
|
/** DynamoDB table for storing document processing metadata and workflow state */
|
|
126
94
|
readonly documentProcessingTable: Table;
|
|
127
95
|
/** Configuration properties for the document processing pipeline */
|
|
128
96
|
private readonly props;
|
|
129
|
-
/** Dead letter queue */
|
|
130
|
-
readonly deadLetterQueue: Queue;
|
|
131
97
|
/** KMS key */
|
|
132
98
|
readonly encryptionKey: Key;
|
|
133
|
-
/**
|
|
134
|
-
readonly
|
|
99
|
+
/** Ingress adapter, responsible for triggering workflow */
|
|
100
|
+
readonly ingressAdapter: IAdapter;
|
|
135
101
|
/**
|
|
136
102
|
* Creates a new BaseDocumentProcessing construct.
|
|
137
103
|
*
|
|
@@ -144,8 +110,6 @@ export declare abstract class BaseDocumentProcessing extends Construct implement
|
|
|
144
110
|
*/
|
|
145
111
|
constructor(scope: Construct, id: string, props: BaseDocumentProcessingProps);
|
|
146
112
|
protected handleStateMachineCreation(stateMachineId: string): StateMachine;
|
|
147
|
-
protected handleWorkflowTrigger(stateMachine: StateMachine): void;
|
|
148
|
-
private createSQSConsumerLambda;
|
|
149
113
|
private createStateMachineRole;
|
|
150
114
|
private createMoveToFailedChain;
|
|
151
115
|
private createMoveToProcessedChain;
|