@rio-cloud/cdk-v2-constructs 4.13.2 → 4.14.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.jsii +7470 -3337
- package/CHANGELOG.md +14 -0
- package/lib/datadog/datadog-log-index-monitoring.d.ts +2 -2
- package/lib/datadog/datadog-log-index-monitoring.js +1 -1
- package/lib/datadogv2/datadog-dashboard.d.ts +28 -0
- package/lib/datadogv2/datadog-dashboard.js +31 -0
- package/lib/datadogv2/datadog-log-index-monitoring.d.ts +61 -0
- package/lib/datadogv2/datadog-log-index-monitoring.js +115 -0
- package/lib/datadogv2/datadog-monitor.d.ts +234 -0
- package/lib/datadogv2/datadog-monitor.js +134 -0
- package/lib/datadogv2/datadog-notification.d.ts +100 -0
- package/lib/datadogv2/datadog-notification.js +94 -0
- package/lib/datadogv2/index.d.ts +4 -0
- package/lib/datadogv2/index.js +17 -0
- package/lib/fargate/datadog.js +2 -2
- package/lib/fargate/index.d.ts +1 -0
- package/lib/fargate/index.js +14 -0
- package/lib/fargate/rio-fargate-service.d.ts +18 -18
- package/lib/fargate/rio-fargate-service.js +1 -1
- package/lib/index.d.ts +7 -13
- package/lib/index.js +8 -14
- package/lib/kafka/kafka-topic.d.ts +5 -5
- package/lib/kafka/kafka-topic.js +1 -1
- package/lib/kafka/rio-kafka-event-source.d.ts +1 -1
- package/lib/kafka/rio-kafka-event-source.js +1 -1
- package/lib/pipeline/rio-backup-secrets-restore-stage.d.ts +1 -1
- package/lib/pipeline/rio-backup-secrets-restore-stage.js +1 -1
- package/lib/pipeline/rio-bitbucket-source-action.d.ts +1 -1
- package/lib/pipeline/rio-bitbucket-source-action.js +1 -1
- package/lib/rio-landing-zone.d.ts +6 -0
- package/lib/rio-landing-zone.js +19 -1
- package/lib/watchful/alb.d.ts +2 -2
- package/lib/watchful/alb.js +1 -1
- package/lib/watchful/aspect.d.ts +6 -6
- package/lib/watchful/aspect.js +12 -7
- package/lib/watchful/cloudfront.d.ts +2 -2
- package/lib/watchful/cloudfront.js +1 -1
- package/lib/watchful/docdb.d.ts +4 -4
- package/lib/watchful/docdb.js +1 -1
- package/lib/watchful/dynamodb.d.ts +5 -5
- package/lib/watchful/dynamodb.js +1 -1
- package/lib/watchful/ecs.d.ts +2 -2
- package/lib/watchful/ecs.js +1 -1
- package/lib/watchful/index.d.ts +13 -0
- package/lib/watchful/index.js +26 -0
- package/lib/watchful/lambda.d.ts +3 -3
- package/lib/watchful/lambda.js +1 -1
- package/lib/watchful/metric-alarm.d.ts +16 -16
- package/lib/watchful/metric-alarm.js +1 -1
- package/lib/watchful/rds.d.ts +2 -2
- package/lib/watchful/rds.js +1 -1
- package/lib/watchful/targetgroup.d.ts +2 -2
- package/lib/watchful/targetgroup.js +1 -1
- package/lib/watchful/watchful.d.ts +52 -48
- package/lib/watchful/watchful.js +103 -98
- package/lib/watchfulv2/datadog-log-alarm.d.ts +37 -0
- package/lib/watchfulv2/datadog-log-alarm.js +61 -0
- package/lib/watchfulv2/datadog-metric-alarm.d.ts +29 -0
- package/lib/watchfulv2/datadog-metric-alarm.js +132 -0
- package/lib/watchfulv2/index.d.ts +1 -0
- package/lib/watchfulv2/index.js +14 -0
- package/lib/watchfulv2/watchful.d.ts +53 -0
- package/lib/watchfulv2/watchful.js +76 -0
- package/package.json +18 -8
- package/version.json +1 -1
package/CHANGELOG.md
CHANGED
|
@@ -2,6 +2,20 @@
|
|
|
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
|
+
## [4.14.0](https://collaboration.msi.audi.com/stash/projects/RIODEV/repos/cdk-v2-constructs/compare/commits?targetBranch=refs%2Ftags%2Fv4.13.1&sourceBranch=refs%2Ftags%2Fv4.14.0) (2023-07-20)
|
|
6
|
+
|
|
7
|
+
|
|
8
|
+
### Features
|
|
9
|
+
|
|
10
|
+
* added DatadogDashboard construct ([e87ca30](https://collaboration.msi.audi.com/stash/projects/RIODEV/repos/cdk-v2-constructs/commits/e87ca3030ae8d9558832cd114aacc2460acefee6))
|
|
11
|
+
* added RioPagingNotification ([d8e023d](https://collaboration.msi.audi.com/stash/projects/RIODEV/repos/cdk-v2-constructs/commits/d8e023d1502697c4e0e055a517368630fd4aed66))
|
|
12
|
+
* **watchfulv2,datadogv2:** ready to release ([c69ca6b](https://collaboration.msi.audi.com/stash/projects/RIODEV/repos/cdk-v2-constructs/commits/c69ca6b7fea331eb18da1e6ea0db4c55da394e95))
|
|
13
|
+
|
|
14
|
+
|
|
15
|
+
### Bug Fixes
|
|
16
|
+
|
|
17
|
+
* :bug: Fixed issue related to wrong service name set in case of lambda functions ([2bfd814](https://collaboration.msi.audi.com/stash/projects/RIODEV/repos/cdk-v2-constructs/commits/2bfd814e0108e2a2b6ef1680a0841368bf26a4ba))
|
|
18
|
+
|
|
5
19
|
### [4.13.2](https://collaboration.msi.audi.com/stash/projects/RIODEV/repos/cdk-v2-constructs/compare/commits?targetBranch=refs%2Ftags%2Fv4.13.1&sourceBranch=refs%2Ftags%2Fv4.13.2) (2023-07-06)
|
|
6
20
|
|
|
7
21
|
|
|
@@ -21,12 +21,12 @@ export interface DatadogLogIndexMonitoringProps {
|
|
|
21
21
|
/**
|
|
22
22
|
* This value should be used if there is no log message within 24 hours or missing historical data.
|
|
23
23
|
* Without setting this value it will lead to an empty result and therefore trigger an alert.
|
|
24
|
-
* @
|
|
24
|
+
* @defaultValue - false
|
|
25
25
|
*/
|
|
26
26
|
readonly sparseLogging?: boolean;
|
|
27
27
|
/**
|
|
28
28
|
* The Datadog organization, e.g., 'EU' or 'LATAM'
|
|
29
|
-
* @
|
|
29
|
+
* @defaultValue - 'EU'
|
|
30
30
|
*/
|
|
31
31
|
readonly organization?: DatadogOrganization;
|
|
32
32
|
}
|
|
@@ -137,4 +137,4 @@ function datadogLogsSearchWithIndexFilterDeepLink(indexName, organization) {
|
|
|
137
137
|
function datadogLogsIndexesConfigurationDeepLink(organization) {
|
|
138
138
|
return `https://${getOrganizationUrlPrefix(organization)}.datadoghq.eu/logs/pipelines/indexes`;
|
|
139
139
|
}
|
|
140
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"datadog-log-index-monitoring.js","sourceRoot":"","sources":["../../src/datadog/datadog-log-index-monitoring.ts"],"names":[],"mappings":"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","sourcesContent":["import { Construct } from 'constructs';\nimport { DatadogMonitor, DatadogAlertType } from './datadog-monitor';\n\nexport interface DatadogLogIndexMonitoringProps {\n\n  /**\n   * The name of your service in Datadog. Can be used to query the monitors.\n   */\n  readonly serviceName: string;\n\n  /**\n   * The name of the Datadog index.\n   */\n  readonly indexName: string;\n\n  /**\n   * The integration to use for alerting.\n   * For OpsGenie, you need to install and configure the 'opsgenie-integration' account module.\n   */\n  readonly alertType: DatadogAlertType;\n\n  /**\n   * The daily log quota settings.\n   */\n  readonly dailyLogQuota: DatadogLogQuotaProps;\n\n  /**\n   * This value should be used if there is no log message within 24 hours or missing historical data.\n   * Without setting this value it will lead to an empty result and therefore trigger an alert.\n   * @default - false\n   */\n  readonly sparseLogging?: boolean;\n\n  /**\n   * The Datadog organization, e.g., 'EU' or 'LATAM'\n   * @default - 'EU'\n   */\n  readonly organization?: DatadogOrganization;\n}\n\nexport interface DatadogLogQuotaProps {\n  /**\n   * The daily log quota for the team-specific index in million events.\n   */\n  readonly valueInMillionEvents: number;\n\n  /**\n   * The warning threshold for the daily log quota monitor in percent.\n   */\n  readonly warningThresholdInPercent: number;\n\n  /**\n   * The alarm threshold for the daily log quota monitor in percent.\n   * The value must be between 0 and 100 and greater than or equal to the warning threshold.\n   */\n  readonly alertThresholdInPercent: number;\n}\n\n/**\n * Basic monitoring and alerting for a Datadog logs index.\n * It follows the Datadog guide at https://docs.datadoghq.com/logs/guide/logs-monitors-on-volumes/\n * and consists of the following three monitors.\n *   1. A metric alert that fires when you reach a certain threshold of your daily log quota.\n *   2. An anomaly monitor that detects log amount spikes.\n *   3. An event alert that fires when you hit the daily log quota.\n *\n * A datadog log index is team specific and thus the monitor needs to be deployed just once per index/team.\n */\nexport class DatadogLogIndexMonitoring extends Construct {\n  constructor(scope: Construct, id: string, props: DatadogLogIndexMonitoringProps) {\n    super(scope, id);\n\n    new DatadogLogDailyQuotaMonitor(this, 'DatadogLogDailyQuotaMonitor', props);\n    new DatadogLogsAnomalyMonitor(this, 'DatadogLogAlertAnomalyMonitor', props);\n    new DatadogLogDailyQuotaReachedMonitor(this, 'DatadogLogDailyQuotaReachedV2Monitor', props);\n\n    this.node.addValidation({\n      validate: (): string[] => {\n        const result = [];\n        if (props.dailyLogQuota.warningThresholdInPercent < 0 || props.dailyLogQuota.warningThresholdInPercent > 100) {\n          result.push('Invalid [warningThresholdInPercent]: expecting a number between 0 and 100');\n        }\n        if (props.dailyLogQuota.alertThresholdInPercent < 0 || props.dailyLogQuota.alertThresholdInPercent > 100) {\n          result.push('Invalid [alertThresholdInPercent]: expecting a number between 0 and 100');\n        }\n        if (props.dailyLogQuota.alertThresholdInPercent < props.dailyLogQuota.warningThresholdInPercent) {\n          result.push('Invalid [alertThresholdInPercent]: must be greater than or equal to [warningThresholdInPercent]');\n        }\n        return result;\n      },\n    });\n  }\n}\n\nconst commonMonitorNamePrefix = (indexName: string) => `'${indexName}' index`;\nconst commonTags = (indexName: string) => [`logIndex:${indexName}`];\n\nclass DatadogLogDailyQuotaMonitor extends DatadogMonitor {\n  constructor(scope: Construct, id: string, props: DatadogLogIndexMonitoringProps) {\n    const logQuotaInEvents = Math.floor(props.dailyLogQuota.valueInMillionEvents * 1_000_000);\n    const warningThreshold = Math.floor(logQuotaInEvents * (props.dailyLogQuota.warningThresholdInPercent / 100));\n    const alertThreshold = Math.floor(logQuotaInEvents * (props.dailyLogQuota.alertThresholdInPercent / 100));\n\n    super(scope, id, {\n      serviceName: props.serviceName,\n      alertTypes: [props.alertType],\n      monitor: {\n        name: `${commonMonitorNamePrefix(props.indexName)}: Log quota almost reached`,\n        type: 'metric alert',\n        query: `sum(last_1d):sum:datadog.estimated_usage.logs.ingested_events{datadog_is_excluded:false,datadog_index:${props.indexName}}.as_count() >= ${alertThreshold}`,\n        message: `[P3] Inspect your log volume at ${datadogLogsSearchWithIndexFilterDeepLink(\n          props.indexName, props.organization,\n        )} or adapt the log quota at ${datadogLogsIndexesConfigurationDeepLink(props.organization)}`,\n        options: {\n          thresholds: {\n            critical: alertThreshold,\n            warning: warningThreshold,\n          },\n          include_tags: false,\n          notify_no_data: !props.sparseLogging,\n          no_data_timeframe: 24 * 60, // 1 day\n        },\n        tags: [\n          ...commonTags(props.indexName),\n        ],\n      },\n    });\n  }\n}\n\nclass DatadogLogsAnomalyMonitor extends DatadogMonitor {\n  constructor(scope: Construct, id: string, props: DatadogLogIndexMonitoringProps) {\n    super(scope, id, {\n      serviceName: props.serviceName,\n      alertTypes: [props.alertType],\n      monitor: {\n        name: `${commonMonitorNamePrefix(props.indexName)}: Log amount is rising to unexpected levels`,\n        type: 'metric alert',\n        query: `avg(last_4h):anomalies(sum:datadog.estimated_usage.logs.ingested_events{datadog_is_excluded:false,datadog_index:${props.indexName}}.as_count(), 'robust', 2, direction='above', alert_window='last_15m', interval=60, count_default_zero='true', seasonality='weekly', timezone='europe/berlin') >= 1`,\n        message: `[P4] Logs are getting out of control: ${datadogLogsSearchWithIndexFilterDeepLink(\n          props.indexName, props.organization,\n        )}`,\n        options: {\n          include_tags: false,\n          notify_no_data: !props.sparseLogging,\n          threshold_windows: {\n            trigger_window: 'last_15m',\n            recovery_window: 'last_15m',\n          },\n        },\n        tags: [\n          ...commonTags(props.indexName),\n        ],\n      },\n    });\n  }\n}\n\nclass DatadogLogDailyQuotaReachedMonitor extends DatadogMonitor {\n  constructor(scope: Construct, id: string, props: DatadogLogIndexMonitoringProps) {\n    super(scope, id, {\n      serviceName: props.serviceName,\n      alertTypes: [props.alertType],\n      monitor: {\n        name: `${commonMonitorNamePrefix(props.indexName)}: Log quota reached`,\n        type: 'event-v2 alert',\n        query: `events(\"source:datadog datadog_index:${props.indexName} Daily quota reached\").rollup(\"count\").last(\"12h\") > 0`,\n        message: `[P2] The log quote for the index has been reached. If you do not increase the quota at ${datadogLogsIndexesConfigurationDeepLink(props.organization)}, no new logs will be indexed until the quota is reset at 2:00pm UTC.`,\n        options: {\n          timeout_h: 24,\n          thresholds: {\n            critical: 0,\n          },\n        },\n        tags: [\n          ...commonTags(props.indexName),\n        ],\n      },\n    });\n  }\n}\n\nexport type DatadogOrganization = 'EU' | 'LATAM' | 'Landing Zone';\n\nexport function getOrganizationUrlPrefix(organization: DatadogOrganization | undefined): string {\n  switch (organization ?? 'EU') {\n    case 'EU':\n      return 'app';\n    case 'LATAM':\n      return 'rio-brazil';\n    case 'Landing Zone':\n      return 'rio-landingzone';\n  }\n}\n\nfunction datadogLogsSearchWithIndexFilterDeepLink(indexName: string, organization: DatadogOrganization | undefined) {\n  return `https://${getOrganizationUrlPrefix(organization)}.datadoghq.eu/logs?index=${indexName}`;\n}\n\nfunction datadogLogsIndexesConfigurationDeepLink(organization: DatadogOrganization | undefined) {\n  return `https://${getOrganizationUrlPrefix(organization)}.datadoghq.eu/logs/pipelines/indexes`;\n}\n"]}
|
|
140
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"datadog-log-index-monitoring.js","sourceRoot":"","sources":["../../src/datadog/datadog-log-index-monitoring.ts"],"names":[],"mappings":"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","sourcesContent":["import { Construct } from 'constructs';\nimport { DatadogMonitor, DatadogAlertType } from './datadog-monitor';\n\nexport interface DatadogLogIndexMonitoringProps {\n\n  /**\n   * The name of your service in Datadog. Can be used to query the monitors.\n   */\n  readonly serviceName: string;\n\n  /**\n   * The name of the Datadog index.\n   */\n  readonly indexName: string;\n\n  /**\n   * The integration to use for alerting.\n   * For OpsGenie, you need to install and configure the 'opsgenie-integration' account module.\n   */\n  readonly alertType: DatadogAlertType;\n\n  /**\n   * The daily log quota settings.\n   */\n  readonly dailyLogQuota: DatadogLogQuotaProps;\n\n  /**\n   * This value should be used if there is no log message within 24 hours or missing historical data.\n   * Without setting this value it will lead to an empty result and therefore trigger an alert.\n   * @defaultValue - false\n   */\n  readonly sparseLogging?: boolean;\n\n  /**\n   * The Datadog organization, e.g., 'EU' or 'LATAM'\n   * @defaultValue - 'EU'\n   */\n  readonly organization?: DatadogOrganization;\n}\n\nexport interface DatadogLogQuotaProps {\n  /**\n   * The daily log quota for the team-specific index in million events.\n   */\n  readonly valueInMillionEvents: number;\n\n  /**\n   * The warning threshold for the daily log quota monitor in percent.\n   */\n  readonly warningThresholdInPercent: number;\n\n  /**\n   * The alarm threshold for the daily log quota monitor in percent.\n   * The value must be between 0 and 100 and greater than or equal to the warning threshold.\n   */\n  readonly alertThresholdInPercent: number;\n}\n\n/**\n * Basic monitoring and alerting for a Datadog logs index.\n * It follows the Datadog guide at https://docs.datadoghq.com/logs/guide/logs-monitors-on-volumes/\n * and consists of the following three monitors.\n *   1. A metric alert that fires when you reach a certain threshold of your daily log quota.\n *   2. An anomaly monitor that detects log amount spikes.\n *   3. An event alert that fires when you hit the daily log quota.\n *\n * A datadog log index is team specific and thus the monitor needs to be deployed just once per index/team.\n */\nexport class DatadogLogIndexMonitoring extends Construct {\n  constructor(scope: Construct, id: string, props: DatadogLogIndexMonitoringProps) {\n    super(scope, id);\n\n    new DatadogLogDailyQuotaMonitor(this, 'DatadogLogDailyQuotaMonitor', props);\n    new DatadogLogsAnomalyMonitor(this, 'DatadogLogAlertAnomalyMonitor', props);\n    new DatadogLogDailyQuotaReachedMonitor(this, 'DatadogLogDailyQuotaReachedV2Monitor', props);\n\n    this.node.addValidation({\n      validate: (): string[] => {\n        const result = [];\n        if (props.dailyLogQuota.warningThresholdInPercent < 0 || props.dailyLogQuota.warningThresholdInPercent > 100) {\n          result.push('Invalid [warningThresholdInPercent]: expecting a number between 0 and 100');\n        }\n        if (props.dailyLogQuota.alertThresholdInPercent < 0 || props.dailyLogQuota.alertThresholdInPercent > 100) {\n          result.push('Invalid [alertThresholdInPercent]: expecting a number between 0 and 100');\n        }\n        if (props.dailyLogQuota.alertThresholdInPercent < props.dailyLogQuota.warningThresholdInPercent) {\n          result.push('Invalid [alertThresholdInPercent]: must be greater than or equal to [warningThresholdInPercent]');\n        }\n        return result;\n      },\n    });\n  }\n}\n\nconst commonMonitorNamePrefix = (indexName: string) => `'${indexName}' index`;\nconst commonTags = (indexName: string) => [`logIndex:${indexName}`];\n\nclass DatadogLogDailyQuotaMonitor extends DatadogMonitor {\n  constructor(scope: Construct, id: string, props: DatadogLogIndexMonitoringProps) {\n    const logQuotaInEvents = Math.floor(props.dailyLogQuota.valueInMillionEvents * 1_000_000);\n    const warningThreshold = Math.floor(logQuotaInEvents * (props.dailyLogQuota.warningThresholdInPercent / 100));\n    const alertThreshold = Math.floor(logQuotaInEvents * (props.dailyLogQuota.alertThresholdInPercent / 100));\n\n    super(scope, id, {\n      serviceName: props.serviceName,\n      alertTypes: [props.alertType],\n      monitor: {\n        name: `${commonMonitorNamePrefix(props.indexName)}: Log quota almost reached`,\n        type: 'metric alert',\n        query: `sum(last_1d):sum:datadog.estimated_usage.logs.ingested_events{datadog_is_excluded:false,datadog_index:${props.indexName}}.as_count() >= ${alertThreshold}`,\n        message: `[P3] Inspect your log volume at ${datadogLogsSearchWithIndexFilterDeepLink(\n          props.indexName, props.organization,\n        )} or adapt the log quota at ${datadogLogsIndexesConfigurationDeepLink(props.organization)}`,\n        options: {\n          thresholds: {\n            critical: alertThreshold,\n            warning: warningThreshold,\n          },\n          include_tags: false,\n          notify_no_data: !props.sparseLogging,\n          no_data_timeframe: 24 * 60, // 1 day\n        },\n        tags: [\n          ...commonTags(props.indexName),\n        ],\n      },\n    });\n  }\n}\n\nclass DatadogLogsAnomalyMonitor extends DatadogMonitor {\n  constructor(scope: Construct, id: string, props: DatadogLogIndexMonitoringProps) {\n    super(scope, id, {\n      serviceName: props.serviceName,\n      alertTypes: [props.alertType],\n      monitor: {\n        name: `${commonMonitorNamePrefix(props.indexName)}: Log amount is rising to unexpected levels`,\n        type: 'metric alert',\n        query: `avg(last_4h):anomalies(sum:datadog.estimated_usage.logs.ingested_events{datadog_is_excluded:false,datadog_index:${props.indexName}}.as_count(), 'robust', 2, direction='above', alert_window='last_15m', interval=60, count_default_zero='true', seasonality='weekly', timezone='europe/berlin') >= 1`,\n        message: `[P4] Logs are getting out of control: ${datadogLogsSearchWithIndexFilterDeepLink(\n          props.indexName, props.organization,\n        )}`,\n        options: {\n          include_tags: false,\n          notify_no_data: !props.sparseLogging,\n          threshold_windows: {\n            trigger_window: 'last_15m',\n            recovery_window: 'last_15m',\n          },\n        },\n        tags: [\n          ...commonTags(props.indexName),\n        ],\n      },\n    });\n  }\n}\n\nclass DatadogLogDailyQuotaReachedMonitor extends DatadogMonitor {\n  constructor(scope: Construct, id: string, props: DatadogLogIndexMonitoringProps) {\n    super(scope, id, {\n      serviceName: props.serviceName,\n      alertTypes: [props.alertType],\n      monitor: {\n        name: `${commonMonitorNamePrefix(props.indexName)}: Log quota reached`,\n        type: 'event-v2 alert',\n        query: `events(\"source:datadog datadog_index:${props.indexName} Daily quota reached\").rollup(\"count\").last(\"12h\") > 0`,\n        message: `[P2] The log quote for the index has been reached. If you do not increase the quota at ${datadogLogsIndexesConfigurationDeepLink(props.organization)}, no new logs will be indexed until the quota is reset at 2:00pm UTC.`,\n        options: {\n          timeout_h: 24,\n          thresholds: {\n            critical: 0,\n          },\n        },\n        tags: [\n          ...commonTags(props.indexName),\n        ],\n      },\n    });\n  }\n}\n\nexport type DatadogOrganization = 'EU' | 'LATAM' | 'Landing Zone';\n\nexport function getOrganizationUrlPrefix(organization: DatadogOrganization | undefined): string {\n  switch (organization ?? 'EU') {\n    case 'EU':\n      return 'app';\n    case 'LATAM':\n      return 'rio-brazil';\n    case 'Landing Zone':\n      return 'rio-landingzone';\n  }\n}\n\nfunction datadogLogsSearchWithIndexFilterDeepLink(indexName: string, organization: DatadogOrganization | undefined) {\n  return `https://${getOrganizationUrlPrefix(organization)}.datadoghq.eu/logs?index=${indexName}`;\n}\n\nfunction datadogLogsIndexesConfigurationDeepLink(organization: DatadogOrganization | undefined) {\n  return `https://${getOrganizationUrlPrefix(organization)}.datadoghq.eu/logs/pipelines/indexes`;\n}\n"]}
|
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
import { Construct } from 'constructs';
|
|
2
|
+
export interface DatadogDashboardProps {
|
|
3
|
+
/**
|
|
4
|
+
* The Dashboard definition as exported from Datadog:
|
|
5
|
+
* ```ts
|
|
6
|
+
* import dashboardDefinition from './dashboard.json';
|
|
7
|
+
*
|
|
8
|
+
* new DatadogDashboard(this, 'SampleDashboard', { dashboardDefinition })
|
|
9
|
+
* ```
|
|
10
|
+
*
|
|
11
|
+
* References to {@link [DatadogMonitor](./datadog-monitor.ts)} will not be resolved at the moment.
|
|
12
|
+
*
|
|
13
|
+
* @alpha
|
|
14
|
+
*/
|
|
15
|
+
readonly dashboardDefinition: object;
|
|
16
|
+
}
|
|
17
|
+
/**
|
|
18
|
+
* Wrapper around Datadog's CloudFormation `Datadog::Monitors::Dashboard` with some RIO defaults applied.
|
|
19
|
+
*
|
|
20
|
+
* To use this the Datadog resources have to be {@link https://docs.datadoghq.com/integrations/guide/amazon_cloudformation/
|
|
21
|
+
* |enabled in the CloudFormation registry} of the corresponding AWS account. Usually this is done
|
|
22
|
+
* automatically by the Datadog Integrations (account module AND ServiceCatalog) provided by CLAID.
|
|
23
|
+
*
|
|
24
|
+
* @alpha
|
|
25
|
+
*/
|
|
26
|
+
export declare class DatadogDashboard extends Construct {
|
|
27
|
+
constructor(scope: Construct, id: string, props: DatadogDashboardProps);
|
|
28
|
+
}
|
|
@@ -0,0 +1,31 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var _a;
|
|
3
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
4
|
+
exports.DatadogDashboard = void 0;
|
|
5
|
+
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
|
6
|
+
const cdk = require("aws-cdk-lib");
|
|
7
|
+
const constructs_1 = require("constructs");
|
|
8
|
+
/**
|
|
9
|
+
* Wrapper around Datadog's CloudFormation `Datadog::Monitors::Dashboard` with some RIO defaults applied.
|
|
10
|
+
*
|
|
11
|
+
* To use this the Datadog resources have to be {@link https://docs.datadoghq.com/integrations/guide/amazon_cloudformation/
|
|
12
|
+
* |enabled in the CloudFormation registry} of the corresponding AWS account. Usually this is done
|
|
13
|
+
* automatically by the Datadog Integrations (account module AND ServiceCatalog) provided by CLAID.
|
|
14
|
+
*
|
|
15
|
+
* @alpha
|
|
16
|
+
*/
|
|
17
|
+
class DatadogDashboard extends constructs_1.Construct {
|
|
18
|
+
constructor(scope, id, props) {
|
|
19
|
+
super(scope, id);
|
|
20
|
+
new cdk.CfnResource(scope, id + 'Dashboard', {
|
|
21
|
+
type: 'Datadog::Dashboards::Dashboard',
|
|
22
|
+
properties: {
|
|
23
|
+
DashboardDefinition: JSON.stringify(props.dashboardDefinition),
|
|
24
|
+
},
|
|
25
|
+
});
|
|
26
|
+
}
|
|
27
|
+
}
|
|
28
|
+
exports.DatadogDashboard = DatadogDashboard;
|
|
29
|
+
_a = JSII_RTTI_SYMBOL_1;
|
|
30
|
+
DatadogDashboard[_a] = { fqn: "@rio-cloud/cdk-v2-constructs.datadogv2.DatadogDashboard", version: "0.0.0" };
|
|
31
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGF0YWRvZy1kYXNoYm9hcmQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvZGF0YWRvZ3YyL2RhdGFkb2ctZGFzaGJvYXJkLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7O0FBQUEsbUNBQW1DO0FBQ25DLDJDQUF1QztBQWtCdkM7Ozs7Ozs7O0dBUUc7QUFDSCxNQUFhLGdCQUFpQixTQUFRLHNCQUFTO0lBRTdDLFlBQVksS0FBZ0IsRUFBRSxFQUFVLEVBQUUsS0FBNEI7UUFDcEUsS0FBSyxDQUFDLEtBQUssRUFBRSxFQUFFLENBQUMsQ0FBQztRQUVqQixJQUFJLEdBQUcsQ0FBQyxXQUFXLENBQUMsS0FBSyxFQUFFLEVBQUUsR0FBRyxXQUFXLEVBQUU7WUFDM0MsSUFBSSxFQUFFLGdDQUFnQztZQUN0QyxVQUFVLEVBQUU7Z0JBQ1YsbUJBQW1CLEVBQUUsSUFBSSxDQUFDLFNBQVMsQ0FBQyxLQUFLLENBQUMsbUJBQW1CLENBQUM7YUFDL0Q7U0FDRixDQUFDLENBQUM7SUFDTCxDQUFDOztBQVhILDRDQVlDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0ICogYXMgY2RrIGZyb20gJ2F3cy1jZGstbGliJztcbmltcG9ydCB7IENvbnN0cnVjdCB9IGZyb20gJ2NvbnN0cnVjdHMnO1xuXG5leHBvcnQgaW50ZXJmYWNlIERhdGFkb2dEYXNoYm9hcmRQcm9wcyB7XG4gIC8qKlxuICAgKiBUaGUgRGFzaGJvYXJkIGRlZmluaXRpb24gYXMgZXhwb3J0ZWQgZnJvbSBEYXRhZG9nOlxuICAgKiBgYGB0c1xuICAgKiBpbXBvcnQgZGFzaGJvYXJkRGVmaW5pdGlvbiBmcm9tICcuL2Rhc2hib2FyZC5qc29uJztcbiAgICpcbiAgICogbmV3IERhdGFkb2dEYXNoYm9hcmQodGhpcywgJ1NhbXBsZURhc2hib2FyZCcsIHsgZGFzaGJvYXJkRGVmaW5pdGlvbiB9KVxuICAgKiBgYGBcbiAgICpcbiAgICogUmVmZXJlbmNlcyB0byB7QGxpbmsgW0RhdGFkb2dNb25pdG9yXSguL2RhdGFkb2ctbW9uaXRvci50cyl9IHdpbGwgbm90IGJlIHJlc29sdmVkIGF0IHRoZSBtb21lbnQuXG4gICAqXG4gICAqIEBhbHBoYVxuICAgKi9cbiAgcmVhZG9ubHkgZGFzaGJvYXJkRGVmaW5pdGlvbjogb2JqZWN0O1xufVxuXG4vKipcbiAqIFdyYXBwZXIgYXJvdW5kIERhdGFkb2cncyBDbG91ZEZvcm1hdGlvbiBgRGF0YWRvZzo6TW9uaXRvcnM6OkRhc2hib2FyZGAgd2l0aCBzb21lIFJJTyBkZWZhdWx0cyBhcHBsaWVkLlxuICpcbiAqIFRvIHVzZSB0aGlzIHRoZSBEYXRhZG9nIHJlc291cmNlcyBoYXZlIHRvIGJlIHtAbGluayBodHRwczovL2RvY3MuZGF0YWRvZ2hxLmNvbS9pbnRlZ3JhdGlvbnMvZ3VpZGUvYW1hem9uX2Nsb3VkZm9ybWF0aW9uL1xuICogfGVuYWJsZWQgaW4gdGhlIENsb3VkRm9ybWF0aW9uIHJlZ2lzdHJ5fSBvZiB0aGUgY29ycmVzcG9uZGluZyBBV1MgYWNjb3VudC4gVXN1YWxseSB0aGlzIGlzIGRvbmVcbiAqIGF1dG9tYXRpY2FsbHkgYnkgdGhlIERhdGFkb2cgSW50ZWdyYXRpb25zIChhY2NvdW50IG1vZHVsZSBBTkQgU2VydmljZUNhdGFsb2cpIHByb3ZpZGVkIGJ5IENMQUlELlxuICpcbiAqIEBhbHBoYVxuICovXG5leHBvcnQgY2xhc3MgRGF0YWRvZ0Rhc2hib2FyZCBleHRlbmRzIENvbnN0cnVjdCB7XG5cbiAgY29uc3RydWN0b3Ioc2NvcGU6IENvbnN0cnVjdCwgaWQ6IHN0cmluZywgcHJvcHM6IERhdGFkb2dEYXNoYm9hcmRQcm9wcykge1xuICAgIHN1cGVyKHNjb3BlLCBpZCk7XG5cbiAgICBuZXcgY2RrLkNmblJlc291cmNlKHNjb3BlLCBpZCArICdEYXNoYm9hcmQnLCB7XG4gICAgICB0eXBlOiAnRGF0YWRvZzo6RGFzaGJvYXJkczo6RGFzaGJvYXJkJyxcbiAgICAgIHByb3BlcnRpZXM6IHtcbiAgICAgICAgRGFzaGJvYXJkRGVmaW5pdGlvbjogSlNPTi5zdHJpbmdpZnkocHJvcHMuZGFzaGJvYXJkRGVmaW5pdGlvbiksXG4gICAgICB9LFxuICAgIH0pO1xuICB9XG59XG4iXX0=
|
|
@@ -0,0 +1,61 @@
|
|
|
1
|
+
import { Construct } from 'constructs';
|
|
2
|
+
import { INotification } from './datadog-notification';
|
|
3
|
+
export interface DatadogLogIndexMonitoringProps {
|
|
4
|
+
/**
|
|
5
|
+
* The name of your service in Datadog. Can be used to query the monitors.
|
|
6
|
+
*/
|
|
7
|
+
readonly serviceName: string;
|
|
8
|
+
/**
|
|
9
|
+
* The name of the Datadog index.
|
|
10
|
+
*/
|
|
11
|
+
readonly indexName: string;
|
|
12
|
+
/**
|
|
13
|
+
* The integration to use for alerting.
|
|
14
|
+
*/
|
|
15
|
+
readonly notification?: INotification;
|
|
16
|
+
/**
|
|
17
|
+
* The daily log quota settings.
|
|
18
|
+
*/
|
|
19
|
+
readonly dailyLogQuota: DatadogLogQuotaProps;
|
|
20
|
+
/**
|
|
21
|
+
* This value should be used if there is no log message within 24 hours or missing historical data.
|
|
22
|
+
* Without setting this value it will lead to an empty result and therefore trigger an alert.
|
|
23
|
+
* @defaultValue - false
|
|
24
|
+
*/
|
|
25
|
+
readonly sparseLogging?: boolean;
|
|
26
|
+
/**
|
|
27
|
+
* The Datadog organization, e.g., 'EU' or 'LATAM'
|
|
28
|
+
* @defaultValue - 'EU'
|
|
29
|
+
*/
|
|
30
|
+
readonly organization?: DatadogOrganization;
|
|
31
|
+
}
|
|
32
|
+
export interface DatadogLogQuotaProps {
|
|
33
|
+
/**
|
|
34
|
+
* The daily log quota for the team-specific index in million events.
|
|
35
|
+
*/
|
|
36
|
+
readonly valueInMillionEvents: number;
|
|
37
|
+
/**
|
|
38
|
+
* The warning threshold for the daily log quota monitor in percent.
|
|
39
|
+
*/
|
|
40
|
+
readonly warningThresholdInPercent: number;
|
|
41
|
+
/**
|
|
42
|
+
* The alarm threshold for the daily log quota monitor in percent.
|
|
43
|
+
* The value must be between 0 and 100 and greater than or equal to the warning threshold.
|
|
44
|
+
*/
|
|
45
|
+
readonly alertThresholdInPercent: number;
|
|
46
|
+
}
|
|
47
|
+
/**
|
|
48
|
+
* Basic monitoring and alerting for a Datadog logs index.
|
|
49
|
+
* It follows the Datadog guide at https://docs.datadoghq.com/logs/guide/logs-monitors-on-volumes/
|
|
50
|
+
* and consists of the following three monitors.
|
|
51
|
+
* 1. A metric alert that fires when you reach a certain threshold of your daily log quota.
|
|
52
|
+
* 2. An anomaly monitor that detects log amount spikes.
|
|
53
|
+
* 3. An event alert that fires when you hit the daily log quota.
|
|
54
|
+
*
|
|
55
|
+
* A datadog log index is team specific and thus the monitor needs to be deployed just once per index/team.
|
|
56
|
+
*/
|
|
57
|
+
export declare class DatadogLogIndexMonitoring extends Construct {
|
|
58
|
+
constructor(scope: Construct, id: string, props: DatadogLogIndexMonitoringProps);
|
|
59
|
+
}
|
|
60
|
+
export declare type DatadogOrganization = 'EU' | 'LATAM' | 'Landing Zone';
|
|
61
|
+
export declare function getOrganizationUrlPrefix(organization: DatadogOrganization | undefined): string;
|
|
@@ -0,0 +1,115 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var _a;
|
|
3
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
4
|
+
exports.getOrganizationUrlPrefix = exports.DatadogLogIndexMonitoring = void 0;
|
|
5
|
+
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
|
6
|
+
const constructs_1 = require("constructs");
|
|
7
|
+
const datadog_monitor_1 = require("./datadog-monitor");
|
|
8
|
+
/**
|
|
9
|
+
* Basic monitoring and alerting for a Datadog logs index.
|
|
10
|
+
* It follows the Datadog guide at https://docs.datadoghq.com/logs/guide/logs-monitors-on-volumes/
|
|
11
|
+
* and consists of the following three monitors.
|
|
12
|
+
* 1. A metric alert that fires when you reach a certain threshold of your daily log quota.
|
|
13
|
+
* 2. An anomaly monitor that detects log amount spikes.
|
|
14
|
+
* 3. An event alert that fires when you hit the daily log quota.
|
|
15
|
+
*
|
|
16
|
+
* A datadog log index is team specific and thus the monitor needs to be deployed just once per index/team.
|
|
17
|
+
*/
|
|
18
|
+
class DatadogLogIndexMonitoring extends constructs_1.Construct {
|
|
19
|
+
constructor(scope, id, props) {
|
|
20
|
+
super(scope, id);
|
|
21
|
+
const logQuotaInEvents = Math.floor(props.dailyLogQuota.valueInMillionEvents * 1000000);
|
|
22
|
+
const warningThreshold = Math.floor(logQuotaInEvents * (props.dailyLogQuota.warningThresholdInPercent / 100));
|
|
23
|
+
const alertThreshold = Math.floor(logQuotaInEvents * (props.dailyLogQuota.alertThresholdInPercent / 100));
|
|
24
|
+
new datadog_monitor_1.DatadogMonitor(this, 'DatadogLogDailyQuotaMonitor', {
|
|
25
|
+
serviceName: props.serviceName,
|
|
26
|
+
monitorType: datadog_monitor_1.DatadogMonitorQueryAlertType.METRIC_ALERT,
|
|
27
|
+
name: `${commonMonitorNamePrefix(props.indexName)}: Log quota almost reached`,
|
|
28
|
+
notification: props.notification,
|
|
29
|
+
priority: 3,
|
|
30
|
+
query: `sum(last_1d):sum:datadog.estimated_usage.logs.ingested_events{datadog_is_excluded:false,datadog_index:${props.indexName}}.as_count() >= ${alertThreshold}`,
|
|
31
|
+
message: `Inspect your log volume at ${datadogLogsSearchWithIndexFilterDeepLink(props.indexName, props.organization)} or adapt the log quota at ${datadogLogsIndexesConfigurationDeepLink(props.organization)}`,
|
|
32
|
+
extraTags: [...commonTags(props.indexName)],
|
|
33
|
+
optionOverrides: {
|
|
34
|
+
thresholds: {
|
|
35
|
+
critical: alertThreshold,
|
|
36
|
+
warning: warningThreshold,
|
|
37
|
+
},
|
|
38
|
+
includeTags: false,
|
|
39
|
+
notifyNoData: !props.sparseLogging,
|
|
40
|
+
noDataTimeframe: 24 * 60,
|
|
41
|
+
},
|
|
42
|
+
});
|
|
43
|
+
new datadog_monitor_1.DatadogMonitor(this, 'DatadogLogAlertAnomalyMonitor', {
|
|
44
|
+
serviceName: props.serviceName,
|
|
45
|
+
monitorType: datadog_monitor_1.DatadogMonitorQueryAlertType.METRIC_ALERT,
|
|
46
|
+
priority: 4,
|
|
47
|
+
name: `${commonMonitorNamePrefix(props.indexName)}: Log amount is rising to unexpected levels`,
|
|
48
|
+
query: `avg(last_4h):anomalies(sum:datadog.estimated_usage.logs.ingested_events{datadog_is_excluded:false,datadog_index:${props.indexName}}.as_count(), 'robust', 2, direction='above', alert_window='last_15m', interval=60, count_default_zero='true', seasonality='weekly', timezone='europe/berlin') >= 1`,
|
|
49
|
+
message: `Logs are getting out of control: ${datadogLogsSearchWithIndexFilterDeepLink(props.indexName, props.organization)}`,
|
|
50
|
+
extraTags: [...commonTags(props.indexName)],
|
|
51
|
+
optionOverrides: {
|
|
52
|
+
includeTags: false,
|
|
53
|
+
notifyNoData: !props.sparseLogging,
|
|
54
|
+
thresholdWindows: {
|
|
55
|
+
triggerWindow: 'last_15m',
|
|
56
|
+
recoveryWindow: 'last_15m',
|
|
57
|
+
},
|
|
58
|
+
},
|
|
59
|
+
});
|
|
60
|
+
new datadog_monitor_1.DatadogMonitor(this, 'DatadogLogDailyQuotaReachedMonitor', {
|
|
61
|
+
serviceName: props.serviceName,
|
|
62
|
+
name: `${commonMonitorNamePrefix(props.indexName)}: Log quota reached`,
|
|
63
|
+
notification: props.notification,
|
|
64
|
+
priority: 2,
|
|
65
|
+
monitorType: datadog_monitor_1.DatadogMonitorQueryAlertType.EVENT_V_2_ALERT,
|
|
66
|
+
query: `events("source:datadog datadog_index:${props.indexName} Daily quota reached").rollup("count").last("12h") > 0`,
|
|
67
|
+
message: `The log quote for the index has been reached. If you do not increase the quota at ${datadogLogsIndexesConfigurationDeepLink(props.organization)}, no new logs will be indexed until the quota is reset at 2:00pm UTC.`,
|
|
68
|
+
extraTags: [...commonTags(props.indexName)],
|
|
69
|
+
optionOverrides: {
|
|
70
|
+
timeoutH: 24,
|
|
71
|
+
thresholds: {
|
|
72
|
+
critical: 0,
|
|
73
|
+
},
|
|
74
|
+
},
|
|
75
|
+
});
|
|
76
|
+
this.node.addValidation({
|
|
77
|
+
validate: () => {
|
|
78
|
+
const result = [];
|
|
79
|
+
if (props.dailyLogQuota.warningThresholdInPercent < 0 || props.dailyLogQuota.warningThresholdInPercent > 100) {
|
|
80
|
+
result.push('Invalid [warningThresholdInPercent]: expecting a number between 0 and 100');
|
|
81
|
+
}
|
|
82
|
+
if (props.dailyLogQuota.alertThresholdInPercent < 0 || props.dailyLogQuota.alertThresholdInPercent > 100) {
|
|
83
|
+
result.push('Invalid [alertThresholdInPercent]: expecting a number between 0 and 100');
|
|
84
|
+
}
|
|
85
|
+
if (props.dailyLogQuota.alertThresholdInPercent < props.dailyLogQuota.warningThresholdInPercent) {
|
|
86
|
+
result.push('Invalid [alertThresholdInPercent]: must be greater than or equal to [warningThresholdInPercent]');
|
|
87
|
+
}
|
|
88
|
+
return result;
|
|
89
|
+
},
|
|
90
|
+
});
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
exports.DatadogLogIndexMonitoring = DatadogLogIndexMonitoring;
|
|
94
|
+
_a = JSII_RTTI_SYMBOL_1;
|
|
95
|
+
DatadogLogIndexMonitoring[_a] = { fqn: "@rio-cloud/cdk-v2-constructs.datadogv2.DatadogLogIndexMonitoring", version: "0.0.0" };
|
|
96
|
+
const commonMonitorNamePrefix = (indexName) => `'${indexName}' index`;
|
|
97
|
+
const commonTags = (indexName) => [`logIndex:${indexName}`];
|
|
98
|
+
function getOrganizationUrlPrefix(organization) {
|
|
99
|
+
switch (organization ?? 'EU') {
|
|
100
|
+
case 'EU':
|
|
101
|
+
return 'app';
|
|
102
|
+
case 'LATAM':
|
|
103
|
+
return 'rio-brazil';
|
|
104
|
+
case 'Landing Zone':
|
|
105
|
+
return 'rio-landingzone';
|
|
106
|
+
}
|
|
107
|
+
}
|
|
108
|
+
exports.getOrganizationUrlPrefix = getOrganizationUrlPrefix;
|
|
109
|
+
function datadogLogsSearchWithIndexFilterDeepLink(indexName, organization) {
|
|
110
|
+
return `https://${getOrganizationUrlPrefix(organization)}.datadoghq.eu/logs?index=${indexName}`;
|
|
111
|
+
}
|
|
112
|
+
function datadogLogsIndexesConfigurationDeepLink(organization) {
|
|
113
|
+
return `https://${getOrganizationUrlPrefix(organization)}.datadoghq.eu/logs/pipelines/indexes`;
|
|
114
|
+
}
|
|
115
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"datadog-log-index-monitoring.js","sourceRoot":"","sources":["../../src/datadogv2/datadog-log-index-monitoring.ts"],"names":[],"mappings":";;;;;AAAA,2CAAuC;AACvC,uDAAiF;AAyDjF;;;;;;;;;GASG;AACH,MAAa,yBAA0B,SAAQ,sBAAS;IACtD,YAAY,KAAgB,EAAE,EAAU,EAAE,KAAqC;QAC7E,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEjB,MAAM,gBAAgB,GAAG,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,aAAa,CAAC,oBAAoB,GAAG,OAAS,CAAC,CAAC;QAC1F,MAAM,gBAAgB,GAAG,IAAI,CAAC,KAAK,CAAC,gBAAgB,GAAG,CAAC,KAAK,CAAC,aAAa,CAAC,yBAAyB,GAAG,GAAG,CAAC,CAAC,CAAC;QAC9G,MAAM,cAAc,GAAG,IAAI,CAAC,KAAK,CAAC,gBAAgB,GAAG,CAAC,KAAK,CAAC,aAAa,CAAC,uBAAuB,GAAG,GAAG,CAAC,CAAC,CAAC;QAE1G,IAAI,gCAAc,CAAC,IAAI,EAAE,6BAA6B,EAAE;YACtD,WAAW,EAAE,KAAK,CAAC,WAAW;YAC9B,WAAW,EAAE,8CAA4B,CAAC,YAAY;YACtD,IAAI,EAAE,GAAG,uBAAuB,CAAC,KAAK,CAAC,SAAS,CAAC,4BAA4B;YAC7E,YAAY,EAAE,KAAK,CAAC,YAAY;YAChC,QAAQ,EAAE,CAAC;YACX,KAAK,EAAE,yGAAyG,KAAK,CAAC,SAAS,mBAAmB,cAAc,EAAE;YAClK,OAAO,EAAE,8BAA8B,wCAAwC,CAC7E,KAAK,CAAC,SAAS,EAAE,KAAK,CAAC,YAAY,CACpC,8BAA8B,uCAAuC,CAAC,KAAK,CAAC,YAAY,CAAC,EAAE;YAC5F,SAAS,EAAE,CAAC,GAAG,UAAU,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;YAC3C,eAAe,EAAE;gBACf,UAAU,EAAE;oBACV,QAAQ,EAAE,cAAc;oBACxB,OAAO,EAAE,gBAAgB;iBAC1B;gBACD,WAAW,EAAE,KAAK;gBAClB,YAAY,EAAE,CAAC,KAAK,CAAC,aAAa;gBAClC,eAAe,EAAE,EAAE,GAAG,EAAE;aACzB;SACF,CAAC,CAAC;QAEH,IAAI,gCAAc,CAAC,IAAI,EAAE,+BAA+B,EAAE;YACxD,WAAW,EAAE,KAAK,CAAC,WAAW;YAC9B,WAAW,EAAE,8CAA4B,CAAC,YAAY;YACtD,QAAQ,EAAE,CAAC;YACX,IAAI,EAAE,GAAG,uBAAuB,CAAC,KAAK,CAAC,SAAS,CAAC,6CAA6C;YAC9F,KAAK,EAAE,mHAAmH,KAAK,CAAC,SAAS,qKAAqK;YAC9S,OAAO,EAAE,oCAAoC,wCAAwC,CACnF,KAAK,CAAC,SAAS,EAAE,KAAK,CAAC,YAAY,CACpC,EAAE;YACH,SAAS,EAAE,CAAC,GAAG,UAAU,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;YAC3C,eAAe,EAAE;gBACf,WAAW,EAAE,KAAK;gBAClB,YAAY,EAAE,CAAC,KAAK,CAAC,aAAa;gBAClC,gBAAgB,EAAE;oBAChB,aAAa,EAAE,UAAU;oBACzB,cAAc,EAAE,UAAU;iBAC3B;aACF;SACF,CAAC,CAAC;QAEH,IAAI,gCAAc,CAAC,IAAI,EAAE,oCAAoC,EAAE;YAC7D,WAAW,EAAE,KAAK,CAAC,WAAW;YAC9B,IAAI,EAAE,GAAG,uBAAuB,CAAC,KAAK,CAAC,SAAS,CAAC,qBAAqB;YACtE,YAAY,EAAE,KAAK,CAAC,YAAY;YAChC,QAAQ,EAAE,CAAC;YACX,WAAW,EAAE,8CAA4B,CAAC,eAAe;YACzD,KAAK,EAAE,wCAAwC,KAAK,CAAC,SAAS,wDAAwD;YACtH,OAAO,EAAE,qFAAqF,uCAAuC,CAAC,KAAK,CAAC,YAAY,CAAC,uEAAuE;YAChO,SAAS,EAAE,CAAC,GAAG,UAAU,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;YAC3C,eAAe,EAAE;gBACf,QAAQ,EAAE,EAAE;gBACZ,UAAU,EAAE;oBACV,QAAQ,EAAE,CAAC;iBACZ;aACF;SACF,CAAC,CAAC;QAEH,IAAI,CAAC,IAAI,CAAC,aAAa,CAAC;YACtB,QAAQ,EAAE,GAAa,EAAE;gBACvB,MAAM,MAAM,GAAG,EAAE,CAAC;gBAClB,IAAI,KAAK,CAAC,aAAa,CAAC,yBAAyB,GAAG,CAAC,IAAI,KAAK,CAAC,aAAa,CAAC,yBAAyB,GAAG,GAAG,EAAE;oBAC5G,MAAM,CAAC,IAAI,CAAC,2EAA2E,CAAC,CAAC;iBAC1F;gBACD,IAAI,KAAK,CAAC,aAAa,CAAC,uBAAuB,GAAG,CAAC,IAAI,KAAK,CAAC,aAAa,CAAC,uBAAuB,GAAG,GAAG,EAAE;oBACxG,MAAM,CAAC,IAAI,CAAC,yEAAyE,CAAC,CAAC;iBACxF;gBACD,IAAI,KAAK,CAAC,aAAa,CAAC,uBAAuB,GAAG,KAAK,CAAC,aAAa,CAAC,yBAAyB,EAAE;oBAC/F,MAAM,CAAC,IAAI,CAAC,iGAAiG,CAAC,CAAC;iBAChH;gBACD,OAAO,MAAM,CAAC;YAChB,CAAC;SACF,CAAC,CAAC;IACL,CAAC;;AAlFH,8DAmFC;;;AAED,MAAM,uBAAuB,GAAG,CAAC,SAAiB,EAAE,EAAE,CAAC,IAAI,SAAS,SAAS,CAAC;AAC9E,MAAM,UAAU,GAAG,CAAC,SAAiB,EAAE,EAAE,CAAC,CAAC,YAAY,SAAS,EAAE,CAAC,CAAC;AAIpE,SAAgB,wBAAwB,CAAC,YAA6C;IACpF,QAAQ,YAAY,IAAI,IAAI,EAAE;QAC5B,KAAK,IAAI;YACP,OAAO,KAAK,CAAC;QACf,KAAK,OAAO;YACV,OAAO,YAAY,CAAC;QACtB,KAAK,cAAc;YACjB,OAAO,iBAAiB,CAAC;KAC5B;AACH,CAAC;AATD,4DASC;AAED,SAAS,wCAAwC,CAAC,SAAiB,EAAE,YAA6C;IAChH,OAAO,WAAW,wBAAwB,CAAC,YAAY,CAAC,4BAA4B,SAAS,EAAE,CAAC;AAClG,CAAC;AAED,SAAS,uCAAuC,CAAC,YAA6C;IAC5F,OAAO,WAAW,wBAAwB,CAAC,YAAY,CAAC,sCAAsC,CAAC;AACjG,CAAC","sourcesContent":["import { Construct } from 'constructs';\nimport { DatadogMonitor, DatadogMonitorQueryAlertType } from './datadog-monitor';\nimport { INotification } from './datadog-notification';\n\nexport interface DatadogLogIndexMonitoringProps {\n\n  /**\n   * The name of your service in Datadog. Can be used to query the monitors.\n   */\n  readonly serviceName: string;\n\n  /**\n   * The name of the Datadog index.\n   */\n  readonly indexName: string;\n\n  /**\n   * The integration to use for alerting.\n   */\n  readonly notification?: INotification;\n\n  /**\n   * The daily log quota settings.\n   */\n  readonly dailyLogQuota: DatadogLogQuotaProps;\n\n  /**\n   * This value should be used if there is no log message within 24 hours or missing historical data.\n   * Without setting this value it will lead to an empty result and therefore trigger an alert.\n   * @defaultValue - false\n   */\n  readonly sparseLogging?: boolean;\n\n  /**\n   * The Datadog organization, e.g., 'EU' or 'LATAM'\n   * @defaultValue - 'EU'\n   */\n  readonly organization?: DatadogOrganization;\n}\n\nexport interface DatadogLogQuotaProps {\n  /**\n   * The daily log quota for the team-specific index in million events.\n   */\n  readonly valueInMillionEvents: number;\n\n  /**\n   * The warning threshold for the daily log quota monitor in percent.\n   */\n  readonly warningThresholdInPercent: number;\n\n  /**\n   * The alarm threshold for the daily log quota monitor in percent.\n   * The value must be between 0 and 100 and greater than or equal to the warning threshold.\n   */\n  readonly alertThresholdInPercent: number;\n}\n\n/**\n * Basic monitoring and alerting for a Datadog logs index.\n * It follows the Datadog guide at https://docs.datadoghq.com/logs/guide/logs-monitors-on-volumes/\n * and consists of the following three monitors.\n *   1. A metric alert that fires when you reach a certain threshold of your daily log quota.\n *   2. An anomaly monitor that detects log amount spikes.\n *   3. An event alert that fires when you hit the daily log quota.\n *\n * A datadog log index is team specific and thus the monitor needs to be deployed just once per index/team.\n */\nexport class DatadogLogIndexMonitoring extends Construct {\n  constructor(scope: Construct, id: string, props: DatadogLogIndexMonitoringProps) {\n    super(scope, id);\n\n    const logQuotaInEvents = Math.floor(props.dailyLogQuota.valueInMillionEvents * 1_000_000);\n    const warningThreshold = Math.floor(logQuotaInEvents * (props.dailyLogQuota.warningThresholdInPercent / 100));\n    const alertThreshold = Math.floor(logQuotaInEvents * (props.dailyLogQuota.alertThresholdInPercent / 100));\n\n    new DatadogMonitor(this, 'DatadogLogDailyQuotaMonitor', {\n      serviceName: props.serviceName,\n      monitorType: DatadogMonitorQueryAlertType.METRIC_ALERT,\n      name: `${commonMonitorNamePrefix(props.indexName)}: Log quota almost reached`,\n      notification: props.notification,\n      priority: 3,\n      query: `sum(last_1d):sum:datadog.estimated_usage.logs.ingested_events{datadog_is_excluded:false,datadog_index:${props.indexName}}.as_count() >= ${alertThreshold}`,\n      message: `Inspect your log volume at ${datadogLogsSearchWithIndexFilterDeepLink(\n        props.indexName, props.organization,\n      )} or adapt the log quota at ${datadogLogsIndexesConfigurationDeepLink(props.organization)}`,\n      extraTags: [...commonTags(props.indexName)],\n      optionOverrides: {\n        thresholds: {\n          critical: alertThreshold,\n          warning: warningThreshold,\n        },\n        includeTags: false,\n        notifyNoData: !props.sparseLogging,\n        noDataTimeframe: 24 * 60, // 1 day\n      },\n    });\n\n    new DatadogMonitor(this, 'DatadogLogAlertAnomalyMonitor', {\n      serviceName: props.serviceName,\n      monitorType: DatadogMonitorQueryAlertType.METRIC_ALERT,\n      priority: 4,\n      name: `${commonMonitorNamePrefix(props.indexName)}: Log amount is rising to unexpected levels`,\n      query: `avg(last_4h):anomalies(sum:datadog.estimated_usage.logs.ingested_events{datadog_is_excluded:false,datadog_index:${props.indexName}}.as_count(), 'robust', 2, direction='above', alert_window='last_15m', interval=60, count_default_zero='true', seasonality='weekly', timezone='europe/berlin') >= 1`,\n      message: `Logs are getting out of control: ${datadogLogsSearchWithIndexFilterDeepLink(\n        props.indexName, props.organization,\n      )}`,\n      extraTags: [...commonTags(props.indexName)],\n      optionOverrides: {\n        includeTags: false,\n        notifyNoData: !props.sparseLogging,\n        thresholdWindows: {\n          triggerWindow: 'last_15m',\n          recoveryWindow: 'last_15m',\n        },\n      },\n    });\n\n    new DatadogMonitor(this, 'DatadogLogDailyQuotaReachedMonitor', {\n      serviceName: props.serviceName,\n      name: `${commonMonitorNamePrefix(props.indexName)}: Log quota reached`,\n      notification: props.notification,\n      priority: 2,\n      monitorType: DatadogMonitorQueryAlertType.EVENT_V_2_ALERT,\n      query: `events(\"source:datadog datadog_index:${props.indexName} Daily quota reached\").rollup(\"count\").last(\"12h\") > 0`,\n      message: `The log quote for the index has been reached. If you do not increase the quota at ${datadogLogsIndexesConfigurationDeepLink(props.organization)}, no new logs will be indexed until the quota is reset at 2:00pm UTC.`,\n      extraTags: [...commonTags(props.indexName)],\n      optionOverrides: {\n        timeoutH: 24,\n        thresholds: {\n          critical: 0,\n        },\n      },\n    });\n\n    this.node.addValidation({\n      validate: (): string[] => {\n        const result = [];\n        if (props.dailyLogQuota.warningThresholdInPercent < 0 || props.dailyLogQuota.warningThresholdInPercent > 100) {\n          result.push('Invalid [warningThresholdInPercent]: expecting a number between 0 and 100');\n        }\n        if (props.dailyLogQuota.alertThresholdInPercent < 0 || props.dailyLogQuota.alertThresholdInPercent > 100) {\n          result.push('Invalid [alertThresholdInPercent]: expecting a number between 0 and 100');\n        }\n        if (props.dailyLogQuota.alertThresholdInPercent < props.dailyLogQuota.warningThresholdInPercent) {\n          result.push('Invalid [alertThresholdInPercent]: must be greater than or equal to [warningThresholdInPercent]');\n        }\n        return result;\n      },\n    });\n  }\n}\n\nconst commonMonitorNamePrefix = (indexName: string) => `'${indexName}' index`;\nconst commonTags = (indexName: string) => [`logIndex:${indexName}`];\n\nexport type DatadogOrganization = 'EU' | 'LATAM' | 'Landing Zone';\n\nexport function getOrganizationUrlPrefix(organization: DatadogOrganization | undefined): string {\n  switch (organization ?? 'EU') {\n    case 'EU':\n      return 'app';\n    case 'LATAM':\n      return 'rio-brazil';\n    case 'Landing Zone':\n      return 'rio-landingzone';\n  }\n}\n\nfunction datadogLogsSearchWithIndexFilterDeepLink(indexName: string, organization: DatadogOrganization | undefined) {\n  return `https://${getOrganizationUrlPrefix(organization)}.datadoghq.eu/logs?index=${indexName}`;\n}\n\nfunction datadogLogsIndexesConfigurationDeepLink(organization: DatadogOrganization | undefined) {\n  return `https://${getOrganizationUrlPrefix(organization)}.datadoghq.eu/logs/pipelines/indexes`;\n}\n"]}
|
|
@@ -0,0 +1,234 @@
|
|
|
1
|
+
import { Construct } from 'constructs';
|
|
2
|
+
import { INotification } from './datadog-notification';
|
|
3
|
+
/**
|
|
4
|
+
* Well known monitor types supported by Datadog.
|
|
5
|
+
*
|
|
6
|
+
* @see https://docs.datadoghq.com/monitors/types/
|
|
7
|
+
*/
|
|
8
|
+
export declare enum DatadogMonitorQueryAlertType {
|
|
9
|
+
AUDIT_ALERT = "audit alert",
|
|
10
|
+
COMPOSITE = "composite",
|
|
11
|
+
EVENT_ALERT = "event alert",
|
|
12
|
+
EVENT_V_2_ALERT = "event-v2 alert",
|
|
13
|
+
LOG_ALERT = "log alert",
|
|
14
|
+
METRIC_ALERT = "metric alert",
|
|
15
|
+
PROCESS_ALERT = "process alert",
|
|
16
|
+
QUERY_ALERT = "query alert",
|
|
17
|
+
SERVICE_CHECK = "service check",
|
|
18
|
+
SYNTHETICS_ALERT = "synthetics alert",
|
|
19
|
+
TRACE_ANALYTICS_ALERT = "trace-analytics alert",
|
|
20
|
+
SLO_ALERT = "slo alert",
|
|
21
|
+
RUM_ALERT = "rum alert",
|
|
22
|
+
CI_PIPELINES_ALERT = "ci-pipelines alert",
|
|
23
|
+
ERROR_TRACKING_ALERT = "error-tracking alert",
|
|
24
|
+
CI_TESTS_ALERT = "ci-tests alert"
|
|
25
|
+
}
|
|
26
|
+
/**
|
|
27
|
+
* Properties of {@link DatadogMonitor}
|
|
28
|
+
*/
|
|
29
|
+
export interface DatadogMonitorProps {
|
|
30
|
+
/**
|
|
31
|
+
* A name suffix to be appended to the monitors name.
|
|
32
|
+
*
|
|
33
|
+
* The monitor name is concatenated form `<TeamName> <serviceName property> <name property>`
|
|
34
|
+
*/
|
|
35
|
+
readonly name: string;
|
|
36
|
+
/**
|
|
37
|
+
* The name of the service to which the monitor belongs.
|
|
38
|
+
*
|
|
39
|
+
* Used to generate the monitor name as well a apply the `service` tag.
|
|
40
|
+
*/
|
|
41
|
+
readonly serviceName: string;
|
|
42
|
+
/**
|
|
43
|
+
* The monitors type
|
|
44
|
+
*
|
|
45
|
+
* @see https://docs.datadoghq.com/monitors/types/
|
|
46
|
+
*/
|
|
47
|
+
readonly monitorType: DatadogMonitorQueryAlertType;
|
|
48
|
+
/**
|
|
49
|
+
* The monitors search query
|
|
50
|
+
*
|
|
51
|
+
* @see https://docs.datadoghq.com/monitors/configuration/#define-the-search-query
|
|
52
|
+
*/
|
|
53
|
+
readonly query: string;
|
|
54
|
+
/**
|
|
55
|
+
* The message pattern for the monitor
|
|
56
|
+
*
|
|
57
|
+
* @see https://docs.datadoghq.com/monitors/configuration/#message
|
|
58
|
+
*/
|
|
59
|
+
readonly message: string;
|
|
60
|
+
/**
|
|
61
|
+
* To explicitly disable notifications use {@link NoNotification}.
|
|
62
|
+
*
|
|
63
|
+
* @defaultValue {@link DefaultSlackNotification}
|
|
64
|
+
*
|
|
65
|
+
* @see https://docs.datadoghq.com/monitors/notify
|
|
66
|
+
*/
|
|
67
|
+
readonly notification?: INotification;
|
|
68
|
+
/**
|
|
69
|
+
* The alert priority of the monitor
|
|
70
|
+
*
|
|
71
|
+
* @defaultValue 4
|
|
72
|
+
*/
|
|
73
|
+
readonly priority?: number;
|
|
74
|
+
/**
|
|
75
|
+
* Specific options to override the default monitor options.
|
|
76
|
+
*/
|
|
77
|
+
readonly optionOverrides?: CfnDatadogMonitorOptions;
|
|
78
|
+
/**
|
|
79
|
+
* Tags to be appended to the monitor in addition to the RIO default tags
|
|
80
|
+
*/
|
|
81
|
+
readonly extraTags?: string[];
|
|
82
|
+
}
|
|
83
|
+
/**
|
|
84
|
+
* Well known renotification settings supported by Datadog.
|
|
85
|
+
*
|
|
86
|
+
* @see https://docs.datadoghq.com/monitors/notify/#renotify
|
|
87
|
+
*/
|
|
88
|
+
export declare enum RenotifyStatuses {
|
|
89
|
+
ALERT = "alert",
|
|
90
|
+
NO_DATA = "no data",
|
|
91
|
+
WARN = "warn"
|
|
92
|
+
}
|
|
93
|
+
/**
|
|
94
|
+
* Options to override for Datadog monitors in case more specific settings are required.
|
|
95
|
+
*
|
|
96
|
+
* {@link https://github.com/DataDog/datadog-cloudformation-resources/blob/master/datadog-monitors-monitor-handler/datadog-monitors-monitor.json}
|
|
97
|
+
*/
|
|
98
|
+
export interface CfnDatadogMonitorOptions {
|
|
99
|
+
/**
|
|
100
|
+
* Whether or not to include a sample of the logs
|
|
101
|
+
*/
|
|
102
|
+
readonly enableLogsSample?: boolean;
|
|
103
|
+
/**
|
|
104
|
+
* Message to include with a re-notification when renotify_interval is set
|
|
105
|
+
*/
|
|
106
|
+
readonly escalationMessage?: string;
|
|
107
|
+
/**
|
|
108
|
+
* Time in seconds to delay evaluation
|
|
109
|
+
*
|
|
110
|
+
* @defaultValue 900
|
|
111
|
+
*/
|
|
112
|
+
readonly evaluationDelay?: number;
|
|
113
|
+
/**
|
|
114
|
+
* Whether or not to include triggering tags into notification title'
|
|
115
|
+
*
|
|
116
|
+
* @defaultValue true
|
|
117
|
+
*/
|
|
118
|
+
readonly includeTags?: boolean;
|
|
119
|
+
/**
|
|
120
|
+
* Whether or not changes to this monitor should be restricted to the creator or admins
|
|
121
|
+
*/
|
|
122
|
+
readonly locked?: boolean;
|
|
123
|
+
/**
|
|
124
|
+
* Number of locations allowed to fail before triggering alert
|
|
125
|
+
*/
|
|
126
|
+
readonly minLocationFailed?: number;
|
|
127
|
+
/**
|
|
128
|
+
* Time in seconds to allow a host to start reporting data before starting the evaluation of monitor results
|
|
129
|
+
*/
|
|
130
|
+
readonly newHostDelay?: number;
|
|
131
|
+
/**
|
|
132
|
+
* Number of minutes data stopped reporting before notifying
|
|
133
|
+
*/
|
|
134
|
+
readonly noDataTimeframe?: number;
|
|
135
|
+
/**
|
|
136
|
+
* Whether or not to notify tagged users when changes are made to the monitor
|
|
137
|
+
*/
|
|
138
|
+
readonly notifyAudit?: boolean;
|
|
139
|
+
/**
|
|
140
|
+
* Whether or not to notify when data stops reporting
|
|
141
|
+
*
|
|
142
|
+
* @defaultValue false
|
|
143
|
+
*/
|
|
144
|
+
readonly notifyNoData?: boolean;
|
|
145
|
+
/**
|
|
146
|
+
* Number of minutes after the last notification before the monitor re-notifies on the current status
|
|
147
|
+
*/
|
|
148
|
+
readonly renotifyInterval?: number;
|
|
149
|
+
/**
|
|
150
|
+
* Whether or not the monitor requires a full window of data before it is evaluated
|
|
151
|
+
*
|
|
152
|
+
* @defaultValue false
|
|
153
|
+
*/
|
|
154
|
+
readonly requireFullWindow?: boolean;
|
|
155
|
+
/**
|
|
156
|
+
* ID of the corresponding synthetics check
|
|
157
|
+
*/
|
|
158
|
+
readonly syntheticsCheckID?: number;
|
|
159
|
+
readonly thresholds?: CfnDatadogMonitorThresholds;
|
|
160
|
+
readonly thresholdWindows?: CfnDatadogMonitorThresholdWindows;
|
|
161
|
+
/**
|
|
162
|
+
* Number of hours of the monitor not reporting data before it automatically resolves
|
|
163
|
+
*/
|
|
164
|
+
readonly timeoutH?: number;
|
|
165
|
+
/**
|
|
166
|
+
* The number of times re-notification messages should be sent on the current status at the provided re-notification interval.
|
|
167
|
+
*/
|
|
168
|
+
readonly renotifyOccurrences?: number;
|
|
169
|
+
/**
|
|
170
|
+
* The types of monitor statuses for which re-notification messages are sent.
|
|
171
|
+
*/
|
|
172
|
+
readonly renotifyStatuses?: RenotifyStatuses;
|
|
173
|
+
/**
|
|
174
|
+
* How long the test should be in failure before alerting (integer, number of seconds, max 7200).
|
|
175
|
+
*/
|
|
176
|
+
readonly minFailureDuration?: number;
|
|
177
|
+
/**
|
|
178
|
+
* Time (in seconds) to skip evaluations for new groups. For example, this option can be used to skip evaluations for new hosts while they initialize. Must be a non negative integer.
|
|
179
|
+
*/
|
|
180
|
+
readonly newGroupDelay?: number;
|
|
181
|
+
/**
|
|
182
|
+
* List of requests that can be used in the monitor query.
|
|
183
|
+
*/
|
|
184
|
+
readonly variables?: object[];
|
|
185
|
+
}
|
|
186
|
+
export interface CfnDatadogMonitorThresholds {
|
|
187
|
+
/**
|
|
188
|
+
*Threshold value for triggering an alert
|
|
189
|
+
*/
|
|
190
|
+
readonly critical?: number;
|
|
191
|
+
/**
|
|
192
|
+
*Threshold value for recovering from an alert state
|
|
193
|
+
*/
|
|
194
|
+
readonly criticalRecovery?: number;
|
|
195
|
+
/**
|
|
196
|
+
*Threshold value for recovering from an alert state
|
|
197
|
+
*/
|
|
198
|
+
readonly ok?: number;
|
|
199
|
+
/**
|
|
200
|
+
*Threshold value for triggering a warning
|
|
201
|
+
*/
|
|
202
|
+
readonly warning?: number;
|
|
203
|
+
/**
|
|
204
|
+
* Threshold value for recovering from a warning state
|
|
205
|
+
*/
|
|
206
|
+
readonly warningRecovery?: number;
|
|
207
|
+
}
|
|
208
|
+
export interface CfnDatadogMonitorThresholdWindows {
|
|
209
|
+
/**
|
|
210
|
+
* How long a metric must be anomalous before triggering an alert
|
|
211
|
+
*/
|
|
212
|
+
readonly triggerWindow?: string;
|
|
213
|
+
/**
|
|
214
|
+
* How long an anomalous metric must be normal before recovering from an alert state
|
|
215
|
+
*/
|
|
216
|
+
readonly recoveryWindow?: string;
|
|
217
|
+
}
|
|
218
|
+
/**
|
|
219
|
+
* Wrapper around Datadog's CloudFormation `Datadog::Monitors::Monitor` with some RIO defaults applied.
|
|
220
|
+
*
|
|
221
|
+
* To use this the Datadog resources have to be {@link https://docs.datadoghq.com/integrations/guide/amazon_cloudformation/
|
|
222
|
+
* |enabled in the CloudFormation registry} of the corresponding AWS account. Usually this is done
|
|
223
|
+
* automatically by the Datadog Integrations (account module AND ServiceCatalog) provided by CLAID.
|
|
224
|
+
*/
|
|
225
|
+
export declare class DatadogMonitor extends Construct {
|
|
226
|
+
/**
|
|
227
|
+
* Default priority of a monitor
|
|
228
|
+
*/
|
|
229
|
+
static readonly DEFAULT_PRIORITY = 4;
|
|
230
|
+
readonly name: string;
|
|
231
|
+
readonly priority: number;
|
|
232
|
+
constructor(scope: Construct, id: string, props: DatadogMonitorProps);
|
|
233
|
+
private concatenateMessage;
|
|
234
|
+
}
|