@rio-cloud/cdk-v2-constructs 2.5.5 → 2.5.8

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/.jsii CHANGED
@@ -3044,7 +3044,7 @@
3044
3044
  "assembly": "@rio-cloud/cdk-v2-constructs",
3045
3045
  "base": "constructs.Construct",
3046
3046
  "docs": {
3047
- "remarks": "The construct creates monitoring for AWS Backup jobs",
3047
+ "remarks": "The construct creates monitoring for AWS Backup jobs.",
3048
3048
  "stability": "stable",
3049
3049
  "summary": "This is a beta construct and is likely to have breaking changes."
3050
3050
  },
@@ -3055,7 +3055,7 @@
3055
3055
  },
3056
3056
  "locationInModule": {
3057
3057
  "filename": "src/contributions/team-claid/aws-backup-monitoring/aws-backup-monitoring.ts",
3058
- "line": 15
3058
+ "line": 16
3059
3059
  },
3060
3060
  "parameters": [
3061
3061
  {
@@ -3069,17 +3069,101 @@
3069
3069
  "type": {
3070
3070
  "primitive": "string"
3071
3071
  }
3072
+ },
3073
+ {
3074
+ "name": "props",
3075
+ "type": {
3076
+ "fqn": "@rio-cloud/cdk-v2-constructs.AwsBackupMonitoringProps"
3077
+ }
3072
3078
  }
3073
3079
  ]
3074
3080
  },
3075
3081
  "kind": "class",
3076
3082
  "locationInModule": {
3077
3083
  "filename": "src/contributions/team-claid/aws-backup-monitoring/aws-backup-monitoring.ts",
3078
- "line": 14
3084
+ "line": 15
3079
3085
  },
3080
3086
  "name": "AwsBackupMonitoring",
3081
3087
  "symbolId": "src/contributions/team-claid/aws-backup-monitoring/aws-backup-monitoring:AwsBackupMonitoring"
3082
3088
  },
3089
+ "@rio-cloud/cdk-v2-constructs.AwsBackupMonitoringProps": {
3090
+ "assembly": "@rio-cloud/cdk-v2-constructs",
3091
+ "datatype": true,
3092
+ "docs": {
3093
+ "stability": "stable"
3094
+ },
3095
+ "fqn": "@rio-cloud/cdk-v2-constructs.AwsBackupMonitoringProps",
3096
+ "kind": "interface",
3097
+ "locationInModule": {
3098
+ "filename": "src/contributions/team-claid/aws-backup-monitoring/aws-backup-monitoring.ts",
3099
+ "line": 7
3100
+ },
3101
+ "name": "AwsBackupMonitoringProps",
3102
+ "properties": [
3103
+ {
3104
+ "abstract": true,
3105
+ "docs": {
3106
+ "stability": "stable"
3107
+ },
3108
+ "immutable": true,
3109
+ "locationInModule": {
3110
+ "filename": "src/contributions/team-claid/aws-backup-monitoring/aws-backup-monitoring.ts",
3111
+ "line": 8
3112
+ },
3113
+ "name": "serviceName",
3114
+ "type": {
3115
+ "primitive": "string"
3116
+ }
3117
+ }
3118
+ ],
3119
+ "symbolId": "src/contributions/team-claid/aws-backup-monitoring/aws-backup-monitoring:AwsBackupMonitoringProps"
3120
+ },
3121
+ "@rio-cloud/cdk-v2-constructs.AwsBackupOverallMonitoring": {
3122
+ "assembly": "@rio-cloud/cdk-v2-constructs",
3123
+ "base": "constructs.Construct",
3124
+ "docs": {
3125
+ "remarks": "The construct creates monitoring for all AWS Backup jobs.",
3126
+ "stability": "stable",
3127
+ "summary": "This is a beta construct and is likely to have breaking changes."
3128
+ },
3129
+ "fqn": "@rio-cloud/cdk-v2-constructs.AwsBackupOverallMonitoring",
3130
+ "initializer": {
3131
+ "docs": {
3132
+ "stability": "stable"
3133
+ },
3134
+ "locationInModule": {
3135
+ "filename": "src/contributions/team-claid/aws-backup-monitoring/aws-backup-monitoring.ts",
3136
+ "line": 56
3137
+ },
3138
+ "parameters": [
3139
+ {
3140
+ "name": "scope",
3141
+ "type": {
3142
+ "fqn": "constructs.Construct"
3143
+ }
3144
+ },
3145
+ {
3146
+ "name": "id",
3147
+ "type": {
3148
+ "primitive": "string"
3149
+ }
3150
+ },
3151
+ {
3152
+ "name": "props",
3153
+ "type": {
3154
+ "fqn": "@rio-cloud/cdk-v2-constructs.AwsBackupMonitoringProps"
3155
+ }
3156
+ }
3157
+ ]
3158
+ },
3159
+ "kind": "class",
3160
+ "locationInModule": {
3161
+ "filename": "src/contributions/team-claid/aws-backup-monitoring/aws-backup-monitoring.ts",
3162
+ "line": 55
3163
+ },
3164
+ "name": "AwsBackupOverallMonitoring",
3165
+ "symbolId": "src/contributions/team-claid/aws-backup-monitoring/aws-backup-monitoring:AwsBackupOverallMonitoring"
3166
+ },
3083
3167
  "@rio-cloud/cdk-v2-constructs.Claidometer": {
3084
3168
  "assembly": "@rio-cloud/cdk-v2-constructs",
3085
3169
  "base": "constructs.Construct",
@@ -4107,7 +4191,7 @@
4107
4191
  },
4108
4192
  "locationInModule": {
4109
4193
  "filename": "src/contributions/team-claid/datadog-usage-monitoring/datadog-usage-monitoring.ts",
4110
- "line": 36
4194
+ "line": 37
4111
4195
  },
