@rio-cloud/cdk-v2-constructs 2.4.0 → 2.5.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 +107 -1
- package/CHANGELOG.md +12 -0
- package/lib/contributions/team-claid/datadog/index.d.ts +1 -0
- package/lib/contributions/team-claid/datadog/index.js +2 -1
- package/lib/contributions/team-claid/datadog/usage-monitoring.d.ts +12 -0
- package/lib/contributions/team-claid/datadog/usage-monitoring.js +133 -0
- package/lib/datadog/datadogLogIndexMonitoring.js +15 -4
- package/package.json +1 -1
- package/version.json +1 -1
package/.jsii
CHANGED
|
@@ -6380,6 +6380,112 @@
|
|
|
6380
6380
|
"name": "UnitOfPeriod",
|
|
6381
6381
|
"symbolId": "src/watchful/datadog-log-alarm:UnitOfPeriod"
|
|
6382
6382
|
},
|
|
6383
|
+
"@rio-cloud/cdk-v2-constructs.UsageMonitoring": {
|
|
6384
|
+
"assembly": "@rio-cloud/cdk-v2-constructs",
|
|
6385
|
+
"base": "constructs.Construct",
|
|
6386
|
+
"docs": {
|
|
6387
|
+
"stability": "stable"
|
|
6388
|
+
},
|
|
6389
|
+
"fqn": "@rio-cloud/cdk-v2-constructs.UsageMonitoring",
|
|
6390
|
+
"initializer": {
|
|
6391
|
+
"docs": {
|
|
6392
|
+
"stability": "stable"
|
|
6393
|
+
},
|
|
6394
|
+
"locationInModule": {
|
|
6395
|
+
"filename": "src/contributions/team-claid/datadog/usage-monitoring.ts",
|
|
6396
|
+
"line": 30
|
|
6397
|
+
},
|
|
6398
|
+
"parameters": [
|
|
6399
|
+
{
|
|
6400
|
+
"name": "scope",
|
|
6401
|
+
"type": {
|
|
6402
|
+
"fqn": "constructs.Construct"
|
|
6403
|
+
}
|
|
6404
|
+
},
|
|
6405
|
+
{
|
|
6406
|
+
"name": "id",
|
|
6407
|
+
"type": {
|
|
6408
|
+
"primitive": "string"
|
|
6409
|
+
}
|
|
6410
|
+
},
|
|
6411
|
+
{
|
|
6412
|
+
"name": "props",
|
|
6413
|
+
"type": {
|
|
6414
|
+
"fqn": "@rio-cloud/cdk-v2-constructs.UsageMonitoringProps"
|
|
6415
|
+
}
|
|
6416
|
+
}
|
|
6417
|
+
]
|
|
6418
|
+
},
|
|
6419
|
+
"kind": "class",
|
|
6420
|
+
"locationInModule": {
|
|
6421
|
+
"filename": "src/contributions/team-claid/datadog/usage-monitoring.ts",
|
|
6422
|
+
"line": 29
|
|
6423
|
+
},
|
|
6424
|
+
"name": "UsageMonitoring",
|
|
6425
|
+
"symbolId": "src/contributions/team-claid/datadog/usage-monitoring:UsageMonitoring"
|
|
6426
|
+
},
|
|
6427
|
+
"@rio-cloud/cdk-v2-constructs.UsageMonitoringProps": {
|
|
6428
|
+
"assembly": "@rio-cloud/cdk-v2-constructs",
|
|
6429
|
+
"datatype": true,
|
|
6430
|
+
"docs": {
|
|
6431
|
+
"stability": "stable"
|
|
6432
|
+
},
|
|
6433
|
+
"fqn": "@rio-cloud/cdk-v2-constructs.UsageMonitoringProps",
|
|
6434
|
+
"kind": "interface",
|
|
6435
|
+
"locationInModule": {
|
|
6436
|
+
"filename": "src/contributions/team-claid/datadog/usage-monitoring.ts",
|
|
6437
|
+
"line": 17
|
|
6438
|
+
},
|
|
6439
|
+
"name": "UsageMonitoringProps",
|
|
6440
|
+
"properties": [
|
|
6441
|
+
{
|
|
6442
|
+
"abstract": true,
|
|
6443
|
+
"docs": {
|
|
6444
|
+
"stability": "stable"
|
|
6445
|
+
},
|
|
6446
|
+
"immutable": true,
|
|
6447
|
+
"locationInModule": {
|
|
6448
|
+
"filename": "src/contributions/team-claid/datadog/usage-monitoring.ts",
|
|
6449
|
+
"line": 19
|
|
6450
|
+
},
|
|
6451
|
+
"name": "alertType",
|
|
6452
|
+
"type": {
|
|
6453
|
+
"primitive": "string"
|
|
6454
|
+
}
|
|
6455
|
+
},
|
|
6456
|
+
{
|
|
6457
|
+
"abstract": true,
|
|
6458
|
+
"docs": {
|
|
6459
|
+
"stability": "stable"
|
|
6460
|
+
},
|
|
6461
|
+
"immutable": true,
|
|
6462
|
+
"locationInModule": {
|
|
6463
|
+
"filename": "src/contributions/team-claid/datadog/usage-monitoring.ts",
|
|
6464
|
+
"line": 20
|
|
6465
|
+
},
|
|
6466
|
+
"name": "organization",
|
|
6467
|
+
"type": {
|
|
6468
|
+
"primitive": "string"
|
|
6469
|
+
}
|
|
6470
|
+
},
|
|
6471
|
+
{
|
|
6472
|
+
"abstract": true,
|
|
6473
|
+
"docs": {
|
|
6474
|
+
"stability": "stable"
|
|
6475
|
+
},
|
|
6476
|
+
"immutable": true,
|
|
6477
|
+
"locationInModule": {
|
|
6478
|
+
"filename": "src/contributions/team-claid/datadog/usage-monitoring.ts",
|
|
6479
|
+
"line": 18
|
|
6480
|
+
},
|
|
6481
|
+
"name": "serviceName",
|
|
6482
|
+
"type": {
|
|
6483
|
+
"primitive": "string"
|
|
6484
|
+
}
|
|
6485
|
+
}
|
|
6486
|
+
],
|
|
6487
|
+
"symbolId": "src/contributions/team-claid/datadog/usage-monitoring:UsageMonitoringProps"
|
|
6488
|
+
},
|
|
6383
6489
|
"@rio-cloud/cdk-v2-constructs.WatchApplicationLoadBalancer": {
|
|
6384
6490
|
"assembly": "@rio-cloud/cdk-v2-constructs",
|
|
6385
6491
|
"base": "constructs.Construct",
|
|
@@ -8255,5 +8361,5 @@
|
|
|
8255
8361
|
}
|
|
8256
8362
|
},
|
|
8257
8363
|
"version": "0.0.0",
|
|
8258
|
-
"fingerprint": "
|
|
8364
|
+
"fingerprint": "dte+8vB0DqG7X/HrsuaP7sCljQ0lY8wacZCfjzyVKZg="
|
|
8259
8365
|
}
|
package/CHANGELOG.md
CHANGED
|
@@ -2,6 +2,18 @@
|
|
|
2
2
|
|
|
3
3
|
All notable changes to this project will be documented in this file. See [standard-version](https://github.com/conventional-changelog/standard-version) for commit guidelines.
|
|
4
4
|
|
|
5
|
+
## [2.5.0](https://collaboration.msi.audi.com/stash/projects/RIODEV/repos/cdk-v2-constructs/compare/commits?targetBranch=refs%2Ftags%2Fv2.4.2&sourceBranch=refs%2Ftags%2Fv2.5.0) (2022-07-28)
|
|
6
|
+
|
|
7
|
+
|
|
8
|
+
### Features
|
|
9
|
+
|
|
10
|
+
* **contributions:** Add support for Datadog usage monitoring ([d9e9375](https://collaboration.msi.audi.com/stash/projects/RIODEV/repos/cdk-v2-constructs/commits/d9e9375b299ad3d12ad8cb2361c54402c5a877b1))
|
|
11
|
+
* **contributions:** Add support for Datadog usage monitoring ([6b729a7](https://collaboration.msi.audi.com/stash/projects/RIODEV/repos/cdk-v2-constructs/commits/6b729a77e63ec10fd4b3bd792d1008751e3d120b))
|
|
12
|
+
|
|
13
|
+
### [2.4.2](https://collaboration.msi.audi.com/stash/projects/RIODEV/repos/cdk-v2-constructs/compare/commits?targetBranch=refs%2Ftags%2Fv2.4.1&sourceBranch=refs%2Ftags%2Fv2.4.2) (2022-07-20)
|
|
14
|
+
|
|
15
|
+
### [2.4.1](https://collaboration.msi.audi.com/stash/projects/RIODEV/repos/cdk-v2-constructs/compare/commits?targetBranch=refs%2Ftags%2Fv2.4.0&sourceBranch=refs%2Ftags%2Fv2.4.1) (2022-07-20)
|
|
16
|
+
|
|
5
17
|
## [2.4.0](https://collaboration.msi.audi.com/stash/projects/RIODEV/repos/cdk-v2-constructs/compare/commits?targetBranch=refs%2Ftags%2Fv2.3.3&sourceBranch=refs%2Ftags%2Fv2.4.0) (2022-07-14)
|
|
6
18
|
|
|
7
19
|
|
|
@@ -11,4 +11,5 @@ var __exportStar = (this && this.__exportStar) || function(m, exports) {
|
|
|
11
11
|
};
|
|
12
12
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
13
13
|
__exportStar(require("./backup-monitor"), exports);
|
|
14
|
-
|
|
14
|
+
__exportStar(require("./usage-monitoring"), exports);
|
|
15
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvY29udHJpYnV0aW9ucy90ZWFtLWNsYWlkL2RhdGFkb2cvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7O0FBQUEsbURBQWlDO0FBQ2pDLHFEQUFtQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vYmFja3VwLW1vbml0b3InO1xuZXhwb3J0ICogZnJvbSAnLi91c2FnZS1tb25pdG9yaW5nJztcbiJdfQ==
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
import * as constructs from 'constructs';
|
|
2
|
+
import { DatadogAlertType } from '../../../datadog/datadogMonitor';
|
|
3
|
+
export declare type DatadogOrganization = 'EU' | 'LATAM' | 'Landing Zone';
|
|
4
|
+
export interface UsageMonitoringProps {
|
|
5
|
+
readonly serviceName: string;
|
|
6
|
+
readonly alertType: DatadogAlertType;
|
|
7
|
+
readonly organization: DatadogOrganization;
|
|
8
|
+
}
|
|
9
|
+
export declare class UsageMonitoring extends constructs.Construct {
|
|
10
|
+
constructor(scope: constructs.Construct, id: string, props: UsageMonitoringProps);
|
|
11
|
+
private createAnomalyMonitorForSpikes;
|
|
12
|
+
}
|
|
@@ -0,0 +1,133 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var _a;
|
|
3
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
4
|
+
exports.UsageMonitoring = void 0;
|
|
5
|
+
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
|
6
|
+
const constructs = require("constructs");
|
|
7
|
+
const datadogMonitor_1 = require("../../../datadog/datadogMonitor");
|
|
8
|
+
function getOrganizationUrlPrefix(organization) {
|
|
9
|
+
switch (organization) {
|
|
10
|
+
case 'EU':
|
|
11
|
+
return 'app';
|
|
12
|
+
case 'LATAM':
|
|
13
|
+
return 'rio-brazil';
|
|
14
|
+
case 'Landing Zone':
|
|
15
|
+
return 'rio-landingzone';
|
|
16
|
+
}
|
|
17
|
+
}
|
|
18
|
+
class UsageMonitoring extends constructs.Construct {
|
|
19
|
+
constructor(scope, id, props) {
|
|
20
|
+
super(scope, id);
|
|
21
|
+
const additionalMessageForLogIndexScopedMonitors = `Take a look at https://${getOrganizationUrlPrefix(props.organization)}.datadoghq.eu/dash/integration/465/log-management-estimated-usage?tpl_var_datadog_index={{datadog_index.name}} for more details.`;
|
|
22
|
+
const additionalMessageForEnvScopedMonitors = `Take a look at https://${getOrganizationUrlPrefix(props.organization)}.datadoghq.eu/dash/integration/489/apm-traces-estimated-usage?tpl_var_env={{env.name}} for more details.`;
|
|
23
|
+
const infrastructureUsageSpikeMonitors = [
|
|
24
|
+
{
|
|
25
|
+
humanReadableName: 'infrastructure hosts',
|
|
26
|
+
query: 'sum:datadog.estimated_usage.hosts{*}',
|
|
27
|
+
},
|
|
28
|
+
{
|
|
29
|
+
humanReadableName: "infrastructure hosts for '{{child_org_name.name}}'",
|
|
30
|
+
query: 'sum:datadog.estimated_usage.hosts{*} by {child_org_name}',
|
|
31
|
+
},
|
|
32
|
+
{ humanReadableName: 'Fargate tasks', query: 'sum:datadog.apm.fargate_task_instance{*}' },
|
|
33
|
+
{ humanReadableName: 'profiled Fargate tasks', query: 'sum:datadog.profiling.fargate{*}' },
|
|
34
|
+
{
|
|
35
|
+
humanReadableName: 'custom metrics',
|
|
36
|
+
query: 'sum:datadog.estimated_usage.metrics.custom{*}',
|
|
37
|
+
},
|
|
38
|
+
{
|
|
39
|
+
humanReadableName: "custom metrics for '{{child_org_name.name}}'",
|
|
40
|
+
query: 'sum:datadog.estimated_usage.metrics.custom{*} by {child_org_name}',
|
|
41
|
+
},
|
|
42
|
+
];
|
|
43
|
+
const logManagementUsageSpikeMonitors = [
|
|
44
|
+
{
|
|
45
|
+
humanReadableName: 'log ingestion',
|
|
46
|
+
query: 'sum:datadog.estimated_usage.logs.ingested_bytes{*}.as_count()',
|
|
47
|
+
},
|
|
48
|
+
{
|
|
49
|
+
humanReadableName: "log ingestion for '{{datadog_index.name}}'",
|
|
50
|
+
query: 'sum:datadog.estimated_usage.logs.ingested_bytes{*} by {datadog_index}.as_count()',
|
|
51
|
+
additionalMessage: additionalMessageForLogIndexScopedMonitors,
|
|
52
|
+
},
|
|
53
|
+
{
|
|
54
|
+
humanReadableName: 'log indexing',
|
|
55
|
+
query: 'sum:datadog.estimated_usage.logs.ingested_events{datadog_is_excluded:false}.as_count()',
|
|
56
|
+
},
|
|
57
|
+
{
|
|
58
|
+
humanReadableName: "log indexing for '{{datadog_index.name}}'",
|
|
59
|
+
query: 'sum:datadog.estimated_usage.logs.ingested_events{datadog_is_excluded:false} by {datadog_index}.as_count()',
|
|
60
|
+
additionalMessage: additionalMessageForLogIndexScopedMonitors,
|
|
61
|
+
},
|
|
62
|
+
];
|
|
63
|
+
const apmUsageSpikeMonitors = [
|
|
64
|
+
{ humanReadableName: 'APM hosts', query: 'sum:datadog.estimated_usage.apm_hosts{*}' },
|
|
65
|
+
{
|
|
66
|
+
humanReadableName: "APM hosts for '{{child_org_name.name}}'",
|
|
67
|
+
query: 'sum:datadog.estimated_usage.apm_hosts{*} by {child_org_name}',
|
|
68
|
+
},
|
|
69
|
+
{
|
|
70
|
+
humanReadableName: 'span ingestion',
|
|
71
|
+
query: 'sum:datadog.estimated_usage.apm.ingested_bytes{*}.as_count()',
|
|
72
|
+
},
|
|
73
|
+
{
|
|
74
|
+
humanReadableName: "span ingestion for '{{env.name}}'",
|
|
75
|
+
query: 'sum:datadog.estimated_usage.apm.ingested_bytes{*} by {env}.as_count()',
|
|
76
|
+
additionalMessage: additionalMessageForEnvScopedMonitors,
|
|
77
|
+
},
|
|
78
|
+
{
|
|
79
|
+
humanReadableName: 'span indexing',
|
|
80
|
+
query: 'sum:datadog.estimated_usage.apm.indexed_spans{*}.as_count()',
|
|
81
|
+
},
|
|
82
|
+
{
|
|
83
|
+
humanReadableName: "span indexing for '{{env.name}}'",
|
|
84
|
+
query: 'sum:datadog.estimated_usage.apm.indexed_spans{*} by {env}.as_count()',
|
|
85
|
+
additionalMessage: additionalMessageForEnvScopedMonitors,
|
|
86
|
+
},
|
|
87
|
+
];
|
|
88
|
+
const syntheticUsageSpikeMonitors = [
|
|
89
|
+
{
|
|
90
|
+
humanReadableName: 'Synthetic API tests',
|
|
91
|
+
query: 'sum:datadog.estimated_usage.synthetics.api_test_runs{*}.as_count()',
|
|
92
|
+
},
|
|
93
|
+
];
|
|
94
|
+
const serverlessUsageSpikeMonitors = [
|
|
95
|
+
{
|
|
96
|
+
humanReadableName: 'serverless functions',
|
|
97
|
+
query: 'sum:datadog.estimated_usage.serverless.aws_lambda_functions{*}',
|
|
98
|
+
},
|
|
99
|
+
{
|
|
100
|
+
humanReadableName: "serverless functions for '{{child_org_name.name}}'",
|
|
101
|
+
query: 'sum:datadog.estimated_usage.serverless.aws_lambda_functions{*} by {child_org_name}',
|
|
102
|
+
},
|
|
103
|
+
];
|
|
104
|
+
for (const usageSpikeMonitor of [].concat(infrastructureUsageSpikeMonitors, logManagementUsageSpikeMonitors, apmUsageSpikeMonitors, serverlessUsageSpikeMonitors, syntheticUsageSpikeMonitors)) {
|
|
105
|
+
this.createAnomalyMonitorForSpikes(props.serviceName, props.alertType, usageSpikeMonitor);
|
|
106
|
+
}
|
|
107
|
+
}
|
|
108
|
+
createAnomalyMonitorForSpikes(serviceName, alertType, usageSpikeMonitor) {
|
|
109
|
+
var _b;
|
|
110
|
+
return new datadogMonitor_1.DatadogMonitor(this, usageSpikeMonitor.humanReadableName, {
|
|
111
|
+
serviceName,
|
|
112
|
+
alertTypes: [alertType],
|
|
113
|
+
monitor: {
|
|
114
|
+
name: `${usageSpikeMonitor.humanReadableName} spike`,
|
|
115
|
+
type: 'query alert',
|
|
116
|
+
query: `avg(last_4h):anomalies(${usageSpikeMonitor.query}, \'agile\', 2, direction=\'above\', interval=60, alert_window=\'last_15m\', seasonality=\'weekly\', timezone=\'Europe/Berlin\', count_default_zero=\'true\') >= 1`,
|
|
117
|
+
message: `${usageSpikeMonitor.humanReadableName} are getting out-of-control. ${(_b = usageSpikeMonitor.additionalMessage) !== null && _b !== void 0 ? _b : ''}`,
|
|
118
|
+
options: {
|
|
119
|
+
include_tags: false,
|
|
120
|
+
notify_no_data: true,
|
|
121
|
+
threshold_windows: {
|
|
122
|
+
trigger_window: 'last_15m',
|
|
123
|
+
recovery_window: 'last_15m',
|
|
124
|
+
},
|
|
125
|
+
},
|
|
126
|
+
},
|
|
127
|
+
});
|
|
128
|
+
}
|
|
129
|
+
}
|
|
130
|
+
exports.UsageMonitoring = UsageMonitoring;
|
|
131
|
+
_a = JSII_RTTI_SYMBOL_1;
|
|
132
|
+
UsageMonitoring[_a] = { fqn: "@rio-cloud/cdk-v2-constructs.UsageMonitoring", version: "0.0.0" };
|
|
133
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"usage-monitoring.js","sourceRoot":"","sources":["../../../../src/contributions/team-claid/datadog/usage-monitoring.ts"],"names":[],"mappings":"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","sourcesContent":["import * as constructs from 'constructs';\nimport { DatadogAlertType, DatadogMonitor } from '../../../datadog/datadogMonitor';\n\nexport type DatadogOrganization = 'EU' | 'LATAM' | 'Landing Zone';\n\nfunction getOrganizationUrlPrefix(organization: DatadogOrganization): string {\n  switch (organization) {\n    case 'EU':\n      return 'app';\n    case 'LATAM':\n      return 'rio-brazil';\n    case 'Landing Zone':\n      return 'rio-landingzone';\n  }\n}\n\nexport interface UsageMonitoringProps {\n  readonly serviceName: string;\n  readonly alertType: DatadogAlertType;\n  readonly organization: DatadogOrganization;\n}\n\ninterface UsageSpikeMonitor {\n  humanReadableName: string;\n  query: string;\n  additionalMessage?: string;\n}\n\nexport class UsageMonitoring extends constructs.Construct {\n  constructor(scope: constructs.Construct, id: string, props: UsageMonitoringProps) {\n    super(scope, id);\n\n    const additionalMessageForLogIndexScopedMonitors = `Take a look at https://${getOrganizationUrlPrefix(\n      props.organization,\n    )}.datadoghq.eu/dash/integration/465/log-management-estimated-usage?tpl_var_datadog_index={{datadog_index.name}} for more details.`;\n    const additionalMessageForEnvScopedMonitors = `Take a look at https://${getOrganizationUrlPrefix(\n      props.organization,\n    )}.datadoghq.eu/dash/integration/489/apm-traces-estimated-usage?tpl_var_env={{env.name}} for more details.`;\n\n    const infrastructureUsageSpikeMonitors: UsageSpikeMonitor[] = [\n      {\n        humanReadableName: 'infrastructure hosts',\n        query: 'sum:datadog.estimated_usage.hosts{*}',\n      },\n      {\n        humanReadableName: \"infrastructure hosts for '{{child_org_name.name}}'\",\n        query: 'sum:datadog.estimated_usage.hosts{*} by {child_org_name}',\n      },\n      { humanReadableName: 'Fargate tasks', query: 'sum:datadog.apm.fargate_task_instance{*}' },\n      { humanReadableName: 'profiled Fargate tasks', query: 'sum:datadog.profiling.fargate{*}' },\n      {\n        humanReadableName: 'custom metrics',\n        query: 'sum:datadog.estimated_usage.metrics.custom{*}',\n      },\n      {\n        humanReadableName: \"custom metrics for '{{child_org_name.name}}'\",\n        query: 'sum:datadog.estimated_usage.metrics.custom{*} by {child_org_name}',\n      },\n    ];\n\n    const logManagementUsageSpikeMonitors: UsageSpikeMonitor[] = [\n      {\n        humanReadableName: 'log ingestion',\n        query: 'sum:datadog.estimated_usage.logs.ingested_bytes{*}.as_count()',\n      },\n      {\n        humanReadableName: \"log ingestion for '{{datadog_index.name}}'\",\n        query: 'sum:datadog.estimated_usage.logs.ingested_bytes{*} by {datadog_index}.as_count()',\n        additionalMessage: additionalMessageForLogIndexScopedMonitors,\n      },\n      {\n        humanReadableName: 'log indexing',\n        query: 'sum:datadog.estimated_usage.logs.ingested_events{datadog_is_excluded:false}.as_count()',\n      },\n      {\n        humanReadableName: \"log indexing for '{{datadog_index.name}}'\",\n        query:\n          'sum:datadog.estimated_usage.logs.ingested_events{datadog_is_excluded:false} by {datadog_index}.as_count()',\n        additionalMessage: additionalMessageForLogIndexScopedMonitors,\n      },\n    ];\n\n    const apmUsageSpikeMonitors: UsageSpikeMonitor[] = [\n      { humanReadableName: 'APM hosts', query: 'sum:datadog.estimated_usage.apm_hosts{*}' },\n      {\n        humanReadableName: \"APM hosts for '{{child_org_name.name}}'\",\n        query: 'sum:datadog.estimated_usage.apm_hosts{*} by {child_org_name}',\n      },\n      {\n        humanReadableName: 'span ingestion',\n        query: 'sum:datadog.estimated_usage.apm.ingested_bytes{*}.as_count()',\n      },\n      {\n        humanReadableName: \"span ingestion for '{{env.name}}'\",\n        query: 'sum:datadog.estimated_usage.apm.ingested_bytes{*} by {env}.as_count()',\n        additionalMessage: additionalMessageForEnvScopedMonitors,\n      },\n      {\n        humanReadableName: 'span indexing',\n        query: 'sum:datadog.estimated_usage.apm.indexed_spans{*}.as_count()',\n      },\n      {\n        humanReadableName: \"span indexing for '{{env.name}}'\",\n        query: 'sum:datadog.estimated_usage.apm.indexed_spans{*} by {env}.as_count()',\n        additionalMessage: additionalMessageForEnvScopedMonitors,\n      },\n    ];\n\n    const syntheticUsageSpikeMonitors: UsageSpikeMonitor[] = [\n      {\n        humanReadableName: 'Synthetic API tests',\n        query: 'sum:datadog.estimated_usage.synthetics.api_test_runs{*}.as_count()',\n      },\n    ];\n\n    const serverlessUsageSpikeMonitors: UsageSpikeMonitor[] = [\n      {\n        humanReadableName: 'serverless functions',\n        query: 'sum:datadog.estimated_usage.serverless.aws_lambda_functions{*}',\n      },\n      {\n        humanReadableName: \"serverless functions for '{{child_org_name.name}}'\",\n        query: 'sum:datadog.estimated_usage.serverless.aws_lambda_functions{*} by {child_org_name}',\n      },\n    ];\n\n    for (const usageSpikeMonitor of ([] as UsageSpikeMonitor[]).concat(\n      infrastructureUsageSpikeMonitors,\n      logManagementUsageSpikeMonitors,\n      apmUsageSpikeMonitors,\n      serverlessUsageSpikeMonitors,\n      syntheticUsageSpikeMonitors,\n    )) {\n      this.createAnomalyMonitorForSpikes(props.serviceName, props.alertType, usageSpikeMonitor);\n    }\n  }\n\n  private createAnomalyMonitorForSpikes(\n    serviceName: string,\n    alertType: DatadogAlertType,\n    usageSpikeMonitor: UsageSpikeMonitor,\n  ): DatadogMonitor {\n    return new DatadogMonitor(this, usageSpikeMonitor.humanReadableName, {\n      serviceName,\n      alertTypes: [alertType],\n      monitor: {\n        name: `${usageSpikeMonitor.humanReadableName} spike`,\n        type: 'query alert',\n        query: `avg(last_4h):anomalies(${usageSpikeMonitor.query}, \\'agile\\', 2, direction=\\'above\\', interval=60, alert_window=\\'last_15m\\', seasonality=\\'weekly\\', timezone=\\'Europe/Berlin\\', count_default_zero=\\'true\\') >= 1`,\n        message: `${usageSpikeMonitor.humanReadableName} are getting out-of-control. ${\n          usageSpikeMonitor.additionalMessage ?? ''\n        }`,\n        options: {\n          include_tags: false,\n          notify_no_data: true,\n          threshold_windows: {\n            trigger_window: 'last_15m',\n            recovery_window: 'last_15m',\n          },\n        },\n      },\n    });\n  }\n}\n"]}
|
|
@@ -41,6 +41,8 @@ class DatadogLogIndexMonitoring extends constructs_1.Construct {
|
|
|
41
41
|
exports.DatadogLogIndexMonitoring = DatadogLogIndexMonitoring;
|
|
42
42
|
_a = JSII_RTTI_SYMBOL_1;
|
|
43
43
|
DatadogLogIndexMonitoring[_a] = { fqn: "@rio-cloud/cdk-v2-constructs.DatadogLogIndexMonitoring", version: "0.0.0" };
|
|
44
|
+
const commonMonitorNamePrefix = (indexName) => `'${indexName}' index`;
|
|
45
|
+
const commonTags = (indexName) => [`logIndex:${indexName}`];
|
|
44
46
|
class DatadogLogDailyQuotaMonitor extends datadogMonitor_1.DatadogMonitor {
|
|
45
47
|
constructor(scope, id, props) {
|
|
46
48
|
const logQuotaInEvents = Math.floor(props.dailyLogQuota.valueInMillionEvents * 1000000);
|
|
@@ -50,7 +52,7 @@ class DatadogLogDailyQuotaMonitor extends datadogMonitor_1.DatadogMonitor {
|
|
|
50
52
|
serviceName: props.serviceName,
|
|
51
53
|
alertTypes: [props.alertType],
|
|
52
54
|
monitor: {
|
|
53
|
-
name:
|
|
55
|
+
name: `${commonMonitorNamePrefix(props.indexName)}: Log quota almost reached`,
|
|
54
56
|
type: 'metric alert',
|
|
55
57
|
query: `sum(last_1d):sum:datadog.estimated_usage.logs.ingested_events{datadog_is_excluded:false,datadog_index:${props.indexName}}.as_count() >= ${alertThreshold}`,
|
|
56
58
|
message: `[P3] Inspect your log volume at ${datadogLogsSearchWithIndexFilterDeepLink(props.indexName)} or adapt the log quota at ${datadogLogsIndexesConfigurationDeepLink}`,
|
|
@@ -63,6 +65,9 @@ class DatadogLogDailyQuotaMonitor extends datadogMonitor_1.DatadogMonitor {
|
|
|
63
65
|
notify_no_data: !props.sparseLogging,
|
|
64
66
|
no_data_timeframe: 24 * 60,
|
|
65
67
|
},
|
|
68
|
+
tags: [
|
|
69
|
+
...commonTags(props.indexName),
|
|
70
|
+
],
|
|
66
71
|
},
|
|
67
72
|
});
|
|
68
73
|
}
|
|
@@ -73,7 +78,7 @@ class DatadogLogsAnomalyMonitor extends datadogMonitor_1.DatadogMonitor {
|
|
|
73
78
|
serviceName: props.serviceName,
|
|
74
79
|
alertTypes: [props.alertType],
|
|
75
80
|
monitor: {
|
|
76
|
-
name:
|
|
81
|
+
name: `${commonMonitorNamePrefix(props.indexName)}: Log amount is rising to unexpected levels`,
|
|
77
82
|
type: 'metric alert',
|
|
78
83
|
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`,
|
|
79
84
|
message: `[P4] Logs are getting out of control: ${datadogLogsSearchWithIndexFilterDeepLink(props.indexName)}`,
|
|
@@ -85,6 +90,9 @@ class DatadogLogsAnomalyMonitor extends datadogMonitor_1.DatadogMonitor {
|
|
|
85
90
|
recovery_window: 'last_15m',
|
|
86
91
|
},
|
|
87
92
|
},
|
|
93
|
+
tags: [
|
|
94
|
+
...commonTags(props.indexName),
|
|
95
|
+
],
|
|
88
96
|
},
|
|
89
97
|
});
|
|
90
98
|
}
|
|
@@ -95,7 +103,7 @@ class DatadogLogDailyQuotaReachedMonitor extends datadogMonitor_1.DatadogMonitor
|
|
|
95
103
|
serviceName: props.serviceName,
|
|
96
104
|
alertTypes: [props.alertType],
|
|
97
105
|
monitor: {
|
|
98
|
-
name:
|
|
106
|
+
name: `${commonMonitorNamePrefix(props.indexName)}: Log quota reached`,
|
|
99
107
|
type: 'event-v2 alert',
|
|
100
108
|
query: `events("source:datadog datadog_index:${props.indexName} Daily quota reached").rollup("count").last("5m") > 0`,
|
|
101
109
|
message: `[P2] The log quote for the index has been reached. If you do not increase the quota at ${datadogLogsIndexesConfigurationDeepLink}, no new logs will be indexed until the quota is reset at 2:00pm UTC.`,
|
|
@@ -105,6 +113,9 @@ class DatadogLogDailyQuotaReachedMonitor extends datadogMonitor_1.DatadogMonitor
|
|
|
105
113
|
critical: 0,
|
|
106
114
|
},
|
|
107
115
|
},
|
|
116
|
+
tags: [
|
|
117
|
+
...commonTags(props.indexName),
|
|
118
|
+
],
|
|
108
119
|
},
|
|
109
120
|
});
|
|
110
121
|
}
|
|
@@ -113,4 +124,4 @@ function datadogLogsSearchWithIndexFilterDeepLink(indexName) {
|
|
|
113
124
|
return `https://app.datadoghq.eu/logs?index=${indexName}`;
|
|
114
125
|
}
|
|
115
126
|
const datadogLogsIndexesConfigurationDeepLink = 'https://app.datadoghq.eu/logs/pipelines/indexes';
|
|
116
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"datadogLogIndexMonitoring.js","sourceRoot":"","sources":["../../src/datadog/datadogLogIndexMonitoring.ts"],"names":[],"mappings":"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","sourcesContent":["import { Construct } from 'constructs';\nimport { DatadogMonitor, DatadogAlertType } from './datadogMonitor';\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\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\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: '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,\n        )} or adapt the log quota at ${datadogLogsIndexesConfigurationDeepLink}`,\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      },\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: '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,\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      },\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: 'Log quota reached',\n        type: 'event-v2 alert',\n        query: `events(\"source:datadog datadog_index:${props.indexName} Daily quota reached\").rollup(\"count\").last(\"5m\") > 0`,\n        message: `[P2] The log quote for the index has been reached. If you do not increase the quota at ${datadogLogsIndexesConfigurationDeepLink}, 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      },\n    });\n  }\n}\n\nfunction datadogLogsSearchWithIndexFilterDeepLink(indexName: string) {\n  return `https://app.datadoghq.eu/logs?index=${indexName}`;\n}\n\nconst datadogLogsIndexesConfigurationDeepLink = 'https://app.datadoghq.eu/logs/pipelines/indexes';\n"]}
|
|
127
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"datadogLogIndexMonitoring.js","sourceRoot":"","sources":["../../src/datadog/datadogLogIndexMonitoring.ts"],"names":[],"mappings":"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","sourcesContent":["import { Construct } from 'constructs';\nimport { DatadogMonitor, DatadogAlertType } from './datadogMonitor';\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\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,\n        )} or adapt the log quota at ${datadogLogsIndexesConfigurationDeepLink}`,\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,\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(\"5m\") > 0`,\n        message: `[P2] The log quote for the index has been reached. If you do not increase the quota at ${datadogLogsIndexesConfigurationDeepLink}, 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\nfunction datadogLogsSearchWithIndexFilterDeepLink(indexName: string) {\n  return `https://app.datadoghq.eu/logs?index=${indexName}`;\n}\n\nconst datadogLogsIndexesConfigurationDeepLink = 'https://app.datadoghq.eu/logs/pipelines/indexes';\n"]}
|
package/package.json
CHANGED
package/version.json
CHANGED