@cdklabs/cdk-appmod-catalog-blueprints 1.3.0 → 1.4.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (48) hide show
  1. package/.jsii +4 -4
  2. package/README.md +76 -98
  3. package/lib/document-processing/adapter/queued-s3-adapter.js +1 -1
  4. package/lib/document-processing/agentic-document-processing.js +1 -1
  5. package/lib/document-processing/base-document-processing.js +1 -1
  6. package/lib/document-processing/bedrock-document-processing.js +1 -1
  7. package/lib/document-processing/default-document-processing-config.js +1 -1
  8. package/lib/document-processing/tests/agentic-document-processing.test.js +104 -60
  9. package/lib/document-processing/tests/base-document-processing-nag.test.d.ts +1 -0
  10. package/lib/document-processing/tests/base-document-processing-nag.test.js +161 -0
  11. package/lib/document-processing/tests/base-document-processing.test.d.ts +1 -0
  12. package/lib/document-processing/tests/base-document-processing.test.js +499 -0
  13. package/lib/document-processing/tests/bedrock-document-processing.test.js +212 -36
  14. package/lib/document-processing/tests/queued-s3-adapter-nag.test.d.ts +1 -0
  15. package/lib/document-processing/tests/queued-s3-adapter-nag.test.js +122 -0
  16. package/lib/document-processing/tests/queued-s3-adapter.test.d.ts +1 -0
  17. package/lib/document-processing/tests/queued-s3-adapter.test.js +276 -0
  18. package/lib/framework/agents/base-agent.js +1 -1
  19. package/lib/framework/agents/batch-agent.js +1 -1
  20. package/lib/framework/agents/default-agent-config.js +1 -1
  21. package/lib/framework/bedrock/bedrock.js +1 -1
  22. package/lib/framework/custom-resource/default-runtimes.js +1 -1
  23. package/lib/framework/foundation/access-log.js +1 -1
  24. package/lib/framework/foundation/eventbridge-broker.js +1 -1
  25. package/lib/framework/foundation/network.js +1 -1
  26. package/lib/framework/tests/access-log.test.d.ts +1 -0
  27. package/lib/framework/tests/access-log.test.js +146 -0
  28. package/lib/framework/tests/batch-agent.test.d.ts +1 -0
  29. package/lib/framework/tests/batch-agent.test.js +164 -0
  30. package/lib/framework/tests/bedrock.test.d.ts +1 -0
  31. package/lib/framework/tests/bedrock.test.js +68 -0
  32. package/lib/framework/tests/eventbridge-broker.test.d.ts +1 -0
  33. package/lib/framework/tests/eventbridge-broker.test.js +73 -0
  34. package/lib/framework/tests/framework-nag.test.d.ts +1 -0
  35. package/lib/framework/tests/framework-nag.test.js +155 -0
  36. package/lib/framework/tests/network.test.d.ts +1 -0
  37. package/lib/framework/tests/network.test.js +120 -0
  38. package/lib/tsconfig.tsbuildinfo +1 -1
  39. package/lib/utilities/data-loader.js +1 -1
  40. package/lib/utilities/lambda-iam-utils.js +1 -1
  41. package/lib/utilities/observability/cloudfront-distribution-observability-property-injector.js +1 -1
  42. package/lib/utilities/observability/default-observability-config.js +1 -1
  43. package/lib/utilities/observability/lambda-observability-property-injector.js +1 -1
  44. package/lib/utilities/observability/log-group-data-protection-utils.js +1 -1
  45. package/lib/utilities/observability/powertools-config.js +1 -1
  46. package/lib/utilities/observability/state-machine-observability-property-injector.js +1 -1
  47. package/lib/webapp/frontend-construct.js +1 -1
  48. package/package.json +8 -8
