@rio-cloud/cdk-v2-constructs 5.1.0 → 6.0.1

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 (34) hide show
  1. package/.jsii +41 -24
  2. package/API.md +26 -0
  3. package/CHANGELOG.md +20 -0
  4. package/lib/contributions/team-claid/datadog-usage-monitoring/datadog-usage-monitoring.js +1 -1
  5. package/lib/contributions/team-transport-two/datadog/datadog-dashboard-downloader.js +1 -1
  6. package/lib/contributions/team-transport-two/pipeline/pipeline-stack.js +1 -1
  7. package/lib/datadog/datadog-log-index-monitoring.js +1 -1
  8. package/lib/datadog/datadog-monitor.js +1 -1
  9. package/lib/datadogv2/datadog-log-index-monitoring.js +1 -1
  10. package/lib/datadogv2/datadog-monitor.js +1 -1
  11. package/lib/datadogv2/datadog-notification.js +1 -1
  12. package/lib/fargate/datadog.js +1 -1
  13. package/lib/fargate/rio-fargate-service.js +2 -2
  14. package/lib/kafka/kafka-event-spec.js +1 -1
  15. package/lib/kafka/kafka-topic.js +1 -1
  16. package/lib/pipeline/rio-backup-secrets-restore-stage.js +1 -1
  17. package/lib/toggle.js +1 -1
  18. package/lib/watchful/alb.d.ts +13 -13
  19. package/lib/watchful/alb.js +5 -2
  20. package/lib/watchful/aspect.js +1 -1
  21. package/lib/watchful/datadog-log-alarm.js +3 -3
  22. package/lib/watchful/datadog-metric-alarm.js +3 -3
  23. package/lib/watchful/ecs.js +2 -2
  24. package/lib/watchful/lambda.js +4 -4
  25. package/lib/watchful/metric-alarm.d.ts +4 -0
  26. package/lib/watchful/metric-alarm.js +1 -1
  27. package/lib/watchful/watchful.js +3 -3
  28. package/lib/watchfulv2/datadog-log-alarm.d.ts +1 -0
  29. package/lib/watchfulv2/datadog-log-alarm.js +19 -6
  30. package/lib/watchfulv2/datadog-metric-alarm.d.ts +4 -4
  31. package/lib/watchfulv2/datadog-metric-alarm.js +14 -7
  32. package/lib/watchfulv2/watchful.js +19 -9
  33. package/package.json +4 -4
  34. package/version.json +1 -1
@@ -4,6 +4,7 @@ exports.DataDogLogAlarm = exports.ComparisonOperator = exports.UnitOfPeriod = vo
4
4
  const aws_cdk_lib_1 = require("aws-cdk-lib");
5
5
  const constructs_1 = require("constructs");
6
6
  const datadogv2_1 = require("../datadogv2");
7
+ const upperToLower_1 = require("../watchful/upperToLower");
7
8
  var UnitOfPeriod;