4112
4196
  "parameters": [
4113
4197
  {
@@ -4133,7 +4217,7 @@
4133
4217
  "kind": "class",
4134
4218
  "locationInModule": {
4135
4219
  "filename": "src/contributions/team-claid/datadog-usage-monitoring/datadog-usage-monitoring.ts",
4136
- "line": 35
4220
+ "line": 36
4137
4221
  },
4138
4222
  "name": "DatadogUsageMonitoring",
4139
4223
  "symbolId": "src/contributions/team-claid/datadog-usage-monitoring/datadog-usage-monitoring:DatadogUsageMonitoring"
@@ -4148,7 +4232,7 @@
4148
4232
  "kind": "interface",
4149
4233
  "locationInModule": {
4150
4234
  "filename": "src/contributions/team-claid/datadog-usage-monitoring/datadog-usage-monitoring.ts",
4151
- "line": 25
4235
+ "line": 26
4152
4236
  },
4153
4237
  "name": "DatadogUsageMonitoringProps",
4154
4238
  "properties": [
@@ -4160,7 +4244,7 @@
4160
4244
  "immutable": true,
4161
4245
  "locationInModule": {
4162
4246
  "filename": "src/contributions/team-claid/datadog-usage-monitoring/datadog-usage-monitoring.ts",
4163
- "line": 27
4247
+ "line": 28
4164
4248
  },
4165
4249
  "name": "alertType",
4166
4250
  "type": {
@@ -4175,7 +4259,7 @@
4175
4259
  "immutable": true,
4176
4260
  "locationInModule": {
4177
4261
  "filename": "src/contributions/team-claid/datadog-usage-monitoring/datadog-usage-monitoring.ts",
4178
- "line": 28
4262
+ "line": 29
4179
4263
  },
4180
4264
  "name": "organization",
4181
4265
  "type": {
@@ -4190,7 +4274,7 @@
4190
4274
  "immutable": true,
4191
4275
  "locationInModule": {
4192
4276
  "filename": "src/contributions/team-claid/datadog-usage-monitoring/datadog-usage-monitoring.ts",
4193
- "line": 26
4277
+ "line": 27
4194
4278
  },
4195
4279
  "name": "serviceName",
4196
4280
  "type": {
@@ -8363,5 +8447,5 @@
8363
8447
  }
8364
8448
  },
8365
8449
  "version": "0.0.0",
8366
- "fingerprint": "vOr//f16Fo35AhSD4dRPQBwBHnZXUUIsEjV00WBbqRE="
8450
+ "fingerprint": "OIkW1e+znFP7aRfsj3BMjhlsdgTx8lf1pc1HNxRZeq8="
8367
8451
  }
package/CHANGELOG.md CHANGED
@@ -2,6 +2,17 @@
2
2
 
3
3
  All notable changes to this project will be documented in this file. See [standard-version](https://github.com/conventional-changelog/standard-version) for commit guidelines.
4
4
 
5
+ ### [2.5.8](https://collaboration.msi.audi.com/stash/projects/RIODEV/repos/cdk-v2-constructs/compare/commits?targetBranch=refs%2Ftags%2Fv2.5.7&sourceBranch=refs%2Ftags%2Fv2.5.8) (2022-09-02)
6
+
7
+ ### [2.5.7](https://collaboration.msi.audi.com/stash/projects/RIODEV/repos/cdk-v2-constructs/compare/commits?targetBranch=refs%2Ftags%2Fv2.5.6&sourceBranch=refs%2Ftags%2Fv2.5.7) (2022-08-22)
8
+
9
+
10
+ ### Bug Fixes
11
+
12
+ * :bug: Fix issue with construct `KafkaTopic` failing when more than 1 topic is created in the same stack ([34cbcaa](https://collaboration.msi.audi.com/stash/projects/RIODEV/repos/cdk-v2-constructs/commits/34cbcaa5cc4513a7353a9035e5df7a93993ba8bb))
13
+
14
+ ### [2.5.6](https://collaboration.msi.audi.com/stash/projects/RIODEV/repos/cdk-v2-constructs/compare/commits?targetBranch=refs%2Ftags%2Fv2.5.5&sourceBranch=refs%2Ftags%2Fv2.5.6) (2022-08-02)
15
+
5
16
  ### [2.5.5](https://collaboration.msi.audi.com/stash/projects/RIODEV/repos/cdk-v2-constructs/compare/commits?targetBranch=refs%2Ftags%2Fv2.5.4&sourceBranch=refs%2Ftags%2Fv2.5.5) (2022-07-29)
6
17
 
7
18
  ### [2.5.4](https://collaboration.msi.audi.com/stash/projects/RIODEV/repos/cdk-v2-constructs/compare/commits?targetBranch=refs%2Ftags%2Fv2.5.3&sourceBranch=refs%2Ftags%2Fv2.5.4) (2022-07-28)
@@ -1,8 +1,18 @@
1
- import { Construct } from 'constructs';
1
+ import * as constructs from 'constructs';
2
+ export interface AwsBackupMonitoringProps {
3
+ readonly serviceName: string;
4
+ }
5
+ /**
6
+ * This is a beta construct and is likely to have breaking changes.
7
+ * The construct creates monitoring for AWS Backup jobs.
8
+ */
9
+ export declare class AwsBackupMonitoring extends constructs.Construct {
10
+ constructor(scope: constructs.Construct, id: string, props: AwsBackupMonitoringProps);
11
+ }
2
12
  /**
3
13
  * This is a beta construct and is likely to have breaking changes.
4
- * The construct creates monitoring for AWS Backup jobs
14
+ * The construct creates monitoring for all AWS Backup jobs.
5
15
  */
6
- export declare class AwsBackupMonitoring extends Construct {
7
- constructor(scope: Construct, id: string);
16
+ export declare class AwsBackupOverallMonitoring extends constructs.Construct {
17
+ constructor(scope: constructs.Construct, id: string, props: AwsBackupMonitoringProps);
8
18
  }
@@ -1,48 +1,96 @@
1
1
  "use strict";
2
- var _a;
2
+ var _a, _b;
3
3
  Object.defineProperty(exports, "__esModule", { value: true });
4
- exports.AwsBackupMonitoring = void 0;
4
+ exports.AwsBackupOverallMonitoring = exports.AwsBackupMonitoring = void 0;
5
5
  const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
6
6
  const path = require("path");
7
- const aws_cdk_lib_1 = require("aws-cdk-lib");
8
- const constructs_1 = require("constructs");
7
+ const cdk = require("aws-cdk-lib");
8
+ const ssm = require("aws-cdk-lib/aws-ssm");
9
+ const constructs = require("constructs");
9
10
  const rio_claidometer_1 = require("../../../rio-claidometer");
10
11
  /**
11
12
  * This is a beta construct and is likely to have breaking changes.
12
- * The construct creates monitoring for AWS Backup jobs
13
+ * The construct creates monitoring for AWS Backup jobs.
13
14
  */
14
- class AwsBackupMonitoring extends constructs_1.Construct {
15
- constructor(scope, id) {
15
+ class AwsBackupMonitoring extends constructs.Construct {
16
+ constructor(scope, id, props) {
16
17
  super(scope, id);
17
- const teamName = aws_cdk_lib_1.aws_ssm.StringParameter.fromStringParameterName(this, 'TeamName', '/config/team/name').stringValue;
18
- new aws_cdk_lib_1.CfnResource(this, 'BackupMonitoringCopyJobs', {
18
+ const teamName = ssm.StringParameter.fromStringParameterName(this, 'TeamName', '/config/team/name').stringValue;
19
+ const commonTags = [
20
+ `service:${props.serviceName}`,
21
+ `team:${teamName}`,
22
+ ];
23
+ new cdk.CfnResource(this, 'BackupMonitoringCopyJobs', {
19
24
  type: 'Datadog::Monitors::Monitor',
20
25
  properties: {
21
- Name: `[P4] : ${teamName} AWS Backup Copy Job to central account failed`,
26
+ Name: 'AWS Backup Copy Job to central account failed',
22
27
  Type: 'query alert',
23
- Query: `sum(last_1d):sum:aws.backup.number_of_copy_jobs_failed{account_id:${aws_cdk_lib_1.Stack.of(this).account}}.as_count() > 0`,
24
- Message: `AWS Backup Copy job to central backup account has failed @opsgenie-${teamName}.\nCheck the job under AWS Backup -> Jobs -> Copy Jobs`,
28
+ Query: `sum(last_1d):sum:aws.backup.number_of_copy_jobs_failed{account_id:${cdk.Stack.of(this).account}}.as_count() > 0`,
29
+ Message: `[P4] AWS Backup Copy job to central backup account has failed @opsgenie-${teamName}.\nCheck the job under AWS Backup -> Jobs -> Copy Jobs`,
30
+ Tags: commonTags,
25
31
  },
26
32
  });
27
- new aws_cdk_lib_1.CfnResource(this, 'BackupMonitoringJob', {
33
+ new cdk.CfnResource(this, 'BackupMonitoringJob', {
28
34
  type: 'Datadog::Monitors::Monitor',
29
35
  properties: {
30
- Name: `[P4] : ${teamName} AWS Backup Job Monitor`,
36
+ Name: 'AWS Backup Job Monitor',
31
37
  Type: 'query alert',
32
- Query: `sum(last_1d):sum:aws.backup.number_of_backup_jobs_failed{account_id:${aws_cdk_lib_1.Stack.of(this).account}}.as_count() > 0`,
33
- Message: `AWS Backup job has failed @opsgenie-${teamName}.\nCheck the job under AWS Backup -> Jobs -> Backup Jobs`,
38
+ Query: `sum(last_1d):sum:aws.backup.number_of_backup_jobs_failed{account_id:${cdk.Stack.of(this).account}}.as_count() > 0`,
39
+ Message: `[P4] AWS Backup job has failed @opsgenie-${teamName}.\nCheck the job under AWS Backup -> Jobs -> Backup Jobs`,
40
+ Tags: commonTags,
34
41
  },
35
42
  });
36
- const [feature, packageName] = __dirname.split(path.sep).reverse();
37
- new rio_claidometer_1.Claidometer(this, 'Claidometer', {
38
- product: rio_claidometer_1.ClaidometerProduct.CDK_CONTRIBUTION,
39
- packageName,
40
- feature,
41
- version: '0.0.4',
42
- });
43
+ addClaidometer(this);
43
44
  }
44
45
  }
45
46
  exports.AwsBackupMonitoring = AwsBackupMonitoring;
46
47
  _a = JSII_RTTI_SYMBOL_1;
47
48
  AwsBackupMonitoring[_a] = { fqn: "@rio-cloud/cdk-v2-constructs.AwsBackupMonitoring", version: "0.0.0" };
48
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXdzLWJhY2t1cC1tb25pdG9yaW5nLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vc3JjL2NvbnRyaWJ1dGlvbnMvdGVhbS1jbGFpZC9hd3MtYmFja3VwLW1vbml0b3JpbmcvYXdzLWJhY2t1cC1tb25pdG9yaW5nLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7O0FBQUEsNkJBQTZCO0FBQzdCLDZDQUlxQjtBQUNyQiwyQ0FBdUM7QUFDdkMsOERBQTJFO0FBRTNFOzs7R0FHRztBQUNILE1BQWEsbUJBQW9CLFNBQVEsc0JBQVM7SUFDaEQsWUFBWSxLQUFnQixFQUFFLEVBQVU7UUFDdEMsS0FBSyxDQUFDLEtBQUssRUFBRSxFQUFFLENBQUMsQ0FBQztRQUVqQixNQUFNLFFBQVEsR0FBRyxxQkFBTyxDQUFDLGVBQWUsQ0FBQyx1QkFBdUIsQ0FBQyxJQUFJLEVBQUUsVUFBVSxFQUFFLG1CQUFtQixDQUFDLENBQUMsV0FBVyxDQUFDO1FBRXBILElBQUkseUJBQVcsQ0FBQyxJQUFJLEVBQUUsMEJBQTBCLEVBQUU7WUFDaEQsSUFBSSxFQUFFLDRCQUE0QjtZQUNsQyxVQUFVLEVBQUU7Z0JBQ1YsSUFBSSxFQUFFLFVBQVUsUUFBUSxnREFBZ0Q7Z0JBQ3hFLElBQUksRUFBRSxhQUFhO2dCQUNuQixLQUFLLEVBQUUscUVBQXFFLG1CQUFLLENBQUMsRUFBRSxDQUFDLElBQUksQ0FBQyxDQUFDLE9BQU8sa0JBQWtCO2dCQUNwSCxPQUFPLEVBQUUsc0VBQXNFLFFBQVEsd0RBQXdEO2FBQ2hKO1NBQ0YsQ0FBQyxDQUFDO1FBRUgsSUFBSSx5QkFBVyxDQUFDLElBQUksRUFBRSxxQkFBcUIsRUFBRTtZQUMzQyxJQUFJLEVBQUUsNEJBQTRCO1lBQ2xDLFVBQVUsRUFBRTtnQkFDVixJQUFJLEVBQUUsVUFBVSxRQUFRLHlCQUF5QjtnQkFDakQsSUFBSSxFQUFFLGFBQWE7Z0JBQ25CLEtBQUssRUFBRSx1RUFBdUUsbUJBQUssQ0FBQyxFQUFFLENBQUMsSUFBSSxDQUFDLENBQUMsT0FBTyxrQkFBa0I7Z0JBQ3RILE9BQU8sRUFBRSx1Q0FBdUMsUUFBUSwwREFBMEQ7YUFDbkg7U0FDRixDQUFDLENBQUM7UUFFSCxNQUFNLENBQUMsT0FBTyxFQUFFLFdBQVcsQ0FBQyxHQUFHLFNBQVMsQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxDQUFDLE9BQU8sRUFBRSxDQUFDO1FBQ25FLElBQUksNkJBQVcsQ0FBQyxJQUFJLEVBQUUsYUFBYSxFQUFFO1lBQ25DLE9BQU8sRUFBRSxvQ0FBa0IsQ0FBQyxnQkFBZ0I7WUFDNUMsV0FBVztZQUNYLE9BQU87WUFDUCxPQUFPLEVBQUUsT0FBTztTQUNqQixDQUFDLENBQUM7SUFDTCxDQUFDOztBQWpDSCxrREFrQ0MiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyBwYXRoIGZyb20gJ3BhdGgnO1xuaW1wb3J0IHtcbiAgQ2ZuUmVzb3VyY2UsXG4gIFN0YWNrLFxuICBhd3Nfc3NtLFxufSBmcm9tICdhd3MtY2RrLWxpYic7XG5pbXBvcnQgeyBDb25zdHJ1Y3QgfSBmcm9tICdjb25zdHJ1Y3RzJztcbmltcG9ydCB7IENsYWlkb21ldGVyLCBDbGFpZG9tZXRlclByb2R1Y3QgfSBmcm9tICcuLi8uLi8uLi9yaW8tY2xhaWRvbWV0ZXInO1xuXG4vKipcbiAqIFRoaXMgaXMgYSBiZXRhIGNvbnN0cnVjdCBhbmQgaXMgbGlrZWx5IHRvIGhhdmUgYnJlYWtpbmcgY2hhbmdlcy5cbiAqIFRoZSBjb25zdHJ1Y3QgY3JlYXRlcyBtb25pdG9yaW5nIGZvciBBV1MgQmFja3VwIGpvYnNcbiAqL1xuZXhwb3J0IGNsYXNzIEF3c0JhY2t1cE1vbml0b3JpbmcgZXh0ZW5kcyBDb25zdHJ1Y3Qge1xuICBjb25zdHJ1Y3RvcihzY29wZTogQ29uc3RydWN0LCBpZDogc3RyaW5nKSB7XG4gICAgc3VwZXIoc2NvcGUsIGlkKTtcblxuICAgIGNvbnN0IHRlYW1OYW1lID0gYXdzX3NzbS5TdHJpbmdQYXJhbWV0ZXIuZnJvbVN0cmluZ1BhcmFtZXRlck5hbWUodGhpcywgJ1RlYW1OYW1lJywgJy9jb25maWcvdGVhbS9uYW1lJykuc3RyaW5nVmFsdWU7XG5cbiAgICBuZXcgQ2ZuUmVzb3VyY2UodGhpcywgJ0JhY2t1cE1vbml0b3JpbmdDb3B5Sm9icycsIHtcbiAgICAgIHR5cGU6ICdEYXRhZG9nOjpNb25pdG9yczo6TW9uaXRvcicsXG4gICAgICBwcm9wZXJ0aWVzOiB7XG4gICAgICAgIE5hbWU6IGBbUDRdIDogJHt0ZWFtTmFtZX0gQVdTIEJhY2t1cCBDb3B5IEpvYiB0byBjZW50cmFsIGFjY291bnQgZmFpbGVkYCxcbiAgICAgICAgVHlwZTogJ3F1ZXJ5IGFsZXJ0JyxcbiAgICAgICAgUXVlcnk6IGBzdW0obGFzdF8xZCk6c3VtOmF3cy5iYWNrdXAubnVtYmVyX29mX2NvcHlfam9ic19mYWlsZWR7YWNjb3VudF9pZDoke1N0YWNrLm9mKHRoaXMpLmFjY291bnR9fS5hc19jb3VudCgpID4gMGAsXG4gICAgICAgIE1lc3NhZ2U6IGBBV1MgQmFja3VwIENvcHkgam9iIHRvIGNlbnRyYWwgYmFja3VwIGFjY291bnQgaGFzIGZhaWxlZCBAb3BzZ2VuaWUtJHt0ZWFtTmFtZX0uXFxuQ2hlY2sgdGhlIGpvYiB1bmRlciBBV1MgQmFja3VwIC0+IEpvYnMgLT4gQ29weSBKb2JzYCxcbiAgICAgIH0sXG4gICAgfSk7XG5cbiAgICBuZXcgQ2ZuUmVzb3VyY2UodGhpcywgJ0JhY2t1cE1vbml0b3JpbmdKb2InLCB7XG4gICAgICB0eXBlOiAnRGF0YWRvZzo6TW9uaXRvcnM6Ok1vbml0b3InLFxuICAgICAgcHJvcGVydGllczoge1xuICAgICAgICBOYW1lOiBgW1A0XSA6ICR7dGVhbU5hbWV9IEFXUyBCYWNrdXAgSm9iIE1vbml0b3JgLFxuICAgICAgICBUeXBlOiAncXVlcnkgYWxlcnQnLFxuICAgICAgICBRdWVyeTogYHN1bShsYXN0XzFkKTpzdW06YXdzLmJhY2t1cC5udW1iZXJfb2ZfYmFja3VwX2pvYnNfZmFpbGVke2FjY291bnRfaWQ6JHtTdGFjay5vZih0aGlzKS5hY2NvdW50fX0uYXNfY291bnQoKSA+IDBgLFxuICAgICAgICBNZXNzYWdlOiBgQVdTIEJhY2t1cCBqb2IgaGFzIGZhaWxlZCBAb3BzZ2VuaWUtJHt0ZWFtTmFtZX0uXFxuQ2hlY2sgdGhlIGpvYiB1bmRlciBBV1MgQmFja3VwIC0+IEpvYnMgLT4gQmFja3VwIEpvYnNgLFxuICAgICAgfSxcbiAgICB9KTtcblxuICAgIGNvbnN0IFtmZWF0dXJlLCBwYWNrYWdlTmFtZV0gPSBfX2Rpcm5hbWUuc3BsaXQocGF0aC5zZXApLnJldmVyc2UoKTtcbiAgICBuZXcgQ2xhaWRvbWV0ZXIodGhpcywgJ0NsYWlkb21ldGVyJywge1xuICAgICAgcHJvZHVjdDogQ2xhaWRvbWV0ZXJQcm9kdWN0LkNES19DT05UUklCVVRJT04sXG4gICAgICBwYWNrYWdlTmFtZSxcbiAgICAgIGZlYXR1cmUsXG4gICAgICB2ZXJzaW9uOiAnMC4wLjQnLFxuICAgIH0pO1xuICB9XG59XG4iXX0=
49
+ /**
50
+ * This is a beta construct and is likely to have breaking changes.
51
+ * The construct creates monitoring for all AWS Backup jobs.
52
+ */
53
+ class AwsBackupOverallMonitoring extends constructs.Construct {
54
+ constructor(scope, id, props) {
55
+ super(scope, id);
56
+ const teamName = ssm.StringParameter.fromStringParameterName(this, 'TeamName', '/config/team/name').stringValue;
57
+ const commonTags = [
58
+ `service:${props.serviceName}`,
59
+ `team:${teamName}`,
60
+ ];
61
+ new cdk.CfnResource(this, 'OverallBackupMonitoringCopyJobs', {
62
+ type: 'Datadog::Monitors::Monitor',
63
+ properties: {
64
+ Name: 'AWS Backup overall Copy Job to central account failed',
65
+ Type: 'query alert',
66
+ Query: 'sum(last_1d):sum:aws.backup.number_of_copy_jobs_failed{*} by {account_name}.as_count() > 0',
67
+ Message: `[P4] AWS Backup Copy job to central backup account has failed @opsgenie-${teamName}.\nCheck the job under AWS Backup -> Jobs -> Copy Jobs`,
68
+ Tags: commonTags,
69
+ },
70
+ });
71
+ new cdk.CfnResource(this, 'OverallBackupMonitoringJob', {
72
+ type: 'Datadog::Monitors::Monitor',
73
+ properties: {
74
+ Name: 'AWS Backup overall Job error',
75
+ Type: 'query alert',
76
+ Query: 'sum(last_1d):sum:aws.backup.number_of_backup_jobs_failed{*} by {account_name}.as_count() > 0',
77
+ Message: `[P4] AWS Backup job has failed @opsgenie-${teamName}.\nCheck the job under AWS Backup -> Jobs -> Backup Jobs`,
78
+ Tags: commonTags,
79
+ },
80
+ });
81
+ addClaidometer(this);
82
+ }
83
+ }
84
+ exports.AwsBackupOverallMonitoring = AwsBackupOverallMonitoring;
85
+ _b = JSII_RTTI_SYMBOL_1;
86
+ AwsBackupOverallMonitoring[_b] = { fqn: "@rio-cloud/cdk-v2-constructs.AwsBackupOverallMonitoring", version: "0.0.0" };
87
+ function addClaidometer(scope) {
88
+ const [feature, packageName] = __dirname.split(path.sep).reverse();
89
+ new rio_claidometer_1.Claidometer(scope, 'Claidometer', {
90
+ product: rio_claidometer_1.ClaidometerProduct.CDK_CONTRIBUTION,
91
+ packageName,
92
+ feature,
93
+ version: '0.0.3',
94
+ });
95
+ }
96
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"aws-backup-monitoring.js","sourceRoot":"","sources":["../../../../src/contributions/team-claid/aws-backup-monitoring/aws-backup-monitoring.ts"],"names":[],"mappings":";;;;;AAAA,6BAA6B;AAC7B,mCAAmC;AACnC,2CAA2C;AAC3C,yCAAyC;AACzC,8DAA2E;AAM3E;;;GAGG;AACH,MAAa,mBAAoB,SAAQ,UAAU,CAAC,SAAS;IAC3D,YAAY,KAA2B,EAAE,EAAU,EAAE,KAA+B;QAClF,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEjB,MAAM,QAAQ,GAAG,GAAG,CAAC,eAAe,CAAC,uBAAuB,CAAC,IAAI,EAAE,UAAU,EAAE,mBAAmB,CAAC,CAAC,WAAW,CAAC;QAChH,MAAM,UAAU,GAAG;YACjB,WAAW,KAAK,CAAC,WAAW,EAAE;YAC9B,QAAQ,QAAQ,EAAE;SACnB,CAAC;QAEF,IAAI,GAAG,CAAC,WAAW,CAAC,IAAI,EAAE,0BAA0B,EAAE;YACpD,IAAI,EAAE,4BAA4B;YAClC,UAAU,EAAE;gBACV,IAAI,EAAE,+CAA+C;gBACrD,IAAI,EAAE,aAAa;gBACnB,KAAK,EAAE,qEAAqE,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,OAAO,kBAAkB;gBACxH,OAAO,EAAE,2EAA2E,QAAQ,wDAAwD;gBACpJ,IAAI,EAAE,UAAU;aACjB;SACF,CAAC,CAAC;QAEH,IAAI,GAAG,CAAC,WAAW,CAAC,IAAI,EAAE,qBAAqB,EAAE;YAC/C,IAAI,EAAE,4BAA4B;YAClC,UAAU,EAAE;gBACV,IAAI,EAAE,wBAAwB;gBAC9B,IAAI,EAAE,aAAa;gBACnB,KAAK,EAAE,uEAAuE,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,OAAO,kBAAkB;gBAC1H,OAAO,EAAE,4CAA4C,QAAQ,0DAA0D;gBACvH,IAAI,EAAE,UAAU;aACjB;SACF,CAAC,CAAC;QAEH,cAAc,CAAC,IAAI,CAAC,CAAC;IACvB,CAAC;;AAjCH,kDAkCC;;;AAED;;;GAGG;AACH,MAAa,0BAA2B,SAAQ,UAAU,CAAC,SAAS;IAClE,YAAY,KAA2B,EAAE,EAAU,EAAE,KAA+B;QAClF,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEjB,MAAM,QAAQ,GAAG,GAAG,CAAC,eAAe,CAAC,uBAAuB,CAAC,IAAI,EAAE,UAAU,EAAE,mBAAmB,CAAC,CAAC,WAAW,CAAC;QAChH,MAAM,UAAU,GAAG;YACjB,WAAW,KAAK,CAAC,WAAW,EAAE;YAC9B,QAAQ,QAAQ,EAAE;SACnB,CAAC;QAEF,IAAI,GAAG,CAAC,WAAW,CAAC,IAAI,EAAE,iCAAiC,EAAE;YAC3D,IAAI,EAAE,4BAA4B;YAClC,UAAU,EAAE;gBACV,IAAI,EAAE,uDAAuD;gBAC7D,IAAI,EAAE,aAAa;gBACnB,KAAK,EAAE,4FAA4F;gBACnG,OAAO,EAAE,2EAA2E,QAAQ,wDAAwD;gBACpJ,IAAI,EAAE,UAAU;aACjB;SACF,CAAC,CAAC;QAEH,IAAI,GAAG,CAAC,WAAW,CAAC,IAAI,EAAE,4BAA4B,EAAE;YACtD,IAAI,EAAE,4BAA4B;YAClC,UAAU,EAAE;gBACV,IAAI,EAAE,8BAA8B;gBACpC,IAAI,EAAE,aAAa;gBACnB,KAAK,EAAE,8FAA8F;gBACrG,OAAO,EAAE,4CAA4C,QAAQ,0DAA0D;gBACvH,IAAI,EAAE,UAAU;aACjB;SACF,CAAC,CAAC;QAEH,cAAc,CAAC,IAAI,CAAC,CAAC;IACvB,CAAC;;AAjCH,gEAkCC;;;AAED,SAAS,cAAc,CAAC,KAA2B;IACjD,MAAM,CAAC,OAAO,EAAE,WAAW,CAAC,GAAG,SAAS,CAAC,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,OAAO,EAAE,CAAC;IACnE,IAAI,6BAAW,CAAC,KAAK,EAAE,aAAa,EAAE;QACpC,OAAO,EAAE,oCAAkB,CAAC,gBAAgB;QAC5C,WAAW;QACX,OAAO;QACP,OAAO,EAAE,OAAO;KACjB,CAAC,CAAC;AACL,CAAC","sourcesContent":["import * as path from 'path';\nimport * as cdk from 'aws-cdk-lib';\nimport * as ssm from 'aws-cdk-lib/aws-ssm';\nimport * as constructs from 'constructs';\nimport { Claidometer, ClaidometerProduct } from '../../../rio-claidometer';\n\nexport interface AwsBackupMonitoringProps {\n  readonly serviceName: string;\n}\n\n/**\n * This is a beta construct and is likely to have breaking changes.\n * The construct creates monitoring for AWS Backup jobs.\n */\nexport class AwsBackupMonitoring extends constructs.Construct {\n  constructor(scope: constructs.Construct, id: string, props: AwsBackupMonitoringProps) {\n    super(scope, id);\n\n    const teamName = ssm.StringParameter.fromStringParameterName(this, 'TeamName', '/config/team/name').stringValue;\n    const commonTags = [\n      `service:${props.serviceName}`,\n      `team:${teamName}`,\n    ];\n\n    new cdk.CfnResource(this, 'BackupMonitoringCopyJobs', {\n      type: 'Datadog::Monitors::Monitor',\n      properties: {\n        Name: 'AWS Backup Copy Job to central account failed',\n        Type: 'query alert',\n        Query: `sum(last_1d):sum:aws.backup.number_of_copy_jobs_failed{account_id:${cdk.Stack.of(this).account}}.as_count() > 0`,\n        Message: `[P4] AWS Backup Copy job to central backup account has failed @opsgenie-${teamName}.\\nCheck the job under AWS Backup -> Jobs -> Copy Jobs`,\n        Tags: commonTags,\n      },\n    });\n\n    new cdk.CfnResource(this, 'BackupMonitoringJob', {\n      type: 'Datadog::Monitors::Monitor',\n      properties: {\n        Name: 'AWS Backup Job Monitor',\n        Type: 'query alert',\n        Query: `sum(last_1d):sum:aws.backup.number_of_backup_jobs_failed{account_id:${cdk.Stack.of(this).account}}.as_count() > 0`,\n        Message: `[P4] AWS Backup job has failed @opsgenie-${teamName}.\\nCheck the job under AWS Backup -> Jobs -> Backup Jobs`,\n        Tags: commonTags,\n      },\n    });\n\n    addClaidometer(this);\n  }\n}\n\n/**\n * This is a beta construct and is likely to have breaking changes.\n * The construct creates monitoring for all AWS Backup jobs.\n */\nexport class AwsBackupOverallMonitoring extends constructs.Construct {\n  constructor(scope: constructs.Construct, id: string, props: AwsBackupMonitoringProps) {\n    super(scope, id);\n\n    const teamName = ssm.StringParameter.fromStringParameterName(this, 'TeamName', '/config/team/name').stringValue;\n    const commonTags = [\n      `service:${props.serviceName}`,\n      `team:${teamName}`,\n    ];\n\n    new cdk.CfnResource(this, 'OverallBackupMonitoringCopyJobs', {\n      type: 'Datadog::Monitors::Monitor',\n      properties: {\n        Name: 'AWS Backup overall Copy Job to central account failed',\n        Type: 'query alert',\n        Query: 'sum(last_1d):sum:aws.backup.number_of_copy_jobs_failed{*} by {account_name}.as_count() > 0',\n        Message: `[P4] AWS Backup Copy job to central backup account has failed @opsgenie-${teamName}.\\nCheck the job under AWS Backup -> Jobs -> Copy Jobs`,\n        Tags: commonTags,\n      },\n    });\n\n    new cdk.CfnResource(this, 'OverallBackupMonitoringJob', {\n      type: 'Datadog::Monitors::Monitor',\n      properties: {\n        Name: 'AWS Backup overall Job error',\n        Type: 'query alert',\n        Query: 'sum(last_1d):sum:aws.backup.number_of_backup_jobs_failed{*} by {account_name}.as_count() > 0',\n        Message: `[P4] AWS Backup job has failed @opsgenie-${teamName}.\\nCheck the job under AWS Backup -> Jobs -> Backup Jobs`,\n        Tags: commonTags,\n      },\n    });\n\n    addClaidometer(this);\n  }\n}\n\nfunction addClaidometer(scope: constructs.Construct) {\n  const [feature, packageName] = __dirname.split(path.sep).reverse();\n  new Claidometer(scope, 'Claidometer', {\n    product: ClaidometerProduct.CDK_CONTRIBUTION,\n    packageName,\n    feature,\n    version: '0.0.3',\n  });\n}\n"]}
@@ -4,6 +4,7 @@ Object.defineProperty(exports, "__esModule", { value: true });
4
4
  exports.DatadogUsageMonitoring = void 0;
5
5
  const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
6
6
  const path = require("path");
7
+ const ssm = require("aws-cdk-lib/aws-ssm");
7
8
  const constructs = require("constructs");
8
9
  const datadogMonitor_1 = require("../../../datadog/datadogMonitor");
9
10
  const rio_claidometer_1 = require("../../../rio-claidometer");
@@ -24,6 +25,7 @@ function getOrganizationUrlPrefix(organization) {
24
25
  class DatadogUsageMonitoring extends constructs.Construct {
25
26
  constructor(scope, id, props) {
26
27
  super(scope, id);
28
+ const teamName = ssm.StringParameter.fromStringParameterName(this, 'TeamName', '/config/team/name').stringValue;
27
29
  const infrastructureUsageSpikeMonitors = [
28
30
  {
29
31
  humanReadableName: 'infrastructure hosts',
@@ -114,20 +116,11 @@ class DatadogUsageMonitoring extends constructs.Construct {
114
116
  },
115
117
  ];
116
118
  for (const usageSpikeMonitor of [].concat(infrastructureUsageSpikeMonitors, logManagementUsageSpikeMonitors, apmUsageSpikeMonitors, serverlessUsageSpikeMonitors, syntheticUsageSpikeMonitors)) {
117
- this.createAnomalyMonitorForSpikes(props.serviceName, props.alertType, usageSpikeMonitor);
118
- }
119
- // There is no Claidometer support in LATAM
120
- if (props.organization !== 'LATAM') {
121
- const [feature, packageName] = __dirname.split(path.sep).reverse();
122
- new rio_claidometer_1.Claidometer(this, 'Claidometer', {
123
- product: rio_claidometer_1.ClaidometerProduct.CDK_CONTRIBUTION,
124
- packageName,
125
- feature,
126
- version: '0.0.6',
127
- });
119
+ this.createAnomalyMonitorForSpikes(props.serviceName, teamName, props.alertType, usageSpikeMonitor);
128
120
  }
121
+ addClaidometer(this);
129
122
  }
130
- createAnomalyMonitorForSpikes(serviceName, alertType, usageSpikeMonitor) {
123
+ createAnomalyMonitorForSpikes(serviceName, teamName, alertType, usageSpikeMonitor) {
131
124
  var _b;
132
125
  return new datadogMonitor_1.DatadogMonitor(this, usageSpikeMonitor.humanReadableName, {
133
126
  serviceName,
@@ -136,7 +129,7 @@ class DatadogUsageMonitoring extends constructs.Construct {
136
129
  name: `${usageSpikeMonitor.humanReadableName} spike`,
137
130
  type: 'query alert',
138
131
  query: `avg(last_12h):anomalies(${usageSpikeMonitor.query}, 'agile', 3, direction='above', interval=120, alert_window='last_30m', seasonality='weekly', timezone='europe/berlin', count_default_zero='true') >= 1`,
139
- message: `${usageSpikeMonitor.humanReadableName} are getting out-of-control. ${(_b = usageSpikeMonitor.additionalMessage) !== null && _b !== void 0 ? _b : ''}`,
132
+ message: `[P4] ${usageSpikeMonitor.humanReadableName} are getting out-of-control. ${(_b = usageSpikeMonitor.additionalMessage) !== null && _b !== void 0 ? _b : ''}`,
140
133
  options: {
141
134
  include_tags: false,
142
135
  notify_no_data: false,
@@ -145,6 +138,10 @@ class DatadogUsageMonitoring extends constructs.Construct {
145
138
  recovery_window: 'last_15m',
146
139
  },
147
140
  },
141
+ tags: [
142
+ `service:${serviceName}`,
143
+ `team:${teamName}`,
144
+ ],
148
145
  },
149
146
  });
150
147
  }
@@ -152,4 +149,13 @@ class DatadogUsageMonitoring extends constructs.Construct {
152
149
  exports.DatadogUsageMonitoring = DatadogUsageMonitoring;
153
150
  _a = JSII_RTTI_SYMBOL_1;
154
151
  DatadogUsageMonitoring[_a] = { fqn: "@rio-cloud/cdk-v2-constructs.DatadogUsageMonitoring", version: "0.0.0" };
155
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"datadog-usage-monitoring.js","sourceRoot":"","sources":["../../../../src/contributions/team-claid/datadog-usage-monitoring/datadog-usage-monitoring.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport * as constructs from 'constructs';\nimport { DatadogAlertType, DatadogMonitor } from '../../../datadog/datadogMonitor';\nimport { Claidometer, ClaidometerProduct } from '../../../rio-claidometer';\n\nexport type DatadogOrganization = 'EU' | 'LATAM' | 'Landing Zone';\n\nfunction getOrganizationUrlPrefix(organization: DatadogOrganization): string {\n  switch (organization) {\n    case 'EU':\n      return 'app';\n    case 'LATAM':\n      return 'rio-brazil';\n    case 'Landing Zone':\n      return 'rio-landingzone';\n  }\n}\n\ninterface UsageSpikeMonitor {\n  humanReadableName: string;\n  query: string;\n  additionalMessage?: string;\n}\n\nexport interface DatadogUsageMonitoringProps {\n  readonly serviceName: string;\n  readonly alertType: DatadogAlertType;\n  readonly organization: DatadogOrganization;\n}\n\n/**\n * This is a beta construct and is likely to have breaking changes.\n * The construct creates monitoring (based on anomaly detection) for the usage of the most relevant Datadog resources\n */\nexport class DatadogUsageMonitoring extends constructs.Construct {\n  constructor(scope: constructs.Construct, id: string, props: DatadogUsageMonitoringProps) {\n    super(scope, id);\n\n    const infrastructureUsageSpikeMonitors: UsageSpikeMonitor[] = [\n      {\n        humanReadableName: 'infrastructure hosts',\n        query: 'sum:datadog.estimated_usage.hosts{*}',\n      },\n      {\n        humanReadableName: \"infrastructure hosts for '{{child_org_name.name}}'\",\n        query: 'sum:datadog.estimated_usage.hosts{*} by {child_org_name}',\n      },\n      { humanReadableName: 'Fargate tasks', query: 'sum:datadog.apm.fargate_task_instance{*}' },\n      { humanReadableName: 'profiled Fargate tasks', query: 'sum:datadog.profiling.fargate{*}' },\n      {\n        humanReadableName: 'custom metrics',\n        query: 'sum:datadog.estimated_usage.metrics.custom{*}',\n      },\n      {\n        humanReadableName: \"custom metrics for '{{child_org_name.name}}'\",\n        query: 'sum:datadog.estimated_usage.metrics.custom{*} by {child_org_name}',\n      },\n    ];\n\n    const additionalMessageLogManagementForGloballyScopedMonitors = `Take a look at https://${getOrganizationUrlPrefix(\n      props.organization,\n    )}.datadoghq.eu/dash/integration/465/log-management-estimated-usage for more details.`;\n    const additionalMessageLogManagementForLogIndexScopedMonitors = `Take a look at https://${getOrganizationUrlPrefix(\n      props.organization,\n    )}.datadoghq.eu/dash/integration/465/log-management-estimated-usage?tpl_var_datadog_index={{datadog_index.name}} for more details.`;\n    const logManagementUsageSpikeMonitors: UsageSpikeMonitor[] = [\n      {\n        humanReadableName: 'log ingestion',\n        query: 'sum:datadog.estimated_usage.logs.ingested_bytes{*}.as_count()',\n        additionalMessage: additionalMessageLogManagementForGloballyScopedMonitors,\n      },\n      {\n        humanReadableName: \"log ingestion for '{{datadog_index.name}}'\",\n        query: 'sum:datadog.estimated_usage.logs.ingested_bytes{*} by {datadog_index}.as_count()',\n        additionalMessage: additionalMessageLogManagementForLogIndexScopedMonitors,\n      },\n      {\n        humanReadableName: 'log indexing',\n        query: 'sum:datadog.estimated_usage.logs.ingested_events{datadog_is_excluded:false}.as_count()',\n        additionalMessage: additionalMessageLogManagementForGloballyScopedMonitors,\n      },\n      {\n        humanReadableName: \"log indexing for '{{datadog_index.name}}'\",\n        query: 'sum:datadog.estimated_usage.logs.ingested_events{datadog_is_excluded:false} by {datadog_index}.as_count()',\n        additionalMessage: additionalMessageLogManagementForLogIndexScopedMonitors,\n      },\n    ];\n\n    const additionalMessageApmForGloballyScopedMonitors = `Take a look at https://${getOrganizationUrlPrefix(\n      props.organization,\n    )}.datadoghq.eu/dash/integration/489/apm-traces-estimated-usage for more details.`;\n    const additionalMessageApmForEnvScopedMonitors = `Take a look at https://${getOrganizationUrlPrefix(\n      props.organization,\n    )}.datadoghq.eu/dash/integration/489/apm-traces-estimated-usage?tpl_var_env={{env.name}} for more details.`;\n    const apmUsageSpikeMonitors: UsageSpikeMonitor[] = [\n      { humanReadableName: 'APM hosts', query: 'sum:datadog.estimated_usage.apm_hosts{*}' },\n      {\n        humanReadableName: \"APM hosts for '{{child_org_name.name}}'\",\n        query: 'sum:datadog.estimated_usage.apm_hosts{*} by {child_org_name}',\n        additionalMessage: additionalMessageApmForGloballyScopedMonitors,\n      },\n      {\n        humanReadableName: 'span ingestion',\n        query: 'sum:datadog.estimated_usage.apm.ingested_bytes{*}.as_count()',\n      },\n      {\n        humanReadableName: \"span ingestion for '{{env.name}}'\",\n        query: 'sum:datadog.estimated_usage.apm.ingested_bytes{*} by {env}.as_count()',\n        additionalMessage: additionalMessageApmForEnvScopedMonitors,\n      },\n      {\n        humanReadableName: 'span indexing',\n        query: 'sum:datadog.estimated_usage.apm.indexed_spans{*}.as_count()',\n        additionalMessage: additionalMessageApmForGloballyScopedMonitors,\n      },\n      {\n        humanReadableName: \"span indexing for '{{env.name}}'\",\n        query: 'sum:datadog.estimated_usage.apm.indexed_spans{*} by {env}.as_count()',\n        additionalMessage: additionalMessageApmForEnvScopedMonitors,\n      },\n    ];\n\n    const syntheticUsageSpikeMonitors: UsageSpikeMonitor[] = [\n      {\n        humanReadableName: 'Synthetic API tests',\n        query: 'sum:datadog.estimated_usage.synthetics.api_test_runs{*}.as_count()',\n      },\n    ];\n\n    const serverlessUsageSpikeMonitors: UsageSpikeMonitor[] = [\n      {\n        humanReadableName: 'serverless functions',\n        query: 'sum:datadog.estimated_usage.serverless.aws_lambda_functions{*}',\n      },\n      {\n        humanReadableName: \"serverless functions for '{{child_org_name.name}}'\",\n        query: 'sum:datadog.estimated_usage.serverless.aws_lambda_functions{*} by {child_org_name}',\n      },\n    ];\n\n    for (const usageSpikeMonitor of ([] as UsageSpikeMonitor[]).concat(\n      infrastructureUsageSpikeMonitors,\n      logManagementUsageSpikeMonitors,\n      apmUsageSpikeMonitors,\n      serverlessUsageSpikeMonitors,\n      syntheticUsageSpikeMonitors,\n    )) {\n      this.createAnomalyMonitorForSpikes(props.serviceName, props.alertType, usageSpikeMonitor);\n    }\n\n    // There is no Claidometer support in LATAM\n    if (props.organization !== 'LATAM') {\n      const [feature, packageName] = __dirname.split(path.sep).reverse();\n      new Claidometer(this, 'Claidometer', {\n        product: ClaidometerProduct.CDK_CONTRIBUTION,\n        packageName,\n        feature,\n        version: '0.0.6',\n      });\n    }\n  }\n\n  private createAnomalyMonitorForSpikes(\n    serviceName: string,\n    alertType: DatadogAlertType,\n    usageSpikeMonitor: UsageSpikeMonitor,\n  ): DatadogMonitor {\n    return new DatadogMonitor(this, usageSpikeMonitor.humanReadableName, {\n      serviceName,\n      alertTypes: [alertType],\n      monitor: {\n        name: `${usageSpikeMonitor.humanReadableName} spike`,\n        type: 'query alert',\n        query: `avg(last_12h):anomalies(${usageSpikeMonitor.query}, 'agile', 3, direction='above', interval=120, alert_window='last_30m', seasonality='weekly', timezone='europe/berlin', count_default_zero='true') >= 1`,\n        message: `${usageSpikeMonitor.humanReadableName} are getting out-of-control. ${\n          usageSpikeMonitor.additionalMessage ?? ''\n        }`,\n        options: {\n          include_tags: false,\n          notify_no_data: false,\n          threshold_windows: {\n            trigger_window: 'last_30m',\n            recovery_window: 'last_15m',\n          },\n        },\n      },\n    });\n  }\n}\n"]}
152
+ function addClaidometer(scope) {
153
+ const [feature, packageName] = __dirname.split(path.sep).reverse();
154
+ new rio_claidometer_1.Claidometer(scope, 'Claidometer', {
155
+ product: rio_claidometer_1.ClaidometerProduct.CDK_CONTRIBUTION,
156
+ packageName,
157
+ feature,
158
+ version: '0.0.7',
159
+ });
160
+ }
161
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"datadog-usage-monitoring.js","sourceRoot":"","sources":["../../../../src/contributions/team-claid/datadog-usage-monitoring/datadog-usage-monitoring.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport * as ssm from 'aws-cdk-lib/aws-ssm';\nimport * as constructs from 'constructs';\nimport { DatadogAlertType, DatadogMonitor } from '../../../datadog/datadogMonitor';\nimport { Claidometer, ClaidometerProduct } from '../../../rio-claidometer';\n\nexport type DatadogOrganization = 'EU' | 'LATAM' | 'Landing Zone';\n\nfunction getOrganizationUrlPrefix(organization: DatadogOrganization): string {\n  switch (organization) {\n    case 'EU':\n      return 'app';\n    case 'LATAM':\n      return 'rio-brazil';\n    case 'Landing Zone':\n      return 'rio-landingzone';\n  }\n}\n\ninterface UsageSpikeMonitor {\n  humanReadableName: string;\n  query: string;\n  additionalMessage?: string;\n}\n\nexport interface DatadogUsageMonitoringProps {\n  readonly serviceName: string;\n  readonly alertType: DatadogAlertType;\n  readonly organization: DatadogOrganization;\n}\n\n/**\n * This is a beta construct and is likely to have breaking changes.\n * The construct creates monitoring (based on anomaly detection) for the usage of the most relevant Datadog resources\n */\nexport class DatadogUsageMonitoring extends constructs.Construct {\n  constructor(scope: constructs.Construct, id: string, props: DatadogUsageMonitoringProps) {\n    super(scope, id);\n\n    const teamName = ssm.StringParameter.fromStringParameterName(this, 'TeamName', '/config/team/name').stringValue;\n\n    const infrastructureUsageSpikeMonitors: UsageSpikeMonitor[] = [\n      {\n        humanReadableName: 'infrastructure hosts',\n        query: 'sum:datadog.estimated_usage.hosts{*}',\n      },\n      {\n        humanReadableName: \"infrastructure hosts for '{{child_org_name.name}}'\",\n        query: 'sum:datadog.estimated_usage.hosts{*} by {child_org_name}',\n      },\n      { humanReadableName: 'Fargate tasks', query: 'sum:datadog.apm.fargate_task_instance{*}' },\n      { humanReadableName: 'profiled Fargate tasks', query: 'sum:datadog.profiling.fargate{*}' },\n      {\n        humanReadableName: 'custom metrics',\n        query: 'sum:datadog.estimated_usage.metrics.custom{*}',\n      },\n      {\n        humanReadableName: \"custom metrics for '{{child_org_name.name}}'\",\n        query: 'sum:datadog.estimated_usage.metrics.custom{*} by {child_org_name}',\n      },\n    ];\n\n    const additionalMessageLogManagementForGloballyScopedMonitors = `Take a look at https://${getOrganizationUrlPrefix(\n      props.organization,\n    )}.datadoghq.eu/dash/integration/465/log-management-estimated-usage for more details.`;\n    const additionalMessageLogManagementForLogIndexScopedMonitors = `Take a look at https://${getOrganizationUrlPrefix(\n      props.organization,\n    )}.datadoghq.eu/dash/integration/465/log-management-estimated-usage?tpl_var_datadog_index={{datadog_index.name}} for more details.`;\n    const logManagementUsageSpikeMonitors: UsageSpikeMonitor[] = [\n      {\n        humanReadableName: 'log ingestion',\n        query: 'sum:datadog.estimated_usage.logs.ingested_bytes{*}.as_count()',\n        additionalMessage: additionalMessageLogManagementForGloballyScopedMonitors,\n      },\n      {\n        humanReadableName: \"log ingestion for '{{datadog_index.name}}'\",\n        query: 'sum:datadog.estimated_usage.logs.ingested_bytes{*} by {datadog_index}.as_count()',\n        additionalMessage: additionalMessageLogManagementForLogIndexScopedMonitors,\n      },\n      {\n        humanReadableName: 'log indexing',\n        query: 'sum:datadog.estimated_usage.logs.ingested_events{datadog_is_excluded:false}.as_count()',\n        additionalMessage: additionalMessageLogManagementForGloballyScopedMonitors,\n      },\n      {\n        humanReadableName: \"log indexing for '{{datadog_index.name}}'\",\n        query: 'sum:datadog.estimated_usage.logs.ingested_events{datadog_is_excluded:false} by {datadog_index}.as_count()',\n        additionalMessage: additionalMessageLogManagementForLogIndexScopedMonitors,\n      },\n    ];\n\n    const additionalMessageApmForGloballyScopedMonitors = `Take a look at https://${getOrganizationUrlPrefix(\n      props.organization,\n    )}.datadoghq.eu/dash/integration/489/apm-traces-estimated-usage for more details.`;\n    const additionalMessageApmForEnvScopedMonitors = `Take a look at https://${getOrganizationUrlPrefix(\n      props.organization,\n    )}.datadoghq.eu/dash/integration/489/apm-traces-estimated-usage?tpl_var_env={{env.name}} for more details.`;\n    const apmUsageSpikeMonitors: UsageSpikeMonitor[] = [\n      { humanReadableName: 'APM hosts', query: 'sum:datadog.estimated_usage.apm_hosts{*}' },\n      {\n        humanReadableName: \"APM hosts for '{{child_org_name.name}}'\",\n        query: 'sum:datadog.estimated_usage.apm_hosts{*} by {child_org_name}',\n        additionalMessage: additionalMessageApmForGloballyScopedMonitors,\n      },\n      {\n        humanReadableName: 'span ingestion',\n        query: 'sum:datadog.estimated_usage.apm.ingested_bytes{*}.as_count()',\n      },\n      {\n        humanReadableName: \"span ingestion for '{{env.name}}'\",\n        query: 'sum:datadog.estimated_usage.apm.ingested_bytes{*} by {env}.as_count()',\n        additionalMessage: additionalMessageApmForEnvScopedMonitors,\n      },\n      {\n        humanReadableName: 'span indexing',\n        query: 'sum:datadog.estimated_usage.apm.indexed_spans{*}.as_count()',\n        additionalMessage: additionalMessageApmForGloballyScopedMonitors,\n      },\n      {\n        humanReadableName: \"span indexing for '{{env.name}}'\",\n        query: 'sum:datadog.estimated_usage.apm.indexed_spans{*} by {env}.as_count()',\n        additionalMessage: additionalMessageApmForEnvScopedMonitors,\n      },\n    ];\n\n    const syntheticUsageSpikeMonitors: UsageSpikeMonitor[] = [\n      {\n        humanReadableName: 'Synthetic API tests',\n        query: 'sum:datadog.estimated_usage.synthetics.api_test_runs{*}.as_count()',\n      },\n    ];\n\n    const serverlessUsageSpikeMonitors: UsageSpikeMonitor[] = [\n      {\n        humanReadableName: 'serverless functions',\n        query: 'sum:datadog.estimated_usage.serverless.aws_lambda_functions{*}',\n      },\n      {\n        humanReadableName: \"serverless functions for '{{child_org_name.name}}'\",\n        query: 'sum:datadog.estimated_usage.serverless.aws_lambda_functions{*} by {child_org_name}',\n      },\n    ];\n\n    for (const usageSpikeMonitor of ([] as UsageSpikeMonitor[]).concat(\n      infrastructureUsageSpikeMonitors,\n      logManagementUsageSpikeMonitors,\n      apmUsageSpikeMonitors,\n      serverlessUsageSpikeMonitors,\n      syntheticUsageSpikeMonitors,\n    )) {\n      this.createAnomalyMonitorForSpikes(props.serviceName, teamName, props.alertType, usageSpikeMonitor);\n    }\n\n    addClaidometer(this);\n  }\n\n  private createAnomalyMonitorForSpikes(\n    serviceName: string,\n    teamName: string,\n    alertType: DatadogAlertType,\n    usageSpikeMonitor: UsageSpikeMonitor,\n  ): DatadogMonitor {\n    return new DatadogMonitor(this, usageSpikeMonitor.humanReadableName, {\n      serviceName,\n      alertTypes: [alertType],\n      monitor: {\n        name: `${usageSpikeMonitor.humanReadableName} spike`,\n        type: 'query alert',\n        query: `avg(last_12h):anomalies(${usageSpikeMonitor.query}, 'agile', 3, direction='above', interval=120, alert_window='last_30m', seasonality='weekly', timezone='europe/berlin', count_default_zero='true') >= 1`,\n        message: `[P4] ${usageSpikeMonitor.humanReadableName} are getting out-of-control. ${\n          usageSpikeMonitor.additionalMessage ?? ''\n        }`,\n        options: {\n          include_tags: false,\n          notify_no_data: false,\n          threshold_windows: {\n            trigger_window: 'last_30m',\n            recovery_window: 'last_15m',\n          },\n        },\n        tags: [\n          `service:${serviceName}`,\n          `team:${teamName}`,\n        ],\n      },\n    });\n  }\n}\n\nfunction addClaidometer(scope: constructs.Construct) {\n  const [feature, packageName] = __dirname.split(path.sep).reverse();\n  new Claidometer(scope, 'Claidometer', {\n    product: ClaidometerProduct.CDK_CONTRIBUTION,\n    packageName,\n    feature,\n    version: '0.0.7',\n  });\n}\n"]}
@@ -14,7 +14,17 @@ class KafkaTopic extends constructs_1.Construct {
14
14
  constructor(scope, id, props) {
15
15
  var _b, _c, _d, _e, _f;
16
16
  super(scope, id);
17
- const teamEmail = ssm.StringParameter.fromStringParameterName(scope, 'TeamEmail', '/config/team/mail').stringValue;
17
+ /**
18
+ SSM Parameter references dont generate logical names by appending IDs of the caller constructs. Instead simply use ID passed during function call 'fromStringParameterName'.
19
+ This means that if this construct is called more than once in the same stack, it would fail with error 'Construct name already used'.
20
+ Checking for existing SSM Parameter used to avoid this error
21
+ */
22
+ const teamEmailSSMCDKIdentifier = 'TeamEmail';
23
+ const teamEmail = aws_cdk_lib_1.Stack.of(this).node.tryFindChild(teamEmailSSMCDKIdentifier)
24
+ ?
25
+ aws_cdk_lib_1.Stack.of(this).node.findChild(teamEmailSSMCDKIdentifier).stringValue
26
+ :
27
+ ssm.StringParameter.fromStringParameterName(scope, teamEmailSSMCDKIdentifier, '/config/team/mail').stringValue;
18
28
  const events = props.metadata.events.map(source => {
19
29
  new kafka_event_spec_1.KafkaEventSpec(this, `${source.eventName.replace(/\./g, '_')}-EventSpec`, { source });
20
30
  return source.eventName;
@@ -67,4 +77,4 @@ class KafkaTopic extends constructs_1.Construct {
67
77
  exports.KafkaTopic = KafkaTopic;
68
78
  _a = JSII_RTTI_SYMBOL_1;
69
79
  KafkaTopic[_a] = { fqn: "@rio-cloud/cdk-v2-constructs.KafkaTopic", version: "0.0.0" };
70
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"kafka-topic.js","sourceRoot":"","sources":["../../src/kafka/kafka-topic.ts"],"names":[],"mappings":";;;;;AAAA,6CAAoD;AACpD,2CAA2C;AAC3C,2CAAuC;AAEvC,yDAA0E;AA6G1E;;GAEG;AACH,MAAa,UAAW,SAAQ,sBAAS;IAEvC,YAAY,KAAgB,EAAE,EAAU,EAAE,KAAsB;;QAC9D,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEjB,MAAM,SAAS,GAAG,GAAG,CAAC,eAAe,CAAC,uBAAuB,CAC3D,KAAK,EACL,WAAW,EACX,mBAAmB,CACpB,CAAC,WAAW,CAAC;QAEd,MAAM,MAAM,GAAG,KAAK,CAAC,QAAQ,CAAC,MAAM,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE;YAChD,IAAI,iCAAc,CAAC,IAAI,EAAE,GAAG,MAAM,CAAC,SAAS,CAAC,OAAO,CAAC,KAAK,EAAE,GAAG,CAAC,YAAY,EAAE,EAAE,MAAM,EAAE,CAAC,CAAC;YAC1F,OAAO,MAAM,CAAC,SAAS,CAAC;QAC1B,CAAC,CAAC,CAAC;QAEH,IAAI,yBAAW,CAAC,IAAI,EAAE,YAAY,EAAE;YAClC,IAAI,EAAE,oBAAoB;YAC1B,UAAU,EAAE;gBACV,YAAY,EAAE,+DAA+D;gBAC7E,OAAO,EAAE,GAAG;gBACZ,SAAS,EAAE,KAAK,CAAC,IAAI;gBACrB,WAAW,EAAE,KAAK,CAAC,QAAQ,CAAC,WAAW;gBACvC,MAAM,EAAE,MAAM;gBACd,kBAAkB,EAAE,KAAK,CAAC,kBAAkB;gBAC5C,iBAAiB,QAAE,KAAK,CAAC,iBAAiB,mCAAI,CAAC;gBAC/C,WAAW,EAAE,OAAC,KAAK,CAAC,SAAS,mCAAI,sBAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,cAAc,EAAE;gBACnE,iBAAiB,EAAE,KAAK,CAAC,cAAc,CAAC,CAAC,aAAC,KAAK,CAAC,uBAAuB,0CAAE,eAAe,mCAAI,sBAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,cAAc,EAAE,CAAC,CAAC,CAAC,SAAS;gBACzI,sBAAsB,QAAE,KAAK,CAAC,sBAAsB,mCAAI,KAAK;gBAC7D,aAAa,EAAE,KAAK,CAAC,cAAc,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,QAAQ;gBAC1D,SAAS,EAAE,SAAS;gBACpB,GAAG,EAAE;oBACH,IAAI,EAAE,KAAK,CAAC,GAAG,CAAC,IAAI;oBACpB,KAAK,EAAE,KAAK,CAAC,GAAG,CAAC,KAAK;iBACvB;aACF;SACF,CAAC,CAAC;QAEH,IAAI,CAAC,IAAI,CAAC,aAAa,CAAC;YACtB,QAAQ,EAAE,GAAa,EAAE;gBACvB,MAAM,MAAM,GAAG,EAAE,CAAC;gBAElB,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,kBAAkB,CAAC,EAAE;oBACzC,MAAM,CAAC,IAAI,CAAC,6EAA6E,CAAC,CAAC;iBAC5F;gBAED,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,KAAK,CAAC,kBAAkB,CAAC,IAAI,KAAK,CAAC,kBAAkB,GAAG,CAAC,EAAE;oBAC/E,MAAM,CAAC,IAAI,CAAC,iGAAiG,KAAK,CAAC,kBAAkB,GAAG,CAAC,CAAC;iBAC3I;gBAED,IAAI,KAAK,CAAC,iBAAiB,KAAK,SAAS;oBACrC,CAAC,CAAC,MAAM,CAAC,SAAS,CAAC,KAAK,CAAC,iBAAiB,CAAC,IAAI,KAAK,CAAC,iBAAiB,GAAG,CAAC,IAAI,KAAK,CAAC,iBAAiB,GAAG,CAAC,CAAC,EAAE;oBAC9G,MAAM,CAAC,IAAI,CAAC,oEAAoE,CAAC,CAAC;iBACnF;gBAED,IAAI,KAAK,CAAC,QAAQ,CAAC,MAAM,CAAC,MAAM,KAAK,CAAC,EAAE;oBACtC,MAAM,CAAC,IAAI,CAAC,qEAAqE,CAAC,CAAC;iBACpF;gBAED,IAAI,CAAC,KAAK,CAAC,cAAc,IAAI,KAAK,CAAC,uBAAuB,EAAE;oBAC1D,MAAM,CAAC,IAAI,CAAC,gGAAgG,CAAC,CAAC;iBAC/G;gBAED,OAAO,MAAM,CAAC;YAChB,CAAC;SACF,CAAC,CAAC;IACL,CAAC;;AAlEH,gCAmEC","sourcesContent":["import { Duration, CfnResource } from 'aws-cdk-lib';\nimport * as ssm from 'aws-cdk-lib/aws-ssm';\nimport { Construct } from 'constructs';\n\nimport { KafkaEventSpec, KafkaEventSpecSource } from './kafka-event-spec';\n\n/**\n * Wrapper for all meta data of a topic.\n */\nexport interface KafkaTopicMetaData {\n  /**\n   * The event specs of the events on that topic\n   */\n  readonly events: KafkaEventSpecSource[];\n\n  /**\n   * Describes the intent of the topic. This might be information about the events on that topic\n   * or additional information about the producer and the context of the events.\n   */\n  readonly description: string;\n}\n\n/**\n * Read and write permissions for the topic.\n *\n * strings are matched against the CNAME of the certificate of the Kafka clients.\n */\nexport interface KafkaAclStatement {\n  /**\n  * List of clients that should get write permissions\n  */\n  readonly write: string[];\n\n  /**\n   * List of clients that should get read permissions\n   */\n  readonly read: string[];\n}\n\n/**\n * Wrapper for all log compaction related properties.\n */\nexport interface LogCompactionProperties {\n  /**\n   * The amount of time to retain delete tombstone markers for log compacted topics;\n   * The soft limit is 10 days but can be increased upon requests.\n   * @default 1 day\n   */\n  readonly deleteRetention?: Duration;\n}\n\n/**\n * The current service limits can be found in the topic limits configuration of the topic manager.\n *\n * @see https://collaboration.msi.audi.com/stash/projects/RSEVTBU/repos/topic-manager/browse/config/topic-service-limits.yaml\n */\nexport interface KafkaTopicProps{\n  /**\n   * The unique name of the topic; has to match pattern [A-Za-z0-9.-]+\n   */\n  readonly name: string;\n\n  /**\n   * Defines degree of parallelism of the topic; should be increased for\n   * large expected loads (e.g., 25 for rio.asset-iot-events).\n   * The soft limit is 10 partitions but can be increased upon requests.\n   */\n  readonly numberOfPartitions: number;\n\n  /**\n   * Defines degree of replication of messages; has to be between 1 and the number of brokers (currently 3).\n   * @default 3\n   */\n  readonly replicationFactor?: number;\n\n  /**\n   * Time how long messages are retained on Kafka cluster;\n   *\n   * We recommend 30 days for general events and 7 days for sensor events and other high-load cases.\n   * The soft limit is 3 to 30 days but can be increased upon requests.\n   * @default 7 days\n   */\n  readonly retention?: Duration;\n\n  /**\n   * If set to 'false, the topic's data will be deleted on Kafka 10 days after stack deletion.\n   * If set to 'true', it will be deleted immediately after stack deletion.\n   * @default false\n   */\n  readonly instantDeletionEnabled?: boolean;\n\n  /**\n   * The metadata of the topic.\n   */\n  readonly metadata: KafkaTopicMetaData;\n\n  /**\n   * The permissions to access to the topic's data.\n   */\n  readonly acl: KafkaAclStatement;\n\n  /**\n   * Specify whether the topic is log compacted or not.\n   */\n  readonly isLogCompacted: boolean;\n\n  /**\n   * Must only be provided if 'isLogCompacted' is 'true'.\n   * @default undefined\n   */\n  readonly logCompactionProperties?: LogCompactionProperties;\n}\n\n/**\n * Construct to create a kafka topic.\n */\nexport class KafkaTopic extends Construct {\n\n  constructor(scope: Construct, id: string, props: KafkaTopicProps) {\n    super(scope, id);\n\n    const teamEmail = ssm.StringParameter.fromStringParameterName(\n      scope,\n      'TeamEmail',\n      '/config/team/mail',\n    ).stringValue;\n\n    const events = props.metadata.events.map(source => {\n      new KafkaEventSpec(this, `${source.eventName.replace(/\\./g, '_')}-EventSpec`, { source });\n      return source.eventName;\n    });\n\n    new CfnResource(this, 'KafkaTopic', {\n      type: 'Custom::KafkaTopic',\n      properties: {\n        ServiceToken: 'arn:aws:sns:eu-west-1:186993757734:dp-topic-manager-sns-topic',\n        Version: '3',\n        TopicName: props.name,\n        Description: props.metadata.description,\n        Events: events,\n        NumberOfPartitions: props.numberOfPartitions,\n        ReplicationFactor: props.replicationFactor ?? 3,\n        RetentionMs: (props.retention ?? Duration.days(7)).toMilliseconds(),\n        DeleteRetentionMs: props.isLogCompacted ? props.logCompactionProperties?.deleteRetention ?? Duration.days(7).toMilliseconds() : undefined,\n        InstantDeletionEnabled: props.instantDeletionEnabled ?? false,\n        CleanupPolicy: props.isLogCompacted ? 'compact' : 'delete',\n        TeamEmail: teamEmail,\n        ACL: {\n          Read: props.acl.read,\n          Write: props.acl.write,\n        },\n      },\n    });\n\n    this.node.addValidation({\n      validate: (): string[] => {\n        const result = [];\n\n        if (!props.name.match(/^[A-Za-z0-9.-]+$/)) {\n          result.push('Invalid [name]: expecting topic name to match the reg exp `[A-Za-z0-9.-]+`.');\n        }\n\n        if (!Number.isInteger(props.numberOfPartitions) || props.numberOfPartitions < 1) {\n          result.push(`Invalid [numberOfPartitions]: expecting number of partitions to be a positive integer but got ${props.numberOfPartitions}.`);\n        }\n\n        if (props.replicationFactor !== undefined &&\n            (!Number.isInteger(props.replicationFactor) || props.replicationFactor < 1 || props.replicationFactor > 3)) {\n          result.push('Invalid [replicationFactor]: expecting an integer between 1 and 3.');\n        }\n\n        if (props.metadata.events.length === 0) {\n          result.push('Invalid [metadata]: expecting [events] to have at least on element.');\n        }\n\n        if (!props.isLogCompacted && props.logCompactionProperties) {\n          result.push('Invalid [logCompactionProperties]: log compaction is disabled so properties must be undefined.');\n        }\n\n        return result;\n      },\n    });\n  }\n}\n"]}
80
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"kafka-topic.js","sourceRoot":"","sources":["../../src/kafka/kafka-topic.ts"],"names":[],"mappings":";;;;;AAAA,6CAA2D;AAC3D,2CAA2C;AAC3C,2CAAuC;AAEvC,yDAA0E;AA6G1E;;GAEG;AACH,MAAa,UAAW,SAAQ,sBAAS;IAEvC,YAAY,KAAgB,EAAE,EAAU,EAAE,KAAsB;;QAC9D,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEjB;;;;UAIE;QACF,MAAM,yBAAyB,GAAG,WAAW,CAAC;QAC9C,MAAM,SAAS,GAAG,mBAAK,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC,yBAAyB,CAAC;YAC3E,CAAC;gBACA,mBAAK,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,yBAAyB,CAAyB,CAAC,WAAW;YAC7F,CAAC;gBACD,GAAG,CAAC,eAAe,CAAC,uBAAuB,CACzC,KAAK,EACL,yBAAyB,EACzB,mBAAmB,CACpB,CAAC,WAAW,CAAC;QAEhB,MAAM,MAAM,GAAG,KAAK,CAAC,QAAQ,CAAC,MAAM,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE;YAChD,IAAI,iCAAc,CAAC,IAAI,EAAE,GAAG,MAAM,CAAC,SAAS,CAAC,OAAO,CAAC,KAAK,EAAE,GAAG,CAAC,YAAY,EAAE,EAAE,MAAM,EAAE,CAAC,CAAC;YAC1F,OAAO,MAAM,CAAC,SAAS,CAAC;QAC1B,CAAC,CAAC,CAAC;QAEH,IAAI,yBAAW,CAAC,IAAI,EAAE,YAAY,EAAE;YAClC,IAAI,EAAE,oBAAoB;YAC1B,UAAU,EAAE;gBACV,YAAY,EAAE,+DAA+D;gBAC7E,OAAO,EAAE,GAAG;gBACZ,SAAS,EAAE,KAAK,CAAC,IAAI;gBACrB,WAAW,EAAE,KAAK,CAAC,QAAQ,CAAC,WAAW;gBACvC,MAAM,EAAE,MAAM;gBACd,kBAAkB,EAAE,KAAK,CAAC,kBAAkB;gBAC5C,iBAAiB,QAAE,KAAK,CAAC,iBAAiB,mCAAI,CAAC;gBAC/C,WAAW,EAAE,OAAC,KAAK,CAAC,SAAS,mCAAI,sBAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,cAAc,EAAE;gBACnE,iBAAiB,EAAE,KAAK,CAAC,cAAc,CAAC,CAAC,aAAC,KAAK,CAAC,uBAAuB,0CAAE,eAAe,mCAAI,sBAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,cAAc,EAAE,CAAC,CAAC,CAAC,SAAS;gBACzI,sBAAsB,QAAE,KAAK,CAAC,sBAAsB,mCAAI,KAAK;gBAC7D,aAAa,EAAE,KAAK,CAAC,cAAc,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,QAAQ;gBAC1D,SAAS,EAAE,SAAS;gBACpB,GAAG,EAAE;oBACH,IAAI,EAAE,KAAK,CAAC,GAAG,CAAC,IAAI;oBACpB,KAAK,EAAE,KAAK,CAAC,GAAG,CAAC,KAAK;iBACvB;aACF;SACF,CAAC,CAAC;QAEH,IAAI,CAAC,IAAI,CAAC,aAAa,CAAC;YACtB,QAAQ,EAAE,GAAa,EAAE;gBACvB,MAAM,MAAM,GAAG,EAAE,CAAC;gBAElB,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,kBAAkB,CAAC,EAAE;oBACzC,MAAM,CAAC,IAAI,CAAC,6EAA6E,CAAC,CAAC;iBAC5F;gBAED,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,KAAK,CAAC,kBAAkB,CAAC,IAAI,KAAK,CAAC,kBAAkB,GAAG,CAAC,EAAE;oBAC/E,MAAM,CAAC,IAAI,CAAC,iGAAiG,KAAK,CAAC,kBAAkB,GAAG,CAAC,CAAC;iBAC3I;gBAED,IAAI,KAAK,CAAC,iBAAiB,KAAK,SAAS;oBACrC,CAAC,CAAC,MAAM,CAAC,SAAS,CAAC,KAAK,CAAC,iBAAiB,CAAC,IAAI,KAAK,CAAC,iBAAiB,GAAG,CAAC,IAAI,KAAK,CAAC,iBAAiB,GAAG,CAAC,CAAC,EAAE;oBAC9G,MAAM,CAAC,IAAI,CAAC,oEAAoE,CAAC,CAAC;iBACnF;gBAED,IAAI,KAAK,CAAC,QAAQ,CAAC,MAAM,CAAC,MAAM,KAAK,CAAC,EAAE;oBACtC,MAAM,CAAC,IAAI,CAAC,qEAAqE,CAAC,CAAC;iBACpF;gBAED,IAAI,CAAC,KAAK,CAAC,cAAc,IAAI,KAAK,CAAC,uBAAuB,EAAE;oBAC1D,MAAM,CAAC,IAAI,CAAC,gGAAgG,CAAC,CAAC;iBAC/G;gBAED,OAAO,MAAM,CAAC;YAChB,CAAC;SACF,CAAC,CAAC;IACL,CAAC;;AA5EH,gCA6EC","sourcesContent":["import { Duration, CfnResource, Stack } from 'aws-cdk-lib';\nimport * as ssm from 'aws-cdk-lib/aws-ssm';\nimport { Construct } from 'constructs';\n\nimport { KafkaEventSpec, KafkaEventSpecSource } from './kafka-event-spec';\n\n/**\n * Wrapper for all meta data of a topic.\n */\nexport interface KafkaTopicMetaData {\n  /**\n   * The event specs of the events on that topic\n   */\n  readonly events: KafkaEventSpecSource[];\n\n  /**\n   * Describes the intent of the topic. This might be information about the events on that topic\n   * or additional information about the producer and the context of the events.\n   */\n  readonly description: string;\n}\n\n/**\n * Read and write permissions for the topic.\n *\n * strings are matched against the CNAME of the certificate of the Kafka clients.\n */\nexport interface KafkaAclStatement {\n  /**\n  * List of clients that should get write permissions\n  */\n  readonly write: string[];\n\n  /**\n   * List of clients that should get read permissions\n   */\n  readonly read: string[];\n}\n\n/**\n * Wrapper for all log compaction related properties.\n */\nexport interface LogCompactionProperties {\n  /**\n   * The amount of time to retain delete tombstone markers for log compacted topics;\n   * The soft limit is 10 days but can be increased upon requests.\n   * @default 1 day\n   */\n  readonly deleteRetention?: Duration;\n}\n\n/**\n * The current service limits can be found in the topic limits configuration of the topic manager.\n *\n * @see https://collaboration.msi.audi.com/stash/projects/RSEVTBU/repos/topic-manager/browse/config/topic-service-limits.yaml\n */\nexport interface KafkaTopicProps{\n  /**\n   * The unique name of the topic; has to match pattern [A-Za-z0-9.-]+\n   */\n  readonly name: string;\n\n  /**\n   * Defines degree of parallelism of the topic; should be increased for\n   * large expected loads (e.g., 25 for rio.asset-iot-events).\n   * The soft limit is 10 partitions but can be increased upon requests.\n   */\n  readonly numberOfPartitions: number;\n\n  /**\n   * Defines degree of replication of messages; has to be between 1 and the number of brokers (currently 3).\n   * @default 3\n   */\n  readonly replicationFactor?: number;\n\n  /**\n   * Time how long messages are retained on Kafka cluster;\n   *\n   * We recommend 30 days for general events and 7 days for sensor events and other high-load cases.\n   * The soft limit is 3 to 30 days but can be increased upon requests.\n   * @default 7 days\n   */\n  readonly retention?: Duration;\n\n  /**\n   * If set to 'false, the topic's data will be deleted on Kafka 10 days after stack deletion.\n   * If set to 'true', it will be deleted immediately after stack deletion.\n   * @default false\n   */\n  readonly instantDeletionEnabled?: boolean;\n\n  /**\n   * The metadata of the topic.\n   */\n  readonly metadata: KafkaTopicMetaData;\n\n  /**\n   * The permissions to access to the topic's data.\n   */\n  readonly acl: KafkaAclStatement;\n\n  /**\n   * Specify whether the topic is log compacted or not.\n   */\n  readonly isLogCompacted: boolean;\n\n  /**\n   * Must only be provided if 'isLogCompacted' is 'true'.\n   * @default undefined\n   */\n  readonly logCompactionProperties?: LogCompactionProperties;\n}\n\n/**\n * Construct to create a kafka topic.\n */\nexport class KafkaTopic extends Construct {\n\n  constructor(scope: Construct, id: string, props: KafkaTopicProps) {\n    super(scope, id);\n\n    /**\n    SSM Parameter references dont generate logical names by appending IDs of the caller constructs. Instead simply use ID passed during function call 'fromStringParameterName'.\n    This means that if this construct is called more than once in the same stack, it would fail with error 'Construct name already used'.\n    Checking for existing SSM Parameter used to avoid this error\n    */\n    const teamEmailSSMCDKIdentifier = 'TeamEmail';\n    const teamEmail = Stack.of(this).node.tryFindChild(teamEmailSSMCDKIdentifier)\n      ?\n      (Stack.of(this).node.findChild(teamEmailSSMCDKIdentifier) as ssm.StringParameter).stringValue\n      :\n      ssm.StringParameter.fromStringParameterName(\n        scope,\n        teamEmailSSMCDKIdentifier,\n        '/config/team/mail',\n      ).stringValue;\n\n    const events = props.metadata.events.map(source => {\n      new KafkaEventSpec(this, `${source.eventName.replace(/\\./g, '_')}-EventSpec`, { source });\n      return source.eventName;\n    });\n\n    new CfnResource(this, 'KafkaTopic', {\n      type: 'Custom::KafkaTopic',\n      properties: {\n        ServiceToken: 'arn:aws:sns:eu-west-1:186993757734:dp-topic-manager-sns-topic',\n        Version: '3',\n        TopicName: props.name,\n        Description: props.metadata.description,\n        Events: events,\n        NumberOfPartitions: props.numberOfPartitions,\n        ReplicationFactor: props.replicationFactor ?? 3,\n        RetentionMs: (props.retention ?? Duration.days(7)).toMilliseconds(),\n        DeleteRetentionMs: props.isLogCompacted ? props.logCompactionProperties?.deleteRetention ?? Duration.days(7).toMilliseconds() : undefined,\n        InstantDeletionEnabled: props.instantDeletionEnabled ?? false,\n        CleanupPolicy: props.isLogCompacted ? 'compact' : 'delete',\n        TeamEmail: teamEmail,\n        ACL: {\n          Read: props.acl.read,\n          Write: props.acl.write,\n        },\n      },\n    });\n\n    this.node.addValidation({\n      validate: (): string[] => {\n        const result = [];\n\n        if (!props.name.match(/^[A-Za-z0-9.-]+$/)) {\n          result.push('Invalid [name]: expecting topic name to match the reg exp `[A-Za-z0-9.-]+`.');\n        }\n\n        if (!Number.isInteger(props.numberOfPartitions) || props.numberOfPartitions < 1) {\n          result.push(`Invalid [numberOfPartitions]: expecting number of partitions to be a positive integer but got ${props.numberOfPartitions}.`);\n        }\n\n        if (props.replicationFactor !== undefined &&\n            (!Number.isInteger(props.replicationFactor) || props.replicationFactor < 1 || props.replicationFactor > 3)) {\n          result.push('Invalid [replicationFactor]: expecting an integer between 1 and 3.');\n        }\n\n        if (props.metadata.events.length === 0) {\n          result.push('Invalid [metadata]: expecting [events] to have at least on element.');\n        }\n\n        if (!props.isLogCompacted && props.logCompactionProperties) {\n          result.push('Invalid [logCompactionProperties]: log compaction is disabled so properties must be undefined.');\n        }\n\n        return result;\n      },\n    });\n  }\n}\n"]}
@@ -18,7 +18,7 @@ class UpperToLower extends constructs_1.Construct {
18
18
  };
19
19
  };`),
20
20
  handler: 'index.handler',
21
- runtime: lambda.Runtime.NODEJS_12_X,
21
+ runtime: lambda.Runtime.NODEJS_14_X,
22
22
  });
23
23
  this.provider = new cr.Provider(this, 'MyProvider', {
24
24
  onEventHandler: crLambda,
@@ -32,4 +32,4 @@ class UpperToLower extends constructs_1.Construct {
32
32
  }
33
33
  }
34
34
  exports.UpperToLower = UpperToLower;
35
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXBwZXJUb0xvd2VyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL3dhdGNoZnVsL3VwcGVyVG9Mb3dlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSw2Q0FBb0M7QUFDcEMsaURBQWlEO0FBQ2pELDZDQUE2QztBQUM3QyxtREFBbUQ7QUFDbkQsMkNBQXVDO0FBR3ZDLE1BQWEsWUFBYSxTQUFRLHNCQUFTO0lBVXpDLFlBQVksS0FBZ0IsRUFBRSxFQUFVO1FBQ3RDLEtBQUssQ0FBQyxLQUFLLEVBQUUsRUFBRSxDQUFDLENBQUM7UUFFakIsTUFBTSxRQUFRLEdBQUcsSUFBSSxNQUFNLENBQUMsUUFBUSxDQUFDLElBQUksRUFBRSxjQUFjLEVBQUU7WUFDekQsSUFBSSxFQUFFLE1BQU0sQ0FBQyxJQUFJLENBQUMsVUFBVSxDQUFDOzs7Ozs7U0FNMUIsQ0FBQztZQUNKLE9BQU8sRUFBRSxlQUFlO1lBQ3hCLE9BQU8sRUFBRSxNQUFNLENBQUMsT0FBTyxDQUFDLFdBQVc7U0FDcEMsQ0FBQyxDQUFDO1FBQ0gsSUFBSSxDQUFDLFFBQVEsR0FBRyxJQUFJLEVBQUUsQ0FBQyxRQUFRLENBQUMsSUFBSSxFQUFFLFlBQVksRUFBRTtZQUNsRCxjQUFjLEVBQUUsUUFBUTtZQUN4QixZQUFZLEVBQUUsSUFBSSxDQUFDLGFBQWEsQ0FBQyxRQUFRO1NBQzFDLENBQUMsQ0FBQztJQUNMLENBQUM7SUExQk0sTUFBTSxDQUFDLFdBQVcsQ0FBQyxLQUFnQjtRQUN4QyxNQUFNLEtBQUssR0FBRyxtQkFBSyxDQUFDLEVBQUUsQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUM5QixNQUFNLEVBQUUsR0FBRyxnQ0FBZ0MsQ0FBQztRQUM1QyxPQUFPLEtBQUssQ0FBQyxJQUFJLENBQUMsWUFBWSxDQUFDLEVBQUUsQ0FBaUIsSUFBSSxJQUFJLFlBQVksQ0FBQyxLQUFLLEVBQUUsRUFBRSxDQUFDLENBQUM7SUFDcEYsQ0FBQztDQXVCRjtBQTdCRCxvQ0E2QkMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBTdGFjayB9IGZyb20gJ2F3cy1jZGstbGliJztcbmltcG9ydCAqIGFzIGxhbWJkYSBmcm9tICdhd3MtY2RrLWxpYi9hd3MtbGFtYmRhJztcbmltcG9ydCAqIGFzIGxvZ3MgZnJvbSAnYXdzLWNkay1saWIvYXdzLWxvZ3MnO1xuaW1wb3J0ICogYXMgY3IgZnJvbSAnYXdzLWNkay1saWIvY3VzdG9tLXJlc291cmNlcyc7XG5pbXBvcnQgeyBDb25zdHJ1Y3QgfSBmcm9tICdjb25zdHJ1Y3RzJztcblxuXG5leHBvcnQgY2xhc3MgVXBwZXJUb0xvd2VyIGV4dGVuZHMgQ29uc3RydWN0IHtcblxuICBwdWJsaWMgc3RhdGljIGdldE9yQ3JlYXRlKHNjb3BlOiBDb25zdHJ1Y3QpIHtcbiAgICBjb25zdCBzdGFjayA9IFN0YWNrLm9mKHNjb3BlKTtcbiAgICBjb25zdCBpZCA9ICd1cHBlci10by1sb3dlci1jdXN0b20tcmVzb3VyY2UnO1xuICAgIHJldHVybiBzdGFjay5ub2RlLnRyeUZpbmRDaGlsZChpZCkgYXMgVXBwZXJUb0xvd2VyIHx8IG5ldyBVcHBlclRvTG93ZXIoc3RhY2ssIGlkKTtcbiAgfVxuXG4gIHB1YmxpYyByZWFkb25seSBwcm92aWRlcjogY3IuUHJvdmlkZXI7XG5cbiAgY29uc3RydWN0b3Ioc2NvcGU6IENvbnN0cnVjdCwgaWQ6IHN0cmluZykge1xuICAgIHN1cGVyKHNjb3BlLCBpZCk7XG5cbiAgICBjb25zdCBjckxhbWJkYSA9IG5ldyBsYW1iZGEuRnVuY3Rpb24odGhpcywgJ1VwcGVyVG9Mb3dlcicsIHtcbiAgICAgIGNvZGU6IGxhbWJkYS5Db2RlLmZyb21JbmxpbmUoYGV4cG9ydHMuaGFuZGxlciA9IGFzeW5jIChlKSA9PiB7XG4gICAgICAgIHJldHVybiB7XG4gICAgICAgICAgRGF0YToge1xuICAgICAgICAgICAgTG93ZXI6IGUuUmVzb3VyY2VQcm9wZXJ0aWVzLlVwcGVyLnRvTG93ZXJDYXNlKCksXG4gICAgICAgICAgfSxcbiAgICAgICAgfTtcbiAgICAgIH07YCksXG4gICAgICBoYW5kbGVyOiAnaW5kZXguaGFuZGxlcicsXG4gICAgICBydW50aW1lOiBsYW1iZGEuUnVudGltZS5OT0RFSlNfMTJfWCxcbiAgICB9KTtcbiAgICB0aGlzLnByb3ZpZGVyID0gbmV3IGNyLlByb3ZpZGVyKHRoaXMsICdNeVByb3ZpZGVyJywge1xuICAgICAgb25FdmVudEhhbmRsZXI6IGNyTGFtYmRhLFxuICAgICAgbG9nUmV0ZW50aW9uOiBsb2dzLlJldGVudGlvbkRheXMuT05FX1dFRUssXG4gICAgfSk7XG4gIH1cbn1cbiJdfQ==
35
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXBwZXJUb0xvd2VyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL3dhdGNoZnVsL3VwcGVyVG9Mb3dlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSw2Q0FBb0M7QUFDcEMsaURBQWlEO0FBQ2pELDZDQUE2QztBQUM3QyxtREFBbUQ7QUFDbkQsMkNBQXVDO0FBR3ZDLE1BQWEsWUFBYSxTQUFRLHNCQUFTO0lBVXpDLFlBQVksS0FBZ0IsRUFBRSxFQUFVO1FBQ3RDLEtBQUssQ0FBQyxLQUFLLEVBQUUsRUFBRSxDQUFDLENBQUM7UUFFakIsTUFBTSxRQUFRLEdBQUcsSUFBSSxNQUFNLENBQUMsUUFBUSxDQUFDLElBQUksRUFBRSxjQUFjLEVBQUU7WUFDekQsSUFBSSxFQUFFLE1BQU0sQ0FBQyxJQUFJLENBQUMsVUFBVSxDQUFDOzs7Ozs7U0FNMUIsQ0FBQztZQUNKLE9BQU8sRUFBRSxlQUFlO1lBQ3hCLE9BQU8sRUFBRSxNQUFNLENBQUMsT0FBTyxDQUFDLFdBQVc7U0FDcEMsQ0FBQyxDQUFDO1FBQ0gsSUFBSSxDQUFDLFFBQVEsR0FBRyxJQUFJLEVBQUUsQ0FBQyxRQUFRLENBQUMsSUFBSSxFQUFFLFlBQVksRUFBRTtZQUNsRCxjQUFjLEVBQUUsUUFBUTtZQUN4QixZQUFZLEVBQUUsSUFBSSxDQUFDLGFBQWEsQ0FBQyxRQUFRO1NBQzFDLENBQUMsQ0FBQztJQUNMLENBQUM7SUExQk0sTUFBTSxDQUFDLFdBQVcsQ0FBQyxLQUFnQjtRQUN4QyxNQUFNLEtBQUssR0FBRyxtQkFBSyxDQUFDLEVBQUUsQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUM5QixNQUFNLEVBQUUsR0FBRyxnQ0FBZ0MsQ0FBQztRQUM1QyxPQUFPLEtBQUssQ0FBQyxJQUFJLENBQUMsWUFBWSxDQUFDLEVBQUUsQ0FBaUIsSUFBSSxJQUFJLFlBQVksQ0FBQyxLQUFLLEVBQUUsRUFBRSxDQUFDLENBQUM7SUFDcEYsQ0FBQztDQXVCRjtBQTdCRCxvQ0E2QkMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBTdGFjayB9IGZyb20gJ2F3cy1jZGstbGliJztcbmltcG9ydCAqIGFzIGxhbWJkYSBmcm9tICdhd3MtY2RrLWxpYi9hd3MtbGFtYmRhJztcbmltcG9ydCAqIGFzIGxvZ3MgZnJvbSAnYXdzLWNkay1saWIvYXdzLWxvZ3MnO1xuaW1wb3J0ICogYXMgY3IgZnJvbSAnYXdzLWNkay1saWIvY3VzdG9tLXJlc291cmNlcyc7XG5pbXBvcnQgeyBDb25zdHJ1Y3QgfSBmcm9tICdjb25zdHJ1Y3RzJztcblxuXG5leHBvcnQgY2xhc3MgVXBwZXJUb0xvd2VyIGV4dGVuZHMgQ29uc3RydWN0IHtcblxuICBwdWJsaWMgc3RhdGljIGdldE9yQ3JlYXRlKHNjb3BlOiBDb25zdHJ1Y3QpIHtcbiAgICBjb25zdCBzdGFjayA9IFN0YWNrLm9mKHNjb3BlKTtcbiAgICBjb25zdCBpZCA9ICd1cHBlci10by1sb3dlci1jdXN0b20tcmVzb3VyY2UnO1xuICAgIHJldHVybiBzdGFjay5ub2RlLnRyeUZpbmRDaGlsZChpZCkgYXMgVXBwZXJUb0xvd2VyIHx8IG5ldyBVcHBlclRvTG93ZXIoc3RhY2ssIGlkKTtcbiAgfVxuXG4gIHB1YmxpYyByZWFkb25seSBwcm92aWRlcjogY3IuUHJvdmlkZXI7XG5cbiAgY29uc3RydWN0b3Ioc2NvcGU6IENvbnN0cnVjdCwgaWQ6IHN0cmluZykge1xuICAgIHN1cGVyKHNjb3BlLCBpZCk7XG5cbiAgICBjb25zdCBjckxhbWJkYSA9IG5ldyBsYW1iZGEuRnVuY3Rpb24odGhpcywgJ1VwcGVyVG9Mb3dlcicsIHtcbiAgICAgIGNvZGU6IGxhbWJkYS5Db2RlLmZyb21JbmxpbmUoYGV4cG9ydHMuaGFuZGxlciA9IGFzeW5jIChlKSA9PiB7XG4gICAgICAgIHJldHVybiB7XG4gICAgICAgICAgRGF0YToge1xuICAgICAgICAgICAgTG93ZXI6IGUuUmVzb3VyY2VQcm9wZXJ0aWVzLlVwcGVyLnRvTG93ZXJDYXNlKCksXG4gICAgICAgICAgfSxcbiAgICAgICAgfTtcbiAgICAgIH07YCksXG4gICAgICBoYW5kbGVyOiAnaW5kZXguaGFuZGxlcicsXG4gICAgICBydW50aW1lOiBsYW1iZGEuUnVudGltZS5OT0RFSlNfMTRfWCxcbiAgICB9KTtcbiAgICB0aGlzLnByb3ZpZGVyID0gbmV3IGNyLlByb3ZpZGVyKHRoaXMsICdNeVByb3ZpZGVyJywge1xuICAgICAgb25FdmVudEhhbmRsZXI6IGNyTGFtYmRhLFxuICAgICAgbG9nUmV0ZW50aW9uOiBsb2dzLlJldGVudGlvbkRheXMuT05FX1dFRUssXG4gICAgfSk7XG4gIH1cbn1cbiJdfQ==
package/package.json CHANGED
@@ -15,7 +15,7 @@
15
15
  ],
16
16
  "main": "lib/index.js",
17
17
  "license": "Apache-2.0",
18
- "version": "2.5.5",
18
+ "version": "2.5.8",
19
19
  "types": "lib/index.d.ts",
20
20
  "stability": "stable",
21
21
  "jsii": {
package/version.json CHANGED
@@ -1,3 +1,3 @@
1
1
  {
2
- "version": "2.5.5"
2
+ "version": "2.5.8"
3
3
  }