@@ -0,0 +1,276 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ const aws_cdk_lib_1 = require("aws-cdk-lib");
4
+ const assertions_1 = require("aws-cdk-lib/assertions");
5
+ const aws_lambda_1 = require("aws-cdk-lib/aws-lambda");
6
+ const aws_s3_1 = require("aws-cdk-lib/aws-s3");
7
+ const aws_stepfunctions_tasks_1 = require("aws-cdk-lib/aws-stepfunctions-tasks");
8
+ const framework_1 = require("../../framework");
9
+ const adapter_1 = require("../adapter");
10
+ const base_document_processing_1 = require("../base-document-processing");
11
+ class TestDocumentProcessing extends base_document_processing_1.BaseDocumentProcessing {
12
+ constructor(scope, id, props) {
13
+ super(scope, id, props);
14
+ this.classificationFn = new aws_lambda_1.Function(this, 'ClassificationFn', {
15
+ runtime: aws_lambda_1.Runtime.NODEJS_20_X,
16
+ handler: 'index.handler',
17
+ code: aws_lambda_1.Code.fromInline('exports.handler = async () => ({ documentClassification: "TEST" });'),
18
+ });
19
+ this.processingFn = new aws_lambda_1.Function(this, 'ProcessingFn', {
20
+ runtime: aws_lambda_1.Runtime.NODEJS_20_X,
21
+ handler: 'index.handler',
22
+ code: aws_lambda_1.Code.fromInline('exports.handler = async () => ({ result: {} });'),
23
+ });
24
+ }
25
+ classificationStep() {
26
+ return new aws_stepfunctions_tasks_1.LambdaInvoke(this, 'MockClassification', {
27
+ lambdaFunction: this.classificationFn,
28
+ resultPath: '$.classificationResult',
29
+ });
30
+ }
31
+ processingStep() {
32
+ return new aws_stepfunctions_tasks_1.LambdaInvoke(this, 'MockProcessing', {
33
+ lambdaFunction: this.processingFn,
34
+ resultPath: '$.processingResult',
35
+ });
36
+ }
37
+ enrichmentStep() {
38
+ return undefined;
39
+ }
40
+ postProcessingStep() {
41
+ return undefined;
42
+ }
43
+ createStateMachine() {
44
+ return this.handleStateMachineCreation('test-state-machine');
45
+ }
46
+ }
47
+ describe('QueuedS3Adapter', () => {
48
+ let defaultStack;
49
+ let customBucketStack;
50
+ let customPrefixStack;
51
+ let customQueueStack;
52
+ let defaultTemplate;
53
+ let customBucketTemplate;
54
+ let customPrefixTemplate;
55
+ let customQueueTemplate;
56
+ beforeAll(() => {
57
+ defaultStack = new aws_cdk_lib_1.Stack();
58
+ const defaultAdapter = new adapter_1.QueuedS3Adapter();
59
+ const defaultConstruct = new TestDocumentProcessing(defaultStack, 'DefaultTest', {
60
+ ingressAdapter: defaultAdapter,
61
+ });
62
+ defaultConstruct.createStateMachine();
63
+ customBucketStack = new aws_cdk_lib_1.Stack();
64
+ const accessLog = new framework_1.AccessLog(customBucketStack, 'AccessLog');
65
+ const customBucket = new aws_s3_1.Bucket(customBucketStack, 'CustomBucket', {
66
+ serverAccessLogsBucket: accessLog.bucket,
67
+ serverAccessLogsPrefix: accessLog.bucketPrefix,
68
+ enforceSSL: true,
69
+ });
70
+ const customBucketAdapter = new adapter_1.QueuedS3Adapter({ bucket: customBucket });
71
+ const customBucketConstruct = new TestDocumentProcessing(customBucketStack, 'CustomBucketTest', {
72
+ ingressAdapter: customBucketAdapter,
73
+ });
74
+ customBucketConstruct.createStateMachine();
75
+ customPrefixStack = new aws_cdk_lib_1.Stack();
76
+ const customPrefixAdapter = new adapter_1.QueuedS3Adapter({
77
+ rawPrefix: 'input/',
78
+ processedPrefix: 'output/',
79
+ failedPrefix: 'errors/',
80
+ });
81
+ const customPrefixConstruct = new TestDocumentProcessing(customPrefixStack, 'CustomPrefixTest', {
82
+ ingressAdapter: customPrefixAdapter,
83
+ });
84
+ customPrefixConstruct.createStateMachine();
85
+ customQueueStack = new aws_cdk_lib_1.Stack();
86
+ const customQueueAdapter = new adapter_1.QueuedS3Adapter({
87
+ queueVisibilityTimeout: aws_cdk_lib_1.Duration.minutes(10),
88
+ dlqMaxReceiveCount: 10,
89
+ });
90
+ const customQueueConstruct = new TestDocumentProcessing(customQueueStack, 'CustomQueueTest', {
91
+ ingressAdapter: customQueueAdapter,
92
+ });
93
+ customQueueConstruct.createStateMachine();
94
+ defaultTemplate = assertions_1.Template.fromStack(defaultStack);
95
+ customBucketTemplate = assertions_1.Template.fromStack(customBucketStack);
96
+ customPrefixTemplate = assertions_1.Template.fromStack(customPrefixStack);
97
+ customQueueTemplate = assertions_1.Template.fromStack(customQueueStack);
98
+ });
99
+ describe('S3 bucket configuration', () => {
100
+ test('creates default S3 bucket', () => {
101
+ defaultTemplate.hasResourceProperties('AWS::S3::Bucket', {
102
+ BucketEncryption: {
103
+ ServerSideEncryptionConfiguration: [{
104
+ ServerSideEncryptionByDefault: {
105
+ SSEAlgorithm: 'aws:kms',
106
+ },
107
+ }],
108
+ },
109
+ });
110
+ });
111
+ test('uses provided custom bucket', () => {
112
+ customBucketTemplate.resourceCountIs('AWS::S3::Bucket', 2);
113
+ });
114
+ test('enables bucket key for cost optimization', () => {
115
+ defaultTemplate.hasResourceProperties('AWS::S3::Bucket', {
116
+ BucketEncryption: {
117
+ ServerSideEncryptionConfiguration: [{
118
+ BucketKeyEnabled: true,
119
+ }],
120
+ },
121
+ });
122
+ });
123
+ });
124
+ describe('SQS queue configuration', () => {
125
+ test('creates SQS queue with default visibility timeout', () => {
126
+ defaultTemplate.hasResourceProperties('AWS::SQS::Queue', {
127
+ VisibilityTimeout: 300,
128
+ });
129
+ });
130
+ test('creates dead letter queue', () => {
131
+ defaultTemplate.hasResourceProperties('AWS::SQS::Queue', {
132
+ RedrivePolicy: {
133
+ deadLetterTargetArn: assertions_1.Match.anyValue(),
134
+ maxReceiveCount: 5,
135
+ },
136
+ });
137
+ });
138
+ test('uses custom visibility timeout', () => {
139
+ customQueueTemplate.hasResourceProperties('AWS::SQS::Queue', {
140
+ VisibilityTimeout: 600,
141
+ });
142
+ });
143
+ test('uses custom DLQ max receive count', () => {
144
+ customQueueTemplate.hasResourceProperties('AWS::SQS::Queue', {
145
+ RedrivePolicy: {
146
+ maxReceiveCount: 10,
147
+ },
148
+ });
149
+ });
150
+ test('encrypts queue with KMS', () => {
151
+ defaultTemplate.hasResourceProperties('AWS::SQS::Queue', {
152
+ KmsMasterKeyId: assertions_1.Match.anyValue(),
153
+ });
154
+ });
155
+ });
156
+ describe('S3 event notifications', () => {
157
+ test('creates S3 event notification for raw prefix', () => {
158
+ defaultTemplate.hasResourceProperties('AWS::Lambda::Function', {
159
+ Environment: {
160
+ Variables: {
161
+ RAW_PREFIX: 'raw/',
162
+ },
163
+ },
164
+ });
165
+ });
166
+ test('uses custom prefix for event notification', () => {
167
+ customPrefixTemplate.hasResourceProperties('AWS::Lambda::Function', {
168
+ Environment: {
169
+ Variables: {
170
+ RAW_PREFIX: 'input/',
171
+ },
172
+ },
173
+ });
174
+ });
175
+ });
176
+ describe('SQS consumer Lambda', () => {
177
+ test('creates SQS consumer Lambda function', () => {
178
+ defaultTemplate.hasResourceProperties('AWS::Lambda::Function', {
179
+ Runtime: 'python3.13',
180
+ Timeout: 300,
181
+ Environment: {
182
+ Variables: {
183
+ STATE_MACHINE_ARN: assertions_1.Match.anyValue(),
184
+ RAW_PREFIX: 'raw/',
185
+ },
186
+ },
187
+ });
188
+ });
189
+ test('configures Lambda event source mapping', () => {
190
+ defaultTemplate.hasResourceProperties('AWS::Lambda::EventSourceMapping', {
191
+ BatchSize: 10,
192
+ MaximumBatchingWindowInSeconds: 5,
193
+ FunctionResponseTypes: ['ReportBatchItemFailures'],
194
+ });
195
+ });
196
+ test('uses custom prefix in Lambda environment', () => {
197
+ customPrefixTemplate.hasResourceProperties('AWS::Lambda::Function', {
198
+ Environment: {
199
+ Variables: {
200
+ RAW_PREFIX: 'input/',
201
+ },
202
+ },
203
+ });
204
+ });
205
+ });
206
+ describe('IAM permissions', () => {
207
+ test('grants Lambda permission to start Step Functions execution', () => {
208
+ defaultTemplate.hasResourceProperties('AWS::IAM::Role', {
209
+ Policies: assertions_1.Match.arrayWith([
210
+ assertions_1.Match.objectLike({
211
+ PolicyDocument: {
212
+ Statement: assertions_1.Match.arrayWith([
213
+ assertions_1.Match.objectLike({
214
+ Action: 'states:StartExecution',
215
+ Effect: 'Allow',
216
+ }),
217
+ ]),
218
+ },
219
+ }),
220
+ ]),
221
+ });
222
+ });
223
+ test('grants KMS decrypt permissions', () => {
224
+ defaultTemplate.hasResourceProperties('AWS::IAM::Policy', {
225
+ PolicyDocument: {
226
+ Statement: assertions_1.Match.arrayWith([
227
+ assertions_1.Match.objectLike({
228
+ Action: assertions_1.Match.arrayWith(['kms:Decrypt']),
229
+ Effect: 'Allow',
230
+ }),
231
+ ]),
232
+ },
233
+ });
234
+ });
235
+ });
236
+ describe('State machine chains', () => {
237
+ test('creates success chain with copy and delete operations', () => {
238
+ defaultTemplate.hasResourceProperties('AWS::StepFunctions::StateMachine', {
239
+ DefinitionString: assertions_1.Match.objectLike({
240
+ 'Fn::Join': assertions_1.Match.arrayWith(['']),
241
+ }),
242
+ });
243
+ });
244
+ test('creates failed chain with copy and delete operations', () => {
245
+ defaultTemplate.hasResourceProperties('AWS::StepFunctions::StateMachine', {
246
+ DefinitionString: assertions_1.Match.objectLike({
247
+ 'Fn::Join': assertions_1.Match.arrayWith(['']),
248
+ }),
249
+ });
250
+ });
251
+ test('uses custom prefixes in state machine', () => {
252
+ customPrefixTemplate.hasResourceProperties('AWS::StepFunctions::StateMachine', {
253
+ DefinitionString: assertions_1.Match.objectLike({
254
+ 'Fn::Join': assertions_1.Match.arrayWith(['']),
255
+ }),
256
+ });
257
+ });
258
+ });
259
+ describe('Removal policy', () => {
260
+ test('applies removal policy to resources', () => {
261
+ const stack = new aws_cdk_lib_1.Stack();
262
+ const adapter = new adapter_1.QueuedS3Adapter();
263
+ const construct = new TestDocumentProcessing(stack, 'RemovalTest', {
264
+ ingressAdapter: adapter,
265
+ removalPolicy: aws_cdk_lib_1.RemovalPolicy.RETAIN,
266
+ });
267
+ construct.createStateMachine();
268
+ const template = assertions_1.Template.fromStack(stack);
269
+ template.hasResource('AWS::SQS::Queue', {
270
+ DeletionPolicy: 'Retain',
271
+ UpdateReplacePolicy: 'Retain',
272
+ });
273
+ });
274
+ });
275
+ });
276
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"queued-s3-adapter.test.js","sourceRoot":"","sources":["../../../use-cases/document-processing/tests/queued-s3-adapter.test.ts"],"names":[],"mappings":"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","sourcesContent":["import { Duration, RemovalPolicy, Stack } from 'aws-cdk-lib';\nimport { Match, Template } from 'aws-cdk-lib/assertions';\nimport { Code, Function, Runtime } from 'aws-cdk-lib/aws-lambda';\nimport { Bucket } from 'aws-cdk-lib/aws-s3';\nimport { LambdaInvoke } from 'aws-cdk-lib/aws-stepfunctions-tasks';\nimport { AccessLog } from '../../framework';\nimport { QueuedS3Adapter } from '../adapter';\nimport { BaseDocumentProcessing, DocumentProcessingStepType } from '../base-document-processing';\n\nclass TestDocumentProcessing extends BaseDocumentProcessing {\n  private classificationFn: Function;\n  private processingFn: Function;\n\n  constructor(scope: any, id: string, props: any) {\n    super(scope, id, props);\n\n    this.classificationFn = new Function(this, 'ClassificationFn', {\n      runtime: Runtime.NODEJS_20_X,\n      handler: 'index.handler',\n      code: Code.fromInline('exports.handler = async () => ({ documentClassification: \"TEST\" });'),\n    });\n\n    this.processingFn = new Function(this, 'ProcessingFn', {\n      runtime: Runtime.NODEJS_20_X,\n      handler: 'index.handler',\n      code: Code.fromInline('exports.handler = async () => ({ result: {} });'),\n    });\n  }\n\n  protected classificationStep(): DocumentProcessingStepType {\n    return new LambdaInvoke(this, 'MockClassification', {\n      lambdaFunction: this.classificationFn,\n      resultPath: '$.classificationResult',\n    });\n  }\n\n  protected processingStep(): DocumentProcessingStepType {\n    return new LambdaInvoke(this, 'MockProcessing', {\n      lambdaFunction: this.processingFn,\n      resultPath: '$.processingResult',\n    });\n  }\n\n  protected enrichmentStep(): DocumentProcessingStepType | undefined {\n    return undefined;\n  }\n\n  protected postProcessingStep(): DocumentProcessingStepType | undefined {\n    return undefined;\n  }\n\n  public createStateMachine() {\n    return this.handleStateMachineCreation('test-state-machine');\n  }\n}\n\ndescribe('QueuedS3Adapter', () => {\n  let defaultStack: Stack;\n  let customBucketStack: Stack;\n  let customPrefixStack: Stack;\n  let customQueueStack: Stack;\n  let defaultTemplate: Template;\n  let customBucketTemplate: Template;\n  let customPrefixTemplate: Template;\n  let customQueueTemplate: Template;\n\n  beforeAll(() => {\n    defaultStack = new Stack();\n    const defaultAdapter = new QueuedS3Adapter();\n    const defaultConstruct = new TestDocumentProcessing(defaultStack, 'DefaultTest', {\n      ingressAdapter: defaultAdapter,\n    });\n    defaultConstruct.createStateMachine();\n\n    customBucketStack = new Stack();\n    const accessLog = new AccessLog(customBucketStack, 'AccessLog');\n    const customBucket = new Bucket(customBucketStack, 'CustomBucket', {\n      serverAccessLogsBucket: accessLog.bucket,\n      serverAccessLogsPrefix: accessLog.bucketPrefix,\n      enforceSSL: true,\n    });\n    const customBucketAdapter = new QueuedS3Adapter({ bucket: customBucket });\n    const customBucketConstruct = new TestDocumentProcessing(customBucketStack, 'CustomBucketTest', {\n      ingressAdapter: customBucketAdapter,\n    });\n    customBucketConstruct.createStateMachine();\n\n    customPrefixStack = new Stack();\n    const customPrefixAdapter = new QueuedS3Adapter({\n      rawPrefix: 'input/',\n      processedPrefix: 'output/',\n      failedPrefix: 'errors/',\n    });\n    const customPrefixConstruct = new TestDocumentProcessing(customPrefixStack, 'CustomPrefixTest', {\n      ingressAdapter: customPrefixAdapter,\n    });\n    customPrefixConstruct.createStateMachine();\n\n    customQueueStack = new Stack();\n    const customQueueAdapter = new QueuedS3Adapter({\n      queueVisibilityTimeout: Duration.minutes(10),\n      dlqMaxReceiveCount: 10,\n    });\n    const customQueueConstruct = new TestDocumentProcessing(customQueueStack, 'CustomQueueTest', {\n      ingressAdapter: customQueueAdapter,\n    });\n    customQueueConstruct.createStateMachine();\n\n    defaultTemplate = Template.fromStack(defaultStack);\n    customBucketTemplate = Template.fromStack(customBucketStack);\n    customPrefixTemplate = Template.fromStack(customPrefixStack);\n    customQueueTemplate = Template.fromStack(customQueueStack);\n  });\n\n  describe('S3 bucket configuration', () => {\n    test('creates default S3 bucket', () => {\n      defaultTemplate.hasResourceProperties('AWS::S3::Bucket', {\n        BucketEncryption: {\n          ServerSideEncryptionConfiguration: [{\n            ServerSideEncryptionByDefault: {\n              SSEAlgorithm: 'aws:kms',\n            },\n          }],\n        },\n      });\n    });\n\n    test('uses provided custom bucket', () => {\n      customBucketTemplate.resourceCountIs('AWS::S3::Bucket', 2);\n    });\n\n    test('enables bucket key for cost optimization', () => {\n      defaultTemplate.hasResourceProperties('AWS::S3::Bucket', {\n        BucketEncryption: {\n          ServerSideEncryptionConfiguration: [{\n            BucketKeyEnabled: true,\n          }],\n        },\n      });\n    });\n  });\n\n  describe('SQS queue configuration', () => {\n    test('creates SQS queue with default visibility timeout', () => {\n      defaultTemplate.hasResourceProperties('AWS::SQS::Queue', {\n        VisibilityTimeout: 300,\n      });\n    });\n\n    test('creates dead letter queue', () => {\n      defaultTemplate.hasResourceProperties('AWS::SQS::Queue', {\n        RedrivePolicy: {\n          deadLetterTargetArn: Match.anyValue(),\n          maxReceiveCount: 5,\n        },\n      });\n    });\n\n    test('uses custom visibility timeout', () => {\n      customQueueTemplate.hasResourceProperties('AWS::SQS::Queue', {\n        VisibilityTimeout: 600,\n      });\n    });\n\n    test('uses custom DLQ max receive count', () => {\n      customQueueTemplate.hasResourceProperties('AWS::SQS::Queue', {\n        RedrivePolicy: {\n          maxReceiveCount: 10,\n        },\n      });\n    });\n\n    test('encrypts queue with KMS', () => {\n      defaultTemplate.hasResourceProperties('AWS::SQS::Queue', {\n        KmsMasterKeyId: Match.anyValue(),\n      });\n    });\n  });\n\n  describe('S3 event notifications', () => {\n    test('creates S3 event notification for raw prefix', () => {\n      defaultTemplate.hasResourceProperties('AWS::Lambda::Function', {\n        Environment: {\n          Variables: {\n            RAW_PREFIX: 'raw/',\n          },\n        },\n      });\n    });\n\n    test('uses custom prefix for event notification', () => {\n      customPrefixTemplate.hasResourceProperties('AWS::Lambda::Function', {\n        Environment: {\n          Variables: {\n            RAW_PREFIX: 'input/',\n          },\n        },\n      });\n    });\n  });\n\n  describe('SQS consumer Lambda', () => {\n    test('creates SQS consumer Lambda function', () => {\n      defaultTemplate.hasResourceProperties('AWS::Lambda::Function', {\n        Runtime: 'python3.13',\n        Timeout: 300,\n        Environment: {\n          Variables: {\n            STATE_MACHINE_ARN: Match.anyValue(),\n            RAW_PREFIX: 'raw/',\n          },\n        },\n      });\n    });\n\n    test('configures Lambda event source mapping', () => {\n      defaultTemplate.hasResourceProperties('AWS::Lambda::EventSourceMapping', {\n        BatchSize: 10,\n        MaximumBatchingWindowInSeconds: 5,\n        FunctionResponseTypes: ['ReportBatchItemFailures'],\n      });\n    });\n\n    test('uses custom prefix in Lambda environment', () => {\n      customPrefixTemplate.hasResourceProperties('AWS::Lambda::Function', {\n        Environment: {\n          Variables: {\n            RAW_PREFIX: 'input/',\n          },\n        },\n      });\n    });\n  });\n\n  describe('IAM permissions', () => {\n    test('grants Lambda permission to start Step Functions execution', () => {\n      defaultTemplate.hasResourceProperties('AWS::IAM::Role', {\n        Policies: Match.arrayWith([\n          Match.objectLike({\n            PolicyDocument: {\n              Statement: Match.arrayWith([\n                Match.objectLike({\n                  Action: 'states:StartExecution',\n                  Effect: 'Allow',\n                }),\n              ]),\n            },\n          }),\n        ]),\n      });\n    });\n\n    test('grants KMS decrypt permissions', () => {\n      defaultTemplate.hasResourceProperties('AWS::IAM::Policy', {\n        PolicyDocument: {\n          Statement: Match.arrayWith([\n            Match.objectLike({\n              Action: Match.arrayWith(['kms:Decrypt']),\n              Effect: 'Allow',\n            }),\n          ]),\n        },\n      });\n    });\n  });\n\n  describe('State machine chains', () => {\n    test('creates success chain with copy and delete operations', () => {\n      defaultTemplate.hasResourceProperties('AWS::StepFunctions::StateMachine', {\n        DefinitionString: Match.objectLike({\n          'Fn::Join': Match.arrayWith(['']),\n        }),\n      });\n    });\n\n    test('creates failed chain with copy and delete operations', () => {\n      defaultTemplate.hasResourceProperties('AWS::StepFunctions::StateMachine', {\n        DefinitionString: Match.objectLike({\n          'Fn::Join': Match.arrayWith(['']),\n        }),\n      });\n    });\n\n    test('uses custom prefixes in state machine', () => {\n      customPrefixTemplate.hasResourceProperties('AWS::StepFunctions::StateMachine', {\n        DefinitionString: Match.objectLike({\n          'Fn::Join': Match.arrayWith(['']),\n        }),\n      });\n    });\n  });\n\n  describe('Removal policy', () => {\n    test('applies removal policy to resources', () => {\n      const stack = new Stack();\n      const adapter = new QueuedS3Adapter();\n      const construct = new TestDocumentProcessing(stack, 'RemovalTest', {\n        ingressAdapter: adapter,\n        removalPolicy: RemovalPolicy.RETAIN,\n      });\n      construct.createStateMachine();\n      const template = Template.fromStack(stack);\n\n      template.hasResource('AWS::SQS::Queue', {\n        DeletionPolicy: 'Retain',\n        UpdateReplacePolicy: 'Retain',\n      });\n    });\n  });\n});\n"]}
@@ -51,5 +51,5 @@ class BaseAgent extends constructs_1.Construct {
51
51
  }
