@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,447 @@
1
+ "use strict";
2
+ var _a;
3
+ Object.defineProperty(exports, "__esModule", { value: true });
4
+ exports.DataLoader = exports.FileType = exports.DatabaseEngine = 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("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_lambda_1 = require("aws-cdk-lib/aws-lambda");
14
+ const aws_s3_1 = require("aws-cdk-lib/aws-s3");
15
+ const aws_s3_deployment_1 = require("aws-cdk-lib/aws-s3-deployment");
16
+ const aws_stepfunctions_1 = require("aws-cdk-lib/aws-stepfunctions");
17
+ const aws_stepfunctions_tasks_1 = require("aws-cdk-lib/aws-stepfunctions-tasks");
18
+ const custom_resources_1 = require("aws-cdk-lib/custom-resources");
19
+ const constructs_1 = require("constructs");
20
+ /**
21
+ * Supported database engines
22
+ */
23
+ var DatabaseEngine;
24
+ (function (DatabaseEngine) {
25
+ DatabaseEngine["MYSQL"] = "mysql";
26
+ DatabaseEngine["POSTGRESQL"] = "postgresql";
27
+ })(DatabaseEngine || (exports.DatabaseEngine = DatabaseEngine = {}));
28
+ /**
29
+ * Supported file types for data loading
30
+ */
31
+ var FileType;
32
+ (function (FileType) {
33
+ /** Standard SQL file */
34
+ FileType["SQL"] = "sql";
35
+ /** MySQL dump file generated by mysqldump */
36
+ FileType["MYSQLDUMP"] = "mysqldump";
37
+ /** PostgreSQL dump file generated by pg_dump */
38
+ FileType["PGDUMP"] = "pgdump";
39
+ })(FileType || (exports.FileType = FileType = {}));
40
+ /**
41
+ * DataLoader construct for loading data into Aurora/RDS databases
42
+ *
43
+ * This construct provides a simplified solution for loading data from various file formats
44
+ * (SQL, mysqldump, pg_dump) into MySQL or PostgreSQL databases. It uses S3 for file storage,
45
+ * Step Functions for orchestration, and Lambda for processing.
46
+ *
47
+ * Architecture:
48
+ * 1. Files are uploaded to S3 bucket
49
+ * 2. Step Function is triggered with list of S3 keys
50
+ * 3. Step Function iterates over files in execution order
51
+ * 4. Lambda function processes each file against the database
52
+ *
53
+ * Example usage:
54
+ * Create a DataLoader with database configuration and file inputs.
55
+ * The construct will handle uploading files to S3, creating a Step Function
56
+ * to orchestrate processing, and executing the data loading pipeline.
57
+ */
58
+ class DataLoader extends constructs_1.Construct {
59
+ constructor(scope, id, props) {
60
+ super(scope, id);
61
+ // Store file inputs for later use
62
+ this.fileInputs = props.fileInputs;
63
+ // Validate props
64
+ this._validateProps(props);
65
+ // Get removal policy with default
66
+ const removalPolicy = props.removalPolicy || aws_cdk_lib_1.RemovalPolicy.DESTROY;
67
+ // Create S3 bucket for storing files
68
+ this.bucket = this._createBucket(removalPolicy);
69
+ // Create Lambda function for processing
70
+ this.processorFunction = this._createProcessorFunction(props);
71
+ // Create Step Functions state machine
72
+ this.stateMachine = this._createStateMachine();
73
+ // Create custom resource provider for triggering execution
74
+ this.customResourceProvider = this._createCustomResourceProvider();
75
+ // Upload files to S3
76
+ this._setupFileProcessing(props);
77
+ // Create custom resource to trigger execution after files are uploaded
78
+ this.executionTrigger = this._createExecutionTrigger();
79
+ }
80
+ /**
81
+ * Grants additional IAM permissions to the execution trigger Lambda function
82
+ * @param statement The IAM policy statement to add
83
+ */
84
+ grantExecutionTriggerPermissions(statement) {
85
+ // Get the Lambda function from the custom resource provider
86
+ const triggerFunction = this.customResourceProvider.onEventHandler;
87
+ if (triggerFunction) {
88
+ triggerFunction.addToRolePolicy(statement);
89
+ }
90
+ }
91
+ /**
92
+ * Validates the construct properties
93
+ * @param props The DataLoader properties
94
+ * @private
95
+ */
96
+ _validateProps(props) {
97
+ if (!props.databaseConfig) {
98
+ throw new Error('databaseConfig is required');
99
+ }
100
+ if (!props.databaseConfig.cluster && !props.databaseConfig.instance) {
101
+ throw new Error('Either cluster or instance must be provided in databaseConfig');
102
+ }
103
+ if (!props.fileInputs || props.fileInputs.length === 0) {
104
+ throw new Error('At least one file input is required');
105
+ }
106
+ // Validate file inputs
107
+ for (const fileInput of props.fileInputs) {
108
+ if (!fileInput.filePath) {
109
+ throw new Error('filePath is required for each file input');
110
+ }
111
+ if (!fileInput.fileType) {
112
+ throw new Error('fileType is required for each file input');
113
+ }
114
+ }
115
+ // Validate engine compatibility
116
+ for (const fileInput of props.fileInputs) {
117
+ if (props.databaseConfig.engine === DatabaseEngine.MYSQL &&
118
+ fileInput.fileType === FileType.PGDUMP) {
119
+ throw new Error('PostgreSQL dump files cannot be used with MySQL databases');
120
+ }
121
+ if (props.databaseConfig.engine === DatabaseEngine.POSTGRESQL &&
122
+ fileInput.fileType === FileType.MYSQLDUMP) {
123
+ throw new Error('MySQL dump files cannot be used with PostgreSQL databases');
124
+ }
125
+ }
126
+ }
127
+ /**
128
+ * Creates the S3 bucket for storing files
129
+ * @param removalPolicy The removal policy to apply
130
+ * @returns The created S3 bucket
131
+ * @private
132
+ */
133
+ _createBucket(removalPolicy) {
134
+ const bucket = new aws_s3_1.Bucket(this, 'DataLoaderBucket', {
135
+ encryption: aws_s3_1.BucketEncryption.S3_MANAGED,
136
+ blockPublicAccess: aws_s3_1.BlockPublicAccess.BLOCK_ALL,
137
+ removalPolicy: removalPolicy,
138
+ autoDeleteObjects: removalPolicy === aws_cdk_lib_1.RemovalPolicy.DESTROY,
139
+ });
140
+ return bucket;
141
+ }
142
+ /**
143
+ * Creates the Lambda function for processing data loading
144
+ * @param props The DataLoader properties
145
+ * @returns The created Lambda function
146
+ * @private
147
+ */
148
+ _createProcessorFunction(props) {
149
+ // Create a dedicated security group for the Lambda function
150
+ const lambdaSecurityGroup = new aws_ec2_1.SecurityGroup(this, 'DataLoaderProcessorSecurityGroup', {
151
+ vpc: props.databaseConfig.vpc,
152
+ description: 'Security group for DataLoader processor Lambda function',
153
+ allowAllOutbound: true, // Lambda needs outbound access for AWS services and internet
154
+ });
155
+ // Allow Lambda to connect to the database
156
+ // Add ingress rule to database security group to allow connections from Lambda
157
+ props.databaseConfig.securityGroup.addIngressRule(lambdaSecurityGroup, aws_ec2_1.Port.tcp(props.databaseConfig.engine === DatabaseEngine.MYSQL ? 3306 : 5432), `Allow DataLoader Lambda to connect to ${props.databaseConfig.engine} database`);
158
+ // Create Lambda function with automatic dependency bundling
159
+ const lambdaFunction = new aws_lambda_python_alpha_1.PythonFunction(this, 'DataLoaderProcessor', {
160
+ entry: path.join(__dirname, 'data-loader-lambda'),
161
+ runtime: aws_lambda_1.Runtime.PYTHON_3_13,
162
+ handler: 'handler',
163
+ index: 'index.py',
164
+ timeout: props.timeout || aws_cdk_lib_1.Duration.minutes(15),
165
+ memorySize: props.memorySize || 1024,
166
+ architecture: aws_lambda_1.Architecture.ARM_64,
167
+ vpc: props.databaseConfig.vpc,
168
+ securityGroups: [lambdaSecurityGroup], // Use the dedicated Lambda security group
169
+ environment: {
170
+ DATABASE_ENGINE: props.databaseConfig.engine,
171
+ DATABASE_SECRET_ARN: props.databaseConfig.secret.secretArn,
172
+ DATABASE_NAME: props.databaseConfig.databaseName,
173
+ S3_BUCKET: this.bucket.bucketName,
174
+ },
175
+ bundling: {
176
+ // Use custom bundling commands to avoid Poetry conflicts
177
+ command: [
178
+ 'bash', '-c', [
179
+ // Create a clean virtual environment
180
+ 'python -m venv /tmp/venv',
181
+ // Activate the virtual environment
182
+ 'source /tmp/venv/bin/activate',
183
+ // Install dependencies in the virtual environment
184
+ 'pip install --upgrade pip',
185
+ 'pip install -r requirements.txt -t /asset-output',
186
+ // Copy source files
187
+ 'cp -r . /asset-output',
188
+ // Clean up __pycache__ and other unnecessary files
189
+ 'find /asset-output -type d -name __pycache__ -exec rm -rf {} + || true',
190
+ 'find /asset-output -name "*.pyc" -delete || true',
191
+ ].join(' && '),
192
+ ],
193
+ // Use the standard Python image to avoid pre-installed packages
194
+ image: aws_cdk_lib_1.DockerImage.fromRegistry('public.ecr.aws/docker/library/python:3.13.5-bullseye'),
195
+ },
196
+ });
197
+ // Grant permissions
198
+ this._grantPermissions(lambdaFunction, props);
199
+ return lambdaFunction;
200
+ }
201
+ /**
202
+ * Creates the Step Functions state machine
203
+ * @returns The created state machine
204
+ * @private
205
+ */
206
+ _createStateMachine() {
207
+ // Create a Map state to iterate over file keys
208
+ const processFileTask = new aws_stepfunctions_tasks_1.LambdaInvoke(this, 'ProcessFileTask', {
209
+ lambdaFunction: this.processorFunction,
210
+ payload: aws_stepfunctions_1.TaskInput.fromObject({
211
+ 's3Key.$': '$',
212
+ 'bucketName': this.bucket.bucketName,
213
+ }),
214
+ });
215
+ // Create the state machine definition
216
+ const definition = new aws_stepfunctions_1.Map(this, 'ProcessFilesMap', {
217
+ itemsPath: '$.fileKeys',
218
+ maxConcurrency: 1, // Process files sequentially to maintain order
219
+ }).itemProcessor(processFileTask);
220
+ // Create the state machine
221
+ const stateMachine = new aws_stepfunctions_1.StateMachine(this, 'DataLoaderStateMachine', {
222
+ definitionBody: aws_stepfunctions_1.DefinitionBody.fromChainable(definition),
223
+ timeout: aws_cdk_lib_1.Duration.hours(2), // Allow up to 2 hours for the entire process
224
+ });
225
+ // Grant the state machine permission to invoke the Lambda function
226
+ this.processorFunction.grantInvoke(stateMachine);
227
+ return stateMachine;
228
+ }
229
+ /**
230
+ * Grants necessary permissions to the Lambda function
231
+ * @param lambdaFunction The Lambda function
232
+ * @param props The DataLoader properties
233
+ * @private
234
+ */
235
+ _grantPermissions(lambdaFunction, props) {
236
+ // Grant S3 permissions
237
+ this.bucket.grantRead(lambdaFunction);
238
+ // Grant Secrets Manager permissions
239
+ props.databaseConfig.secret.grantRead(lambdaFunction);
240
+ }
241
+ /**
242
+ * Sets up file processing by uploading files to S3
243
+ * @param props The DataLoader properties
244
+ * @private
245
+ */
246
+ _setupFileProcessing(props) {
247
+ // Separate local files from S3 URIs
248
+ const localFiles = props.fileInputs.filter(f => !f.filePath.startsWith('s3://'));
249
+ // Upload local files to S3 if any
250
+ if (localFiles.length > 0) {
251
+ // Process each file individually to handle both files and directories
252
+ localFiles.forEach((fileInput, index) => {
253
+ const filePath = fileInput.filePath;
254
+ // eslint-disable-next-line @typescript-eslint/no-require-imports
255
+ const fs = require('fs');
256
+ // eslint-disable-next-line @typescript-eslint/no-require-imports
257
+ const pathModule = require('path');
258
+ let resolvedPath;
259
+ if (pathModule.isAbsolute(filePath)) {
260
+ resolvedPath = filePath;
261
+ }
262
+ else {
263
+ // Resolve relative paths from the current working directory
264
+ resolvedPath = pathModule.resolve(filePath);
265
+ }
266
+ // Check if the path exists
267
+ if (!fs.existsSync(resolvedPath)) {
268
+ throw new Error(`File not found: ${resolvedPath}`);
269
+ }
270
+ if (fs.statSync(resolvedPath).isFile()) {
271
+ // For individual files, deploy from parent directory with include filter
272
+ const parentDir = pathModule.dirname(resolvedPath);
273
+ const fileName = pathModule.basename(resolvedPath);
274
+ new aws_s3_deployment_1.BucketDeployment(this, `DataLoaderFileDeployment${index}`, {
275
+ sources: [aws_s3_deployment_1.Source.asset(parentDir)],
276
+ destinationBucket: this.bucket,
277
+ destinationKeyPrefix: 'data-files/',
278
+ include: [fileName],
279
+ });
280
+ }
281
+ else {
282
+ // For directories, deploy entire directory
283
+ new aws_s3_deployment_1.BucketDeployment(this, `DataLoaderFileDeployment${index}`, {
284
+ sources: [aws_s3_deployment_1.Source.asset(resolvedPath)],
285
+ destinationBucket: this.bucket,
286
+ destinationKeyPrefix: 'data-files/',
287
+ });
288
+ }
289
+ });
290
+ }
291
+ }
292
+ /**
293
+ * Creates a custom resource provider for triggering state machine execution
294
+ * @returns The custom resource provider
295
+ * @private
296
+ */
297
+ _createCustomResourceProvider() {
298
+ // Create IAM role for the custom resource Lambda function with all necessary policies
299
+ const customResourceRole = new aws_iam_1.Role(this, 'StateMachineExecutionTriggerRole', {
300
+ assumedBy: new aws_iam_1.ServicePrincipal('lambda.amazonaws.com'),
301
+ description: 'IAM role for DataLoader custom resource Lambda function',
302
+ managedPolicies: [
303
+ aws_iam_1.ManagedPolicy.fromAwsManagedPolicyName('service-role/AWSLambdaBasicExecutionRole'),
304
+ ],
305
+ inlinePolicies: {
306
+ StateMachineExecutionPolicy: new aws_iam_1.PolicyDocument({
307
+ statements: [
308
+ new aws_iam_1.PolicyStatement({
309
+ effect: aws_iam_1.Effect.ALLOW,
310
+ actions: [
311
+ 'states:StartExecution',
312
+ 'states:DescribeExecution',
313
+ 'states:StopExecution',
314
+ ],
315
+ resources: [
316
+ this.stateMachine.stateMachineArn,
317
+ `${this.stateMachine.stateMachineArn}:*`, // For execution ARNs
318
+ ],
319
+ }),
320
+ ],
321
+ }),
322
+ },
323
+ });
324
+ // Create Lambda function for custom resource with the pre-configured role
325
+ const customResourceFunction = new aws_lambda_1.Function(this, 'StateMachineExecutionTrigger', {
326
+ runtime: aws_lambda_1.Runtime.PYTHON_3_11,
327
+ handler: 'index.handler',
328
+ role: customResourceRole, // Attach the role during creation
329
+ code: aws_lambda_1.Code.fromInline(`
330
+ import json
331
+ import boto3
332
+ import logging
333
+
334
+ logger = logging.getLogger()
335
+ logger.setLevel(logging.INFO)
336
+
337
+ stepfunctions = boto3.client('stepfunctions')
338
+
339
+ def handler(event, context):
340
+ logger.info(f"Received event: {json.dumps(event)}")
341
+
342
+ request_type = event['RequestType']
343
+
344
+ if request_type == 'Create' or request_type == 'Update':
345
+ try:
346
+ # Get parameters from event
347
+ state_machine_arn = event['ResourceProperties']['StateMachineArn']
348
+ file_keys = event['ResourceProperties']['FileKeys']
349
+
350
+ # Start execution
351
+ response = stepfunctions.start_execution(
352
+ stateMachineArn=state_machine_arn,
353
+ input=json.dumps({
354
+ 'fileKeys': file_keys
355
+ })
356
+ )
357
+
358
+ execution_arn = response['executionArn']
359
+ logger.info(f"Started execution: {execution_arn}")
360
+
361
+ return {
362
+ 'Status': 'SUCCESS',
363
+ 'PhysicalResourceId': execution_arn,
364
+ 'Data': {
365
+ 'ExecutionArn': execution_arn
366
+ }
367
+ }
368
+
369
+ except Exception as e:
370
+ logger.error(f"Error starting execution: {str(e)}")
371
+ return {
372
+ 'Status': 'FAILED',
373
+ 'Reason': str(e),
374
+ 'PhysicalResourceId': 'failed'
375
+ }
376
+
377
+ elif request_type == 'Delete':
378
+ # For delete, we don't need to do anything special
379
+ return {
380
+ 'Status': 'SUCCESS',
381
+ 'PhysicalResourceId': event.get('PhysicalResourceId', 'none')
382
+ }
383
+
384
+ return {
385
+ 'Status': 'SUCCESS',
386
+ 'PhysicalResourceId': 'none'
387
+ }
388
+ `),
389
+ timeout: aws_cdk_lib_1.Duration.minutes(5),
390
+ });
391
+ // Create provider with the Lambda function that already has the correct role
392
+ const provider = new custom_resources_1.Provider(this, 'StateMachineExecutionProvider', {
393
+ onEventHandler: customResourceFunction,
394
+ });
395
+ return provider;
396
+ }
397
+ /**
398
+ * Creates a custom resource to trigger state machine execution
399
+ * @returns The custom resource
400
+ * @private
401
+ */
402
+ _createExecutionTrigger() {
403
+ // Get ordered file keys
404
+ const orderedFileKeys = this._getOrderedFileKeys();
405
+ const customResource = new aws_cdk_lib_1.CustomResource(this, 'ExecutionTriggerResource', {
406
+ serviceToken: this.customResourceProvider.serviceToken,
407
+ properties: {
408
+ StateMachineArn: this.stateMachine.stateMachineArn,
409
+ FileKeys: orderedFileKeys,
410
+ // Add a timestamp to force updates when needed
411
+ Timestamp: Date.now().toString(),
412
+ },
413
+ });
414
+ // Ensure the custom resource runs after bucket deployment (if exists)
415
+ if (this.bucketDeployment) {
416
+ customResource.node.addDependency(this.stateMachine);
417
+ customResource.node.addDependency(this.bucketDeployment);
418
+ }
419
+ return customResource;
420
+ }
421
+ /**
422
+ * Gets the ordered file keys for execution
423
+ * @returns Array of S3 keys in execution order
424
+ * @private
425
+ */
426
+ _getOrderedFileKeys() {
427
+ // Sort files by execution order
428
+ const sortedFiles = [...this.fileInputs].sort((a, b) => (a.executionOrder || 0) - (b.executionOrder || 0));
429
+ // Convert file paths to S3 keys
430
+ return sortedFiles.map(file => {
431
+ if (file.filePath.startsWith('s3://')) {
432
+ // Extract key from S3 URI
433
+ const parts = file.filePath.replace('s3://', '').split('/');
434
+ return parts.slice(1).join('/'); // Remove bucket name
435
+ }
436
+ else {
437
+ // Local files are uploaded with data-files/ prefix
438
+ const fileName = path.basename(file.filePath);
439
+ return `data-files/${fileName}`;
440
+ }
441
+ });
442
+ }
443
+ }
444
+ exports.DataLoader = DataLoader;
445
+ _a = JSII_RTTI_SYMBOL_1;
446
+ DataLoader[_a] = { fqn: "@cdklabs/cdk-appmod-catalog-blueprints.DataLoader", version: "1.0.0" };
447
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"data-loader.js","sourceRoot":"","sources":["../../use-cases/utilities/data-loader.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 'path';\nimport { PythonFunction } from '@aws-cdk/aws-lambda-python-alpha';\nimport { Duration, RemovalPolicy, CustomResource, DockerImage } from 'aws-cdk-lib';\nimport { IVpc, SecurityGroup, Port, ISecurityGroup } from 'aws-cdk-lib/aws-ec2';\nimport { PolicyStatement, Role, ServicePrincipal, ManagedPolicy, PolicyDocument, Effect } from 'aws-cdk-lib/aws-iam';\nimport { Architecture, Code, Function as LambdaFunction, Runtime } from 'aws-cdk-lib/aws-lambda';\nimport { IDatabaseCluster, IDatabaseInstance } from 'aws-cdk-lib/aws-rds';\nimport { Bucket, BucketEncryption, BlockPublicAccess } from 'aws-cdk-lib/aws-s3';\nimport { BucketDeployment, Source } from 'aws-cdk-lib/aws-s3-deployment';\nimport { ISecret } from 'aws-cdk-lib/aws-secretsmanager';\nimport { StateMachine, DefinitionBody, Map, TaskInput } from 'aws-cdk-lib/aws-stepfunctions';\nimport { LambdaInvoke } from 'aws-cdk-lib/aws-stepfunctions-tasks';\nimport { Provider } from 'aws-cdk-lib/custom-resources';\nimport { Construct } from 'constructs';\n\n/**\n * Supported database engines\n */\nexport enum DatabaseEngine {\n  MYSQL = 'mysql',\n  POSTGRESQL = 'postgresql',\n}\n\n/**\n * Supported file types for data loading\n */\nexport enum FileType {\n  /** Standard SQL file */\n  SQL = 'sql',\n  /** MySQL dump file generated by mysqldump */\n  MYSQLDUMP = 'mysqldump',\n  /** PostgreSQL dump file generated by pg_dump */\n  PGDUMP = 'pgdump',\n}\n\n/**\n * Database connection configuration\n */\nexport interface DatabaseConfig {\n  /** Database engine type */\n  readonly engine: DatabaseEngine;\n  /** Database cluster (for Aurora) */\n  readonly cluster?: IDatabaseCluster;\n  /** Database instance (for RDS) */\n  readonly instance?: IDatabaseInstance;\n  /** Database credentials secret */\n  readonly secret: ISecret;\n  /** Database name to connect to */\n  readonly databaseName: string;\n  /** VPC where the database is located */\n  readonly vpc: IVpc;\n  /** Security group for database access */\n  readonly securityGroup: ISecurityGroup;\n}\n\n/**\n * File input configuration\n */\nexport interface FileInput {\n  /** Path to the file (local path or S3 URI) */\n  readonly filePath: string;\n  /** Type of file */\n  readonly fileType: FileType;\n  /** Execution order (lower numbers execute first) */\n  readonly executionOrder?: number;\n  /** Whether to continue on error */\n  readonly continueOnError?: boolean;\n}\n\n/**\n * Properties for the DataLoader construct\n */\nexport interface DataLoaderProps {\n  /** Database configuration */\n  readonly databaseConfig: DatabaseConfig;\n  /** List of files to load */\n  readonly fileInputs: FileInput[];\n  /** Optional removal policy for resources (defaults to DESTROY) */\n  readonly removalPolicy?: RemovalPolicy;\n  /** Optional timeout for Lambda function (defaults to 15 minutes) */\n  readonly timeout?: Duration;\n  /** Optional memory size for Lambda function (defaults to 1024 MB) */\n  readonly memorySize?: number;\n}\n\n/**\n * DataLoader construct for loading data into Aurora/RDS databases\n *\n * This construct provides a simplified solution for loading data from various file formats\n * (SQL, mysqldump, pg_dump) into MySQL or PostgreSQL databases. It uses S3 for file storage,\n * Step Functions for orchestration, and Lambda for processing.\n *\n * Architecture:\n * 1. Files are uploaded to S3 bucket\n * 2. Step Function is triggered with list of S3 keys\n * 3. Step Function iterates over files in execution order\n * 4. Lambda function processes each file against the database\n *\n * Example usage:\n * Create a DataLoader with database configuration and file inputs.\n * The construct will handle uploading files to S3, creating a Step Function\n * to orchestrate processing, and executing the data loading pipeline.\n */\nexport class DataLoader extends Construct {\n  /** The S3 bucket used for storing files */\n  public readonly bucket: Bucket;\n  /** The Step Functions state machine for orchestration */\n  public readonly stateMachine: StateMachine;\n  /** The Lambda function that processes the data loading */\n  public readonly processorFunction: LambdaFunction;\n  /** The bucket deployment for uploading files */\n  public bucketDeployment?: BucketDeployment;\n  /** The custom resource provider for triggering state machine execution */\n  public readonly customResourceProvider: Provider;\n  /** The custom resource that triggers the state machine */\n  public readonly executionTrigger: CustomResource;\n  /** The file inputs configuration */\n  private readonly fileInputs: FileInput[];\n\n  constructor(scope: Construct, id: string, props: DataLoaderProps) {\n    super(scope, id);\n\n    // Store file inputs for later use\n    this.fileInputs = props.fileInputs;\n\n    // Validate props\n    this._validateProps(props);\n\n    // Get removal policy with default\n    const removalPolicy = props.removalPolicy || RemovalPolicy.DESTROY;\n\n    // Create S3 bucket for storing files\n    this.bucket = this._createBucket(removalPolicy);\n\n    // Create Lambda function for processing\n    this.processorFunction = this._createProcessorFunction(props);\n\n    // Create Step Functions state machine\n    this.stateMachine = this._createStateMachine();\n\n    // Create custom resource provider for triggering execution\n    this.customResourceProvider = this._createCustomResourceProvider();\n\n    // Upload files to S3\n    this._setupFileProcessing(props);\n\n    // Create custom resource to trigger execution after files are uploaded\n    this.executionTrigger = this._createExecutionTrigger();\n  }\n\n  /**\n   * Grants additional IAM permissions to the execution trigger Lambda function\n   * @param statement The IAM policy statement to add\n   */\n  public grantExecutionTriggerPermissions(statement: PolicyStatement): void {\n    // Get the Lambda function from the custom resource provider\n    const triggerFunction = this.customResourceProvider.onEventHandler;\n    if (triggerFunction) {\n      triggerFunction.addToRolePolicy(statement);\n    }\n  }\n\n  /**\n   * Validates the construct properties\n   * @param props The DataLoader properties\n   * @private\n   */\n  private _validateProps(props: DataLoaderProps): void {\n    if (!props.databaseConfig) {\n      throw new Error('databaseConfig is required');\n    }\n\n    if (!props.databaseConfig.cluster && !props.databaseConfig.instance) {\n      throw new Error('Either cluster or instance must be provided in databaseConfig');\n    }\n\n    if (!props.fileInputs || props.fileInputs.length === 0) {\n      throw new Error('At least one file input is required');\n    }\n\n    // Validate file inputs\n    for (const fileInput of props.fileInputs) {\n      if (!fileInput.filePath) {\n        throw new Error('filePath is required for each file input');\n      }\n      if (!fileInput.fileType) {\n        throw new Error('fileType is required for each file input');\n      }\n    }\n\n    // Validate engine compatibility\n    for (const fileInput of props.fileInputs) {\n      if (props.databaseConfig.engine === DatabaseEngine.MYSQL &&\n          fileInput.fileType === FileType.PGDUMP) {\n        throw new Error('PostgreSQL dump files cannot be used with MySQL databases');\n      }\n      if (props.databaseConfig.engine === DatabaseEngine.POSTGRESQL &&\n          fileInput.fileType === FileType.MYSQLDUMP) {\n        throw new Error('MySQL dump files cannot be used with PostgreSQL databases');\n      }\n    }\n  }\n\n  /**\n   * Creates the S3 bucket for storing files\n   * @param removalPolicy The removal policy to apply\n   * @returns The created S3 bucket\n   * @private\n   */\n  private _createBucket(removalPolicy: RemovalPolicy): Bucket {\n    const bucket = new Bucket(this, 'DataLoaderBucket', {\n      encryption: BucketEncryption.S3_MANAGED,\n      blockPublicAccess: BlockPublicAccess.BLOCK_ALL,\n      removalPolicy: removalPolicy,\n      autoDeleteObjects: removalPolicy === RemovalPolicy.DESTROY,\n    });\n\n    return bucket;\n  }\n\n  /**\n   * Creates the Lambda function for processing data loading\n   * @param props The DataLoader properties\n   * @returns The created Lambda function\n   * @private\n   */\n  private _createProcessorFunction(props: DataLoaderProps): LambdaFunction {\n    // Create a dedicated security group for the Lambda function\n    const lambdaSecurityGroup = new SecurityGroup(this, 'DataLoaderProcessorSecurityGroup', {\n      vpc: props.databaseConfig.vpc,\n      description: 'Security group for DataLoader processor Lambda function',\n      allowAllOutbound: true, // Lambda needs outbound access for AWS services and internet\n    });\n\n    // Allow Lambda to connect to the database\n    // Add ingress rule to database security group to allow connections from Lambda\n    props.databaseConfig.securityGroup.addIngressRule(\n      lambdaSecurityGroup,\n      Port.tcp(props.databaseConfig.engine === DatabaseEngine.MYSQL ? 3306 : 5432),\n      `Allow DataLoader Lambda to connect to ${props.databaseConfig.engine} database`,\n    );\n\n    // Create Lambda function with automatic dependency bundling\n    const lambdaFunction = new PythonFunction(this, 'DataLoaderProcessor', {\n      entry: path.join(__dirname, 'data-loader-lambda'),\n      runtime: Runtime.PYTHON_3_13,\n      handler: 'handler',\n      index: 'index.py',\n      timeout: props.timeout || Duration.minutes(15),\n      memorySize: props.memorySize || 1024,\n      architecture: Architecture.ARM_64,\n      vpc: props.databaseConfig.vpc,\n      securityGroups: [lambdaSecurityGroup], // Use the dedicated Lambda security group\n      environment: {\n        DATABASE_ENGINE: props.databaseConfig.engine,\n        DATABASE_SECRET_ARN: props.databaseConfig.secret.secretArn,\n        DATABASE_NAME: props.databaseConfig.databaseName,\n        S3_BUCKET: this.bucket.bucketName,\n      },\n      bundling: {\n        // Use custom bundling commands to avoid Poetry conflicts\n        command: [\n          'bash', '-c', [\n            // Create a clean virtual environment\n            'python -m venv /tmp/venv',\n            // Activate the virtual environment\n            'source /tmp/venv/bin/activate',\n            // Install dependencies in the virtual environment\n            'pip install --upgrade pip',\n            'pip install -r requirements.txt -t /asset-output',\n            // Copy source files\n            'cp -r . /asset-output',\n            // Clean up __pycache__ and other unnecessary files\n            'find /asset-output -type d -name __pycache__ -exec rm -rf {} + || true',\n            'find /asset-output -name \"*.pyc\" -delete || true',\n          ].join(' && '),\n        ],\n        // Use the standard Python image to avoid pre-installed packages\n        image: DockerImage.fromRegistry('public.ecr.aws/docker/library/python:3.13.5-bullseye'),\n      },\n    });\n\n    // Grant permissions\n    this._grantPermissions(lambdaFunction, props);\n\n    return lambdaFunction;\n  }\n\n  /**\n   * Creates the Step Functions state machine\n   * @returns The created state machine\n   * @private\n   */\n  private _createStateMachine(): StateMachine {\n    // Create a Map state to iterate over file keys\n    const processFileTask = new LambdaInvoke(this, 'ProcessFileTask', {\n      lambdaFunction: this.processorFunction,\n      payload: TaskInput.fromObject({\n        's3Key.$': '$',\n        'bucketName': this.bucket.bucketName,\n      }),\n    });\n\n    // Create the state machine definition\n    const definition = new Map(this, 'ProcessFilesMap', {\n      itemsPath: '$.fileKeys',\n      maxConcurrency: 1, // Process files sequentially to maintain order\n    }).itemProcessor(processFileTask);\n\n    // Create the state machine\n    const stateMachine = new StateMachine(this, 'DataLoaderStateMachine', {\n      definitionBody: DefinitionBody.fromChainable(definition),\n      timeout: Duration.hours(2), // Allow up to 2 hours for the entire process\n    });\n\n    // Grant the state machine permission to invoke the Lambda function\n    this.processorFunction.grantInvoke(stateMachine);\n\n    return stateMachine;\n  }\n\n  /**\n   * Grants necessary permissions to the Lambda function\n   * @param lambdaFunction The Lambda function\n   * @param props The DataLoader properties\n   * @private\n   */\n  private _grantPermissions(lambdaFunction: LambdaFunction, props: DataLoaderProps): void {\n    // Grant S3 permissions\n    this.bucket.grantRead(lambdaFunction);\n\n    // Grant Secrets Manager permissions\n    props.databaseConfig.secret.grantRead(lambdaFunction);\n  }\n\n  /**\n   * Sets up file processing by uploading files to S3\n   * @param props The DataLoader properties\n   * @private\n   */\n  private _setupFileProcessing(props: DataLoaderProps): void {\n    // Separate local files from S3 URIs\n    const localFiles = props.fileInputs.filter(f => !f.filePath.startsWith('s3://'));\n\n    // Upload local files to S3 if any\n    if (localFiles.length > 0) {\n      // Process each file individually to handle both files and directories\n      localFiles.forEach((fileInput, index) => {\n        const filePath = fileInput.filePath;\n        // eslint-disable-next-line @typescript-eslint/no-require-imports\n        const fs = require('fs');\n        // eslint-disable-next-line @typescript-eslint/no-require-imports\n        const pathModule = require('path');\n\n        let resolvedPath: string;\n        if (pathModule.isAbsolute(filePath)) {\n          resolvedPath = filePath;\n        } else {\n          // Resolve relative paths from the current working directory\n          resolvedPath = pathModule.resolve(filePath);\n        }\n\n        // Check if the path exists\n        if (!fs.existsSync(resolvedPath)) {\n          throw new Error(`File not found: ${resolvedPath}`);\n        }\n\n        if (fs.statSync(resolvedPath).isFile()) {\n          // For individual files, deploy from parent directory with include filter\n          const parentDir = pathModule.dirname(resolvedPath);\n          const fileName = pathModule.basename(resolvedPath);\n\n          new BucketDeployment(this, `DataLoaderFileDeployment${index}`, {\n            sources: [Source.asset(parentDir)],\n            destinationBucket: this.bucket,\n            destinationKeyPrefix: 'data-files/',\n            include: [fileName],\n          });\n        } else {\n          // For directories, deploy entire directory\n          new BucketDeployment(this, `DataLoaderFileDeployment${index}`, {\n            sources: [Source.asset(resolvedPath)],\n            destinationBucket: this.bucket,\n            destinationKeyPrefix: 'data-files/',\n          });\n        }\n      });\n    }\n  }\n\n  /**\n   * Creates a custom resource provider for triggering state machine execution\n   * @returns The custom resource provider\n   * @private\n   */\n  private _createCustomResourceProvider(): Provider {\n    // Create IAM role for the custom resource Lambda function with all necessary policies\n    const customResourceRole = new Role(this, 'StateMachineExecutionTriggerRole', {\n      assumedBy: new ServicePrincipal('lambda.amazonaws.com'),\n      description: 'IAM role for DataLoader custom resource Lambda function',\n      managedPolicies: [\n        ManagedPolicy.fromAwsManagedPolicyName('service-role/AWSLambdaBasicExecutionRole'),\n      ],\n      inlinePolicies: {\n        StateMachineExecutionPolicy: new PolicyDocument({\n          statements: [\n            new PolicyStatement({\n              effect: Effect.ALLOW,\n              actions: [\n                'states:StartExecution',\n                'states:DescribeExecution',\n                'states:StopExecution',\n              ],\n              resources: [\n                this.stateMachine.stateMachineArn,\n                `${this.stateMachine.stateMachineArn}:*`, // For execution ARNs\n              ],\n            }),\n          ],\n        }),\n      },\n    });\n\n    // Create Lambda function for custom resource with the pre-configured role\n    const customResourceFunction = new LambdaFunction(this, 'StateMachineExecutionTrigger', {\n      runtime: Runtime.PYTHON_3_11,\n      handler: 'index.handler',\n      role: customResourceRole, // Attach the role during creation\n      code: Code.fromInline(`\nimport json\nimport boto3\nimport logging\n\nlogger = logging.getLogger()\nlogger.setLevel(logging.INFO)\n\nstepfunctions = boto3.client('stepfunctions')\n\ndef handler(event, context):\n    logger.info(f\"Received event: {json.dumps(event)}\")\n    \n    request_type = event['RequestType']\n    \n    if request_type == 'Create' or request_type == 'Update':\n        try:\n            # Get parameters from event\n            state_machine_arn = event['ResourceProperties']['StateMachineArn']\n            file_keys = event['ResourceProperties']['FileKeys']\n            \n            # Start execution\n            response = stepfunctions.start_execution(\n                stateMachineArn=state_machine_arn,\n                input=json.dumps({\n                    'fileKeys': file_keys\n                })\n            )\n            \n            execution_arn = response['executionArn']\n            logger.info(f\"Started execution: {execution_arn}\")\n            \n            return {\n                'Status': 'SUCCESS',\n                'PhysicalResourceId': execution_arn,\n                'Data': {\n                    'ExecutionArn': execution_arn\n                }\n            }\n            \n        except Exception as e:\n            logger.error(f\"Error starting execution: {str(e)}\")\n            return {\n                'Status': 'FAILED',\n                'Reason': str(e),\n                'PhysicalResourceId': 'failed'\n            }\n    \n    elif request_type == 'Delete':\n        # For delete, we don't need to do anything special\n        return {\n            'Status': 'SUCCESS',\n            'PhysicalResourceId': event.get('PhysicalResourceId', 'none')\n        }\n    \n    return {\n        'Status': 'SUCCESS',\n        'PhysicalResourceId': 'none'\n    }\n      `),\n      timeout: Duration.minutes(5),\n    });\n\n    // Create provider with the Lambda function that already has the correct role\n    const provider = new Provider(this, 'StateMachineExecutionProvider', {\n      onEventHandler: customResourceFunction,\n    });\n\n    return provider;\n  }\n\n  /**\n   * Creates a custom resource to trigger state machine execution\n   * @returns The custom resource\n   * @private\n   */\n  private _createExecutionTrigger(): CustomResource {\n    // Get ordered file keys\n    const orderedFileKeys = this._getOrderedFileKeys();\n\n    const customResource = new CustomResource(this, 'ExecutionTriggerResource', {\n      serviceToken: this.customResourceProvider.serviceToken,\n      properties: {\n        StateMachineArn: this.stateMachine.stateMachineArn,\n        FileKeys: orderedFileKeys,\n        // Add a timestamp to force updates when needed\n        Timestamp: Date.now().toString(),\n      },\n    });\n\n    // Ensure the custom resource runs after bucket deployment (if exists)\n    if (this.bucketDeployment) {\n      customResource.node.addDependency(this.stateMachine);\n      customResource.node.addDependency(this.bucketDeployment);\n    }\n\n    return customResource;\n  }\n\n  /**\n   * Gets the ordered file keys for execution\n   * @returns Array of S3 keys in execution order\n   * @private\n   */\n  private _getOrderedFileKeys(): string[] {\n    // Sort files by execution order\n    const sortedFiles = [...this.fileInputs].sort(\n      (a, b) => (a.executionOrder || 0) - (b.executionOrder || 0),\n    );\n\n    // Convert file paths to S3 keys\n    return sortedFiles.map(file => {\n      if (file.filePath.startsWith('s3://')) {\n        // Extract key from S3 URI\n        const parts = file.filePath.replace('s3://', '').split('/');\n        return parts.slice(1).join('/'); // Remove bucket name\n      } else {\n        // Local files are uploaded with data-files/ prefix\n        const fileName = path.basename(file.filePath);\n        return `data-files/${fileName}`;\n      }\n    });\n  }\n}\n"]}
@@ -0,0 +1,3 @@
1
+ export * from './observability';
2
+ export * from './lambda-iam-utils';
3
+ export * from './data-loader';
@@ -0,0 +1,20 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __exportStar = (this && this.__exportStar) || function(m, exports) {
14
+ for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
15
+ };
16
+ Object.defineProperty(exports, "__esModule", { value: true });
17
+ __exportStar(require("./observability"), exports);
18
+ __exportStar(require("./lambda-iam-utils"), exports);
19
+ __exportStar(require("./data-loader"), exports);
20
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi91c2UtY2FzZXMvdXRpbGl0aWVzL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7QUFBQSxrREFBZ0M7QUFDaEMscURBQW1DO0FBQ25DLGdEQUE4QiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vb2JzZXJ2YWJpbGl0eSc7XG5leHBvcnQgKiBmcm9tICcuL2xhbWJkYS1pYW0tdXRpbHMnO1xuZXhwb3J0ICogZnJvbSAnLi9kYXRhLWxvYWRlcic7XG4iXX0=