@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,278 @@
1
+ "use strict";
2
+ // Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
3
+ // SPDX-License-Identifier: Apache-2.0
4
+ Object.defineProperty(exports, "__esModule", { value: true });
5
+ exports.ServerlessObservability = void 0;
6
+ const cdk = require("aws-cdk-lib");
7
+ const cloudwatch = require("aws-cdk-lib/aws-cloudwatch");
8
+ const iam = require("aws-cdk-lib/aws-iam");
9
+ const lambda = require("aws-cdk-lib/aws-lambda");
10
+ const logs = require("aws-cdk-lib/aws-logs");
11
+ const constructs_1 = require("constructs");
12
+ /**
13
+ * Comprehensive observability construct for serverless use cases
14
+ */
15
+ class ServerlessObservability extends constructs_1.Construct {
16
+ constructor(scope, id, resources, config = {}) {
17
+ super(scope, id);
18
+ // Configuration with defaults
19
+ const observabilityConfig = {
20
+ enableDashboard: true,
21
+ enableAlarms: true,
22
+ enableTracing: true,
23
+ logRetentionDays: logs.RetentionDays.ONE_MONTH,
24
+ dashboardName: `${resources.useCaseName}-${cdk.Stack.of(this).stackName}`,
25
+ metricsNamespace: resources.useCaseName,
26
+ ...config,
27
+ };
28
+ this.metricsNamespace = observabilityConfig.metricsNamespace;
29
+ this.logGroups = [];
30
+ this.alarms = [];
31
+ // Create log groups for Lambda functions
32
+ this.createLogGroups(resources.lambdaFunctions, observabilityConfig);
33
+ // Enable X-Ray tracing
34
+ if (observabilityConfig.enableTracing) {
35
+ this.enableTracing(resources);
36
+ }
37
+ // Create CloudWatch alarms
38
+ if (observabilityConfig.enableAlarms) {
39
+ this.createAlarms(resources, observabilityConfig);
40
+ }
41
+ // Create CloudWatch dashboard
42
+ if (observabilityConfig.enableDashboard) {
43
+ this.dashboard = this.createDashboard(resources, observabilityConfig);
44
+ }
45
+ }
46
+ /**
47
+ * Create log groups for Lambda functions with proper retention
48
+ */
49
+ createLogGroups(lambdaFunctions, config) {
50
+ lambdaFunctions.forEach((fn, index) => {
51
+ const logGroup = new logs.LogGroup(this, `LogGroup-${index}`, {
52
+ logGroupName: `/aws/lambda/${fn.functionName}`,
53
+ retention: config.logRetentionDays,
54
+ removalPolicy: cdk.RemovalPolicy.DESTROY,
55
+ });
56
+ this.logGroups.push(logGroup);
57
+ });
58
+ }
59
+ /**
60
+ * Enable AWS X-Ray tracing for all resources
61
+ */
62
+ enableTracing(resources) {
63
+ // Enable X-Ray tracing for Lambda functions
64
+ resources.lambdaFunctions.forEach(fn => {
65
+ // Enable X-Ray tracing via Lambda configuration (not environment variables)
66
+ const cfnFunction = fn.node.defaultChild;
67
+ cfnFunction.tracingConfig = {
68
+ mode: lambda.Tracing.ACTIVE,
69
+ };
70
+ // Add X-Ray permissions
71
+ fn.addToRolePolicy(new iam.PolicyStatement({
72
+ actions: ['xray:PutTraceSegments', 'xray:PutTelemetryRecords'],
73
+ resources: ['*'],
74
+ }));
75
+ // Note: _X_AMZN_TRACE_ID is automatically managed by Lambda runtime
76
+ });
77
+ // Enable tracing for Step Functions
78
+ // Note: Tracing should be enabled during state machine creation with tracingEnabled: true
79
+ }
80
+ /**
81
+ * Create CloudWatch alarms for monitoring critical metrics
82
+ */
83
+ createAlarms(resources, config) {
84
+ // Step Functions execution failure alarm (if state machine exists)
85
+ if (resources.stateMachine) {
86
+ const stepFunctionFailureAlarm = new cloudwatch.Alarm(this, 'StepFunctionFailureAlarm', {
87
+ alarmName: `${config.dashboardName}-StepFunction-Failures`,
88
+ alarmDescription: 'Step Function execution failures',
89
+ metric: resources.stateMachine.metricFailed({
90
+ period: cdk.Duration.minutes(5),
91
+ statistic: 'Sum',
92
+ }),
93
+ threshold: 1,
94
+ evaluationPeriods: 1,
95
+ treatMissingData: cloudwatch.TreatMissingData.NOT_BREACHING,
96
+ });
97
+ this.alarms.push(stepFunctionFailureAlarm);
98
+ }
99
+ // Lambda function error rate alarms
100
+ resources.lambdaFunctions.forEach((fn, index) => {
101
+ const errorAlarm = new cloudwatch.Alarm(this, `LambdaErrorAlarm-${index}`, {
102
+ alarmName: `${config.dashboardName}-Lambda-${fn.functionName}-Errors`,
103
+ alarmDescription: `High error rate for Lambda function ${fn.functionName}`,
104
+ metric: fn.metricErrors({
105
+ period: cdk.Duration.minutes(5),
106
+ statistic: 'Sum',
107
+ }),
108
+ threshold: 5,
109
+ evaluationPeriods: 2,
110
+ treatMissingData: cloudwatch.TreatMissingData.NOT_BREACHING,
111
+ });
112
+ this.alarms.push(errorAlarm);
113
+ // Lambda function duration alarm
114
+ const durationAlarm = new cloudwatch.Alarm(this, `LambdaDurationAlarm-${index}`, {
115
+ alarmName: `${config.dashboardName}-Lambda-${fn.functionName}-Duration`,
116
+ alarmDescription: `High duration for Lambda function ${fn.functionName}`,
117
+ metric: fn.metricDuration({
118
+ period: cdk.Duration.minutes(5),
119
+ statistic: 'Average',
120
+ }),
121
+ threshold: 30000, // 30 seconds
122
+ evaluationPeriods: 3,
123
+ treatMissingData: cloudwatch.TreatMissingData.NOT_BREACHING,
124
+ });
125
+ this.alarms.push(durationAlarm);
126
+ });
127
+ // SQS queue depth alarm (if queues exist)
128
+ if (resources.sqsQueues) {
129
+ resources.sqsQueues.forEach((queue, index) => {
130
+ const queueDepthAlarm = new cloudwatch.Alarm(this, `SQSDepthAlarm-${index}`, {
131
+ alarmName: `${config.dashboardName}-SQS-${queue.queueName}-Depth`,
132
+ alarmDescription: `High message count in SQS queue ${queue.queueName}`,
133
+ metric: queue.metricApproximateNumberOfMessagesVisible({
134
+ period: cdk.Duration.minutes(5),
135
+ statistic: 'Average',
136
+ }),
137
+ threshold: 100,
138
+ evaluationPeriods: 2,
139
+ treatMissingData: cloudwatch.TreatMissingData.NOT_BREACHING,
140
+ });
141
+ this.alarms.push(queueDepthAlarm);
142
+ });
143
+ }
144
+ // Dead letter queue alarm
145
+ if (resources.deadLetterQueue) {
146
+ const dlqAlarm = new cloudwatch.Alarm(this, 'DeadLetterQueueAlarm', {
147
+ alarmName: `${config.dashboardName}-DeadLetterQueue-Messages`,
148
+ alarmDescription: 'Messages in dead letter queue',
149
+ metric: resources.deadLetterQueue.metricApproximateNumberOfMessagesVisible({
150
+ period: cdk.Duration.minutes(5),
151
+ statistic: 'Sum',
152
+ }),
153
+ threshold: 1,
154
+ evaluationPeriods: 1,
155
+ treatMissingData: cloudwatch.TreatMissingData.NOT_BREACHING,
156
+ });
157
+ this.alarms.push(dlqAlarm);
158
+ }
159
+ }
160
+ /**
161
+ * Create comprehensive CloudWatch dashboard
162
+ */
163
+ createDashboard(resources, config) {
164
+ const dashboard = new cloudwatch.Dashboard(this, 'Dashboard', {
165
+ dashboardName: config.dashboardName,
166
+ });
167
+ // Step Functions metrics (if state machine exists)
168
+ if (resources.stateMachine) {
169
+ dashboard.addWidgets(new cloudwatch.GraphWidget({
170
+ title: 'Step Functions Executions',
171
+ left: [
172
+ resources.stateMachine.metricStarted({
173
+ label: 'Started',
174
+ color: cloudwatch.Color.BLUE,
175
+ }),
176
+ resources.stateMachine.metricSucceeded({
177
+ label: 'Succeeded',
178
+ color: cloudwatch.Color.GREEN,
179
+ }),
180
+ resources.stateMachine.metricFailed({
181
+ label: 'Failed',
182
+ color: cloudwatch.Color.RED,
183
+ }),
184
+ ],
185
+ width: 12,
186
+ height: 6,
187
+ }));
188
+ dashboard.addWidgets(new cloudwatch.GraphWidget({
189
+ title: 'Step Functions Execution Duration',
190
+ left: [
191
+ resources.stateMachine.metricTime({
192
+ label: 'Execution Time',
193
+ statistic: 'Average',
194
+ color: cloudwatch.Color.PURPLE,
195
+ }),
196
+ ],
197
+ width: 12,
198
+ height: 6,
199
+ }));
200
+ }
201
+ // Lambda functions metrics
202
+ const lambdaInvocations = resources.lambdaFunctions.map(fn => fn.metricInvocations({
203
+ label: fn.functionName,
204
+ }));
205
+ const lambdaErrors = resources.lambdaFunctions.map(fn => fn.metricErrors({
206
+ label: `${fn.functionName} Errors`,
207
+ color: cloudwatch.Color.RED,
208
+ }));
209
+ dashboard.addWidgets(new cloudwatch.GraphWidget({
210
+ title: 'Lambda Function Invocations',
211
+ left: lambdaInvocations,
212
+ width: 12,
213
+ height: 6,
214
+ }));
215
+ dashboard.addWidgets(new cloudwatch.GraphWidget({
216
+ title: 'Lambda Function Errors',
217
+ left: lambdaErrors,
218
+ width: 12,
219
+ height: 6,
220
+ }));
221
+ // SQS queue metrics (if queues exist)
222
+ if (resources.sqsQueues && resources.sqsQueues.length > 0) {
223
+ const sqsMessages = resources.sqsQueues.map(queue => queue.metricApproximateNumberOfMessagesVisible({
224
+ label: `${queue.queueName} Messages`,
225
+ }));
226
+ dashboard.addWidgets(new cloudwatch.GraphWidget({
227
+ title: 'SQS Queue Depth',
228
+ left: sqsMessages,
229
+ width: 12,
230
+ height: 6,
231
+ }));
232
+ }
233
+ // Custom metrics widget for use case specific metrics
234
+ dashboard.addWidgets(new cloudwatch.GraphWidget({
235
+ title: `${resources.useCaseName} Metrics`,
236
+ left: [
237
+ new cloudwatch.Metric({
238
+ namespace: this.metricsNamespace,
239
+ metricName: 'ProcessedItems',
240
+ statistic: 'Sum',
241
+ label: 'Items Processed',
242
+ color: cloudwatch.Color.GREEN,
243
+ }),
244
+ new cloudwatch.Metric({
245
+ namespace: this.metricsNamespace,
246
+ metricName: 'FailedItems',
247
+ statistic: 'Sum',
248
+ label: 'Items Failed',
249
+ color: cloudwatch.Color.RED,
250
+ }),
251
+ ],
252
+ width: 12,
253
+ height: 6,
254
+ }));
255
+ return dashboard;
256
+ }
257
+ /**
258
+ * Add custom widget to the dashboard
259
+ */
260
+ addCustomWidget(widget) {
261
+ if (this.dashboard) {
262
+ this.dashboard.addWidgets(widget);
263
+ }
264
+ }
265
+ /**
266
+ * Create a custom metric for use case specific events
267
+ */
268
+ createCustomMetric(metricName, dimensionsMap) {
269
+ return new cloudwatch.Metric({
270
+ namespace: this.metricsNamespace,
271
+ metricName,
272
+ dimensionsMap,
273
+ statistic: 'Sum',
274
+ });
275
+ }
276
+ }
277
+ exports.ServerlessObservability = ServerlessObservability;
278
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"observability.js","sourceRoot":"","sources":["../../../use-cases/utilities/observability/observability.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 cdk from 'aws-cdk-lib';\nimport * as cloudwatch from 'aws-cdk-lib/aws-cloudwatch';\nimport * as iam from 'aws-cdk-lib/aws-iam';\nimport * as lambda from 'aws-cdk-lib/aws-lambda';\nimport * as logs from 'aws-cdk-lib/aws-logs';\nimport * as s3 from 'aws-cdk-lib/aws-s3';\nimport * as sqs from 'aws-cdk-lib/aws-sqs';\nimport * as stepfunctions from 'aws-cdk-lib/aws-stepfunctions';\nimport { Construct } from 'constructs';\n\n/**\n * Configuration for observability features\n */\nexport interface ObservabilityConfig {\n  /** Enable CloudWatch Dashboard */\n  readonly enableDashboard?: boolean;\n\n  /** Enable CloudWatch Alarms */\n  readonly enableAlarms?: boolean;\n\n  /** Enable AWS X-Ray tracing */\n  readonly enableTracing?: boolean;\n\n  /** Log retention period in days */\n  readonly logRetentionDays?: logs.RetentionDays;\n\n  /** Custom dashboard name */\n  readonly dashboardName?: string;\n\n  /** SNS topic ARN for alarm notifications */\n  readonly alarmNotificationTopicArn?: string;\n\n  /** Custom metric namespace */\n  readonly metricsNamespace?: string;\n}\n\n/**\n * Resources that need observability monitoring\n */\nexport interface ObservabilityResources {\n  /** Step Functions state machine (optional) */\n  readonly stateMachine?: stepfunctions.StateMachine;\n\n  /** Lambda functions to monitor */\n  readonly lambdaFunctions: lambda.Function[];\n\n  /** SQS queues to monitor (optional) */\n  readonly sqsQueues?: sqs.Queue[];\n\n  /** S3 buckets for storage (optional) */\n  readonly s3Buckets?: s3.IBucket[];\n\n  /** Dead letter queue (optional) */\n  readonly deadLetterQueue?: sqs.IQueue;\n\n  /** Use case name for labeling */\n  readonly useCaseName: string;\n}\n\n/**\n * Comprehensive observability construct for serverless use cases\n */\nexport class ServerlessObservability extends Construct {\n  /** CloudWatch Dashboard for monitoring */\n  public readonly dashboard?: cloudwatch.Dashboard;\n\n  /** CloudWatch Log Groups */\n  public readonly logGroups: logs.LogGroup[];\n\n  /** CloudWatch Alarms */\n  public readonly alarms: cloudwatch.Alarm[];\n\n  /** Metrics namespace */\n  public readonly metricsNamespace: string;\n\n  constructor(\n    scope: Construct,\n    id: string,\n    resources: ObservabilityResources,\n    config: ObservabilityConfig = {},\n  ) {\n    super(scope, id);\n\n    // Configuration with defaults\n    const observabilityConfig = {\n      enableDashboard: true,\n      enableAlarms: true,\n      enableTracing: true,\n      logRetentionDays: logs.RetentionDays.ONE_MONTH,\n      dashboardName: `${resources.useCaseName}-${cdk.Stack.of(this).stackName}`,\n      metricsNamespace: resources.useCaseName,\n      ...config,\n    };\n\n    this.metricsNamespace = observabilityConfig.metricsNamespace;\n    this.logGroups = [];\n    this.alarms = [];\n\n    // Create log groups for Lambda functions\n    this.createLogGroups(resources.lambdaFunctions, observabilityConfig);\n\n    // Enable X-Ray tracing\n    if (observabilityConfig.enableTracing) {\n      this.enableTracing(resources);\n    }\n\n    // Create CloudWatch alarms\n    if (observabilityConfig.enableAlarms) {\n      this.createAlarms(resources, observabilityConfig);\n    }\n\n    // Create CloudWatch dashboard\n    if (observabilityConfig.enableDashboard) {\n      this.dashboard = this.createDashboard(resources, observabilityConfig);\n    }\n  }\n\n  /**\n   * Create log groups for Lambda functions with proper retention\n   */\n  private createLogGroups(\n    lambdaFunctions: lambda.Function[],\n    config: ObservabilityConfig,\n  ): void {\n    lambdaFunctions.forEach((fn, index) => {\n      const logGroup = new logs.LogGroup(this, `LogGroup-${index}`, {\n        logGroupName: `/aws/lambda/${fn.functionName}`,\n        retention: config.logRetentionDays!,\n        removalPolicy: cdk.RemovalPolicy.DESTROY,\n      });\n\n      this.logGroups.push(logGroup);\n    });\n  }\n\n  /**\n   * Enable AWS X-Ray tracing for all resources\n   */\n  private enableTracing(resources: ObservabilityResources): void {\n    // Enable X-Ray tracing for Lambda functions\n    resources.lambdaFunctions.forEach(fn => {\n      // Enable X-Ray tracing via Lambda configuration (not environment variables)\n      const cfnFunction = fn.node.defaultChild as lambda.CfnFunction;\n      cfnFunction.tracingConfig = {\n        mode: lambda.Tracing.ACTIVE,\n      };\n\n      // Add X-Ray permissions\n      fn.addToRolePolicy(new iam.PolicyStatement({\n        actions: ['xray:PutTraceSegments', 'xray:PutTelemetryRecords'],\n        resources: ['*'],\n      }));\n\n      // Note: _X_AMZN_TRACE_ID is automatically managed by Lambda runtime\n    });\n\n    // Enable tracing for Step Functions\n    // Note: Tracing should be enabled during state machine creation with tracingEnabled: true\n  }\n\n  /**\n   * Create CloudWatch alarms for monitoring critical metrics\n   */\n  private createAlarms(\n    resources: ObservabilityResources,\n    config: ObservabilityConfig,\n  ): void {\n    // Step Functions execution failure alarm (if state machine exists)\n    if (resources.stateMachine) {\n      const stepFunctionFailureAlarm = new cloudwatch.Alarm(this, 'StepFunctionFailureAlarm', {\n        alarmName: `${config.dashboardName}-StepFunction-Failures`,\n        alarmDescription: 'Step Function execution failures',\n        metric: resources.stateMachine.metricFailed({\n          period: cdk.Duration.minutes(5),\n          statistic: 'Sum',\n        }),\n        threshold: 1,\n        evaluationPeriods: 1,\n        treatMissingData: cloudwatch.TreatMissingData.NOT_BREACHING,\n      });\n      this.alarms.push(stepFunctionFailureAlarm);\n    }\n\n    // Lambda function error rate alarms\n    resources.lambdaFunctions.forEach((fn, index) => {\n      const errorAlarm = new cloudwatch.Alarm(this, `LambdaErrorAlarm-${index}`, {\n        alarmName: `${config.dashboardName}-Lambda-${fn.functionName}-Errors`,\n        alarmDescription: `High error rate for Lambda function ${fn.functionName}`,\n        metric: fn.metricErrors({\n          period: cdk.Duration.minutes(5),\n          statistic: 'Sum',\n        }),\n        threshold: 5,\n        evaluationPeriods: 2,\n        treatMissingData: cloudwatch.TreatMissingData.NOT_BREACHING,\n      });\n      this.alarms.push(errorAlarm);\n\n      // Lambda function duration alarm\n      const durationAlarm = new cloudwatch.Alarm(this, `LambdaDurationAlarm-${index}`, {\n        alarmName: `${config.dashboardName}-Lambda-${fn.functionName}-Duration`,\n        alarmDescription: `High duration for Lambda function ${fn.functionName}`,\n        metric: fn.metricDuration({\n          period: cdk.Duration.minutes(5),\n          statistic: 'Average',\n        }),\n        threshold: 30000, // 30 seconds\n        evaluationPeriods: 3,\n        treatMissingData: cloudwatch.TreatMissingData.NOT_BREACHING,\n      });\n      this.alarms.push(durationAlarm);\n    });\n\n    // SQS queue depth alarm (if queues exist)\n    if (resources.sqsQueues) {\n      resources.sqsQueues.forEach((queue, index) => {\n        const queueDepthAlarm = new cloudwatch.Alarm(this, `SQSDepthAlarm-${index}`, {\n          alarmName: `${config.dashboardName}-SQS-${queue.queueName}-Depth`,\n          alarmDescription: `High message count in SQS queue ${queue.queueName}`,\n          metric: queue.metricApproximateNumberOfMessagesVisible({\n            period: cdk.Duration.minutes(5),\n            statistic: 'Average',\n          }),\n          threshold: 100,\n          evaluationPeriods: 2,\n          treatMissingData: cloudwatch.TreatMissingData.NOT_BREACHING,\n        });\n        this.alarms.push(queueDepthAlarm);\n      });\n    }\n\n    // Dead letter queue alarm\n    if (resources.deadLetterQueue) {\n      const dlqAlarm = new cloudwatch.Alarm(this, 'DeadLetterQueueAlarm', {\n        alarmName: `${config.dashboardName}-DeadLetterQueue-Messages`,\n        alarmDescription: 'Messages in dead letter queue',\n        metric: resources.deadLetterQueue.metricApproximateNumberOfMessagesVisible({\n          period: cdk.Duration.minutes(5),\n          statistic: 'Sum',\n        }),\n        threshold: 1,\n        evaluationPeriods: 1,\n        treatMissingData: cloudwatch.TreatMissingData.NOT_BREACHING,\n      });\n      this.alarms.push(dlqAlarm);\n    }\n  }\n\n  /**\n   * Create comprehensive CloudWatch dashboard\n   */\n  private createDashboard(\n    resources: ObservabilityResources,\n    config: ObservabilityConfig,\n  ): cloudwatch.Dashboard {\n    const dashboard = new cloudwatch.Dashboard(this, 'Dashboard', {\n      dashboardName: config.dashboardName,\n    });\n\n    // Step Functions metrics (if state machine exists)\n    if (resources.stateMachine) {\n      dashboard.addWidgets(\n        new cloudwatch.GraphWidget({\n          title: 'Step Functions Executions',\n          left: [\n            resources.stateMachine.metricStarted({\n              label: 'Started',\n              color: cloudwatch.Color.BLUE,\n            }),\n            resources.stateMachine.metricSucceeded({\n              label: 'Succeeded',\n              color: cloudwatch.Color.GREEN,\n            }),\n            resources.stateMachine.metricFailed({\n              label: 'Failed',\n              color: cloudwatch.Color.RED,\n            }),\n          ],\n          width: 12,\n          height: 6,\n        }),\n      );\n\n      dashboard.addWidgets(\n        new cloudwatch.GraphWidget({\n          title: 'Step Functions Execution Duration',\n          left: [\n            resources.stateMachine.metricTime({\n              label: 'Execution Time',\n              statistic: 'Average',\n              color: cloudwatch.Color.PURPLE,\n            }),\n          ],\n          width: 12,\n          height: 6,\n        }),\n      );\n    }\n\n    // Lambda functions metrics\n    const lambdaInvocations = resources.lambdaFunctions.map(fn =>\n      fn.metricInvocations({\n        label: fn.functionName,\n      }),\n    );\n\n    const lambdaErrors = resources.lambdaFunctions.map(fn =>\n      fn.metricErrors({\n        label: `${fn.functionName} Errors`,\n        color: cloudwatch.Color.RED,\n      }),\n    );\n\n    dashboard.addWidgets(\n      new cloudwatch.GraphWidget({\n        title: 'Lambda Function Invocations',\n        left: lambdaInvocations,\n        width: 12,\n        height: 6,\n      }),\n    );\n\n    dashboard.addWidgets(\n      new cloudwatch.GraphWidget({\n        title: 'Lambda Function Errors',\n        left: lambdaErrors,\n        width: 12,\n        height: 6,\n      }),\n    );\n\n    // SQS queue metrics (if queues exist)\n    if (resources.sqsQueues && resources.sqsQueues.length > 0) {\n      const sqsMessages = resources.sqsQueues.map(queue =>\n        queue.metricApproximateNumberOfMessagesVisible({\n          label: `${queue.queueName} Messages`,\n        }),\n      );\n\n      dashboard.addWidgets(\n        new cloudwatch.GraphWidget({\n          title: 'SQS Queue Depth',\n          left: sqsMessages,\n          width: 12,\n          height: 6,\n        }),\n      );\n    }\n\n    // Custom metrics widget for use case specific metrics\n    dashboard.addWidgets(\n      new cloudwatch.GraphWidget({\n        title: `${resources.useCaseName} Metrics`,\n        left: [\n          new cloudwatch.Metric({\n            namespace: this.metricsNamespace,\n            metricName: 'ProcessedItems',\n            statistic: 'Sum',\n            label: 'Items Processed',\n            color: cloudwatch.Color.GREEN,\n          }),\n          new cloudwatch.Metric({\n            namespace: this.metricsNamespace,\n            metricName: 'FailedItems',\n            statistic: 'Sum',\n            label: 'Items Failed',\n            color: cloudwatch.Color.RED,\n          }),\n        ],\n        width: 12,\n        height: 6,\n      }),\n    );\n\n    return dashboard;\n  }\n\n  /**\n   * Add custom widget to the dashboard\n   */\n  public addCustomWidget(widget: cloudwatch.IWidget): void {\n    if (this.dashboard) {\n      this.dashboard.addWidgets(widget);\n    }\n  }\n\n  /**\n   * Create a custom metric for use case specific events\n   */\n  public createCustomMetric(\n    metricName: string,\n    dimensionsMap?: { [key: string]: string },\n  ): cloudwatch.Metric {\n    return new cloudwatch.Metric({\n      namespace: this.metricsNamespace,\n      metricName,\n      dimensionsMap,\n      statistic: 'Sum',\n    });\n  }\n}\n"]}
@@ -0,0 +1,32 @@
1
+ import { IMetric } from 'aws-cdk-lib/aws-cloudwatch';
2
+ import { LogGroupDataProtectionProps } from './log-group-data-protection-props';
3
+ /**
4
+ * Additional properties that constructs implementing the IObservable
5
+ * interface should extend as part of their input props
6
+ */
7
+ export interface ObservableProps {
8
+ /**
9
+ * Business metric service name dimension
10
+ * @default would be defined per use case
11
+ */
12
+ readonly metricServiceName?: string;
13
+ /**
14
+ * Business metric namespace
15
+ * @default would be defined per use case
16
+ */
17
+ readonly metricNamespace?: string;
18
+ /**
19
+ * Data protection related configuration
20
+ * @default a new KMS key would be generated
21
+ */
22
+ readonly logGroupDataProtection?: LogGroupDataProtectionProps;
23
+ }
24
+ /**
25
+ * Interface providing configuration parameters for constructs that support Observability
26
+ */
27
+ export interface IObservable {
28
+ readonly metricServiceName: string;
29
+ readonly metricNamespace: string;
30
+ readonly logGroupDataProtection: LogGroupDataProtectionProps;
31
+ metrics(): IMetric[];
32
+ }
@@ -0,0 +1,3 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoib2JzZXJ2YWJsZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3VzZS1jYXNlcy91dGlsaXRpZXMvb2JzZXJ2YWJpbGl0eS9vYnNlcnZhYmxlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyIvLyBDb3B5cmlnaHQgQW1hem9uLmNvbSwgSW5jLiBvciBpdHMgYWZmaWxpYXRlcy4gQWxsIFJpZ2h0cyBSZXNlcnZlZC5cbi8vIFNQRFgtTGljZW5zZS1JZGVudGlmaWVyOiBBcGFjaGUtMi4wXG5pbXBvcnQgeyBJTWV0cmljIH0gZnJvbSAnYXdzLWNkay1saWIvYXdzLWNsb3Vkd2F0Y2gnO1xuaW1wb3J0IHsgTG9nR3JvdXBEYXRhUHJvdGVjdGlvblByb3BzIH0gZnJvbSAnLi9sb2ctZ3JvdXAtZGF0YS1wcm90ZWN0aW9uLXByb3BzJztcblxuLyoqXG4gKiBBZGRpdGlvbmFsIHByb3BlcnRpZXMgdGhhdCBjb25zdHJ1Y3RzIGltcGxlbWVudGluZyB0aGUgSU9ic2VydmFibGVcbiAqIGludGVyZmFjZSBzaG91bGQgZXh0ZW5kIGFzIHBhcnQgb2YgdGhlaXIgaW5wdXQgcHJvcHNcbiAqL1xuZXhwb3J0IGludGVyZmFjZSBPYnNlcnZhYmxlUHJvcHMge1xuICAvKipcbiAgICogQnVzaW5lc3MgbWV0cmljIHNlcnZpY2UgbmFtZSBkaW1lbnNpb25cbiAgICogQGRlZmF1bHQgd291bGQgYmUgZGVmaW5lZCBwZXIgdXNlIGNhc2VcbiAgICovXG4gIHJlYWRvbmx5IG1ldHJpY1NlcnZpY2VOYW1lPzogc3RyaW5nO1xuXG4gIC8qKlxuICAgKiBCdXNpbmVzcyBtZXRyaWMgbmFtZXNwYWNlXG4gICAqIEBkZWZhdWx0IHdvdWxkIGJlIGRlZmluZWQgcGVyIHVzZSBjYXNlXG4gICAqL1xuICByZWFkb25seSBtZXRyaWNOYW1lc3BhY2U/OiBzdHJpbmc7XG5cbiAgLyoqXG4gICAqIERhdGEgcHJvdGVjdGlvbiByZWxhdGVkIGNvbmZpZ3VyYXRpb25cbiAgICogQGRlZmF1bHQgYSBuZXcgS01TIGtleSB3b3VsZCBiZSBnZW5lcmF0ZWRcbiAgICovXG4gIHJlYWRvbmx5IGxvZ0dyb3VwRGF0YVByb3RlY3Rpb24/OiBMb2dHcm91cERhdGFQcm90ZWN0aW9uUHJvcHM7XG59XG5cbi8qKlxuICogSW50ZXJmYWNlIHByb3ZpZGluZyBjb25maWd1cmF0aW9uIHBhcmFtZXRlcnMgZm9yIGNvbnN0cnVjdHMgdGhhdCBzdXBwb3J0IE9ic2VydmFiaWxpdHlcbiAqL1xuZXhwb3J0IGludGVyZmFjZSBJT2JzZXJ2YWJsZSB7XG4gIHJlYWRvbmx5IG1ldHJpY1NlcnZpY2VOYW1lOiBzdHJpbmc7XG4gIHJlYWRvbmx5IG1ldHJpY05hbWVzcGFjZTogc3RyaW5nO1xuICByZWFkb25seSBsb2dHcm91cERhdGFQcm90ZWN0aW9uOiBMb2dHcm91cERhdGFQcm90ZWN0aW9uUHJvcHM7XG5cbiAgbWV0cmljcygpOiBJTWV0cmljW107XG59Il19
@@ -0,0 +1,3 @@
1
+ export declare class PowertoolsConfig {
2
+ static generateDefaultLambdaConfig(enableObservability?: boolean, metricsNamespace?: string, serviceName?: string): Record<string, string>;
3
+ }
@@ -0,0 +1,25 @@
1
+ "use strict";
2
+ var _a;
3
+ Object.defineProperty(exports, "__esModule", { value: true });
4
+ exports.PowertoolsConfig = void 0;
5
+ const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
6
+ class PowertoolsConfig {
7
+ static generateDefaultLambdaConfig(enableObservability = false, metricsNamespace = 'appmod-catalog', serviceName = 'appmod-catalog-service') {
8
+ if (enableObservability) {
9
+ return {
10
+ POWERTOOLS_METRICS_NAMESPACE: metricsNamespace,
11
+ POWERTOOLS_SERVICE_NAME: serviceName,
12
+ };
13
+ }
14
+ else {
15
+ return {
16
+ POWERTOOLS_METRICS_DISABLED: 'true',
17
+ POWERTOOLS_TRACE_DISABLED: 'true',
18
+ };
19
+ }
20
+ }
21
+ }
22
+ exports.PowertoolsConfig = PowertoolsConfig;
23
+ _a = JSII_RTTI_SYMBOL_1;
24
+ PowertoolsConfig[_a] = { fqn: "@cdklabs/cdk-appmod-catalog-blueprints.PowertoolsConfig", version: "1.0.0" };
25
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicG93ZXJ0b29scy1jb25maWcuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi91c2UtY2FzZXMvdXRpbGl0aWVzL29ic2VydmFiaWxpdHkvcG93ZXJ0b29scy1jb25maWcudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7QUFBQSxNQUFhLGdCQUFnQjtJQUNwQixNQUFNLENBQUMsMkJBQTJCLENBQUMsc0JBQStCLEtBQUssRUFBRSxtQkFBMkIsZ0JBQWdCLEVBQUUsY0FBc0Isd0JBQXdCO1FBQ3pLLElBQUksbUJBQW1CLEVBQUUsQ0FBQztZQUN4QixPQUFPO2dCQUNMLDRCQUE0QixFQUFFLGdCQUFnQjtnQkFDOUMsdUJBQXVCLEVBQUUsV0FBVzthQUNyQyxDQUFDO1FBQ0osQ0FBQzthQUFNLENBQUM7WUFDTixPQUFPO2dCQUNMLDJCQUEyQixFQUFFLE1BQU07Z0JBQ25DLHlCQUF5QixFQUFFLE1BQU07YUFDbEMsQ0FBQztRQUNKLENBQUM7SUFDSCxDQUFDOztBQWJILDRDQWNDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IGNsYXNzIFBvd2VydG9vbHNDb25maWcge1xuICBwdWJsaWMgc3RhdGljIGdlbmVyYXRlRGVmYXVsdExhbWJkYUNvbmZpZyhlbmFibGVPYnNlcnZhYmlsaXR5OiBib29sZWFuID0gZmFsc2UsIG1ldHJpY3NOYW1lc3BhY2U6IHN0cmluZyA9ICdhcHBtb2QtY2F0YWxvZycsIHNlcnZpY2VOYW1lOiBzdHJpbmcgPSAnYXBwbW9kLWNhdGFsb2ctc2VydmljZScpOiBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+IHtcbiAgICBpZiAoZW5hYmxlT2JzZXJ2YWJpbGl0eSkge1xuICAgICAgcmV0dXJuIHtcbiAgICAgICAgUE9XRVJUT09MU19NRVRSSUNTX05BTUVTUEFDRTogbWV0cmljc05hbWVzcGFjZSxcbiAgICAgICAgUE9XRVJUT09MU19TRVJWSUNFX05BTUU6IHNlcnZpY2VOYW1lLFxuICAgICAgfTtcbiAgICB9IGVsc2Uge1xuICAgICAgcmV0dXJuIHtcbiAgICAgICAgUE9XRVJUT09MU19NRVRSSUNTX0RJU0FCTEVEOiAndHJ1ZScsXG4gICAgICAgIFBPV0VSVE9PTFNfVFJBQ0VfRElTQUJMRUQ6ICd0cnVlJyxcbiAgICAgIH07XG4gICAgfVxuICB9XG59Il19
@@ -0,0 +1,27 @@
1
+ import boto3
2
+
3
+ client = boto3.client('bedrock')
4
+
5
+ def handler(event, context):
6
+ request_type = event['RequestType']
7
+
8
+ # Only process CREATE and UPDATE events
9
+ if request_type not in ['Create', 'Update']:
10
+ return
11
+
12
+ current_logging_config = client.get_model_invocation_logging_configuration()
13
+
14
+ # Check if logging is already configured and override is not set
15
+ if current_logging_config.get('loggingConfig') and not event.get('ResourceProperties', {}).get('override', False):
16
+ return
17
+
18
+ # Update logging configuration
19
+ props = event['ResourceProperties']
20
+ client.put_model_invocation_logging_configuration(
21
+ loggingConfig={
22
+ 'cloudWatchConfig': {
23
+ 'logGroupName': props['logGroupName'],
24
+ 'roleArn': props['roleArn']
25
+ }
26
+ }
27
+ )
@@ -0,0 +1,8 @@
1
+ import { InjectionContext, IPropertyInjector } from 'aws-cdk-lib';
2
+ import { LogGroupDataProtectionProps } from './log-group-data-protection-props';
3
+ export declare class StateMachineObservabilityPropertyInjector implements IPropertyInjector {
4
+ readonly constructUniqueId: string;
5
+ readonly logGroupDataProtection: LogGroupDataProtectionProps;
6
+ constructor(logGroupDataProtection: LogGroupDataProtectionProps);
7
+ inject(originalProps: any, _context: InjectionContext): any;
8
+ }
@@ -0,0 +1,49 @@
1
+ "use strict";
2
+ var _a;
3
+ Object.defineProperty(exports, "__esModule", { value: true });
4
+ exports.StateMachineObservabilityPropertyInjector = void 0;
5
+ const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
6
+ const aws_cdk_lib_1 = require("aws-cdk-lib");
7
+ const aws_iam_1 = require("aws-cdk-lib/aws-iam");
8
+ const aws_logs_1 = require("aws-cdk-lib/aws-logs");
9
+ const aws_stepfunctions_1 = require("aws-cdk-lib/aws-stepfunctions");
10
+ class StateMachineObservabilityPropertyInjector {
11
+ constructor(logGroupDataProtection) {
12
+ this.constructUniqueId = aws_stepfunctions_1.StateMachine.PROPERTY_INJECTION_ID;
13
+ this.logGroupDataProtection = logGroupDataProtection;
14
+ }
15
+ inject(originalProps, _context) {
16
+ const namePrefix = aws_cdk_lib_1.Names.uniqueResourceName(_context.scope, {
17
+ maxLength: 100,
18
+ });
19
+ const { region, account } = aws_cdk_lib_1.Stack.of(_context.scope);
20
+ const logGroupName = `/aws/sfn-statemachine/${namePrefix}-statemachine`;
21
+ const logGroupArn = `arn:aws:logs:${region}:${account}:log-group:${logGroupName}`;
22
+ this.logGroupDataProtection.logGroupEncryptionKey?.grantEncryptDecrypt(new aws_iam_1.ServicePrincipal('logs.amazonaws.com', {
23
+ conditions: {
24
+ ArnEquals: {
25
+ 'kms:EncryptionContext:aws:logs:arn': logGroupArn,
26
+ },
27
+ },
28
+ }));
29
+ return {
30
+ tracingEnabled: true,
31
+ logs: {
32
+ destination: new aws_logs_1.LogGroup(_context.scope, `${_context.id}-LogGroup`, {
33
+ logGroupName,
34
+ encryptionKey: this.logGroupDataProtection.logGroupEncryptionKey,
35
+ dataProtectionPolicy: this.logGroupDataProtection.dataProtectionIdentifiers ? new aws_logs_1.DataProtectionPolicy({
36
+ identifiers: this.logGroupDataProtection.dataProtectionIdentifiers,
37
+ }) : undefined,
38
+ removalPolicy: aws_cdk_lib_1.RemovalPolicy.DESTROY,
39
+ }),
40
+ level: aws_stepfunctions_1.LogLevel.ALL,
41
+ },
42
+ ...originalProps,
43
+ };
44
+ }
45
+ }
46
+ exports.StateMachineObservabilityPropertyInjector = StateMachineObservabilityPropertyInjector;
47
+ _a = JSII_RTTI_SYMBOL_1;
48
+ StateMachineObservabilityPropertyInjector[_a] = { fqn: "@cdklabs/cdk-appmod-catalog-blueprints.StateMachineObservabilityPropertyInjector", version: "1.0.0" };
49
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RhdGUtbWFjaGluZS1vYnNlcnZhYmlsaXR5LXByb3BlcnR5LWluamVjdG9yLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vdXNlLWNhc2VzL3V0aWxpdGllcy9vYnNlcnZhYmlsaXR5L3N0YXRlLW1hY2hpbmUtb2JzZXJ2YWJpbGl0eS1wcm9wZXJ0eS1pbmplY3Rvci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7OztBQUFBLDZDQUErRjtBQUMvRixpREFBdUQ7QUFDdkQsbURBQXNFO0FBQ3RFLHFFQUF1RTtBQUd2RSxNQUFhLHlDQUF5QztJQUlwRCxZQUFZLHNCQUFtRDtRQUM3RCxJQUFJLENBQUMsaUJBQWlCLEdBQUcsZ0NBQVksQ0FBQyxxQkFBcUIsQ0FBQztRQUM1RCxJQUFJLENBQUMsc0JBQXNCLEdBQUcsc0JBQXNCLENBQUM7SUFDdkQsQ0FBQztJQUVELE1BQU0sQ0FBQyxhQUFrQixFQUFFLFFBQTBCO1FBQ25ELE1BQU0sVUFBVSxHQUFHLG1CQUFLLENBQUMsa0JBQWtCLENBQUMsUUFBUSxDQUFDLEtBQUssRUFBRTtZQUMxRCxTQUFTLEVBQUUsR0FBRztTQUNmLENBQUMsQ0FBQztRQUNILE1BQU0sRUFBRSxNQUFNLEVBQUUsT0FBTyxFQUFFLEdBQUcsbUJBQUssQ0FBQyxFQUFFLENBQUMsUUFBUSxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBQ3JELE1BQU0sWUFBWSxHQUFHLHlCQUF5QixVQUFVLGVBQWUsQ0FBQztRQUN4RSxNQUFNLFdBQVcsR0FBRyxnQkFBZ0IsTUFBTSxJQUFJLE9BQU8sY0FBYyxZQUFZLEVBQUUsQ0FBQztRQUNsRixJQUFJLENBQUMsc0JBQXNCLENBQUMscUJBQXFCLEVBQUUsbUJBQW1CLENBQUMsSUFBSSwwQkFBZ0IsQ0FBQyxvQkFBb0IsRUFBRTtZQUNoSCxVQUFVLEVBQUU7Z0JBQ1YsU0FBUyxFQUFFO29CQUNULG9DQUFvQyxFQUFFLFdBQVc7aUJBQ2xEO2FBQ0Y7U0FDRixDQUFDLENBQUMsQ0FBQztRQUVKLE9BQU87WUFDTCxjQUFjLEVBQUUsSUFBSTtZQUNwQixJQUFJLEVBQUU7Z0JBQ0osV0FBVyxFQUFFLElBQUksbUJBQVEsQ0FBQyxRQUFRLENBQUMsS0FBSyxFQUFFLEdBQUcsUUFBUSxDQUFDLEVBQUUsV0FBVyxFQUFFO29CQUNuRSxZQUFZO29CQUNaLGFBQWEsRUFBRSxJQUFJLENBQUMsc0JBQXNCLENBQUMscUJBQXFCO29CQUNoRSxvQkFBb0IsRUFBRSxJQUFJLENBQUMsc0JBQXNCLENBQUMseUJBQXlCLENBQUMsQ0FBQyxDQUFDLElBQUksK0JBQW9CLENBQUM7d0JBQ3JHLFdBQVcsRUFBRSxJQUFJLENBQUMsc0JBQXNCLENBQUMseUJBQXlCO3FCQUNuRSxDQUFDLENBQUMsQ0FBQyxDQUFDLFNBQVM7b0JBQ2QsYUFBYSxFQUFFLDJCQUFhLENBQUMsT0FBTztpQkFDckMsQ0FBQztnQkFDRixLQUFLLEVBQUUsNEJBQVEsQ0FBQyxHQUFHO2FBQ3BCO1lBQ0QsR0FBRyxhQUFhO1NBQ2pCLENBQUM7SUFDSixDQUFDOztBQXZDSCw4RkF5Q0MiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBJbmplY3Rpb25Db250ZXh0LCBJUHJvcGVydHlJbmplY3RvciwgTmFtZXMsIFJlbW92YWxQb2xpY3ksIFN0YWNrIH0gZnJvbSAnYXdzLWNkay1saWInO1xuaW1wb3J0IHsgU2VydmljZVByaW5jaXBhbCB9IGZyb20gJ2F3cy1jZGstbGliL2F3cy1pYW0nO1xuaW1wb3J0IHsgRGF0YVByb3RlY3Rpb25Qb2xpY3ksIExvZ0dyb3VwIH0gZnJvbSAnYXdzLWNkay1saWIvYXdzLWxvZ3MnO1xuaW1wb3J0IHsgTG9nTGV2ZWwsIFN0YXRlTWFjaGluZSB9IGZyb20gJ2F3cy1jZGstbGliL2F3cy1zdGVwZnVuY3Rpb25zJztcbmltcG9ydCB7IExvZ0dyb3VwRGF0YVByb3RlY3Rpb25Qcm9wcyB9IGZyb20gJy4vbG9nLWdyb3VwLWRhdGEtcHJvdGVjdGlvbi1wcm9wcyc7XG5cbmV4cG9ydCBjbGFzcyBTdGF0ZU1hY2hpbmVPYnNlcnZhYmlsaXR5UHJvcGVydHlJbmplY3RvciBpbXBsZW1lbnRzIElQcm9wZXJ0eUluamVjdG9yIHtcbiAgcmVhZG9ubHkgY29uc3RydWN0VW5pcXVlSWQ6IHN0cmluZztcbiAgcmVhZG9ubHkgbG9nR3JvdXBEYXRhUHJvdGVjdGlvbjogTG9nR3JvdXBEYXRhUHJvdGVjdGlvblByb3BzO1xuXG4gIGNvbnN0cnVjdG9yKGxvZ0dyb3VwRGF0YVByb3RlY3Rpb246IExvZ0dyb3VwRGF0YVByb3RlY3Rpb25Qcm9wcykge1xuICAgIHRoaXMuY29uc3RydWN0VW5pcXVlSWQgPSBTdGF0ZU1hY2hpbmUuUFJPUEVSVFlfSU5KRUNUSU9OX0lEO1xuICAgIHRoaXMubG9nR3JvdXBEYXRhUHJvdGVjdGlvbiA9IGxvZ0dyb3VwRGF0YVByb3RlY3Rpb247XG4gIH1cblxuICBpbmplY3Qob3JpZ2luYWxQcm9wczogYW55LCBfY29udGV4dDogSW5qZWN0aW9uQ29udGV4dCk6IGFueSB7XG4gICAgY29uc3QgbmFtZVByZWZpeCA9IE5hbWVzLnVuaXF1ZVJlc291cmNlTmFtZShfY29udGV4dC5zY29wZSwge1xuICAgICAgbWF4TGVuZ3RoOiAxMDAsXG4gICAgfSk7XG4gICAgY29uc3QgeyByZWdpb24sIGFjY291bnQgfSA9IFN0YWNrLm9mKF9jb250ZXh0LnNjb3BlKTtcbiAgICBjb25zdCBsb2dHcm91cE5hbWUgPSBgL2F3cy9zZm4tc3RhdGVtYWNoaW5lLyR7bmFtZVByZWZpeH0tc3RhdGVtYWNoaW5lYDtcbiAgICBjb25zdCBsb2dHcm91cEFybiA9IGBhcm46YXdzOmxvZ3M6JHtyZWdpb259OiR7YWNjb3VudH06bG9nLWdyb3VwOiR7bG9nR3JvdXBOYW1lfWA7XG4gICAgdGhpcy5sb2dHcm91cERhdGFQcm90ZWN0aW9uLmxvZ0dyb3VwRW5jcnlwdGlvbktleT8uZ3JhbnRFbmNyeXB0RGVjcnlwdChuZXcgU2VydmljZVByaW5jaXBhbCgnbG9ncy5hbWF6b25hd3MuY29tJywge1xuICAgICAgY29uZGl0aW9uczoge1xuICAgICAgICBBcm5FcXVhbHM6IHtcbiAgICAgICAgICAna21zOkVuY3J5cHRpb25Db250ZXh0OmF3czpsb2dzOmFybic6IGxvZ0dyb3VwQXJuLFxuICAgICAgICB9LFxuICAgICAgfSxcbiAgICB9KSk7XG5cbiAgICByZXR1cm4ge1xuICAgICAgdHJhY2luZ0VuYWJsZWQ6IHRydWUsXG4gICAgICBsb2dzOiB7XG4gICAgICAgIGRlc3RpbmF0aW9uOiBuZXcgTG9nR3JvdXAoX2NvbnRleHQuc2NvcGUsIGAke19jb250ZXh0LmlkfS1Mb2dHcm91cGAsIHtcbiAgICAgICAgICBsb2dHcm91cE5hbWUsXG4gICAgICAgICAgZW5jcnlwdGlvbktleTogdGhpcy5sb2dHcm91cERhdGFQcm90ZWN0aW9uLmxvZ0dyb3VwRW5jcnlwdGlvbktleSxcbiAgICAgICAgICBkYXRhUHJvdGVjdGlvblBvbGljeTogdGhpcy5sb2dHcm91cERhdGFQcm90ZWN0aW9uLmRhdGFQcm90ZWN0aW9uSWRlbnRpZmllcnMgPyBuZXcgRGF0YVByb3RlY3Rpb25Qb2xpY3koe1xuICAgICAgICAgICAgaWRlbnRpZmllcnM6IHRoaXMubG9nR3JvdXBEYXRhUHJvdGVjdGlvbi5kYXRhUHJvdGVjdGlvbklkZW50aWZpZXJzLFxuICAgICAgICAgIH0pIDogdW5kZWZpbmVkLFxuICAgICAgICAgIHJlbW92YWxQb2xpY3k6IFJlbW92YWxQb2xpY3kuREVTVFJPWSxcbiAgICAgICAgfSksXG4gICAgICAgIGxldmVsOiBMb2dMZXZlbC5BTEwsXG4gICAgICB9LFxuICAgICAgLi4ub3JpZ2luYWxQcm9wcyxcbiAgICB9O1xuICB9XG5cbn0iXX0=
@@ -0,0 +1 @@
1
+ export {};