52
52
  exports.BaseAgent = BaseAgent;
53
53
  _a = JSII_RTTI_SYMBOL_1;
54
- BaseAgent[_a] = { fqn: "@cdklabs/cdk-appmod-catalog-blueprints.BaseAgent", version: "1.3.0" };
54
+ BaseAgent[_a] = { fqn: "@cdklabs/cdk-appmod-catalog-blueprints.BaseAgent", version: "1.4.0" };
55
55
  //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"base-agent.js","sourceRoot":"","sources":["../../../use-cases/framework/agents/base-agent.ts"],"names":[],"mappings":";;;;;AAIA,6CAA+D;AAC/D,iDAA8F;AAC9F,iDAA0C;AAG1C,2CAAuC;AACvC,+CAAiK;AACjK,wCAAkE;AAsFlE,MAAsB,SAAU,SAAQ,sBAAS;IAS/C,YAAY,KAAgB,EAAE,EAAU,EAAE,KAAqB;QAC7D,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QACjB,IAAI,CAAC,YAAY,GAAG,KAAK,CAAC,eAAe,CAAC,YAAY,CAAC;QACvD,IAAI,CAAC,aAAa,GAAG,KAAK,CAAC,aAAa,IAAI,IAAI,aAAG,CAAC,IAAI,EAAE,oBAAoB,EAAE;YAC9E,iBAAiB,EAAE,IAAI;YACvB,aAAa,EAAE,KAAK,CAAC,aAAa,IAAI,2BAAa,CAAC,OAAO;SAC5D,CAAC,CAAC;QAGH,MAAM,cAAc,GAAmC,EAAE,CAAC;QAE1D,IAAI,KAAK,CAAC,eAAe,CAAC,kCAAkC,IAAI,KAAK,CAAC,eAAe,CAAC,kCAAkC,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACpI,cAAc,CAAC,eAAe,GAAG,IAAI,wBAAc,CAAC;gBAClD,UAAU,EAAE,KAAK,CAAC,eAAe,CAAC,kCAAkC;aACrE,CAAC,CAAC;QACL,CAAC;QAED,IAAI,CAAC,SAAS,GAAG,IAAI,cAAI,CAAC,IAAI,EAAE,SAAS,KAAK,CAAC,SAAS,OAAO,EAAE;YAC/D,SAAS,EAAE,IAAI,0BAAgB,CAAC,sBAAsB,CAAC;YACvD,cAAc;SACf,CAAC,CAAC;QAEH,IAAI,KAAK,CAAC,OAAO,EAAE,CAAC;YAClB,IAAI,CAAC,SAAS,CAAC,oBAAoB,CAAC,0BAAc,CAAC,4BAA4B,EAAE,CAAC,CAAC;QACrF,CAAC;QAED,IAAI,CAAC,6BAA6B,GAAG,EAAE,CAAC;QAExC,IAAI,KAAK,CAAC,eAAe,CAAC,KAAK,EAAE,CAAC;YAChC,KAAK,MAAM,IAAI,IAAI,KAAK,CAAC,eAAe,CAAC,KAAK,EAAE,CAAC;gBAC/C,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;gBAE/B,IAAI,CAAC,6BAA6B,CAAC,IAAI,CAAC;oBACtC,UAAU,EAAE,IAAI,CAAC,YAAY;oBAC7B,GAAG,EAAE,IAAI,CAAC,WAAW;oBACrB,MAAM,EAAE,IAAI,CAAC,MAAM;oBACnB,YAAY,EAAE,IAAI,CAAC,YAAY;iBAChC,CAAC,CAAC;YACL,CAAC;QACH,CAAC;QAED,IAAI,CAAC,SAAS,CAAC,oBAAoB,CAAC,2BAAiB,CAAC,2BAA2B,CAAC,IAAI,EAAE,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC;QAC5G,IAAI,CAAC,sBAAsB,GAAG,uCAA2B,CAAC,aAAa,CAAC,IAAI,EAAE,KAAK,CAAC,sBAAsB,EAAE,KAAK,CAAC,aAAa,CAAC,CAAC;QAEjI,IAAI,KAAK,CAAC,mBAAmB,EAAE,CAAC;YAC9B,+BAAiB,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,GAAG,CAC5B,IAAI,+CAAmC,CAAC,IAAI,CAAC,sBAAsB,CAAC,CACrE,CAAC;QACJ,CAAC;IACH,CAAC;;AA1DH,8BA2DC","sourcesContent":["// Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.\n// SPDX-License-Identifier: Apache-2.0\n\nimport { PythonFunction } from '@aws-cdk/aws-lambda-python-alpha';\nimport { PropertyInjectors, RemovalPolicy } from 'aws-cdk-lib';\nimport { PolicyDocument, PolicyStatement, Role, ServicePrincipal } from 'aws-cdk-lib/aws-iam';\nimport { Key } from 'aws-cdk-lib/aws-kms';\nimport { LayerVersion } from 'aws-cdk-lib/aws-lambda';\nimport { Asset } from 'aws-cdk-lib/aws-s3-assets';\nimport { Construct } from 'constructs';\nimport { LambdaIamUtils, LambdaObservabilityPropertyInjector, LogGroupDataProtectionProps, LogGroupDataProtectionUtils, ObservableProps } from '../../utilities';\nimport { BedrockModelProps, BedrockModelUtils } from '../bedrock';\nimport { Network } from '../foundation';\n\nexport interface AgentToolsLocationDefinition {\n  readonly bucketName: string;\n  readonly key: string;\n  readonly isFile: boolean;\n  readonly isZipArchive: boolean;\n}\n\n/**\n * Parameters that influences the behavior of the agent\n */\nexport interface AgentDefinitionProps {\n  /**\n     * Configuration for the Bedrock Model to be used\n     */\n  readonly bedrockModel: BedrockModelProps;\n\n  /**\n     * The system prompt of the agent\n     *\n     */\n  readonly systemPrompt: Asset;\n\n  /**\n     * List of tools defined in python files. This tools would automatically\n     * be loaded by the agent. You can also use this to incorporate other specialized\n     * agents as tools.\n     */\n  readonly tools?: Asset[];\n\n  /**\n     * Any dependencies needed by the provided tools\n     */\n  readonly lambdaLayers?: LayerVersion[];\n\n  /**\n     * If tools need additional IAM permissions, these statements\n     * would be attached to the Agent's IAM role\n     */\n  readonly additionalPolicyStatementsForTools?: PolicyStatement[];\n}\n\nexport interface BaseAgentProps extends ObservableProps {\n\n  /**\n     * Name of the agent\n     */\n  readonly agentName: string;\n\n  /**\n     * Agent related parameters\n     */\n  readonly agentDefinition: AgentDefinitionProps;\n\n  /**\n     * Enable observability\n     *\n     * @default false\n     */\n  readonly enableObservability?: boolean;\n\n  /**\n     * If the Agent would be running inside a VPC\n     *\n     * @default Agent would not be in a VPC\n     */\n  readonly network?: Network;\n\n  /**\n     * Encryption key to encrypt agent environment variables\n     *\n     * @default new KMS Key would be created\n     */\n  readonly encryptionKey?: Key;\n\n  /**\n     * Removal policy for resources created by this\n     * construct\n     *\n     * @default RemovalPolicy.DESTROY\n     */\n  readonly removalPolicy?: RemovalPolicy;\n}\n\nexport abstract class BaseAgent extends Construct {\n  public abstract readonly agentFunction: PythonFunction;\n  public readonly bedrockModel?: BedrockModelProps;\n  public readonly agentRole: Role;\n  public readonly encryptionKey: Key;\n  /** log group data protection configuration */\n  protected readonly logGroupDataProtection: LogGroupDataProtectionProps;\n  protected readonly agentToolsLocationDefinitions: AgentToolsLocationDefinition[];\n\n  constructor(scope: Construct, id: string, props: BaseAgentProps) {\n    super(scope, id);\n    this.bedrockModel = props.agentDefinition.bedrockModel;\n    this.encryptionKey = props.encryptionKey || new Key(this, 'AgentEncryptionKey', {\n      enableKeyRotation: true,\n      removalPolicy: props.removalPolicy || RemovalPolicy.DESTROY,\n    });\n\n\n    const inlinePolicies: Record<string, PolicyDocument> = {};\n\n    if (props.agentDefinition.additionalPolicyStatementsForTools && props.agentDefinition.additionalPolicyStatementsForTools.length > 0) {\n      inlinePolicies.ToolPermissions = new PolicyDocument({\n        statements: props.agentDefinition.additionalPolicyStatementsForTools,\n      });\n    }\n\n    this.agentRole = new Role(this, `Agent-${props.agentName}-Role`, {\n      assumedBy: new ServicePrincipal('lambda.amazonaws.com'),\n      inlinePolicies,\n    });\n\n    if (props.network) {\n      this.agentRole.addToPrincipalPolicy(LambdaIamUtils.generateLambdaVPCPermissions());\n    }\n\n    this.agentToolsLocationDefinitions = [];\n\n    if (props.agentDefinition.tools) {\n      for (const tool of props.agentDefinition.tools) {\n        tool.grantRead(this.agentRole);\n\n        this.agentToolsLocationDefinitions.push({\n          bucketName: tool.s3BucketName,\n          key: tool.s3ObjectKey,\n          isFile: tool.isFile,\n          isZipArchive: tool.isZipArchive,\n        });\n      }\n    }\n\n    this.agentRole.addToPrincipalPolicy(BedrockModelUtils.generateModelIAMPermissions(this, this.bedrockModel));\n    this.logGroupDataProtection = LogGroupDataProtectionUtils.handleDefault(this, props.logGroupDataProtection, props.removalPolicy);\n\n    if (props.enableObservability) {\n      PropertyInjectors.of(this).add(\n        new LambdaObservabilityPropertyInjector(this.logGroupDataProtection),\n      );\n    }\n  }\n}"]}
@@ -60,5 +60,5 @@ class BatchAgent extends base_agent_1.BaseAgent {
60
60
  }
61
61
  exports.BatchAgent = BatchAgent;
62
62
  _a = JSII_RTTI_SYMBOL_1;