8
9
  (function (UnitOfPeriod) {
9
10
  UnitOfPeriod["SECONDS"] = "s";
@@ -40,11 +41,23 @@ class DataDogLogAlarm extends constructs_1.Construct {
40
41
  alarmToDataDogQuery(alarm) {
41
42
  const [periodValue, periodUnit] = this.convertToDatadogPeriod(alarm.period, alarm.unitOfPeriod);
42
43
  let comparisonOperator = alarm.periodOperator;
43
- const filters = new Map([
44
- ['service', alarm.serviceName],
45
- ['account_id', aws_cdk_lib_1.Stack.of(this).account],
46
- ['status', alarm.queryFilters],
47
- ]);
44
+ const filters = new Map([]);
45
+ if (alarm.lambdaFunctionName) {
46
+ const upperToLowerCr = new aws_cdk_lib_1.CustomResource(this, 'ConcatenatedFiltersUpperToLower', {
47
+ resourceType: 'Custom::UpperToLower',
48
+ serviceToken: upperToLower_1.UpperToLower.getOrCreate(this).provider.serviceToken,
49
+ properties: {
50
+ Upper: alarm.lambdaFunctionName,
51
+ },
52
+ });
53
+ const lambdaFunctionNameToLower = upperToLowerCr.getAtt('Lower');
54
+ filters.set('functionname', lambdaFunctionNameToLower);
55
+ }
56
+ else {
57
+ filters.set('service', alarm.serviceName);
58
+ }
59
+ filters.set('account_id', aws_cdk_lib_1.Stack.of(this).account);
60
+ filters.set('status', alarm.queryFilters);
48
61
  const concatenatedFilters = Array.from(filters.entries())
49
62
  .map(([filterKey, filterValue]) => `${filterKey}:${filterValue}`)
50
63
  .join(' ');
@@ -65,4 +78,4 @@ class DataDogLogAlarm extends constructs_1.Construct {
65
78
  }
66
79
  }
67
80
  exports.DataDogLogAlarm = DataDogLogAlarm;
68
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"datadog-log-alarm.js","sourceRoot":"","sources":["../../src/watchfulv2/datadog-log-alarm.ts"],"names":[],"mappings":"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","sourcesContent":["import { Duration, Stack } from 'aws-cdk-lib';\nimport { Construct } from 'constructs';\nimport { DatadogMonitor, DatadogMonitorQueryAlertType, INotification } from '../datadogv2';\n\nexport interface DataDogLogQueryAlarmProps extends DataDogProps, LogAlarmProps {\n}\n\nexport enum UnitOfPeriod {\n  SECONDS = 's',\n  MINUTES = 'm',\n  HOURS = 'h',\n  DAYS = 'd',\n}\n\nconst DEFAULT_UNIT_OF_PERIOD = UnitOfPeriod.SECONDS;\n\nexport enum ComparisonOperator {\n  GREATER_THAN_OR_EQUAL_TO_THRESHOLD = '>=',\n  GREATER_THAN_THRESHOLD = '>',\n  LESS_THAN_OR_EQUAL_TO_THRESHOLD = '<=',\n  LESS_THAN_THRESHOLD = '<',\n}\n\nexport interface DataDogProps {\n  readonly serviceName: string; // needs to match the service name used in datadog-integration\n}\n\nexport interface LogAlarmProps {\n  readonly period: Duration;\n  readonly unitOfPeriod?: UnitOfPeriod; // unit which will be used to create the alert. if none is provided the periods unit will default to seconds\n  readonly periodOperator: ComparisonOperator;\n  readonly statistic: 'count'; // aggregation function\n  readonly queryFilters: 'error'|'warn'; // it should either say error or warn\n  readonly alarmDescription: string;\n  readonly threshold: number;\n  readonly priority: number;\n  readonly index: string;\n  readonly notification?: INotification;\n  readonly lambdaCloudwatchGroupName?: string; // Should only be set in case of lambda function. The cloudwatch log group name is same the lambda function name\n  readonly disableAutoClose?: boolean;\n  readonly renotifyInterval?: number;\n}\n\nexport class DataDogLogAlarm extends Construct {\n  constructor(scope: Construct, id: string, props: DataDogLogQueryAlarmProps) {\n    super(scope, id);\n\n    new DatadogMonitor(scope,\n      'ErrorMonitorLogAlert',\n      {\n        serviceName: props.serviceName,\n        monitorType: DatadogMonitorQueryAlertType.LOG_ALERT,\n        name: `${this.node.scope?.node.id} ${id}`,\n        query: this.alarmToDataDogQuery(props),\n        message: `${props.alarmDescription}: {{log.link}}`,\n        notification: props.notification,\n        optionOverrides: {\n          evaluationDelay: 0,\n          thresholds: props.disableAutoClose ? { criticalRecovery: -1 } : undefined,\n          renotifyInterval: props.renotifyInterval,\n        },\n        priority: props.priority,\n      });\n\n  }\n  private alarmToDataDogQuery(alarm: DataDogLogQueryAlarmProps): string {\n\n    const [periodValue, periodUnit] = this.convertToDatadogPeriod(alarm.period, alarm.unitOfPeriod);\n\n    let comparisonOperator = alarm.periodOperator;\n\n    const filters = new Map([\n      ['service', alarm.serviceName],\n      ['account_id', Stack.of(this).account],\n      ['status', alarm.queryFilters],\n    ]);\n\n    const concatenatedFilters = Array.from(filters.entries())\n      .map(([filterKey, filterValue]) => `${filterKey}:${filterValue}`)\n      .join(' ');\n\n    return `logs(\"${concatenatedFilters}\").index(\"${alarm.index}\").rollup(\"${alarm.statistic}\").last(\"${periodValue}${periodUnit}\") ${comparisonOperator} ${alarm.threshold}`;\n  }\n\n  private convertToDatadogPeriod(period: Duration, unitOfPeriod?: UnitOfPeriod): [number, string] {\n    const unit = unitOfPeriod ?? DEFAULT_UNIT_OF_PERIOD;\n    const mapPeriodToDatadogTime = {\n      [UnitOfPeriod.SECONDS]: () => period.toSeconds(),\n      [UnitOfPeriod.MINUTES]: () => period.toMinutes(),\n      [UnitOfPeriod.HOURS]: () => period.toHours(),\n      [UnitOfPeriod.DAYS]: () => period.toDays(),\n    };\n    return [\n      mapPeriodToDatadogTime[unit](),\n      unit.toString(),\n    ];\n  }\n}"]}
81
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"datadog-log-alarm.js","sourceRoot":"","sources":["../../src/watchfulv2/datadog-log-alarm.ts"],"names":[],"mappings":"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","sourcesContent":["import { CustomResource, Duration, Stack } from 'aws-cdk-lib';\nimport { Construct } from 'constructs';\nimport { DatadogMonitor, DatadogMonitorQueryAlertType, INotification } from '../datadogv2';\nimport { UpperToLower } from '../watchful/upperToLower';\n\nexport interface DataDogLogQueryAlarmProps extends DataDogProps, LogAlarmProps {\n}\n\nexport enum UnitOfPeriod {\n  SECONDS = 's',\n  MINUTES = 'm',\n  HOURS = 'h',\n  DAYS = 'd',\n}\n\nconst DEFAULT_UNIT_OF_PERIOD = UnitOfPeriod.SECONDS;\n\nexport enum ComparisonOperator {\n  GREATER_THAN_OR_EQUAL_TO_THRESHOLD = '>=',\n  GREATER_THAN_THRESHOLD = '>',\n  LESS_THAN_OR_EQUAL_TO_THRESHOLD = '<=',\n  LESS_THAN_THRESHOLD = '<',\n}\n\nexport interface DataDogProps {\n  readonly serviceName: string; // needs to match the service name used in datadog-integration\n  readonly lambdaFunctionName?: string;\n}\n\nexport interface LogAlarmProps {\n  readonly period: Duration;\n  readonly unitOfPeriod?: UnitOfPeriod; // unit which will be used to create the alert. if none is provided the periods unit will default to seconds\n  readonly periodOperator: ComparisonOperator;\n  readonly statistic: 'count'; // aggregation function\n  readonly queryFilters: 'error' | 'warn'; // it should either say error or warn\n  readonly alarmDescription: string;\n  readonly threshold: number;\n  readonly priority: number;\n  readonly index: string;\n  readonly notification?: INotification;\n  readonly lambdaCloudwatchGroupName?: string; // Should only be set in case of lambda function. The cloudwatch log group name is same the lambda function name\n  readonly disableAutoClose?: boolean;\n  readonly renotifyInterval?: number;\n}\n\nexport class DataDogLogAlarm extends Construct {\n  constructor(scope: Construct, id: string, props: DataDogLogQueryAlarmProps) {\n    super(scope, id);\n\n    new DatadogMonitor(scope,\n      'ErrorMonitorLogAlert',\n      {\n        serviceName: props.serviceName,\n        monitorType: DatadogMonitorQueryAlertType.LOG_ALERT,\n        name: `${this.node.scope?.node.id} ${id}`,\n        query: this.alarmToDataDogQuery(props),\n        message: `${props.alarmDescription}: {{log.link}}`,\n        notification: props.notification,\n        optionOverrides: {\n          evaluationDelay: 0,\n          thresholds: props.disableAutoClose ? { criticalRecovery: -1 } : undefined,\n          renotifyInterval: props.renotifyInterval,\n        },\n        priority: props.priority,\n      });\n\n  }\n  private alarmToDataDogQuery(alarm: DataDogLogQueryAlarmProps): string {\n\n    const [periodValue, periodUnit] = this.convertToDatadogPeriod(alarm.period, alarm.unitOfPeriod);\n\n    let comparisonOperator = alarm.periodOperator;\n\n    const filters = new Map([]);\n    if (alarm.lambdaFunctionName) {\n      const upperToLowerCr = new CustomResource(this, 'ConcatenatedFiltersUpperToLower', {\n        resourceType: 'Custom::UpperToLower',\n        serviceToken: UpperToLower.getOrCreate(this).provider.serviceToken,\n        properties: {\n          Upper: alarm.lambdaFunctionName,\n        },\n      });\n      const lambdaFunctionNameToLower = upperToLowerCr.getAtt('Lower');\n      filters.set('functionname', lambdaFunctionNameToLower);\n    } else {\n      filters.set('service', alarm.serviceName);\n    }\n    filters.set('account_id', Stack.of(this).account);\n    filters.set('status', alarm.queryFilters);\n\n    const concatenatedFilters = Array.from(filters.entries())\n      .map(([filterKey, filterValue]) => `${filterKey}:${filterValue}`)\n      .join(' ');\n\n    return `logs(\"${concatenatedFilters}\").index(\"${alarm.index}\").rollup(\"${alarm.statistic}\").last(\"${periodValue}${periodUnit}\") ${comparisonOperator} ${alarm.threshold}`;\n  }\n\n  private convertToDatadogPeriod(period: Duration, unitOfPeriod?: UnitOfPeriod): [number, string] {\n    const unit = unitOfPeriod ?? DEFAULT_UNIT_OF_PERIOD;\n    const mapPeriodToDatadogTime = {\n      [UnitOfPeriod.SECONDS]: () => period.toSeconds(),\n      [UnitOfPeriod.MINUTES]: () => period.toMinutes(),\n      [UnitOfPeriod.HOURS]: () => period.toHours(),\n      [UnitOfPeriod.DAYS]: () => period.toDays(),\n    };\n    return [\n      mapPeriodToDatadogTime[unit](),\n      unit.toString(),\n    ];\n  }\n}"]}
@@ -6,12 +6,12 @@ import { MetricAlarmProps } from '../watchful/metric-alarm';
6
6
  */
7
7
  export interface DataDogMetricAlarmProps extends MetricAlarmProps {
8
8
  /**
9
- * The service name used to name and tag the monitors.
10
- */
9
+ * The service name used to name and tag the monitors.
10
+ */
11
11
  readonly serviceName: string;
12
12
  /**
13
- * The notification to use for the monitor.
14
- */
13
+ * The notification to use for the monitor.
14
+ */
15
15
  readonly notification?: INotification;
16
16
  }
17
17
  /**
@@ -24,7 +24,9 @@ class DataDogMetricAlarm extends constructs_1.Construct {
24
24
  notification: props.notification,
25
25
  priority: props.priority,
26
26
  optionOverrides: {
27
- evaluationDelay: isAwsMetric ? cdk.Duration.minutes(15).toSeconds() : undefined,
27
+ evaluationDelay: isAwsMetric
28
+ ? cdk.Duration.minutes(15).toSeconds()
29
+ : undefined,
28
30
  },
29
31
  });
30
32
  }
@@ -57,7 +59,9 @@ ${props.alarmDescription} is back to normal.
57
59
  unit = 'd';
58
60
  }
59
61
  // avg(last_15m):max:aws.fargate.cpu.percent{ecs_container_name:$ServiceName$} > 95
60
- const namespace = alarm.metric.namespace.toLocaleLowerCase().replace('/', '.');
62
+ const namespace = alarm.metric.namespace
63
+ .toLocaleLowerCase()
64
+ .replace('/', '.');
61
65
  let metricName = alarm.metric.metricName;
62
66
  // Special case for ALB / TargetGroup metrics
63
67
  switch (metricName) {
@@ -76,13 +80,15 @@ ${props.alarmDescription} is back to normal.
76
80
  // statistics must be lowerCase and probably also converted (see allowed values in cw.Metric)
77
81
  let comparisonOperator = 'not supported';
78
82
  switch (alarm.comparisonOperator) {
79
- case cdk.aws_cloudwatch.ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD:
83
+ case cdk.aws_cloudwatch.ComparisonOperator
84
+ .GREATER_THAN_OR_EQUAL_TO_THRESHOLD:
80
85
  comparisonOperator = '>=';
81
86
  break;
82
87
  case cdk.aws_cloudwatch.ComparisonOperator.GREATER_THAN_THRESHOLD:
83
88
  comparisonOperator = '>';
84
89
  break;
85
- case cdk.aws_cloudwatch.ComparisonOperator.LESS_THAN_OR_EQUAL_TO_THRESHOLD:
90
+ case cdk.aws_cloudwatch.ComparisonOperator
91
+ .LESS_THAN_OR_EQUAL_TO_THRESHOLD:
86
92
  comparisonOperator = '<=';
87
93
  break;
88
94
  case cdk.aws_cloudwatch.ComparisonOperator.LESS_THAN_THRESHOLD:
@@ -123,15 +129,16 @@ ${props.alarmDescription} is back to normal.
123
129
  dimensions += `,${key.toLocaleLowerCase()}:${value}`;
124
130
  }
125
131
  }
132
+ const asCountAgggregator = alarm.aggregateQueryAsCount ? '.as_count()' : '';
126
133
  switch (alarm.treatMissingData) {
127
134
  case cdk.aws_cloudwatch.TreatMissingData.NOT_BREACHING:
128
- return `${alarmStatistic}(last_${totalPeriod}${unit}):default_zero(${alarmStatistic}:${metric}{${dimensions}}) ${comparisonOperator} ${alarm.threshold}`;
135
+ return `${alarmStatistic}(last_${totalPeriod}${unit}):default_zero(${alarmStatistic}:${metric}{${dimensions}}${asCountAgggregator}) ${comparisonOperator} ${alarm.threshold}`;
129
136
  case cdk.aws_cloudwatch.TreatMissingData.BREACHING:
130
137
  case cdk.aws_cloudwatch.TreatMissingData.IGNORE:
131
138
  default:
132
- return `${alarmStatistic}(last_${totalPeriod}${unit}):${alarmStatistic}:${metric}{${dimensions}} ${comparisonOperator} ${alarm.threshold}`;
139
+ return `${alarmStatistic}(last_${totalPeriod}${unit}):${alarmStatistic}:${metric}{${dimensions}}${asCountAgggregator} ${comparisonOperator} ${alarm.threshold}`;
133
140
  }
134
141
  }
135
142
  }
136
143
  exports.DataDogMetricAlarm = DataDogMetricAlarm;
137
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"datadog-metric-alarm.js","sourceRoot":"","sources":["../../src/watchfulv2/datadog-metric-alarm.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from 'aws-cdk-lib';\nimport { CustomResource, Stack } from 'aws-cdk-lib';\nimport { Construct } from 'constructs';\nimport { DatadogMonitor, DatadogMonitorQueryAlertType, INotification } from '../datadogv2';\nimport { MetricAlarmProps } from '../watchful/metric-alarm';\nimport { UpperToLower } from '../watchful/upperToLower';\n\n\n/**\n * Interface for DataDogMonitor implementation.\n */\nexport interface DataDogMetricAlarmProps extends MetricAlarmProps {\n\n  /**\n     * The service name used to name and tag the monitors.\n     */\n  readonly serviceName: string;\n\n  /**\n     * The notification to use for the monitor.\n     */\n  readonly notification?: INotification;\n}\n\n/**\n * A wrapper for a custom resource that creates a DataDog Monitor.\n *\n * To use this construct, the DataDog integration needs to be installed.\n */\nexport class DataDogMetricAlarm extends Construct {\n\n  constructor(scope: Construct, id: string, props: DataDogMetricAlarmProps) {\n    super(scope, id);\n\n    const isAwsMetric = props.metric.namespace.startsWith('AWS');\n\n    new DatadogMonitor(this,\n      'ErrorMonitorMetricAlert',\n      {\n        serviceName: props.serviceName,\n        name: `${this.node.scope?.node.id} ${id}`,\n        monitorType: DatadogMonitorQueryAlertType.QUERY_ALERT,\n        query: this.alarmToDataDogQuery(props),\n        message: this.getMessage(props),\n        notification: props.notification,\n        priority: props.priority,\n        optionOverrides: {\n          evaluationDelay: isAwsMetric ? cdk.Duration.minutes(15).toSeconds() : undefined,\n        },\n      });\n\n  }\n\n  private getMessage(props: DataDogMetricAlarmProps) {\n    return `\n{{#is_alert}}\n${props.alarmDescription}\n{{/is_alert}} \n\n{{#is_alert_recovery}}\n${props.alarmDescription} is back to normal.\n{{/is_alert_recovery}}`;\n  }\n\n  /**\n   * Helper function to convert a generic alarm into a DataDog query string.\n   */\n  alarmToDataDogQuery(alarm: MetricAlarmProps): string {\n    let unit: string;\n    let totalPeriod: number;\n    if (alarm.metric.period.toSeconds() < 60) {\n      totalPeriod = alarm.metric.period.toSeconds() * alarm.evaluationPeriods;\n      unit = 's';\n    } else if (alarm.metric.period.toMinutes() < 1440) {\n      totalPeriod = alarm.metric.period.toMinutes() * alarm.evaluationPeriods;\n      unit = 'm';\n    } else {\n      totalPeriod = alarm.metric.period.toDays() * alarm.evaluationPeriods;\n      unit = 'd';\n    }\n\n    // avg(last_15m):max:aws.fargate.cpu.percent{ecs_container_name:$ServiceName$} > 95\n    const namespace = alarm.metric.namespace.toLocaleLowerCase().replace('/', '.');\n\n    let metricName = alarm.metric.metricName;\n\n    // Special case for ALB / TargetGroup metrics\n    switch (metricName) {\n      case 'HTTPCode_ELB_5XX_Count':\n        metricName = 'HTTPCode_ELB_5XX';\n        break;\n      case 'HTTPCode_Target_5XX_Count':\n        metricName = 'HTTPCode_Target_5XX';\n        break;\n    }\n    if (!metricName.includes('_')) {\n      metricName = metricName.replace(/([a-z])([A-Z])/g, '$1_$2');\n    }\n    metricName = metricName.toLocaleLowerCase();\n\n    const metric = `${namespace}.${metricName}`;\n\n    // statistics must be lowerCase and probably also converted (see allowed values in cw.Metric)\n    let comparisonOperator = 'not supported';\n    switch (alarm.comparisonOperator) {\n      case cdk.aws_cloudwatch.ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD:\n        comparisonOperator = '>=';\n        break;\n      case cdk.aws_cloudwatch.ComparisonOperator.GREATER_THAN_THRESHOLD:\n        comparisonOperator = '>';\n        break;\n      case cdk.aws_cloudwatch.ComparisonOperator.LESS_THAN_OR_EQUAL_TO_THRESHOLD:\n        comparisonOperator = '<=';\n        break;\n      case cdk.aws_cloudwatch.ComparisonOperator.LESS_THAN_THRESHOLD:\n        comparisonOperator = '<';\n        break;\n    }\n\n    let alarmStatistic = 'not supported';\n    switch (alarm.metric.statistic) {\n      case cdk.aws_cloudwatch.Stats.AVERAGE:\n        alarmStatistic = 'avg';\n        break;\n      case cdk.aws_cloudwatch.Stats.MAXIMUM:\n        alarmStatistic = 'max';\n        break;\n      case cdk.aws_cloudwatch.Stats.MINIMUM:\n        alarmStatistic = 'min';\n        break;\n      case cdk.aws_cloudwatch.Stats.SAMPLE_COUNT:\n        break;\n      case cdk.aws_cloudwatch.Stats.SUM:\n        alarmStatistic = 'sum';\n        break;\n      default:\n        alarmStatistic = 'avg';\n        break;\n    }\n\n    let dimensions = `account_id:${Stack.of(this).account}`;\n\n    if (alarm.metric.dimensions) {\n      for (const key in alarm.metric.dimensions) {\n        const upperToLowerCr = new CustomResource(this, `${key}UpperToLower`, {\n          resourceType: 'Custom::UpperToLower',\n          serviceToken: UpperToLower.getOrCreate(this).provider.serviceToken,\n          properties: {\n            Upper: alarm.metric.dimensions[key],\n          },\n        });\n        const value = upperToLowerCr.getAtt('Lower');\n        dimensions += `,${key.toLocaleLowerCase()}:${value}`;\n      }\n    }\n\n    switch (alarm.treatMissingData) {\n      case cdk.aws_cloudwatch.TreatMissingData.NOT_BREACHING:\n        return `${alarmStatistic}(last_${totalPeriod}${unit}):default_zero(${alarmStatistic}:${metric}{${dimensions}}) ${comparisonOperator} ${alarm.threshold}`;\n      case cdk.aws_cloudwatch.TreatMissingData.BREACHING:\n      case cdk.aws_cloudwatch.TreatMissingData.IGNORE:\n      default:\n        return `${alarmStatistic}(last_${totalPeriod}${unit}):${alarmStatistic}:${metric}{${dimensions}} ${comparisonOperator} ${alarm.threshold}`;\n    }\n  }\n}\n"]}
144
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"datadog-metric-alarm.js","sourceRoot":"","sources":["../../src/watchfulv2/datadog-metric-alarm.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cdk from 'aws-cdk-lib';\nimport { CustomResource, Stack } from 'aws-cdk-lib';\nimport { Construct } from 'constructs';\nimport {\n  DatadogMonitor,\n  DatadogMonitorQueryAlertType,\n  INotification,\n} from '../datadogv2';\nimport { MetricAlarmProps } from '../watchful/metric-alarm';\nimport { UpperToLower } from '../watchful/upperToLower';\n\n/**\n * Interface for DataDogMonitor implementation.\n */\nexport interface DataDogMetricAlarmProps extends MetricAlarmProps {\n  /**\n   * The service name used to name and tag the monitors.\n   */\n  readonly serviceName: string;\n\n  /**\n   * The notification to use for the monitor.\n   */\n  readonly notification?: INotification;\n}\n\n/**\n * A wrapper for a custom resource that creates a DataDog Monitor.\n *\n * To use this construct, the DataDog integration needs to be installed.\n */\nexport class DataDogMetricAlarm extends Construct {\n  constructor(scope: Construct, id: string, props: DataDogMetricAlarmProps) {\n    super(scope, id);\n\n    const isAwsMetric = props.metric.namespace.startsWith('AWS');\n\n    new DatadogMonitor(this, 'ErrorMonitorMetricAlert', {\n      serviceName: props.serviceName,\n      name: `${this.node.scope?.node.id} ${id}`,\n      monitorType: DatadogMonitorQueryAlertType.QUERY_ALERT,\n      query: this.alarmToDataDogQuery(props),\n      message: this.getMessage(props),\n      notification: props.notification,\n      priority: props.priority,\n      optionOverrides: {\n        evaluationDelay: isAwsMetric\n          ? cdk.Duration.minutes(15).toSeconds()\n          : undefined,\n      },\n    });\n  }\n\n  private getMessage(props: DataDogMetricAlarmProps) {\n    return `\n{{#is_alert}}\n${props.alarmDescription}\n{{/is_alert}} \n\n{{#is_alert_recovery}}\n${props.alarmDescription} is back to normal.\n{{/is_alert_recovery}}`;\n  }\n\n  /**\n   * Helper function to convert a generic alarm into a DataDog query string.\n   */\n  alarmToDataDogQuery(alarm: MetricAlarmProps): string {\n    let unit: string;\n    let totalPeriod: number;\n    if (alarm.metric.period.toSeconds() < 60) {\n      totalPeriod = alarm.metric.period.toSeconds() * alarm.evaluationPeriods;\n      unit = 's';\n    } else if (alarm.metric.period.toMinutes() < 1440) {\n      totalPeriod = alarm.metric.period.toMinutes() * alarm.evaluationPeriods;\n      unit = 'm';\n    } else {\n      totalPeriod = alarm.metric.period.toDays() * alarm.evaluationPeriods;\n      unit = 'd';\n    }\n\n    // avg(last_15m):max:aws.fargate.cpu.percent{ecs_container_name:$ServiceName$} > 95\n    const namespace = alarm.metric.namespace\n      .toLocaleLowerCase()\n      .replace('/', '.');\n\n    let metricName = alarm.metric.metricName;\n\n    // Special case for ALB / TargetGroup metrics\n    switch (metricName) {\n      case 'HTTPCode_ELB_5XX_Count':\n        metricName = 'HTTPCode_ELB_5XX';\n        break;\n      case 'HTTPCode_Target_5XX_Count':\n        metricName = 'HTTPCode_Target_5XX';\n        break;\n    }\n    if (!metricName.includes('_')) {\n      metricName = metricName.replace(/([a-z])([A-Z])/g, '$1_$2');\n    }\n    metricName = metricName.toLocaleLowerCase();\n\n    const metric = `${namespace}.${metricName}`;\n\n    // statistics must be lowerCase and probably also converted (see allowed values in cw.Metric)\n    let comparisonOperator = 'not supported';\n    switch (alarm.comparisonOperator) {\n      case cdk.aws_cloudwatch.ComparisonOperator\n        .GREATER_THAN_OR_EQUAL_TO_THRESHOLD:\n        comparisonOperator = '>=';\n        break;\n      case cdk.aws_cloudwatch.ComparisonOperator.GREATER_THAN_THRESHOLD:\n        comparisonOperator = '>';\n        break;\n      case cdk.aws_cloudwatch.ComparisonOperator\n        .LESS_THAN_OR_EQUAL_TO_THRESHOLD:\n        comparisonOperator = '<=';\n        break;\n      case cdk.aws_cloudwatch.ComparisonOperator.LESS_THAN_THRESHOLD:\n        comparisonOperator = '<';\n        break;\n    }\n\n    let alarmStatistic = 'not supported';\n    switch (alarm.metric.statistic) {\n      case cdk.aws_cloudwatch.Stats.AVERAGE:\n        alarmStatistic = 'avg';\n        break;\n      case cdk.aws_cloudwatch.Stats.MAXIMUM:\n        alarmStatistic = 'max';\n        break;\n      case cdk.aws_cloudwatch.Stats.MINIMUM:\n        alarmStatistic = 'min';\n        break;\n      case cdk.aws_cloudwatch.Stats.SAMPLE_COUNT:\n        break;\n      case cdk.aws_cloudwatch.Stats.SUM:\n        alarmStatistic = 'sum';\n        break;\n      default:\n        alarmStatistic = 'avg';\n        break;\n    }\n\n    let dimensions = `account_id:${Stack.of(this).account}`;\n\n    if (alarm.metric.dimensions) {\n      for (const key in alarm.metric.dimensions) {\n        const upperToLowerCr = new CustomResource(this, `${key}UpperToLower`, {\n          resourceType: 'Custom::UpperToLower',\n          serviceToken: UpperToLower.getOrCreate(this).provider.serviceToken,\n          properties: {\n            Upper: alarm.metric.dimensions[key],\n          },\n        });\n        const value = upperToLowerCr.getAtt('Lower');\n        dimensions += `,${key.toLocaleLowerCase()}:${value}`;\n      }\n    }\n    const asCountAgggregator = alarm.aggregateQueryAsCount ? '.as_count()' : '';\n\n    switch (alarm.treatMissingData) {\n      case cdk.aws_cloudwatch.TreatMissingData.NOT_BREACHING:\n        return `${alarmStatistic}(last_${totalPeriod}${unit}):default_zero(${alarmStatistic}:${metric}{${dimensions}}${asCountAgggregator}) ${comparisonOperator} ${alarm.threshold}`;\n      case cdk.aws_cloudwatch.TreatMissingData.BREACHING:\n      case cdk.aws_cloudwatch.TreatMissingData.IGNORE:\n      default:\n        return `${alarmStatistic}(last_${totalPeriod}${unit}):${alarmStatistic}:${metric}{${dimensions}}${asCountAgggregator} ${comparisonOperator} ${alarm.threshold}`;\n    }\n  }\n}\n"]}
@@ -25,9 +25,10 @@ class Watchful extends watchful_1.AbstractWatchful {
25
25
  super(scope, id);
26
26
  this.serviceName = props.serviceName ?? aws_cdk_lib_1.Stack.of(this).stackName;
27
27
  this.notification = props.notification ?? new datadogv2_1.DefaultSlackNotification();
28
- this.createCasesFromErrorLogMonitors = props.logErrorMonitorConfig?.createCasesFromErrorLogMonitors
29
- ?? props.createCasesFromErrorLogMonitors
30
- ?? false;
28
+ this.createCasesFromErrorLogMonitors =
29
+ props.logErrorMonitorConfig?.createCasesFromErrorLogMonitors ??
30
+ props.createCasesFromErrorLogMonitors ??
31
+ false;
31
32
  if (props.createCasesFromErrorLogMonitors) {
32
33
  const annotations = aws_cdk_lib_1.Annotations.of(aws_cdk_lib_1.Stack.of(this));
33
34
  annotations.addWarningV2(scope.node.path, '\n ❌ This property has been deprecated. The propery has been instead nested into logErrorMonitorConfig.\n');
@@ -49,9 +50,12 @@ class Watchful extends watchful_1.AbstractWatchful {
49
50
  comparisonOperator: alarm.comparisonOperator,
50
51
  evaluationPeriods: alarm.evaluationPeriods,
51
52
  metric: alarm.metric,
52
- threshold: shouldOverrideThreshold ? threshold : alarm.threshold,
53
+ threshold: shouldOverrideThreshold
54
+ ? threshold
55
+ : alarm.threshold,
53
56
  treatMissingData: alarm.treatMissingData,
54
57
  priority: this.queryErrorMonitorConfig?.priority ?? alarm.priority,
58
+ aggregateQueryAsCount: alarm.aggregateQueryAsCount,
55
59
  });
56
60
  }
57
61
  /**
@@ -61,18 +65,24 @@ class Watchful extends watchful_1.AbstractWatchful {
61
65
  createLogAlarm(scope, id, alarm) {
62
66
  const { shouldOverrideThreshold, threshold } = this.shouldOverrideThreshold({ watchfulMonitorScope: scope, watchfulNodeId: id });
63
67
  const notifications = this.createCasesFromErrorLogMonitors
64
- ? new datadogv2_1.NotificationAggregator([this.notification, new datadogv2_1.DatadogCaseNotification()])
68
+ ? new datadogv2_1.NotificationAggregator([
69
+ this.notification,
70
+ new datadogv2_1.DatadogCaseNotification(),
71
+ ])
65
72
  : this.notification;
66
73
  new datadog_log_alarm_1.DataDogLogAlarm(scope, id, {
67
74
  serviceName: alarm.lambdaCloudwatchGroupName ?? this.serviceName,
75
+ lambdaFunctionName: alarm.lambdaCloudwatchGroupName,
68
76
  notification: notifications,
69
77
  period: alarm.period,
70
- unitOfPeriod: alarm.unitOfPeriod,
78
+ unitOfPeriod: alarm.unitOfPeriod, // toDo: move that somewhere else and set defaults there?
71
79
  periodOperator: alarm.periodOperator,
72
80
  statistic: alarm.statistic,
73
- queryFilters: alarm.queryFilters,
81
+ queryFilters: alarm.queryFilters, // toDo: move that somewhere else and set defaults there?
74
82
  alarmDescription: alarm.alarmDescription,
75
- threshold: shouldOverrideThreshold ? threshold : alarm.threshold,
83
+ threshold: shouldOverrideThreshold
84
+ ? threshold
85
+ : alarm.threshold,
76
86
  priority: this.logErrorMonitorConfig?.priority ?? alarm.priority,
77
87
  index: alarm.index,
78
88
  disableAutoClose: this.logErrorMonitorConfig?.disableAutoClose,
@@ -90,4 +100,4 @@ class Watchful extends watchful_1.AbstractWatchful {
90
100
  exports.Watchful = Watchful;
91
101
  _a = JSII_RTTI_SYMBOL_1;
92
102
  Watchful[_a] = { fqn: "@rio-cloud/cdk-v2-constructs.watchfulv2.Watchful", version: "0.0.0" };
93
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"watchful.js","sourceRoot":"","sources":["../../src/watchfulv2/watchful.ts"],"names":[],"mappings":"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","sourcesContent":["import { Aspects, Stack, Annotations } from 'aws-cdk-lib';\nimport { Construct } from 'constructs';\nimport { DataDogLogAlarm, LogAlarmProps } from './datadog-log-alarm';\nimport { DataDogMetricAlarm } from './datadog-metric-alarm';\nimport { DatadogCaseNotification, DefaultSlackNotification, NotificationAggregator, INotification } from '../datadogv2';\nimport { WatchfulAspect, MetricAlarmProps, AbstractWatchful, MonitorType } from '../watchful';\n\n/**\n * Log error monitor props\n */\nexport interface LogErrorMonitorProps {\n  /**\n   * Configure if monitor needs to be resolved manually every time. This option also sets the renotify interval to 2 hours\n   * @defaultValue false\n   */\n  readonly disableAutoClose?: boolean;\n\n  /**\n   * Set the priority of the log error monitor\n   * @defaultValue 3\n   */\n  readonly priority?: number;\n\n  /**\n   * Enable error log monitors creating cases in Datadog.\n   * This property has been deprecated. The propery has been instead nested into logErrorMonitorConfig.\n   * The functionality remains unchanged\n   *\n   * @defaultValue {@link false}\n   */\n  readonly createCasesFromErrorLogMonitors?: boolean;\n\n  /**\n   * Renotification interval for log error monitor in minutes\n   */\n  readonly renotifyInterval?: number;\n}\n\n/**\n * Query error monitor props\n */\nexport interface QueryErrorMonitorProps {\n  /**\n   * Set the priority of the log error monitor\n   * @defaultValue 3\n   */\n  readonly priority?: number;\n}\n\n/**\n * Properties for defining Watchful\n */\nexport interface WatchfulProps {\n  /**\n   * The service name.\n   *\n   * @defaultValue The stack name.\n   */\n  readonly serviceName?: string;\n\n  /**\n   * Set the way how monitors should notify in case of an alert.\n   *\n   * @defaultValue {@link DefaultSlackNotification}\n   */\n  readonly notification?: INotification;\n\n  /**\n   * Enable error log monitors creating cases in Datadog.\n   *\n   * @deprecated This property has been deprecated. The propery has been instead nested into logErrorMonitorConfig. The functionality remains unchanged.\n   * @defaultValue {@link false}\n   */\n  readonly createCasesFromErrorLogMonitors?: boolean;\n\n  /**\n   * Configure log error monitor properties\n   */\n  readonly logErrorMonitorConfig?: LogErrorMonitorProps;\n\n  /**\n   * Configure query alert monitors. All monitors created by watchful except for Log error monitor.\n   * Configure property *logErrorMonitor* for Log error monitor.\n   */\n  readonly queryErrorMonitorConfig?: QueryErrorMonitorProps;\n}\n\n/**\n * A construct to watch given scope or resources. Opinionated DataDog alarms are automatically created for watched resources.\n *\n * Usage with non-default notification:\n * ```ts\n * const dw = new watchfulv2.Watchful(this, 'Watchful', {\n *   serviceName: props.serviceName,\n *   notification: new datadogv2.SlackNotification({channel: 'team-alert-channel'})\n * });\n * dw.watchScope(this);\n *```\n */\nexport class Watchful extends AbstractWatchful {\n  private readonly notification: INotification;\n  private readonly createCasesFromErrorLogMonitors: boolean;\n  private readonly serviceName: string;\n  private readonly logErrorMonitorConfig?: LogErrorMonitorProps;\n  private readonly queryErrorMonitorConfig?: QueryErrorMonitorProps;\n\n  constructor(scope: Construct, id: string, props: WatchfulProps = {}) {\n    super(scope, id);\n    this.serviceName = props.serviceName ?? Stack.of(this).stackName;\n\n    this.notification = props.notification ?? new DefaultSlackNotification();\n    this.createCasesFromErrorLogMonitors = props.logErrorMonitorConfig?.createCasesFromErrorLogMonitors\n      ?? props.createCasesFromErrorLogMonitors\n      ?? false;\n    if (props.createCasesFromErrorLogMonitors) {\n      const annotations = Annotations.of(Stack.of(this));\n      annotations.addWarningV2(\n        scope.node.path,\n        '\\n ❌ This property has been deprecated. The propery has been instead nested into logErrorMonitorConfig.\\n',\n      );\n    }\n    this.queryErrorMonitorConfig = props.queryErrorMonitorConfig;\n    this.logErrorMonitorConfig = props.logErrorMonitorConfig;\n  }\n\n  /**\n   * Adds the alarmTopic as alarm action to the given alarm.\n   *\n   * @param alarm - The watchful to add those resources to\n   */\n  public createAlarm(scope: Construct, id: MonitorType, alarm: MetricAlarmProps): void {\n    const { shouldOverrideThreshold, threshold } = this.shouldOverrideThreshold({ watchfulMonitorScope: scope, watchfulNodeId: id });\n    new DataDogMetricAlarm(scope, id, {\n      serviceName: this.serviceName,\n      notification: this.notification,\n      alarmDescription: alarm.alarmDescription,\n      comparisonOperator: alarm.comparisonOperator,\n      evaluationPeriods: alarm.evaluationPeriods,\n      metric: alarm.metric,\n      threshold: shouldOverrideThreshold ? threshold as number : alarm.threshold,\n      treatMissingData: alarm.treatMissingData,\n      priority: this.queryErrorMonitorConfig?.priority ?? alarm.priority,\n    });\n\n  }\n\n  /**\n   * Adds an alert for error logs\n   * @param scope\n   */\n  public createLogAlarm(scope: Construct, id: MonitorType, alarm: LogAlarmProps):void {\n    const { shouldOverrideThreshold, threshold } = this.shouldOverrideThreshold({ watchfulMonitorScope: scope, watchfulNodeId: id });\n    const notifications: INotification = this.createCasesFromErrorLogMonitors\n      ? new NotificationAggregator([this.notification, new DatadogCaseNotification()])\n      : this.notification;\n\n    new DataDogLogAlarm(scope, id, {\n      serviceName: alarm.lambdaCloudwatchGroupName ?? this.serviceName,\n      notification: notifications,\n      period: alarm.period,\n      unitOfPeriod: alarm.unitOfPeriod, // toDo: move that somewhere else and set defaults there?\n      periodOperator: alarm.periodOperator,\n      statistic: alarm.statistic,\n      queryFilters: alarm.queryFilters, // toDo: move that somewhere else and set defaults there?\n      alarmDescription: alarm.alarmDescription,\n      threshold: shouldOverrideThreshold ? threshold as number : alarm.threshold,\n      priority: this.logErrorMonitorConfig?.priority ?? alarm.priority,\n      index: alarm.index,\n      disableAutoClose: this.logErrorMonitorConfig?.disableAutoClose,\n      renotifyInterval: this.logErrorMonitorConfig?.renotifyInterval,\n    });\n  }\n\n  /**\n   * Watches the given scope and adds alarms for known resources.\n   */\n  public watchScope(scope: Construct): void {\n    const aspect = new WatchfulAspect(this);\n    Aspects.of(scope).add(aspect);\n  }\n}\n"]}
103
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"watchful.js","sourceRoot":"","sources":["../../src/watchfulv2/watchful.ts"],"names":[],"mappings":"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","sourcesContent":["import { Aspects, Stack, Annotations } from 'aws-cdk-lib';\nimport { Construct } from 'constructs';\nimport { DataDogLogAlarm, LogAlarmProps } from './datadog-log-alarm';\nimport { DataDogMetricAlarm } from './datadog-metric-alarm';\nimport {\n  DatadogCaseNotification,\n  DefaultSlackNotification,\n  NotificationAggregator,\n  INotification,\n} from '../datadogv2';\nimport {\n  WatchfulAspect,\n  MetricAlarmProps,\n  AbstractWatchful,\n  MonitorType,\n} from '../watchful';\n\n/**\n * Log error monitor props\n */\nexport interface LogErrorMonitorProps {\n  /**\n   * Configure if monitor needs to be resolved manually every time. This option also sets the renotify interval to 2 hours\n   * @defaultValue false\n   */\n  readonly disableAutoClose?: boolean;\n\n  /**\n   * Set the priority of the log error monitor\n   * @defaultValue 3\n   */\n  readonly priority?: number;\n\n  /**\n   * Enable error log monitors creating cases in Datadog.\n   * This property has been deprecated. The propery has been instead nested into logErrorMonitorConfig.\n   * The functionality remains unchanged\n   *\n   * @defaultValue {@link false}\n   */\n  readonly createCasesFromErrorLogMonitors?: boolean;\n\n  /**\n   * Renotification interval for log error monitor in minutes\n   */\n  readonly renotifyInterval?: number;\n}\n\n/**\n * Query error monitor props\n */\nexport interface QueryErrorMonitorProps {\n  /**\n   * Set the priority of the log error monitor\n   * @defaultValue 3\n   */\n  readonly priority?: number;\n}\n\n/**\n * Properties for defining Watchful\n */\nexport interface WatchfulProps {\n  /**\n   * The service name.\n   *\n   * @defaultValue The stack name.\n   */\n  readonly serviceName?: string;\n\n  /**\n   * Set the way how monitors should notify in case of an alert.\n   *\n   * @defaultValue {@link DefaultSlackNotification}\n   */\n  readonly notification?: INotification;\n\n  /**\n   * Enable error log monitors creating cases in Datadog.\n   *\n   * @deprecated This property has been deprecated. The propery has been instead nested into logErrorMonitorConfig. The functionality remains unchanged.\n   * @defaultValue {@link false}\n   */\n  readonly createCasesFromErrorLogMonitors?: boolean;\n\n  /**\n   * Configure log error monitor properties\n   */\n  readonly logErrorMonitorConfig?: LogErrorMonitorProps;\n\n  /**\n   * Configure query alert monitors. All monitors created by watchful except for Log error monitor.\n   * Configure property *logErrorMonitor* for Log error monitor.\n   */\n  readonly queryErrorMonitorConfig?: QueryErrorMonitorProps;\n}\n\n/**\n * A construct to watch given scope or resources. Opinionated DataDog alarms are automatically created for watched resources.\n *\n * Usage with non-default notification:\n * ```ts\n * const dw = new watchfulv2.Watchful(this, 'Watchful', {\n *   serviceName: props.serviceName,\n *   notification: new datadogv2.SlackNotification({channel: 'team-alert-channel'})\n * });\n * dw.watchScope(this);\n *```\n */\nexport class Watchful extends AbstractWatchful {\n  private readonly notification: INotification;\n  private readonly createCasesFromErrorLogMonitors: boolean;\n  private readonly serviceName: string;\n  private readonly logErrorMonitorConfig?: LogErrorMonitorProps;\n  private readonly queryErrorMonitorConfig?: QueryErrorMonitorProps;\n\n  constructor(scope: Construct, id: string, props: WatchfulProps = {}) {\n    super(scope, id);\n    this.serviceName = props.serviceName ?? Stack.of(this).stackName;\n\n    this.notification = props.notification ?? new DefaultSlackNotification();\n    this.createCasesFromErrorLogMonitors =\n      props.logErrorMonitorConfig?.createCasesFromErrorLogMonitors ??\n      props.createCasesFromErrorLogMonitors ??\n      false;\n    if (props.createCasesFromErrorLogMonitors) {\n      const annotations = Annotations.of(Stack.of(this));\n      annotations.addWarningV2(\n        scope.node.path,\n        '\\n ❌ This property has been deprecated. The propery has been instead nested into logErrorMonitorConfig.\\n',\n      );\n    }\n    this.queryErrorMonitorConfig = props.queryErrorMonitorConfig;\n    this.logErrorMonitorConfig = props.logErrorMonitorConfig;\n  }\n\n  /**\n   * Adds the alarmTopic as alarm action to the given alarm.\n   *\n   * @param alarm - The watchful to add those resources to\n   */\n  public createAlarm(\n    scope: Construct,\n    id: MonitorType,\n    alarm: MetricAlarmProps,\n  ): void {\n    const { shouldOverrideThreshold, threshold } = this.shouldOverrideThreshold(\n      { watchfulMonitorScope: scope, watchfulNodeId: id },\n    );\n    new DataDogMetricAlarm(scope, id, {\n      serviceName: this.serviceName,\n      notification: this.notification,\n      alarmDescription: alarm.alarmDescription,\n      comparisonOperator: alarm.comparisonOperator,\n      evaluationPeriods: alarm.evaluationPeriods,\n      metric: alarm.metric,\n      threshold: shouldOverrideThreshold\n        ? (threshold as number)\n        : alarm.threshold,\n      treatMissingData: alarm.treatMissingData,\n      priority: this.queryErrorMonitorConfig?.priority ?? alarm.priority,\n      aggregateQueryAsCount: alarm.aggregateQueryAsCount,\n    });\n  }\n\n  /**\n   * Adds an alert for error logs\n   * @param scope\n   */\n  public createLogAlarm(\n    scope: Construct,\n    id: MonitorType,\n    alarm: LogAlarmProps,\n  ): void {\n    const { shouldOverrideThreshold, threshold } = this.shouldOverrideThreshold(\n      { watchfulMonitorScope: scope, watchfulNodeId: id },\n    );\n    const notifications: INotification = this.createCasesFromErrorLogMonitors\n      ? new NotificationAggregator([\n        this.notification,\n        new DatadogCaseNotification(),\n      ])\n      : this.notification;\n\n    new DataDogLogAlarm(scope, id, {\n      serviceName: alarm.lambdaCloudwatchGroupName ?? this.serviceName,\n      lambdaFunctionName: alarm.lambdaCloudwatchGroupName,\n      notification: notifications,\n      period: alarm.period,\n      unitOfPeriod: alarm.unitOfPeriod, // toDo: move that somewhere else and set defaults there?\n      periodOperator: alarm.periodOperator,\n      statistic: alarm.statistic,\n      queryFilters: alarm.queryFilters, // toDo: move that somewhere else and set defaults there?\n      alarmDescription: alarm.alarmDescription,\n      threshold: shouldOverrideThreshold\n        ? (threshold as number)\n        : alarm.threshold,\n      priority: this.logErrorMonitorConfig?.priority ?? alarm.priority,\n      index: alarm.index,\n      disableAutoClose: this.logErrorMonitorConfig?.disableAutoClose,\n      renotifyInterval: this.logErrorMonitorConfig?.renotifyInterval,\n    });\n  }\n\n  /**\n   * Watches the given scope and adds alarms for known resources.\n   */\n  public watchScope(scope: Construct): void {\n    const aspect = new WatchfulAspect(this);\n    Aspects.of(scope).add(aspect);\n  }\n}\n"]}
package/package.json CHANGED
@@ -15,7 +15,7 @@
15
15
  ],
16
16
  "main": "lib/index.js",
17
17
  "license": "Apache-2.0",
18
- "version": "5.1.0",
18
+ "version": "6.0.1",
19
19
  "types": "lib/index.d.ts",
20
20
  "stability": "stable",
21
21
  "exports": {
@@ -81,9 +81,9 @@
81
81
  "eslint-plugin-tsdoc": "^0.2.17",
82
82
  "jest": "~29.7.0",
83
83
  "jest-junit": "16.0.0",
84
- "jsii": "~5.2.32",
85
- "jsii-diff": "~1.92.0",
86
- "jsii-docgen": "^10.3.5",
84
+ "jsii": "~5.4.26",
85
+ "jsii-diff": "~1.101.0",
86
+ "jsii-docgen": "^10.4.17",
87
87
  "tmp": "^0.2.3",
88
88
  "ts-jest": "^29.1.1",
89
89
  "ts-node": "^10.9.1",
package/version.json CHANGED
@@ -1,3 +1,3 @@
1
1
  {
2
- "version": "5.1.0"
2
+ "version": "6.0.1"
3
3
  }