63
- BatchAgent[_a] = { fqn: "@cdklabs/cdk-appmod-catalog-blueprints.BatchAgent", version: "1.3.0" };
63
+ BatchAgent[_a] = { fqn: "@cdklabs/cdk-appmod-catalog-blueprints.BatchAgent", version: "1.4.0" };
64
64
  //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYmF0Y2gtYWdlbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi91c2UtY2FzZXMvZnJhbWV3b3JrL2FnZW50cy9iYXRjaC1hZ2VudC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7OztBQUFBLHFFQUFxRTtBQUNyRSxzQ0FBc0M7QUFFdEMsNkJBQTZCO0FBQzdCLDhFQUFrRTtBQUNsRSw2Q0FBOEM7QUFDOUMsdURBQXNEO0FBRXRELDZDQUF5RDtBQUN6RCwrQ0FBK0Y7QUFDL0Ysd0NBQStDO0FBQy9DLHdEQUFxRDtBQUNyRCxpRUFBNEQ7QUFPNUQsTUFBYSxVQUFXLFNBQVEsc0JBQVM7SUFHdkMsWUFBWSxLQUFnQixFQUFFLEVBQVUsRUFBRSxLQUFzQjtRQUM5RCxLQUFLLENBQUMsS0FBSyxFQUFFLEVBQUUsRUFBRSxLQUFLLENBQUMsQ0FBQztRQUN4QixNQUFNLE9BQU8sR0FBRywyQkFBaUIsQ0FBQyxtQkFBbUIsQ0FBQyxJQUFJLENBQUMsWUFBWSxDQUFDLENBQUM7UUFDekUsTUFBTSxlQUFlLEdBQUcsS0FBSyxDQUFDLGVBQWUsSUFBSSxzQ0FBMEIsQ0FBQyx3QkFBd0IsQ0FBQztRQUNyRyxNQUFNLGlCQUFpQixHQUFHLEtBQUssQ0FBQyxpQkFBaUIsSUFBSSx5Q0FBa0IsQ0FBQyxxQ0FBcUMsQ0FBQztRQUU5RyxNQUFNLEdBQUcsR0FBMkI7WUFDbEMsNEJBQTRCLEVBQUUsS0FBSyxDQUFDLGVBQWUsQ0FBQyxZQUFZLENBQUMsWUFBWTtZQUM3RSxvQkFBb0IsRUFBRSxLQUFLLENBQUMsZUFBZSxDQUFDLFlBQVksQ0FBQyxXQUFXO1lBQ3BFLFlBQVksRUFBRSxJQUFJLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyw2QkFBNkIsQ0FBQztZQUNoRSxRQUFRLEVBQUUsT0FBTztZQUNqQixXQUFXLEVBQUUsT0FBTztZQUNwQixNQUFNLEVBQUUsS0FBSyxDQUFDLE1BQU07WUFDcEIsV0FBVyxFQUFFLEtBQUssQ0FBQyxVQUFVLENBQUMsQ0FBQyxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsRUFBRTtZQUMzQyxHQUFHLDRCQUFnQixDQUFDLDJCQUEyQixDQUM3QyxLQUFLLENBQUMsbUJBQW1CLEVBQ3pCLGVBQWUsRUFDZixpQkFBaUIsQ0FDbEI7U0FDRixDQUFDO1FBRUYsTUFBTSxFQUFFLE9BQU8sRUFBRSxNQUFNLEVBQUUsR0FBRyxtQkFBSyxDQUFDLEVBQUUsQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUMzQyxNQUFNLCtCQUErQixHQUFHLDBCQUFjLENBQUMscUJBQXFCLENBQUM7WUFDM0UsT0FBTztZQUNQLE1BQU07WUFDTixLQUFLLEVBQUUsSUFBSTtZQUNYLFlBQVksRUFBRSxLQUFLLENBQUMsU0FBUztZQUM3QixtQkFBbUIsRUFBRSxLQUFLLENBQUMsbUJBQW1CO1NBQy9DLENBQUMsQ0FBQztRQUVILElBQUksQ0FBQyxhQUFhLEdBQUcsSUFBSSx3Q0FBYyxDQUFDLElBQUksRUFBRSxvQkFBb0IsRUFBRTtZQUNsRSxZQUFZLEVBQUUsK0JBQStCLENBQUMsa0JBQWtCO1lBQ2hFLFlBQVksRUFBRSx5QkFBWSxDQUFDLE1BQU07WUFDakMsS0FBSyxFQUFFLElBQUksQ0FBQyxJQUFJLENBQUMsU0FBUyxFQUFFLGlDQUFpQyxDQUFDO1lBQzlELElBQUksRUFBRSxJQUFJLENBQUMsU0FBUztZQUNwQixLQUFLLEVBQUUsVUFBVTtZQUNqQixPQUFPLEVBQUUsaUNBQWUsQ0FBQyxNQUFNO1lBQy9CLE1BQU0sRUFBRSxLQUFLLENBQUMsZUFBZSxDQUFDLFlBQVk7WUFDMUMsT0FBTyxFQUFFLHNCQUFRLENBQUMsT0FBTyxDQUFDLEVBQUUsQ0FBQztZQUM3QixVQUFVLEVBQUUsSUFBSTtZQUNoQixXQUFXLEVBQUUsR0FBRztZQUNoQixxQkFBcUIsRUFBRSxJQUFJLENBQUMsYUFBYTtZQUN6QyxHQUFHLEVBQUUsS0FBSyxDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUMsS0FBSyxDQUFDLE9BQU8sQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFDLFNBQVM7WUFDbEQsVUFBVSxFQUFFLEtBQUssQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFDLEtBQUssQ0FBQyxPQUFPLENBQUMsMEJBQTBCLEVBQUUsQ0FBQyxDQUFDLENBQUMsU0FBUztTQUNuRixDQUFDLENBQUM7UUFFSCxLQUFLLE1BQU0sQ0FBQyxJQUFJLCtCQUErQixDQUFDLGdCQUFnQixFQUFFLENBQUM7WUFDakUsSUFBSSxDQUFDLFNBQVMsQ0FBQyxvQkFBb0IsQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUN6QyxDQUFDO0lBQ0gsQ0FBQzs7QUFwREgsZ0NBcURDIiwic291cmNlc0NvbnRlbnQiOlsiLy8gQ29weXJpZ2h0IEFtYXpvbi5jb20sIEluYy4gb3IgaXRzIGFmZmlsaWF0ZXMuIEFsbCBSaWdodHMgUmVzZXJ2ZWQuXG4vLyBTUERYLUxpY2Vuc2UtSWRlbnRpZmllcjogQXBhY2hlLTIuMFxuXG5pbXBvcnQgKiBhcyBwYXRoIGZyb20gJ3BhdGgnO1xuaW1wb3J0IHsgUHl0aG9uRnVuY3Rpb24gfSBmcm9tICdAYXdzLWNkay9hd3MtbGFtYmRhLXB5dGhvbi1hbHBoYSc7XG5pbXBvcnQgeyBEdXJhdGlvbiwgU3RhY2sgfSBmcm9tICdhd3MtY2RrLWxpYic7XG5pbXBvcnQgeyBBcmNoaXRlY3R1cmUgfSBmcm9tICdhd3MtY2RrLWxpYi9hd3MtbGFtYmRhJztcbmltcG9ydCB7IENvbnN0cnVjdCB9IGZyb20gJ2NvbnN0cnVjdHMnO1xuaW1wb3J0IHsgQmFzZUFnZW50LCBCYXNlQWdlbnRQcm9wcyB9IGZyb20gJy4vYmFzZS1hZ2VudCc7XG5pbXBvcnQgeyBEZWZhdWx0T2JzZXJ2YWJpbGl0eUNvbmZpZywgTGFtYmRhSWFtVXRpbHMsIFBvd2VydG9vbHNDb25maWcgfSBmcm9tICcuLi8uLi91dGlsaXRpZXMnO1xuaW1wb3J0IHsgQmVkcm9ja01vZGVsVXRpbHMgfSBmcm9tICcuLi9iZWRyb2NrJztcbmltcG9ydCB7IERlZmF1bHRSdW50aW1lcyB9IGZyb20gJy4uL2N1c3RvbS1yZXNvdXJjZSc7XG5pbXBvcnQgeyBEZWZhdWx0QWdlbnRDb25maWcgfSBmcm9tICcuL2RlZmF1bHQtYWdlbnQtY29uZmlnJztcblxuZXhwb3J0IGludGVyZmFjZSBCYXRjaEFnZW50UHJvcHMgZXh0ZW5kcyBCYXNlQWdlbnRQcm9wcyB7XG4gIHJlYWRvbmx5IHByb21wdDogc3RyaW5nO1xuICByZWFkb25seSBleHBlY3RKc29uPzogYm9vbGVhbjtcbn1cblxuZXhwb3J0IGNsYXNzIEJhdGNoQWdlbnQgZXh0ZW5kcyBCYXNlQWdlbnQge1xuICBwdWJsaWMgcmVhZG9ubHkgYWdlbnRGdW5jdGlvbjogUHl0aG9uRnVuY3Rpb247XG5cbiAgY29uc3RydWN0b3Ioc2NvcGU6IENvbnN0cnVjdCwgaWQ6IHN0cmluZywgcHJvcHM6IEJhdGNoQWdlbnRQcm9wcykge1xuICAgIHN1cGVyKHNjb3BlLCBpZCwgcHJvcHMpO1xuICAgIGNvbnN0IG1vZGVsSWQgPSBCZWRyb2NrTW9kZWxVdGlscy5kZXJpdmVBY3R1YWxNb2RlbElkKHRoaXMuYmVkcm9ja01vZGVsKTtcbiAgICBjb25zdCBtZXRyaWNOYW1lc3BhY2UgPSBwcm9wcy5tZXRyaWNOYW1lc3BhY2UgfHwgRGVmYXVsdE9ic2VydmFiaWxpdHlDb25maWcuREVGQVVMVF9NRVRSSUNfTkFNRVNQQUNFO1xuICAgIGNvbnN0IG1ldHJpY1NlcnZpY2VOYW1lID0gcHJvcHMubWV0cmljU2VydmljZU5hbWUgfHwgRGVmYXVsdEFnZW50Q29uZmlnLkRFRkFVTFRfT0JTRVJWQUJJTElUWV9NRVRSSUNfU1ZDX05BTUU7XG5cbiAgICBjb25zdCBlbnY6IFJlY29yZDxzdHJpbmcsIHN0cmluZz4gPSB7XG4gICAgICBTWVNURU1fUFJPTVBUX1MzX0JVQ0tFVF9OQU1FOiBwcm9wcy5hZ2VudERlZmluaXRpb24uc3lzdGVtUHJvbXB0LnMzQnVja2V0TmFtZSxcbiAgICAgIFNZU1RFTV9QUk9NUFRfUzNfS0VZOiBwcm9wcy5hZ2VudERlZmluaXRpb24uc3lzdGVtUHJvbXB0LnMzT2JqZWN0S2V5LFxuICAgICAgVE9PTFNfQ09ORklHOiBKU09OLnN0cmluZ2lmeSh0aGlzLmFnZW50VG9vbHNMb2NhdGlvbkRlZmluaXRpb25zKSxcbiAgICAgIE1PREVMX0lEOiBtb2RlbElkLFxuICAgICAgSU5WT0tFX1RZUEU6ICdiYXRjaCcsXG4gICAgICBQUk9NUFQ6IHByb3BzLnByb21wdCxcbiAgICAgIEVYUEVDVF9KU09OOiBwcm9wcy5leHBlY3RKc29uID8gJ1RydWUnIDogJycsXG4gICAgICAuLi5Qb3dlcnRvb2xzQ29uZmlnLmdlbmVyYXRlRGVmYXVsdExhbWJkYUNvbmZpZyhcbiAgICAgICAgcHJvcHMuZW5hYmxlT2JzZXJ2YWJpbGl0eSxcbiAgICAgICAgbWV0cmljTmFtZXNwYWNlLFxuICAgICAgICBtZXRyaWNTZXJ2aWNlTmFtZSxcbiAgICAgICksXG4gICAgfTtcblxuICAgIGNvbnN0IHsgYWNjb3VudCwgcmVnaW9uIH0gPSBTdGFjay5vZih0aGlzKTtcbiAgICBjb25zdCBhZ2VudExhbWJkYUxvZ1Blcm1pc3Npb25zUmVzdWx0ID0gTGFtYmRhSWFtVXRpbHMuY3JlYXRlTG9nc1Blcm1pc3Npb25zKHtcbiAgICAgIGFjY291bnQsXG4gICAgICByZWdpb24sXG4gICAgICBzY29wZTogdGhpcyxcbiAgICAgIGZ1bmN0aW9uTmFtZTogcHJvcHMuYWdlbnROYW1lLFxuICAgICAgZW5hYmxlT2JzZXJ2YWJpbGl0eTogcHJvcHMuZW5hYmxlT2JzZXJ2YWJpbGl0eSxcbiAgICB9KTtcblxuICAgIHRoaXMuYWdlbnRGdW5jdGlvbiA9IG5ldyBQeXRob25GdW5jdGlvbih0aGlzLCAnQmF0Y2hBZ2VudEZ1bmN0aW9uJywge1xuICAgICAgZnVuY3Rpb25OYW1lOiBhZ2VudExhbWJkYUxvZ1Blcm1pc3Npb25zUmVzdWx0LnVuaXF1ZUZ1bmN0aW9uTmFtZSxcbiAgICAgIGFyY2hpdGVjdHVyZTogQXJjaGl0ZWN0dXJlLlg4Nl82NCxcbiAgICAgIGVudHJ5OiBwYXRoLmpvaW4oX19kaXJuYW1lLCAncmVzb3VyY2VzL2RlZmF1bHQtc3RyYW5kcy1hZ2VudCcpLFxuICAgICAgcm9sZTogdGhpcy5hZ2VudFJvbGUsXG4gICAgICBpbmRleDogJ2JhdGNoLnB5JyxcbiAgICAgIHJ1bnRpbWU6IERlZmF1bHRSdW50aW1lcy5QWVRIT04sXG4gICAgICBsYXllcnM6IHByb3BzLmFnZW50RGVmaW5pdGlvbi5sYW1iZGFMYXllcnMsXG4gICAgICB0aW1lb3V0OiBEdXJhdGlvbi5taW51dGVzKDEwKSxcbiAgICAgIG1lbW9yeVNpemU6IDEwMjQsXG4gICAgICBlbnZpcm9ubWVudDogZW52LFxuICAgICAgZW52aXJvbm1lbnRFbmNyeXB0aW9uOiB0aGlzLmVuY3J5cHRpb25LZXksXG4gICAgICB2cGM6IHByb3BzLm5ldHdvcmsgPyBwcm9wcy5uZXR3b3JrLnZwYyA6IHVuZGVmaW5lZCxcbiAgICAgIHZwY1N1Ym5ldHM6IHByb3BzLm5ldHdvcmsgPyBwcm9wcy5uZXR3b3JrLmFwcGxpY2F0aW9uU3VibmV0U2VsZWN0aW9uKCkgOiB1bmRlZmluZWQsXG4gICAgfSk7XG5cbiAgICBmb3IgKGNvbnN0IHMgb2YgYWdlbnRMYW1iZGFMb2dQZXJtaXNzaW9uc1Jlc3VsdC5wb2xpY3lTdGF0ZW1lbnRzKSB7XG4gICAgICB0aGlzLmFnZW50Um9sZS5hZGRUb1ByaW5jaXBhbFBvbGljeShzKTtcbiAgICB9XG4gIH1cbn0iXX0=
@@ -7,6 +7,6 @@ class DefaultAgentConfig {
7
7
  }
8
8
  exports.DefaultAgentConfig = DefaultAgentConfig;
9
9
  _a = JSII_RTTI_SYMBOL_1;
10
- DefaultAgentConfig[_a] = { fqn: "@cdklabs/cdk-appmod-catalog-blueprints.DefaultAgentConfig", version: "1.3.0" };
10
+ DefaultAgentConfig[_a] = { fqn: "@cdklabs/cdk-appmod-catalog-blueprints.DefaultAgentConfig", version: "1.4.0" };
11
11
  DefaultAgentConfig.DEFAULT_OBSERVABILITY_METRIC_SVC_NAME = 'agent';
12
12
  //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGVmYXVsdC1hZ2VudC1jb25maWcuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi91c2UtY2FzZXMvZnJhbWV3b3JrL2FnZW50cy9kZWZhdWx0LWFnZW50LWNvbmZpZy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7OztBQUFBLE1BQWEsa0JBQWtCOztBQUEvQixnREFFQzs7O0FBRHdCLHdEQUFxQyxHQUFHLE9BQU8sQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCBjbGFzcyBEZWZhdWx0QWdlbnRDb25maWcge1xuICBwdWJsaWMgc3RhdGljIHJlYWRvbmx5IERFRkFVTFRfT0JTRVJWQUJJTElUWV9NRVRSSUNfU1ZDX05BTUUgPSAnYWdlbnQnO1xufSJdfQ==
@@ -50,5 +50,5 @@ class BedrockModelUtils {
50
50
  }
51
51
  exports.BedrockModelUtils = BedrockModelUtils;
52
52
  _a = JSII_RTTI_SYMBOL_1;
53
- BedrockModelUtils[_a] = { fqn: "@cdklabs/cdk-appmod-catalog-blueprints.BedrockModelUtils", version: "1.3.0" };
53
+ BedrockModelUtils[_a] = { fqn: "@cdklabs/cdk-appmod-catalog-blueprints.BedrockModelUtils", version: "1.4.0" };
54
54
  //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYmVkcm9jay5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3VzZS1jYXNlcy9mcmFtZXdvcmsvYmVkcm9jay9iZWRyb2NrLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7O0FBQUEscUVBQXFFO0FBQ3JFLHNDQUFzQztBQUV0Qyw2Q0FBb0M7QUFDcEMseURBQW9FO0FBQ3BFLGlEQUE4RDtBQUc5RDs7O0dBR0c7QUFDSCxJQUFZLGlDQUtYO0FBTEQsV0FBWSxpQ0FBaUM7SUFDM0MsOENBQThDO0lBQzlDLDhDQUFTLENBQUE7SUFDVCw4Q0FBOEM7SUFDOUMsOENBQVMsQ0FBQTtBQUNYLENBQUMsRUFMVyxpQ0FBaUMsaURBQWpDLGlDQUFpQyxRQUs1QztBQXdCRCxNQUFhLGlCQUFpQjtJQUNyQixNQUFNLENBQUMsbUJBQW1CLENBQUMsS0FBeUI7UUFDekQsTUFBTSxFQUFFLFNBQVMsRUFBRSxpQkFBaUIsRUFBRSxHQUFHLGlCQUFpQixDQUFDLGNBQWMsQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUNqRixPQUFPLEtBQUssRUFBRSx1QkFBdUIsQ0FBQyxDQUFDLENBQUMsR0FBRyxpQkFBaUIsSUFBSSxTQUFTLENBQUMsT0FBTyxFQUFFLENBQUMsQ0FBQyxDQUFDLFNBQVMsQ0FBQyxPQUFPLENBQUM7SUFDMUcsQ0FBQztJQUVNLE1BQU0sQ0FBQywyQkFBMkIsQ0FBQyxLQUFnQixFQUFFLEtBQXlCO1FBQ25GLE1BQU0sRUFBRSxPQUFPLEVBQUUsTUFBTSxFQUFFLEdBQUcsbUJBQUssQ0FBQyxFQUFFLENBQUMsS0FBSyxDQUFDLENBQUM7UUFDNUMsTUFBTSxFQUFFLFNBQVMsRUFBRSxpQkFBaUIsRUFBRSxHQUFHLGlCQUFpQixDQUFDLGNBQWMsQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUVqRixPQUFPLElBQUkseUJBQWUsQ0FBQztZQUN6QixNQUFNLEVBQUUsZ0JBQU0sQ0FBQyxLQUFLO1lBQ3BCLE9BQU8sRUFBRTtnQkFDUCxxQkFBcUI7Z0JBQ3JCLHVDQUF1QzthQUN4QztZQUNELFNBQVMsRUFBRTtnQkFDVCx1Q0FBdUMsU0FBUyxDQUFDLE9BQU8sRUFBRTtnQkFDMUQsbUJBQW1CLE1BQU0sSUFBSSxPQUFPLHNCQUFzQixpQkFBaUIsSUFBSSxTQUFTLENBQUMsT0FBTyxFQUFFO2FBQ25HO1NBQ0YsQ0FBQyxDQUFDO0lBQ0wsQ0FBQztJQUVPLE1BQU0sQ0FBQyxjQUFjLENBQUMsS0FBeUI7UUFDckQsTUFBTSxTQUFTLEdBQUcsS0FBSyxFQUFFLFNBQVMsSUFBSSx1Q0FBeUIsQ0FBQyx1Q0FBdUMsQ0FBQztRQUN4RyxNQUFNLGlCQUFpQixHQUFHLEtBQUssRUFBRSwwQkFBMEIsSUFBSSxpQ0FBaUMsQ0FBQyxFQUFFLENBQUM7UUFFcEcsT0FBTztZQUNMLFNBQVM7WUFDVCxpQkFBaUI7U0FDbEIsQ0FBQztJQUNKLENBQUM7O0FBL0JILDhDQWdDQyIsInNvdXJjZXNDb250ZW50IjpbIi8vIENvcHlyaWdodCBBbWF6b24uY29tLCBJbmMuIG9yIGl0cyBhZmZpbGlhdGVzLiBBbGwgUmlnaHRzIFJlc2VydmVkLlxuLy8gU1BEWC1MaWNlbnNlLUlkZW50aWZpZXI6IEFwYWNoZS0yLjBcblxuaW1wb3J0IHsgU3RhY2sgfSBmcm9tICdhd3MtY2RrLWxpYic7XG5pbXBvcnQgeyBGb3VuZGF0aW9uTW9kZWxJZGVudGlmaWVyIH0gZnJvbSAnYXdzLWNkay1saWIvYXdzLWJlZHJvY2snO1xuaW1wb3J0IHsgRWZmZWN0LCBQb2xpY3lTdGF0ZW1lbnQgfSBmcm9tICdhd3MtY2RrLWxpYi9hd3MtaWFtJztcbmltcG9ydCB7IENvbnN0cnVjdCB9IGZyb20gJ2NvbnN0cnVjdHMnO1xuXG4vKipcbiAqIENyb3NzLXJlZ2lvbiBpbmZlcmVuY2UgcHJlZml4IG9wdGlvbnMgZm9yIEJlZHJvY2sgbW9kZWxzLlxuICogVXNlZCB0byBjb25maWd1cmUgaW5mZXJlbmNlIHByb2ZpbGVzIGZvciBpbXByb3ZlZCBhdmFpbGFiaWxpdHkgYW5kIHBlcmZvcm1hbmNlLlxuICovXG5leHBvcnQgZW51bSBCZWRyb2NrQ3Jvc3NSZWdpb25JbmZlcmVuY2VQcmVmaXgge1xuICAvKiogVVMtYmFzZWQgY3Jvc3MtcmVnaW9uIGluZmVyZW5jZSBwcm9maWxlICovXG4gIFVTID0gJ3VzJyxcbiAgLyoqIEVVLWJhc2VkIGNyb3NzLXJlZ2lvbiBpbmZlcmVuY2UgcHJvZmlsZSAqL1xuICBFVSA9ICdldScsXG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQmVkcm9ja01vZGVsUHJvcHMge1xuICAvKipcbiAgICAgKiBGb3VuZGF0aW9uIG1vZGVsIHRvIHVzZVxuICAgICAqXG4gICAgICogQGRlZmF1bHQgRm91bmRhdGlvbk1vZGVsSWRlbnRpZmllci5BTlRIUk9QSUNfQ0xBVURFX1NPTk5FVF80XzIwMjUwNTE0X1YxXzBcbiAgICAgKi9cbiAgcmVhZG9ubHkgZm1Nb2RlbElkPzogRm91bmRhdGlvbk1vZGVsSWRlbnRpZmllcjtcblxuICAvKipcbiAgICAgKiBFbmFibGUgY3Jvc3MtcmVnaW9uIGluZmVyZW5jZSBmb3IgQmVkcm9jayBtb2RlbHMgdG8gaW1wcm92ZSBhdmFpbGFiaWxpdHkgYW5kIHBlcmZvcm1hbmNlLlxuICAgICAqIFdoZW4gZW5hYmxlZCwgdXNlcyBpbmZlcmVuY2UgcHJvZmlsZXMgaW5zdGVhZCBvZiBkaXJlY3QgbW9kZWwgaW52b2NhdGlvbi5cbiAgICAgKiBAZGVmYXVsdCBmYWxzZVxuICAgICAqL1xuICByZWFkb25seSB1c2VDcm9zc1JlZ2lvbkluZmVyZW5jZT86IGJvb2xlYW47XG4gIC8qKlxuICAgICAqIFByZWZpeCBmb3IgY3Jvc3MtcmVnaW9uIGluZmVyZW5jZSBjb25maWd1cmF0aW9uLlxuICAgICAqIE9ubHkgdXNlZCB3aGVuIHVzZUNyb3NzUmVnaW9uSW5mZXJlbmNlIGlzIHRydWUuXG4gICAgICogQGRlZmF1bHQgQmVkcm9ja0Nyb3NzUmVnaW9uSW5mZXJlbmNlUHJlZml4LlVTXG4gICAgICovXG4gIHJlYWRvbmx5IGNyb3NzUmVnaW9uSW5mZXJlbmNlUHJlZml4PzogQmVkcm9ja0Nyb3NzUmVnaW9uSW5mZXJlbmNlUHJlZml4O1xufVxuXG5leHBvcnQgY2xhc3MgQmVkcm9ja01vZGVsVXRpbHMge1xuICBwdWJsaWMgc3RhdGljIGRlcml2ZUFjdHVhbE1vZGVsSWQocHJvcHM/OiBCZWRyb2NrTW9kZWxQcm9wcyk6IHN0cmluZyB7XG4gICAgY29uc3QgeyBmbU1vZGVsSWQsIGNyb3NzUmVnaW9uUHJlZml4IH0gPSBCZWRyb2NrTW9kZWxVdGlscy5kZXJpdmVEZWZhdWx0cyhwcm9wcyk7XG4gICAgcmV0dXJuIHByb3BzPy51c2VDcm9zc1JlZ2lvbkluZmVyZW5jZSA/IGAke2Nyb3NzUmVnaW9uUHJlZml4fS4ke2ZtTW9kZWxJZC5tb2RlbElkfWAgOiBmbU1vZGVsSWQubW9kZWxJZDtcbiAgfVxuXG4gIHB1YmxpYyBzdGF0aWMgZ2VuZXJhdGVNb2RlbElBTVBlcm1pc3Npb25zKHNjb3BlOiBDb25zdHJ1Y3QsIHByb3BzPzogQmVkcm9ja01vZGVsUHJvcHMpOiBQb2xpY3lTdGF0ZW1lbnQge1xuICAgIGNvbnN0IHsgYWNjb3VudCwgcmVnaW9uIH0gPSBTdGFjay5vZihzY29wZSk7XG4gICAgY29uc3QgeyBmbU1vZGVsSWQsIGNyb3NzUmVnaW9uUHJlZml4IH0gPSBCZWRyb2NrTW9kZWxVdGlscy5kZXJpdmVEZWZhdWx0cyhwcm9wcyk7XG5cbiAgICByZXR1cm4gbmV3IFBvbGljeVN0YXRlbWVudCh7XG4gICAgICBlZmZlY3Q6IEVmZmVjdC5BTExPVyxcbiAgICAgIGFjdGlvbnM6IFtcbiAgICAgICAgJ2JlZHJvY2s6SW52b2tlTW9kZWwnLFxuICAgICAgICAnYmVkcm9jazpJbnZva2VNb2RlbFdpdGhSZXNwb25zZVN0cmVhbScsXG4gICAgICBdLFxuICAgICAgcmVzb3VyY2VzOiBbXG4gICAgICAgIGBhcm46YXdzOmJlZHJvY2s6Kjo6Zm91bmRhdGlvbi1tb2RlbC8ke2ZtTW9kZWxJZC5tb2RlbElkfWAsXG4gICAgICAgIGBhcm46YXdzOmJlZHJvY2s6JHtyZWdpb259OiR7YWNjb3VudH06aW5mZXJlbmNlLXByb2ZpbGUvJHtjcm9zc1JlZ2lvblByZWZpeH0uJHtmbU1vZGVsSWQubW9kZWxJZH1gLFxuICAgICAgXSxcbiAgICB9KTtcbiAgfVxuXG4gIHByaXZhdGUgc3RhdGljIGRlcml2ZURlZmF1bHRzKHByb3BzPzogQmVkcm9ja01vZGVsUHJvcHMpOiBSZWNvcmQ8c3RyaW5nLCBhbnk+IHtcbiAgICBjb25zdCBmbU1vZGVsSWQgPSBwcm9wcz8uZm1Nb2RlbElkIHx8IEZvdW5kYXRpb25Nb2RlbElkZW50aWZpZXIuQU5USFJPUElDX0NMQVVERV9TT05ORVRfNF8yMDI1MDUxNF9WMV8wO1xuICAgIGNvbnN0IGNyb3NzUmVnaW9uUHJlZml4ID0gcHJvcHM/LmNyb3NzUmVnaW9uSW5mZXJlbmNlUHJlZml4IHx8IEJlZHJvY2tDcm9zc1JlZ2lvbkluZmVyZW5jZVByZWZpeC5VUztcblxuICAgIHJldHVybiB7XG4gICAgICBmbU1vZGVsSWQsXG4gICAgICBjcm9zc1JlZ2lvblByZWZpeCxcbiAgICB9O1xuICB9XG59Il19
@@ -13,7 +13,7 @@ class DefaultRuntimes {
13
13
  }
14
14
  exports.DefaultRuntimes = DefaultRuntimes;
15
15
  _a = DefaultRuntimes, _b = JSII_RTTI_SYMBOL_1;
16
- DefaultRuntimes[_b] = { fqn: "@cdklabs/cdk-appmod-catalog-blueprints.DefaultRuntimes", version: "1.3.0" };
16
+ DefaultRuntimes[_b] = { fqn: "@cdklabs/cdk-appmod-catalog-blueprints.DefaultRuntimes", version: "1.4.0" };
17
17
  /**
18
18
  * Default runtime for all Lambda functions in the use cases.
19
19
  */
@@ -117,5 +117,5 @@ class AccessLog extends constructs_1.Construct {
117
117
  }
118
118
  exports.AccessLog = AccessLog;
119
119
  _a = JSII_RTTI_SYMBOL_1;
120
- AccessLog[_a] = { fqn: "@cdklabs/cdk-appmod-catalog-blueprints.AccessLog", version: "1.3.0" };
120
+ AccessLog[_a] = { fqn: "@cdklabs/cdk-appmod-catalog-blueprints.AccessLog", version: "1.4.0" };
121
121
  //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"access-log.js","sourceRoot":"","sources":["../../../use-cases/framework/foundation/access-log.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\nimport * as cdk from 'aws-cdk-lib';\nimport * as iam from 'aws-cdk-lib/aws-iam';\nimport * as s3 from 'aws-cdk-lib/aws-s3';\nimport { Construct } from 'constructs';\n\n/**\n * Configuration options for the AccessLog construct\n */\nexport interface AccessLogProps {\n  /**\n   * The name of the S3 bucket for access logs\n   * @default 'access-logs'\n   */\n  readonly bucketName?: string;\n\n  /**\n   * Lifecycle rules for the access logs\n   * @default Transition to IA after 30 days, delete after 90 days\n   */\n  readonly lifecycleRules?: s3.LifecycleRule[];\n\n  /**\n   * Whether to enable versioning on the access logs bucket\n   * @default false\n   */\n  readonly versioned?: boolean;\n\n  /**\n   * Custom bucket prefix for organizing access logs\n   * @default 'access-logs'\n   */\n  readonly bucketPrefix?: string;\n}\n\n/**\n * AccessLog construct that provides a centralized S3 bucket for storing access logs.\n * This construct creates a secure S3 bucket with appropriate policies for AWS services\n * to deliver access logs.\n *\n * Usage:\n *\n * const accessLog = new AccessLog(this, 'AccessLog');\n * const bucket = accessLog.bucket;\n * const bucketName = accessLog.bucketName;\n */\nexport class AccessLog extends Construct {\n  /**\n   * The S3 bucket for storing access logs\n   */\n  public readonly bucket: s3.Bucket;\n\n  /**\n   * The name of the S3 bucket\n   */\n  public readonly bucketName: string;\n\n  /**\n   * The bucket prefix used for organizing access logs\n   */\n  public readonly bucketPrefix: string;\n\n  constructor(scope: Construct, id: string, props: AccessLogProps = {}) {\n    super(scope, id);\n\n    // Store the bucket prefix\n    this.bucketPrefix = props.bucketPrefix || 'access-logs';\n\n    // Generate unique bucket name with account and region to avoid conflicts\n    const accountId = cdk.Stack.of(this).account;\n    const region = cdk.Stack.of(this).region;\n    const baseName = props.bucketName || 'access-logs';\n    this.bucketName = `${baseName}-${accountId}-${region}`;\n\n    // Default lifecycle rules\n    const defaultLifecycleRules: s3.LifecycleRule[] = [\n      {\n        id: 'AccessLogLifecycle',\n        enabled: true,\n        transitions: [\n          {\n            storageClass: s3.StorageClass.INFREQUENT_ACCESS,\n            transitionAfter: cdk.Duration.days(30),\n          },\n          {\n            storageClass: s3.StorageClass.GLACIER,\n            transitionAfter: cdk.Duration.days(90),\n          },\n        ],\n        expiration: cdk.Duration.days(365), // Delete after 1 year\n      },\n    ];\n\n    // Create the S3 bucket for access logs\n    this.bucket = new s3.Bucket(this, 'AccessLogBucket', {\n      bucketName: this.bucketName,\n      encryption: s3.BucketEncryption.S3_MANAGED,\n      versioned: props.versioned || false,\n      blockPublicAccess: s3.BlockPublicAccess.BLOCK_ALL,\n      lifecycleRules: props.lifecycleRules || defaultLifecycleRules,\n      removalPolicy: cdk.RemovalPolicy.RETAIN, // Retain logs for compliance\n      enforceSSL: true, // Enforce SSL for all requests\n      eventBridgeEnabled: false, // Disable EventBridge for cost optimization\n    });\n\n    // Add bucket policy to allow access log delivery from AWS services\n    this.bucket.addToResourcePolicy(new iam.PolicyStatement({\n      sid: 'AllowAccessLogDelivery',\n      effect: iam.Effect.ALLOW,\n      principals: [\n        new iam.ServicePrincipal('logging.s3.amazonaws.com'),\n        new iam.ServicePrincipal('delivery.logs.amazonaws.com'),\n      ],\n      actions: [\n        's3:PutObject',\n        's3:GetBucketAcl',\n        's3:ListBucket',\n      ],\n      resources: [\n        this.bucket.bucketArn,\n        `${this.bucket.bucketArn}/*`,\n      ],\n      conditions: {\n        StringEquals: {\n          's3:x-amz-acl': 'bucket-owner-full-control',\n        },\n      },\n    }));\n\n    // Allow ELB service to write access logs\n    this.bucket.addToResourcePolicy(new iam.PolicyStatement({\n      sid: 'AllowELBAccessLogDelivery',\n      effect: iam.Effect.ALLOW,\n      principals: [\n        new iam.ServicePrincipal('elasticloadbalancing.amazonaws.com'),\n      ],\n      actions: ['s3:PutObject'],\n      resources: [`${this.bucket.bucketArn}/*`],\n    }));\n  }\n\n  /**\n   * Get the S3 bucket path for a specific service's access logs\n   *\n   * @param serviceName The name of the service (e.g., 'alb', 'cloudfront', 's3')\n   * @param resourceName Optional resource name for further organization\n   * @returns The S3 path for the service's access logs\n   */\n  public getLogPath(serviceName: string, resourceName?: string): string {\n    const basePath = `${this.bucketName}/${this.bucketPrefix}/${serviceName}`;\n    return resourceName ? `${basePath}/${resourceName}` : basePath;\n  }\n\n  /**\n   * Get the S3 URI for a specific service's access logs\n   *\n   * @param serviceName The name of the service (e.g., 'alb', 'cloudfront', 's3')\n   * @param resourceName Optional resource name for further organization\n   * @returns The S3 URI for the service's access logs\n   */\n  public getLogUri(serviceName: string, resourceName?: string): string {\n    return `s3://${this.getLogPath(serviceName, resourceName)}`;\n  }\n}\n"]}
@@ -38,5 +38,5 @@ class EventbridgeBroker extends constructs_1.Construct {
38
38
  }
39
39
  exports.EventbridgeBroker = EventbridgeBroker;
40
40
  _a = JSII_RTTI_SYMBOL_1;
41
- EventbridgeBroker[_a] = { fqn: "@cdklabs/cdk-appmod-catalog-blueprints.EventbridgeBroker", version: "1.3.0" };
41
+ EventbridgeBroker[_a] = { fqn: "@cdklabs/cdk-appmod-catalog-blueprints.EventbridgeBroker", version: "1.4.0" };
42
42
  //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZXZlbnRicmlkZ2UtYnJva2VyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vdXNlLWNhc2VzL2ZyYW1ld29yay9mb3VuZGF0aW9uL2V2ZW50YnJpZGdlLWJyb2tlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7OztBQUFBLDZDQUE0QztBQUM1Qyx1REFBa0Q7QUFDbEQsaURBQTBDO0FBQzFDLHFFQUEwRDtBQUMxRCxpRkFBMkU7QUFDM0UsMkNBQXVDO0FBU3ZDLE1BQWEsaUJBQWtCLFNBQVEsc0JBQVM7SUFLOUMsWUFBWSxLQUFnQixFQUFFLEVBQVUsRUFBRSxLQUE2QjtRQUNyRSxLQUFLLENBQUMsS0FBSyxFQUFFLEVBQUUsQ0FBQyxDQUFDO1FBQ2pCLE1BQU0sYUFBYSxHQUFHLEtBQUssQ0FBQyxhQUFhLElBQUksMkJBQWEsQ0FBQyxPQUFPLENBQUM7UUFDbkUsSUFBSSxDQUFDLE1BQU0sR0FBRyxLQUFLLENBQUMsTUFBTSxJQUFJLElBQUksYUFBRyxDQUFDLElBQUksRUFBRSxtQkFBbUIsRUFBRTtZQUMvRCxpQkFBaUIsRUFBRSxJQUFJO1lBQ3ZCLGFBQWE7U0FDZCxDQUFDLENBQUM7UUFDSCxJQUFJLENBQUMsUUFBUSxHQUFHLElBQUkscUJBQVEsQ0FBQyxJQUFJLEVBQUUsVUFBVSxFQUFFO1lBQzdDLFlBQVksRUFBRSxLQUFLLENBQUMsSUFBSTtZQUN4QixNQUFNLEVBQUUsSUFBSSxDQUFDLE1BQU07U0FDcEIsQ0FBQyxDQUFDO1FBRUgsSUFBSSxDQUFDLFdBQVcsR0FBRyxLQUFLLENBQUMsV0FBVyxDQUFDO0lBQ3ZDLENBQUM7SUFFTSxlQUFlLENBQUMsVUFBa0IsRUFBRSxXQUFnQjtRQUN6RCxPQUFPLElBQUksOENBQW9CLENBQUMsSUFBSSxFQUFFLHNCQUFzQixVQUFVLEVBQUUsRUFBRTtZQUN4RSxPQUFPLEVBQUU7Z0JBQ1A7b0JBQ0UsUUFBUSxFQUFFLElBQUksQ0FBQyxRQUFRO29CQUN2QixNQUFNLEVBQUUsSUFBSSxDQUFDLFdBQVc7b0JBQ3hCLFVBQVU7b0JBQ1YsTUFBTSxFQUFFLDZCQUFTLENBQUMsVUFBVSxDQUFDLFdBQVcsQ0FBQztpQkFDMUM7YUFDRjtTQUNGLENBQUMsQ0FBQztJQUNMLENBQUM7O0FBL0JILDhDQWdDQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IFJlbW92YWxQb2xpY3kgfSBmcm9tICdhd3MtY2RrLWxpYic7XG5pbXBvcnQgeyBFdmVudEJ1cyB9IGZyb20gJ2F3cy1jZGstbGliL2F3cy1ldmVudHMnO1xuaW1wb3J0IHsgS2V5IH0gZnJvbSAnYXdzLWNkay1saWIvYXdzLWttcyc7XG5pbXBvcnQgeyBUYXNrSW5wdXQgfSBmcm9tICdhd3MtY2RrLWxpYi9hd3Mtc3RlcGZ1bmN0aW9ucyc7XG5pbXBvcnQgeyBFdmVudEJyaWRnZVB1dEV2ZW50cyB9IGZyb20gJ2F3cy1jZGstbGliL2F3cy1zdGVwZnVuY3Rpb25zLXRhc2tzJztcbmltcG9ydCB7IENvbnN0cnVjdCB9IGZyb20gJ2NvbnN0cnVjdHMnO1xuXG5leHBvcnQgaW50ZXJmYWNlIEV2ZW50YnJpZGdlQnJva2VyUHJvcHMge1xuICByZWFkb25seSBuYW1lPzogc3RyaW5nO1xuICByZWFkb25seSBldmVudFNvdXJjZTogc3RyaW5nO1xuICByZWFkb25seSBrbXNLZXk/OiBLZXk7XG4gIHJlYWRvbmx5IHJlbW92YWxQb2xpY3k/OiBSZW1vdmFsUG9saWN5O1xufVxuXG5leHBvcnQgY2xhc3MgRXZlbnRicmlkZ2VCcm9rZXIgZXh0ZW5kcyBDb25zdHJ1Y3Qge1xuICBwcml2YXRlIHJlYWRvbmx5IGV2ZW50U291cmNlOiBzdHJpbmc7XG4gIHJlYWRvbmx5IGV2ZW50YnVzOiBFdmVudEJ1cztcbiAgcmVhZG9ubHkga21zS2V5OiBLZXk7XG5cbiAgY29uc3RydWN0b3Ioc2NvcGU6IENvbnN0cnVjdCwgaWQ6IHN0cmluZywgcHJvcHM6IEV2ZW50YnJpZGdlQnJva2VyUHJvcHMpIHtcbiAgICBzdXBlcihzY29wZSwgaWQpO1xuICAgIGNvbnN0IHJlbW92YWxQb2xpY3kgPSBwcm9wcy5yZW1vdmFsUG9saWN5IHx8IFJlbW92YWxQb2xpY3kuREVTVFJPWTtcbiAgICB0aGlzLmttc0tleSA9IHByb3BzLmttc0tleSB8fCBuZXcgS2V5KHRoaXMsICdFdmVudEJyb2tlcktNU0tleScsIHtcbiAgICAgIGVuYWJsZUtleVJvdGF0aW9uOiB0cnVlLFxuICAgICAgcmVtb3ZhbFBvbGljeSxcbiAgICB9KTtcbiAgICB0aGlzLmV2ZW50YnVzID0gbmV3IEV2ZW50QnVzKHRoaXMsICdFdmVudEJ1cycsIHtcbiAgICAgIGV2ZW50QnVzTmFtZTogcHJvcHMubmFtZSxcbiAgICAgIGttc0tleTogdGhpcy5rbXNLZXksXG4gICAgfSk7XG5cbiAgICB0aGlzLmV2ZW50U291cmNlID0gcHJvcHMuZXZlbnRTb3VyY2U7XG4gIH1cblxuICBwdWJsaWMgc2VuZFZpYVNmbkNoYWluKGRldGFpbFR5cGU6IHN0cmluZywgZXZlbnREZXRhaWw6IGFueSkge1xuICAgIHJldHVybiBuZXcgRXZlbnRCcmlkZ2VQdXRFdmVudHModGhpcywgYFNmbkViUHV0RXZlbnRDaGFpbi0ke2RldGFpbFR5cGV9YCwge1xuICAgICAgZW50cmllczogW1xuICAgICAgICB7XG4gICAgICAgICAgZXZlbnRCdXM6IHRoaXMuZXZlbnRidXMsXG4gICAgICAgICAgc291cmNlOiB0aGlzLmV2ZW50U291cmNlLFxuICAgICAgICAgIGRldGFpbFR5cGUsXG4gICAgICAgICAgZGV0YWlsOiBUYXNrSW5wdXQuZnJvbU9iamVjdChldmVudERldGFpbCksXG4gICAgICAgIH0sXG4gICAgICBdLFxuICAgIH0pO1xuICB9XG59Il19
@@ -79,5 +79,5 @@ class Network extends constructs_1.Construct {
79
79
  }
80
80
  exports.Network = Network;
81
81
  _a = JSII_RTTI_SYMBOL_1;
82
- Network[_a] = { fqn: "@cdklabs/cdk-appmod-catalog-blueprints.Network", version: "1.3.0" };
82
+ Network[_a] = { fqn: "@cdklabs/cdk-appmod-catalog-blueprints.Network", version: "1.4.0" };
83
83
  //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibmV0d29yay5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3VzZS1jYXNlcy9mcmFtZXdvcmsvZm91bmRhdGlvbi9uZXR3b3JrLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7O0FBQUEscUVBQXFFO0FBQ3JFLHNDQUFzQztBQUN0QyxpREFBdVI7QUFDdlIsMkNBQXVDO0FBYXZDLE1BQWEsT0FBUSxTQUFRLHNCQUFTO0lBSXBDLFlBQVksS0FBZ0IsRUFBRSxFQUFVLEVBQUUsUUFBc0IsRUFBRTtRQUNoRSxLQUFLLENBQUMsS0FBSyxFQUFFLEVBQUUsQ0FBQyxDQUFDO1FBQ2pCLElBQUksQ0FBQyxLQUFLLEdBQUcsS0FBSyxDQUFDO1FBRW5CLElBQUksS0FBSyxDQUFDLE9BQU8sRUFBRSxDQUFDO1lBQ2xCLElBQUksQ0FBQyxHQUFHLEdBQUcsSUFBSSxhQUFHLENBQUMsSUFBSSxFQUFFLEtBQUssRUFBRTtnQkFDOUIsV0FBVyxFQUFFLEtBQUssQ0FBQyxXQUFXLElBQUkscUJBQVcsQ0FBQyxJQUFJLENBQUMsYUFBYSxDQUFDO2dCQUNqRSxXQUFXLEVBQUUsQ0FBQztnQkFDZCxNQUFNLEVBQUUsS0FBSyxDQUFDLE1BQU07Z0JBQ3BCLE9BQU8sRUFBRSxLQUFLLENBQUMsT0FBTztnQkFDdEIsbUJBQW1CLEVBQUUsS0FBSyxDQUFDLG1CQUFtQixJQUFJO29CQUNoRDt3QkFDRSxJQUFJLEVBQUUsVUFBVTt3QkFDaEIsVUFBVSxFQUFFLG9CQUFVLENBQUMsZ0JBQWdCO3dCQUN2QyxRQUFRLEVBQUUsRUFBRTtxQkFDYjtpQkFDRjthQUNGLENBQUMsQ0FBQztRQUNMLENBQUM7YUFBTSxDQUFDO1lBQ04sSUFBSSxDQUFDLEdBQUcsR0FBRyxJQUFJLGFBQUcsQ0FBQyxJQUFJLEVBQUUsS0FBSyxFQUFFO2dCQUM5QixXQUFXLEVBQUUsS0FBSyxDQUFDLFdBQVcsSUFBSSxxQkFBVyxDQUFDLElBQUksQ0FBQyxhQUFhLENBQUM7Z0JBQ2pFLGtCQUFrQixFQUFFLEtBQUssQ0FBQyxrQkFBa0I7Z0JBQzVDLGlCQUFpQixFQUFFLEtBQUssQ0FBQyxpQkFBaUI7Z0JBQzFDLFdBQVcsRUFBRSxLQUFLLENBQUMsV0FBVyxJQUFJLENBQUM7Z0JBQ25DLE1BQU0sRUFBRSxLQUFLLENBQUMsTUFBTTtnQkFDcEIsT0FBTyxFQUFFLEtBQUssQ0FBQyxPQUFPO2dCQUN0QixtQkFBbUIsRUFBRSxLQUFLLENBQUMsbUJBQW1CLElBQUk7b0JBQ2hEO3dCQUNFLElBQUksRUFBRSxRQUFRO3dCQUNkLFVBQVUsRUFBRSxvQkFBVSxDQUFDLE1BQU07d0JBQzdCLFFBQVEsRUFBRSxFQUFFO3FCQUNiO29CQUNEO3dCQUNFLElBQUksRUFBRSxTQUFTO3dCQUNmLFVBQVUsRUFBRSxvQkFBVSxDQUFDLG1CQUFtQjt3QkFDMUMsUUFBUSxFQUFFLEVBQUU7cUJBQ2I7b0JBQ0Q7d0JBQ0UsSUFBSSxFQUFFLFVBQVU7d0JBQ2hCLFVBQVUsRUFBRSxvQkFBVSxDQUFDLGdCQUFnQjt3QkFDdkMsUUFBUSxFQUFFLEVBQUU7cUJBQ2I7aUJBQ0Y7YUFDRixDQUFDLENBQUM7UUFDTCxDQUFDO0lBQ0gsQ0FBQztJQUVNLHFCQUFxQixDQUFDLEVBQVUsRUFBRSxPQUFvQyxFQUFFLElBQVk7UUFDekYsSUFBSSxPQUFPLEtBQUssd0NBQThCLENBQUMsRUFBRSxFQUFFLENBQUM7WUFDbEQsSUFBSSxDQUFDLEdBQUcsQ0FBQyxrQkFBa0IsQ0FBQyxHQUFHLEVBQUUsVUFBVSxFQUFFO2dCQUMzQyxPQUFPLEVBQUUsc0NBQTRCLENBQUMsRUFBRTtnQkFDeEMsT0FBTyxFQUFFLENBQUMsSUFBSSxDQUFDLDBCQUEwQixFQUFFLENBQUM7YUFDN0MsQ0FBQyxDQUFDO1FBQ0wsQ0FBQztRQUVELE1BQU0sYUFBYSxHQUFHLElBQUksdUJBQWEsQ0FBQyxJQUFJLEVBQUUsNkJBQTZCLEVBQUUsRUFBRSxFQUFFO1lBQy9FLEdBQUcsRUFBRSxJQUFJLENBQUMsR0FBRztTQUNkLENBQUMsQ0FBQztRQUVILGFBQWEsQ0FBQyxjQUFjLENBQUMsSUFBSSxJQUFJLGNBQUksQ0FBQyxPQUFPLEVBQUUsRUFBRSxjQUFJLENBQUMsS0FBSyxDQUFDLENBQUM7UUFFakUsT0FBTyxJQUFJLENBQUMsR0FBRyxDQUFDLG9CQUFvQixDQUFDLDBCQUEwQixFQUFFLEVBQUUsRUFBRTtZQUNuRSxPQUFPO1lBQ1AsY0FBYyxFQUFFLENBQUMsYUFBYSxDQUFDO1lBQy9CLE9BQU8sRUFBRSxJQUFJLENBQUMsMEJBQTBCLEVBQUU7U0FDM0MsQ0FBQyxDQUFDO0lBQ0wsQ0FBQztJQUVNLDBCQUEwQjtRQUMvQixPQUFPO1lBQ0wsVUFBVSxFQUFFLElBQUksQ0FBQyxLQUFLLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQyxvQkFBVSxDQUFDLGdCQUFnQixDQUFDLENBQUMsQ0FBQyxvQkFBVSxDQUFDLG1CQUFtQjtTQUM5RixDQUFDO0lBQ0osQ0FBQzs7QUE1RUgsMEJBNkVDIiwic291cmNlc0NvbnRlbnQiOlsiLy8gQ29weXJpZ2h0IEFtYXpvbi5jb20sIEluYy4gb3IgaXRzIGFmZmlsaWF0ZXMuIEFsbCBSaWdodHMgUmVzZXJ2ZWQuXG4vLyBTUERYLUxpY2Vuc2UtSWRlbnRpZmllcjogQXBhY2hlLTIuMFxuaW1wb3J0IHsgR2F0ZXdheVZwY0VuZHBvaW50QXdzU2VydmljZSwgSUlwQWRkcmVzc2VzLCBJbnRlcmZhY2VWcGNFbmRwb2ludCwgSW50ZXJmYWNlVnBjRW5kcG9pbnRBd3NTZXJ2aWNlLCBJbnRlcmZhY2VWcGNFbmRwb2ludFNlcnZpY2UsIElwQWRkcmVzc2VzLCBJUGVlciwgTmF0UHJvdmlkZXIsIFBlZXIsIFBvcnQsIFNlY3VyaXR5R3JvdXAsIFN1Ym5ldENvbmZpZ3VyYXRpb24sIFN1Ym5ldFNlbGVjdGlvbiwgU3VibmV0VHlwZSwgVnBjIH0gZnJvbSAnYXdzLWNkay1saWIvYXdzLWVjMic7XG5pbXBvcnQgeyBDb25zdHJ1Y3QgfSBmcm9tICdjb25zdHJ1Y3RzJztcblxuZXhwb3J0IGludGVyZmFjZSBOZXR3b3JrUHJvcHMge1xuICByZWFkb25seSBwcml2YXRlPzogYm9vbGVhbjtcbiAgcmVhZG9ubHkgaXBBZGRyZXNzZXM/OiBJSXBBZGRyZXNzZXM7XG4gIHJlYWRvbmx5IG5hdEdhdGV3YXlQcm92aWRlcj86IE5hdFByb3ZpZGVyO1xuICByZWFkb25seSBuYXRHYXRld2F5U3VibmV0cz86IFN1Ym5ldFNlbGVjdGlvbjtcbiAgcmVhZG9ubHkgbmF0R2F0ZXdheXM/OiBudW1iZXI7XG4gIHJlYWRvbmx5IG1heEF6cz86IG51bWJlcjtcbiAgcmVhZG9ubHkgdnBjTmFtZT86IHN0cmluZztcbiAgcmVhZG9ubHkgc3VibmV0Q29uZmlndXJhdGlvbj86IFN1Ym5ldENvbmZpZ3VyYXRpb25bXTtcbn1cblxuZXhwb3J0IGNsYXNzIE5ldHdvcmsgZXh0ZW5kcyBDb25zdHJ1Y3Qge1xuICByZWFkb25seSB2cGM6IFZwYztcbiAgcHJpdmF0ZSByZWFkb25seSBwcm9wczogTmV0d29ya1Byb3BzO1xuXG4gIGNvbnN0cnVjdG9yKHNjb3BlOiBDb25zdHJ1Y3QsIGlkOiBzdHJpbmcsIHByb3BzOiBOZXR3b3JrUHJvcHMgPSB7fSkge1xuICAgIHN1cGVyKHNjb3BlLCBpZCk7XG4gICAgdGhpcy5wcm9wcyA9IHByb3BzO1xuXG4gICAgaWYgKHByb3BzLnByaXZhdGUpIHtcbiAgICAgIHRoaXMudnBjID0gbmV3IFZwYyh0aGlzLCAnVlBDJywge1xuICAgICAgICBpcEFkZHJlc3NlczogcHJvcHMuaXBBZGRyZXNzZXMgfHwgSXBBZGRyZXNzZXMuY2lkcignMTAuMC4wLjAvMTYnKSxcbiAgICAgICAgbmF0R2F0ZXdheXM6IDAsXG4gICAgICAgIG1heEF6czogcHJvcHMubWF4QXpzLFxuICAgICAgICB2cGNOYW1lOiBwcm9wcy52cGNOYW1lLFxuICAgICAgICBzdWJuZXRDb25maWd1cmF0aW9uOiBwcm9wcy5zdWJuZXRDb25maWd1cmF0aW9uIHx8IFtcbiAgICAgICAgICB7XG4gICAgICAgICAgICBuYW1lOiAnSXNvbGF0ZWQnLFxuICAgICAgICAgICAgc3VibmV0VHlwZTogU3VibmV0VHlwZS5QUklWQVRFX0lTT0xBVEVELFxuICAgICAgICAgICAgY2lkck1hc2s6IDI0LFxuICAgICAgICAgIH0sXG4gICAgICAgIF0sXG4gICAgICB9KTtcbiAgICB9IGVsc2Uge1xuICAgICAgdGhpcy52cGMgPSBuZXcgVnBjKHRoaXMsICdWUEMnLCB7XG4gICAgICAgIGlwQWRkcmVzc2VzOiBwcm9wcy5pcEFkZHJlc3NlcyB8fCBJcEFkZHJlc3Nlcy5jaWRyKCcxMC4wLjAuMC8xNicpLFxuICAgICAgICBuYXRHYXRld2F5UHJvdmlkZXI6IHByb3BzLm5hdEdhdGV3YXlQcm92aWRlcixcbiAgICAgICAgbmF0R2F0ZXdheVN1Ym5ldHM6IHByb3BzLm5hdEdhdGV3YXlTdWJuZXRzLFxuICAgICAgICBuYXRHYXRld2F5czogcHJvcHMubmF0R2F0ZXdheXMgfHwgMSxcbiAgICAgICAgbWF4QXpzOiBwcm9wcy5tYXhBenMsXG4gICAgICAgIHZwY05hbWU6IHByb3BzLnZwY05hbWUsXG4gICAgICAgIHN1Ym5ldENvbmZpZ3VyYXRpb246IHByb3BzLnN1Ym5ldENvbmZpZ3VyYXRpb24gfHwgW1xuICAgICAgICAgIHtcbiAgICAgICAgICAgIG5hbWU6ICdQdWJsaWMnLFxuICAgICAgICAgICAgc3VibmV0VHlwZTogU3VibmV0VHlwZS5QVUJMSUMsXG4gICAgICAgICAgICBjaWRyTWFzazogMjQsXG4gICAgICAgICAgfSxcbiAgICAgICAgICB7XG4gICAgICAgICAgICBuYW1lOiAnUHJpdmF0ZScsXG4gICAgICAgICAgICBzdWJuZXRUeXBlOiBTdWJuZXRUeXBlLlBSSVZBVEVfV0lUSF9FR1JFU1MsXG4gICAgICAgICAgICBjaWRyTWFzazogMjQsXG4gICAgICAgICAgfSxcbiAgICAgICAgICB7XG4gICAgICAgICAgICBuYW1lOiAnSXNvbGF0ZWQnLFxuICAgICAgICAgICAgc3VibmV0VHlwZTogU3VibmV0VHlwZS5QUklWQVRFX0lTT0xBVEVELFxuICAgICAgICAgICAgY2lkck1hc2s6IDI0LFxuICAgICAgICAgIH0sXG4gICAgICAgIF0sXG4gICAgICB9KTtcbiAgICB9XG4gIH1cblxuICBwdWJsaWMgY3JlYXRlU2VydmljZUVuZHBvaW50KGlkOiBzdHJpbmcsIHNlcnZpY2U6IEludGVyZmFjZVZwY0VuZHBvaW50U2VydmljZSwgcGVlcj86IElQZWVyKTogSW50ZXJmYWNlVnBjRW5kcG9pbnQge1xuICAgIGlmIChzZXJ2aWNlID09PSBJbnRlcmZhY2VWcGNFbmRwb2ludEF3c1NlcnZpY2UuUzMpIHtcbiAgICAgIHRoaXMudnBjLmFkZEdhdGV3YXlFbmRwb2ludChgJHtpZH0tZ2F0ZXdheWAsIHtcbiAgICAgICAgc2VydmljZTogR2F0ZXdheVZwY0VuZHBvaW50QXdzU2VydmljZS5TMyxcbiAgICAgICAgc3VibmV0czogW3RoaXMuYXBwbGljYXRpb25TdWJuZXRTZWxlY3Rpb24oKV0sXG4gICAgICB9KTtcbiAgICB9XG5cbiAgICBjb25zdCBzZWN1cml0eUdyb3VwID0gbmV3IFNlY3VyaXR5R3JvdXAodGhpcywgYHNnLWZvci1pbnRlcmZhY2UtZW5kcG9pbnQtJHtpZH1gLCB7XG4gICAgICB2cGM6IHRoaXMudnBjLFxuICAgIH0pO1xuXG4gICAgc2VjdXJpdHlHcm91cC5hZGRJbmdyZXNzUnVsZShwZWVyIHx8IFBlZXIuYW55SXB2NCgpLCBQb3J0LkhUVFBTKTtcblxuICAgIHJldHVybiB0aGlzLnZwYy5hZGRJbnRlcmZhY2VFbmRwb2ludChgaW50ZXJmYWNlLWVuZHBvaW50LWZvci0ke2lkfWAsIHtcbiAgICAgIHNlcnZpY2UsXG4gICAgICBzZWN1cml0eUdyb3VwczogW3NlY3VyaXR5R3JvdXBdLFxuICAgICAgc3VibmV0czogdGhpcy5hcHBsaWNhdGlvblN1Ym5ldFNlbGVjdGlvbigpLFxuICAgIH0pO1xuICB9XG5cbiAgcHVibGljIGFwcGxpY2F0aW9uU3VibmV0U2VsZWN0aW9uKCk6IFN1Ym5ldFNlbGVjdGlvbiB7XG4gICAgcmV0dXJuIHtcbiAgICAgIHN1Ym5ldFR5cGU6IHRoaXMucHJvcHMucHJpdmF0ZSA/IFN1Ym5ldFR5cGUuUFJJVkFURV9JU09MQVRFRCA6IFN1Ym5ldFR5cGUuUFJJVkFURV9XSVRIX0VHUkVTUyxcbiAgICB9O1xuICB9XG59Il19
@@ -0,0 +1 @@
1
+ export {};