@rio-cloud/cdk-v2-constructs 2.6.2 → 2.7.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 CHANGED
@@ -3976,7 +3976,7 @@
3976
3976
  "stability": "stable"
3977
3977
  },
3978
3978
  "locationInModule": {
3979
- "filename": "src/datadog/datadogLogIndexMonitoring.ts",
3979
+ "filename": "src/datadog/datadog-log-index-monitoring.ts",
3980
3980
  "line": 70
3981
3981
  },
3982
3982
  "parameters": [
@@ -4002,11 +4002,11 @@
4002
4002
  },
4003
4003
  "kind": "class",
4004
4004
  "locationInModule": {
4005
- "filename": "src/datadog/datadogLogIndexMonitoring.ts",
4005
+ "filename": "src/datadog/datadog-log-index-monitoring.ts",
4006
4006
  "line": 69
4007
4007
  },
4008
4008
  "name": "DatadogLogIndexMonitoring",
4009
- "symbolId": "src/datadog/datadogLogIndexMonitoring:DatadogLogIndexMonitoring"
4009
+ "symbolId": "src/datadog/datadog-log-index-monitoring:DatadogLogIndexMonitoring"
4010
4010
  },
4011
4011
  "@rio-cloud/cdk-v2-constructs.DatadogLogIndexMonitoringProps": {
4012
4012
  "assembly": "@rio-cloud/cdk-v2-constructs",
@@ -4017,7 +4017,7 @@
4017
4017
  "fqn": "@rio-cloud/cdk-v2-constructs.DatadogLogIndexMonitoringProps",
4018
4018
  "kind": "interface",
4019
4019
  "locationInModule": {
4020
- "filename": "src/datadog/datadogLogIndexMonitoring.ts",
4020
+ "filename": "src/datadog/datadog-log-index-monitoring.ts",
4021
4021
  "line": 4
4022
4022
  },
4023
4023
  "name": "DatadogLogIndexMonitoringProps",
@@ -4031,7 +4031,7 @@
4031
4031
  },
4032
4032
  "immutable": true,
4033
4033
  "locationInModule": {
4034
- "filename": "src/datadog/datadogLogIndexMonitoring.ts",
4034
+ "filename": "src/datadog/datadog-log-index-monitoring.ts",
4035
4035
  "line": 20
4036
4036
  },
4037
4037
  "name": "alertType",
@@ -4047,7 +4047,7 @@
4047
4047
  },
4048
4048
  "immutable": true,
4049
4049
  "locationInModule": {
4050
- "filename": "src/datadog/datadogLogIndexMonitoring.ts",
4050
+ "filename": "src/datadog/datadog-log-index-monitoring.ts",
4051
4051
  "line": 25
4052
4052
  },
4053
4053
  "name": "dailyLogQuota",
@@ -4063,7 +4063,7 @@
4063
4063
  },
4064
4064
  "immutable": true,
4065
4065
  "locationInModule": {
4066
- "filename": "src/datadog/datadogLogIndexMonitoring.ts",
4066
+ "filename": "src/datadog/datadog-log-index-monitoring.ts",
4067
4067
  "line": 14
4068
4068
  },
4069
4069
  "name": "indexName",
@@ -4080,7 +4080,7 @@
4080
4080
  },
4081
4081
  "immutable": true,
4082
4082
  "locationInModule": {
4083
- "filename": "src/datadog/datadogLogIndexMonitoring.ts",
4083
+ "filename": "src/datadog/datadog-log-index-monitoring.ts",
4084
4084
  "line": 9
4085
4085
  },
4086
4086
  "name": "serviceName",
@@ -4097,7 +4097,7 @@
4097
4097
  },
4098
4098
  "immutable": true,
4099
4099
  "locationInModule": {
4100
- "filename": "src/datadog/datadogLogIndexMonitoring.ts",
4100
+ "filename": "src/datadog/datadog-log-index-monitoring.ts",
4101
4101
  "line": 38
4102
4102
  },
4103
4103
  "name": "organization",
@@ -4116,7 +4116,7 @@
4116
4116
  },
4117
4117
  "immutable": true,
4118
4118
  "locationInModule": {
4119
- "filename": "src/datadog/datadogLogIndexMonitoring.ts",
4119
+ "filename": "src/datadog/datadog-log-index-monitoring.ts",
4120
4120
  "line": 32
4121
4121
  },
4122
4122
  "name": "sparseLogging",
@@ -4126,7 +4126,7 @@
4126
4126
  }
4127
4127
  }
4128
4128
  ],
4129
- "symbolId": "src/datadog/datadogLogIndexMonitoring:DatadogLogIndexMonitoringProps"
4129
+ "symbolId": "src/datadog/datadog-log-index-monitoring:DatadogLogIndexMonitoringProps"
4130
4130
  },
4131
4131
  "@rio-cloud/cdk-v2-constructs.DatadogLogQuotaProps": {
4132
4132
  "assembly": "@rio-cloud/cdk-v2-constructs",
@@ -4137,7 +4137,7 @@
4137
4137
  "fqn": "@rio-cloud/cdk-v2-constructs.DatadogLogQuotaProps",
4138
4138
  "kind": "interface",
4139
4139
  "locationInModule": {
4140
- "filename": "src/datadog/datadogLogIndexMonitoring.ts",
4140
+ "filename": "src/datadog/datadog-log-index-monitoring.ts",
4141
4141
  "line": 41
4142
4142
  },
4143
4143
  "name": "DatadogLogQuotaProps",
@@ -4151,7 +4151,7 @@
4151
4151
  },
4152
4152
  "immutable": true,
4153
4153
  "locationInModule": {
4154
- "filename": "src/datadog/datadogLogIndexMonitoring.ts",
4154
+ "filename": "src/datadog/datadog-log-index-monitoring.ts",
4155
4155
  "line": 56
4156
4156
  },
4157
4157
  "name": "alertThresholdInPercent",
@@ -4167,7 +4167,7 @@
4167
4167
  },
4168
4168
  "immutable": true,
4169
4169
  "locationInModule": {
4170
- "filename": "src/datadog/datadogLogIndexMonitoring.ts",
4170
+ "filename": "src/datadog/datadog-log-index-monitoring.ts",
4171
4171
  "line": 45
4172
4172
  },
4173
4173
  "name": "valueInMillionEvents",
@@ -4183,7 +4183,7 @@
4183
4183
  },
4184
4184
  "immutable": true,
4185
4185
  "locationInModule": {
4186
- "filename": "src/datadog/datadogLogIndexMonitoring.ts",
4186
+ "filename": "src/datadog/datadog-log-index-monitoring.ts",
4187
4187
  "line": 50
4188
4188
  },
4189
4189
  "name": "warningThresholdInPercent",
@@ -4192,7 +4192,7 @@
4192
4192
  }
4193
4193
  }
4194
4194
  ],
4195
- "symbolId": "src/datadog/datadogLogIndexMonitoring:DatadogLogQuotaProps"
4195
+ "symbolId": "src/datadog/datadog-log-index-monitoring:DatadogLogQuotaProps"
4196
4196
  },
4197
4197
  "@rio-cloud/cdk-v2-constructs.DatadogUsageMonitoring": {
4198
4198
  "assembly": "@rio-cloud/cdk-v2-constructs",
@@ -5889,6 +5889,140 @@
5889
5889
  ],
5890
5890
  "symbolId": "src/pipeline/rio-bitbucket-source-action:RioBitBucketSourceActionProps"
5891
5891
  },
5892
+ "@rio-cloud/cdk-v2-constructs.RioCloudfrontDistribution": {
5893
+ "assembly": "@rio-cloud/cdk-v2-constructs",
5894
+ "base": "constructs.Construct",
5895
+ "docs": {
5896
+ "remarks": "Intended for delivery of html documents. It sets some defaults a recommended by AWS and provides basic security header.\nRequires installation of the custom-resource-certificate.",
5897
+ "stability": "stable",
5898
+ "summary": "A Construct which creates a default cloudfront distribution using a s3 bucket as origin."
5899
+ },
5900
+ "fqn": "@rio-cloud/cdk-v2-constructs.RioCloudfrontDistribution",
5901
+ "initializer": {
5902
+ "docs": {
5903
+ "stability": "stable"
5904
+ },
5905
+ "locationInModule": {
5906
+ "filename": "src/contributions/cop-frontend/cloudfront/rio-cloudfront-distribution.ts",
5907
+ "line": 70
5908
+ },
5909
+ "parameters": [
5910
+ {
5911
+ "name": "scope",
5912
+ "type": {
5913
+ "fqn": "constructs.Construct"
5914
+ }
5915
+ },
5916
+ {
5917
+ "name": "id",
5918
+ "type": {
5919
+ "primitive": "string"
5920
+ }
5921
+ },
5922
+ {
5923
+ "name": "props",
5924
+ "type": {
5925
+ "fqn": "@rio-cloud/cdk-v2-constructs.RioCloudfrontDistributionProps"
5926
+ }
5927
+ }
5928
+ ]
5929
+ },
5930
+ "kind": "class",
5931
+ "locationInModule": {
5932
+ "filename": "src/contributions/cop-frontend/cloudfront/rio-cloudfront-distribution.ts",
5933
+ "line": 69
5934
+ },
5935
+ "name": "RioCloudfrontDistribution",
5936
+ "symbolId": "src/contributions/cop-frontend/cloudfront/rio-cloudfront-distribution:RioCloudfrontDistribution"
5937
+ },
5938
+ "@rio-cloud/cdk-v2-constructs.RioCloudfrontDistributionProps": {
5939
+ "assembly": "@rio-cloud/cdk-v2-constructs",
5940
+ "datatype": true,
5941
+ "docs": {
5942
+ "stability": "stable",
5943
+ "summary": "Properties for defining a RioCloudfrontDistribution."
5944
+ },
5945
+ "fqn": "@rio-cloud/cdk-v2-constructs.RioCloudfrontDistributionProps",
5946
+ "kind": "interface",
5947
+ "locationInModule": {
5948
+ "filename": "src/contributions/cop-frontend/cloudfront/rio-cloudfront-distribution.ts",
5949
+ "line": 16
5950
+ },
5951
+ "name": "RioCloudfrontDistributionProps",
5952
+ "properties": [
5953
+ {
5954
+ "abstract": true,
5955
+ "docs": {
5956
+ "stability": "stable",
5957
+ "summary": "The reference to the S3 bucket containing the files to serve."
5958
+ },
5959
+ "immutable": true,
5960
+ "locationInModule": {
5961
+ "filename": "src/contributions/cop-frontend/cloudfront/rio-cloudfront-distribution.ts",
5962
+ "line": 20
5963
+ },
5964
+ "name": "contentBucketName",
5965
+ "type": {
5966
+ "primitive": "string"
5967
+ }
5968
+ },
5969
+ {
5970
+ "abstract": true,
5971
+ "docs": {
5972
+ "remarks": "If not set, a rio Default will be used. The default should cover connections to split.io, sentry and datadog,\nas well as a hash to support google tagManger. The latter might need to be adapted if you change the minification of\nthe index.html, in which case you can provide your csp here.",
5973
+ "stability": "stable",
5974
+ "summary": "The content Security Policy to be sent as a header when delivering the bucket content."
5975
+ },
5976
+ "immutable": true,
5977
+ "locationInModule": {
5978
+ "filename": "src/contributions/cop-frontend/cloudfront/rio-cloudfront-distribution.ts",
5979
+ "line": 40
5980
+ },
5981
+ "name": "contentSecurityPolicy",
5982
+ "optional": true,
5983
+ "type": {
5984
+ "primitive": "string"
5985
+ }
5986
+ },
5987
+ {
5988
+ "abstract": true,
5989
+ "docs": {
5990
+ "remarks": "If not given, the default hosted Zone\nprovided by RIO setup will be used.",
5991
+ "stability": "stable",
5992
+ "summary": "The hosted zone of the domain under which the content will be served."
5993
+ },
5994
+ "immutable": true,
5995
+ "locationInModule": {
5996
+ "filename": "src/contributions/cop-frontend/cloudfront/rio-cloudfront-distribution.ts",
5997
+ "line": 32
5998
+ },
5999
+ "name": "hostedZone",
6000
+ "optional": true,
6001
+ "type": {
6002
+ "fqn": "aws-cdk-lib.aws_route53.IHostedZone"
6003
+ }
6004
+ },
6005
+ {
6006
+ "abstract": true,
6007
+ "docs": {
6008
+ "remarks": "If not given, an empty string, i.e. no subDomain will be used.\nInstead, the name of the hosted zone will be used directly.",
6009
+ "stability": "stable",
6010
+ "summary": "The subdomain where the content should be served."
6011
+ },
6012
+ "immutable": true,
6013
+ "locationInModule": {
6014
+ "filename": "src/contributions/cop-frontend/cloudfront/rio-cloudfront-distribution.ts",
6015
+ "line": 26
6016
+ },
6017
+ "name": "subDomain",
6018
+ "optional": true,
6019
+ "type": {
6020
+ "primitive": "string"
6021
+ }
6022
+ }
6023
+ ],
6024
+ "symbolId": "src/contributions/cop-frontend/cloudfront/rio-cloudfront-distribution:RioCloudfrontDistributionProps"
6025
+ },
5892
6026
  "@rio-cloud/cdk-v2-constructs.RioFargateService": {
5893
6027
  "assembly": "@rio-cloud/cdk-v2-constructs",
5894
6028
  "base": "constructs.Construct",
@@ -8465,5 +8599,5 @@
8465
8599
  }
8466
8600
  },
8467
8601
  "version": "0.0.0",
8468
- "fingerprint": "dc1LbEEhAJZAZkrbkORSSXzh1HnX8Yh9603ZQb+s8VQ="
8602
+ "fingerprint": "XfxoSTqVvoXthMn4w1xNPq3Q2jsqW1fMpEwQGvyDGZg="
8469
8603
  }
package/CHANGELOG.md CHANGED
@@ -2,6 +2,16 @@
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.7.0](https://collaboration.msi.audi.com/stash/projects/RIODEV/repos/cdk-v2-constructs/compare/commits?targetBranch=refs%2Ftags%2Fv2.6.2&sourceBranch=refs%2Ftags%2Fv2.7.0) (2022-10-26)
6
+
7
+
8
+ ### Features
9
+
10
+ * **contributions:** Adapt documentation, fix typo ([422fb3c](https://collaboration.msi.audi.com/stash/projects/RIODEV/repos/cdk-v2-constructs/commits/422fb3c139faabc9e0b5acd7b0c017949d51078c))
11
+ * **contributions:** Add a cloudfront construct which can be used to deploy FEs via CDK with very little code if sticking to RIO defaults ([095d6e0](https://collaboration.msi.audi.com/stash/projects/RIODEV/repos/cdk-v2-constructs/commits/095d6e02d2fee71dfbe8f12d63159b3284dedaca))
12
+ * **contributions:** Add an option to easily adjust the script src of the csp without ([86f4543](https://collaboration.msi.audi.com/stash/projects/RIODEV/repos/cdk-v2-constructs/commits/86f4543af955a85f3c15a0074f4f096cb74d7b12))
13
+ * **contributions:** Preparing for a release with the new contirbutions ([04a2d81](https://collaboration.msi.audi.com/stash/projects/RIODEV/repos/cdk-v2-constructs/commits/04a2d81a295fdd3675314fc0ed3da7a7cbfc3db4))
14
+
5
15
  ### [2.6.2](https://collaboration.msi.audi.com/stash/projects/RIODEV/repos/cdk-v2-constructs/compare/commits?targetBranch=refs%2Ftags%2Fv2.6.1&sourceBranch=refs%2Ftags%2Fv2.6.2) (2022-09-29)
6
16
 
7
17
  ### [2.6.1](https://collaboration.msi.audi.com/stash/projects/RIODEV/repos/cdk-v2-constructs/compare/commits?targetBranch=refs%2Ftags%2Fv2.6.0&sourceBranch=refs%2Ftags%2Fv2.6.1) (2022-09-28)
@@ -0,0 +1 @@
1
+ export * from './rio-cloudfront-distribution';
@@ -0,0 +1,14 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ Object.defineProperty(o, k2, { enumerable: true, get: function() { return m[k]; } });
5
+ }) : (function(o, m, k, k2) {
6
+ if (k2 === undefined) k2 = k;
7
+ o[k2] = m[k];
8
+ }));
9
+ var __exportStar = (this && this.__exportStar) || function(m, exports) {
10
+ for (var p in m) if (p !== "default" && !exports.hasOwnProperty(p)) __createBinding(exports, m, p);
11
+ };
12
+ Object.defineProperty(exports, "__esModule", { value: true });
13
+ __exportStar(require("./rio-cloudfront-distribution"), exports);
14
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9zcmMvY29udHJpYnV0aW9ucy9jb3AtZnJvbnRlbmQvY2xvdWRmcm9udC9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7QUFBQSxnRUFBOEMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL3Jpby1jbG91ZGZyb250LWRpc3RyaWJ1dGlvbic7XG4iXX0=
@@ -0,0 +1,38 @@
1
+ import * as route53 from 'aws-cdk-lib/aws-route53';
2
+ import { Construct } from 'constructs';
3
+ /**
4
+ * Properties for defining a RioCloudfrontDistribution
5
+ */
6
+ export interface RioCloudfrontDistributionProps {
7
+ /**
8
+ * The reference to the S3 bucket containing the files to serve
9
+ */
10
+ readonly contentBucketName: string;
11
+ /**
12
+ * The subdomain where the content should be served. If not given, an empty string, i.e. no subDomain will be used.
13
+ * Instead, the name of the hosted zone will be used directly.
14
+ */
15
+ readonly subDomain?: string;
16
+ /**
17
+ * The hosted zone of the domain under which the content will be served. If not given, the default hosted Zone
18
+ * provided by RIO setup will be used.
19
+ */
20
+ readonly hostedZone?: route53.IHostedZone;
21
+ /**
22
+ * The content Security Policy to be sent as a header when delivering the bucket content.
23
+ * If not set, a rio Default will be used. The default should cover connections to split.io, sentry and datadog,
24
+ * as well as a hash to support google tagManger. The latter might need to be adapted if you change the minification of
25
+ * the index.html, in which case you can provide your csp here.
26
+ */
27
+ readonly contentSecurityPolicy?: string;
28
+ }
29
+ export declare const createRioDefaultCSP: (additionalScriptSources?: string | undefined) => string;
30
+ /**
31
+ * A Construct which creates a default cloudfront distribution using a s3 bucket as origin.
32
+ * Intended for delivery of html documents. It sets some defaults a recommended by AWS and provides basic security header.
33
+ * Requires installation of the custom-resource-certificate.
34
+ *
35
+ */
36
+ export declare class RioCloudfrontDistribution extends Construct {
37
+ constructor(scope: Construct, id: string, props: RioCloudfrontDistributionProps);
38
+ }
@@ -0,0 +1,159 @@
1
+ "use strict";
2
+ var _a;
3
+ Object.defineProperty(exports, "__esModule", { value: true });
4
+ exports.RioCloudfrontDistribution = exports.createRioDefaultCSP = void 0;
5
+ const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
6
+ const aws_cdk_lib_1 = require("aws-cdk-lib");
7
+ const acm = require("aws-cdk-lib/aws-certificatemanager");
8
+ const cloudfront = require("aws-cdk-lib/aws-cloudfront");
9
+ const origins = require("aws-cdk-lib/aws-cloudfront-origins");
10
+ const iam = require("aws-cdk-lib/aws-iam");
11
+ const route53 = require("aws-cdk-lib/aws-route53");
12
+ const targets = require("aws-cdk-lib/aws-route53-targets");
13
+ const s3 = require("aws-cdk-lib/aws-s3");
14
+ const constructs_1 = require("constructs");
15
+ const rio_landing_zone_1 = require("../../../rio-landing-zone");
16
+ const alwaysAllowedCSPSources = "'self' https://*.rio.cloud https://rio.cloud";
17
+ // The hash in script src is for the google tagManager script included in the index.html
18
+ exports.createRioDefaultCSP = (additionalScriptSources) => `
19
+ default-src ${alwaysAllowedCSPSources};
20
+ connect-src ${alwaysAllowedCSPSources} https://*.sentry.io https://sdk.split.io https://*.datadoghq.eu https://*.browser-intake-datadoghq.eu;
21
+ font-src ${alwaysAllowedCSPSources} data:;
22
+ img-src ${alwaysAllowedCSPSources} www.googletagmanager.com data:;
23
+ object-src 'none';
24
+ script-src ${alwaysAllowedCSPSources} https://www.googletagmanager.com
25
+ 'sha256-HDWAgfTeMAguT8wFw7+92Y7i4Eo/6RUQQuA3TW9JxbA='
26
+ ${additionalScriptSources ? ' ' + additionalScriptSources : ''};
27
+ style-src ${alwaysAllowedCSPSources} 'unsafe-inline';
28
+ base-uri 'self';
29
+ form-action ${alwaysAllowedCSPSources};
30
+ frame-ancestors ${alwaysAllowedCSPSources};
31
+ frame-src 'self' ${alwaysAllowedCSPSources};
32
+ `
33
+ .replace(/[\r\n\s]+/g, ' ')
34
+ .trim();
35
+ /**
36
+ * A Construct which creates a default cloudfront distribution using a s3 bucket as origin.
37
+ * Intended for delivery of html documents. It sets some defaults a recommended by AWS and provides basic security header.
38
+ * Requires installation of the custom-resource-certificate.
39
+ *
40
+ */
41
+ class RioCloudfrontDistribution extends constructs_1.Construct {
42
+ constructor(scope, id, props) {
43
+ var _b, _c, _d;
44
+ super(scope, id);
45
+ const hostedZone = (_b = props.hostedZone) !== null && _b !== void 0 ? _b : rio_landing_zone_1.RioLandingZone.getDefaultHostedZone(this);
46
+ const subDomain = (_c = props.subDomain) !== null && _c !== void 0 ? _c : '';
47
+ const domainName = props.subDomain === undefined ? hostedZone.zoneName : subDomain + '.' + hostedZone.zoneName;
48
+ const defaultContentSecurityPolicy = exports.createRioDefaultCSP();
49
+ const contentSecurityPolicy = (_d = props.contentSecurityPolicy) !== null && _d !== void 0 ? _d : defaultContentSecurityPolicy;
50
+ // Certificate created via Custom Resource due to other region
51
+ const certificate = new aws_cdk_lib_1.CustomResource(this, 'Certificate', {
52
+ resourceType: 'Custom::Certificate',
53
+ serviceToken: aws_cdk_lib_1.Fn.importValue('custom-resource-certificate-function-arn'),
54
+ properties: {
55
+ DomainName: domainName,
56
+ HostedZoneId: hostedZone.hostedZoneId,
57
+ Region: 'us-east-1',
58
+ },
59
+ });
60
+ const originAccessIdentity = new cloudfront.OriginAccessIdentity(this, 'OriginAccessIdentity', {
61
+ comment: `AccessIdentity for ${domainName}`,
62
+ });
63
+ const bucket = new s3.Bucket(this, 'FrontEndBucket', {
64
+ bucketName: props.contentBucketName,
65
+ encryption: s3.BucketEncryption.S3_MANAGED,
66
+ versioned: true,
67
+ removalPolicy: aws_cdk_lib_1.RemovalPolicy.DESTROY,
68
+ autoDeleteObjects: true,
69
+ lifecycleRules: [
70
+ {
71
+ id: 'ExpiresObjects',
72
+ expiration: aws_cdk_lib_1.Duration.days(90),
73
+ noncurrentVersionExpiration: aws_cdk_lib_1.Duration.days(30),
74
+ },
75
+ ],
76
+ enforceSSL: true,
77
+ cors: [
78
+ {
79
+ allowedMethods: [s3.HttpMethods.GET],
80
+ allowedOrigins: ['*'],
81
+ allowedHeaders: ['*'],
82
+ },
83
+ ],
84
+ });
85
+ bucket.grantRead(new iam.CanonicalUserPrincipal(originAccessIdentity.cloudFrontOriginAccessIdentityS3CanonicalUserId), '*');
86
+ const responseHeadersPolicy = new cloudfront.ResponseHeadersPolicy(this, 'ResponseHeadersPolicy', {
87
+ comment: 'Policy for CloudFront responses',
88
+ corsBehavior: {
89
+ accessControlAllowCredentials: false,
90
+ accessControlAllowHeaders: ['*'],
91
+ accessControlAllowMethods: ['GET'],
92
+ accessControlAllowOrigins: [hostedZone.zoneName],
93
+ accessControlMaxAge: aws_cdk_lib_1.Duration.seconds(600),
94
+ originOverride: true,
95
+ },
96
+ securityHeadersBehavior: {
97
+ contentSecurityPolicy: {
98
+ contentSecurityPolicy: contentSecurityPolicy,
99
+ override: true,
100
+ },
101
+ contentTypeOptions: { override: true },
102
+ frameOptions: { frameOption: cloudfront.HeadersFrameOption.DENY, override: true },
103
+ referrerPolicy: {
104
+ referrerPolicy: cloudfront.HeadersReferrerPolicy.STRICT_ORIGIN_WHEN_CROSS_ORIGIN,
105
+ override: true,
106
+ },
107
+ strictTransportSecurity: {
108
+ accessControlMaxAge: aws_cdk_lib_1.Duration.days(365),
109
+ includeSubdomains: true,
110
+ override: true,
111
+ },
112
+ xssProtection: {
113
+ protection: true,
114
+ modeBlock: true,
115
+ override: true,
116
+ },
117
+ },
118
+ });
119
+ const distribution = new cloudfront.Distribution(this, 'Distribution', {
120
+ defaultBehavior: {
121
+ origin: new origins.S3Origin(bucket),
122
+ originRequestPolicy: cloudfront.OriginRequestPolicy.CORS_S3_ORIGIN,
123
+ viewerProtocolPolicy: cloudfront.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,
124
+ cachePolicy: cloudfront.CachePolicy.CACHING_OPTIMIZED,
125
+ responseHeadersPolicy: responseHeadersPolicy,
126
+ compress: true,
127
+ },
128
+ domainNames: [domainName],
129
+ priceClass: cloudfront.PriceClass.PRICE_CLASS_100,
130
+ certificate: acm.Certificate.fromCertificateArn(this, 'AcmCertificate', certificate.ref),
131
+ defaultRootObject: 'index.html',
132
+ minimumProtocolVersion: cloudfront.SecurityPolicyProtocol.TLS_V1_2_2018,
133
+ errorResponses: [
134
+ {
135
+ httpStatus: 404,
136
+ responseHttpStatus: 200,
137
+ responsePagePath: '/index.html',
138
+ ttl: aws_cdk_lib_1.Duration.minutes(5),
139
+ },
140
+ {
141
+ httpStatus: 403,
142
+ responseHttpStatus: 200,
143
+ responsePagePath: '/index.html',
144
+ ttl: aws_cdk_lib_1.Duration.minutes(5),
145
+ },
146
+ ],
147
+ });
148
+ new route53.RecordSet(this, 'RecordSet', {
149
+ recordName: subDomain,
150
+ target: route53.RecordTarget.fromAlias(new targets.CloudFrontTarget(distribution)),
151
+ zone: hostedZone,
152
+ recordType: route53.RecordType.A,
153
+ });
154
+ }
155
+ }
156
+ exports.RioCloudfrontDistribution = RioCloudfrontDistribution;
157
+ _a = JSII_RTTI_SYMBOL_1;
158
+ RioCloudfrontDistribution[_a] = { fqn: "@rio-cloud/cdk-v2-constructs.RioCloudfrontDistribution", version: "0.0.0" };
159
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"rio-cloudfront-distribution.js","sourceRoot":"","sources":["../../../../src/contributions/cop-frontend/cloudfront/rio-cloudfront-distribution.ts"],"names":[],"mappings":"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","sourcesContent":["import { CustomResource, Duration, Fn, RemovalPolicy } from 'aws-cdk-lib';\nimport * as acm from 'aws-cdk-lib/aws-certificatemanager';\nimport * as cloudfront from 'aws-cdk-lib/aws-cloudfront';\nimport * as origins from 'aws-cdk-lib/aws-cloudfront-origins';\nimport * as iam from 'aws-cdk-lib/aws-iam';\nimport * as route53 from 'aws-cdk-lib/aws-route53';\nimport * as targets from 'aws-cdk-lib/aws-route53-targets';\nimport * as s3 from 'aws-cdk-lib/aws-s3';\nimport { Construct } from 'constructs';\nimport { RioLandingZone } from '../../../rio-landing-zone';\n\n\n/**\n * Properties for defining a RioCloudfrontDistribution\n */\nexport interface RioCloudfrontDistributionProps {\n  /**\n     * The reference to the S3 bucket containing the files to serve\n     */\n  readonly contentBucketName: string;\n\n  /**\n     * The subdomain where the content should be served. If not given, an empty string, i.e. no subDomain will be used.\n     * Instead, the name of the hosted zone will be used directly.\n     */\n  readonly subDomain?: string;\n\n  /**\n     * The hosted zone of the domain under which the content will be served. If not given, the default hosted Zone\n     * provided by RIO setup will be used.\n     */\n  readonly hostedZone?: route53.IHostedZone;\n\n  /**\n     * The content Security Policy to be sent as a header when delivering the bucket content.\n     * If not set, a rio Default will be used. The default should cover connections to split.io, sentry and datadog,\n     * as well as a hash to support google tagManger. The latter might need to be adapted if you change the minification of\n     * the index.html, in which case you can provide your csp here.\n     */\n  readonly contentSecurityPolicy?: string;\n}\n\nconst alwaysAllowedCSPSources = \"'self' https://*.rio.cloud https://rio.cloud\";\n// The hash in script src is for the google tagManager script included in the index.html\nexport const createRioDefaultCSP = (additionalScriptSources?: string) => `\n       default-src ${alwaysAllowedCSPSources};\n       connect-src ${alwaysAllowedCSPSources} https://*.sentry.io https://sdk.split.io https://*.datadoghq.eu https://*.browser-intake-datadoghq.eu;\n       font-src ${alwaysAllowedCSPSources} data:;\n       img-src ${alwaysAllowedCSPSources} www.googletagmanager.com data:;\n       object-src 'none';\n       script-src ${alwaysAllowedCSPSources} https://www.googletagmanager.com\n        'sha256-HDWAgfTeMAguT8wFw7+92Y7i4Eo/6RUQQuA3TW9JxbA='\n        ${additionalScriptSources ? ' ' + additionalScriptSources : ''};\n       style-src ${alwaysAllowedCSPSources} 'unsafe-inline';\n       base-uri 'self';\n       form-action ${alwaysAllowedCSPSources};\n       frame-ancestors ${alwaysAllowedCSPSources};\n       frame-src 'self' ${alwaysAllowedCSPSources};\n       `\n  .replace(/[\\r\\n\\s]+/g, ' ')\n  .trim();\n\n/**\n * A Construct which creates a default cloudfront distribution using a s3 bucket as origin.\n * Intended for delivery of html documents. It sets some defaults a recommended by AWS and provides basic security header.\n * Requires installation of the custom-resource-certificate.\n *\n */\nexport class RioCloudfrontDistribution extends Construct {\n  constructor(scope: Construct, id: string, props: RioCloudfrontDistributionProps) {\n    super(scope, id);\n\n    const hostedZone = props.hostedZone ?? RioLandingZone.getDefaultHostedZone(this);\n    const subDomain = props.subDomain ?? '';\n    const domainName = props.subDomain === undefined ? hostedZone.zoneName : subDomain + '.' + hostedZone.zoneName;\n\n    const defaultContentSecurityPolicy = createRioDefaultCSP();\n    const contentSecurityPolicy = props.contentSecurityPolicy ?? defaultContentSecurityPolicy;\n\n    // Certificate created via Custom Resource due to other region\n    const certificate = new CustomResource(this, 'Certificate', {\n      resourceType: 'Custom::Certificate',\n      serviceToken: Fn.importValue('custom-resource-certificate-function-arn'),\n      properties: {\n        DomainName: domainName,\n        HostedZoneId: hostedZone.hostedZoneId,\n        Region: 'us-east-1',\n      },\n    });\n\n    const originAccessIdentity = new cloudfront.OriginAccessIdentity(this, 'OriginAccessIdentity', {\n      comment: `AccessIdentity for ${domainName}`,\n    });\n\n    const bucket = new s3.Bucket(this, 'FrontEndBucket', {\n      bucketName: props.contentBucketName,\n      encryption: s3.BucketEncryption.S3_MANAGED,\n      versioned: true,\n      removalPolicy: RemovalPolicy.DESTROY,\n      autoDeleteObjects: true,\n      lifecycleRules: [\n        {\n          id: 'ExpiresObjects',\n          expiration: Duration.days(90),\n          noncurrentVersionExpiration: Duration.days(30),\n        },\n      ],\n      enforceSSL: true,\n      cors: [\n        {\n          allowedMethods: [s3.HttpMethods.GET],\n          allowedOrigins: ['*'],\n          allowedHeaders: ['*'],\n        },\n      ],\n    });\n    bucket.grantRead(\n      new iam.CanonicalUserPrincipal(originAccessIdentity.cloudFrontOriginAccessIdentityS3CanonicalUserId),\n      '*',\n    );\n\n    const responseHeadersPolicy = new cloudfront.ResponseHeadersPolicy(this, 'ResponseHeadersPolicy', {\n      comment: 'Policy for CloudFront responses',\n      corsBehavior: {\n        accessControlAllowCredentials: false,\n        accessControlAllowHeaders: ['*'],\n        accessControlAllowMethods: ['GET'],\n        accessControlAllowOrigins: [hostedZone.zoneName],\n        accessControlMaxAge: Duration.seconds(600),\n        originOverride: true,\n      },\n      securityHeadersBehavior: {\n        contentSecurityPolicy: {\n          contentSecurityPolicy: contentSecurityPolicy,\n          override: true,\n        },\n        contentTypeOptions: { override: true },\n        frameOptions: { frameOption: cloudfront.HeadersFrameOption.DENY, override: true },\n        referrerPolicy: {\n          referrerPolicy: cloudfront.HeadersReferrerPolicy.STRICT_ORIGIN_WHEN_CROSS_ORIGIN,\n          override: true,\n        },\n        strictTransportSecurity: {\n          accessControlMaxAge: Duration.days(365),\n          includeSubdomains: true,\n          override: true,\n        },\n        xssProtection: {\n          protection: true,\n          modeBlock: true,\n          override: true,\n        },\n      },\n    });\n\n    const distribution = new cloudfront.Distribution(this, 'Distribution', {\n      defaultBehavior: {\n        origin: new origins.S3Origin(bucket),\n        originRequestPolicy: cloudfront.OriginRequestPolicy.CORS_S3_ORIGIN,\n        viewerProtocolPolicy: cloudfront.ViewerProtocolPolicy.REDIRECT_TO_HTTPS,\n        cachePolicy: cloudfront.CachePolicy.CACHING_OPTIMIZED,\n        responseHeadersPolicy: responseHeadersPolicy,\n        compress: true,\n      },\n      domainNames: [domainName],\n      priceClass: cloudfront.PriceClass.PRICE_CLASS_100,\n      certificate: acm.Certificate.fromCertificateArn(this, 'AcmCertificate', certificate.ref),\n      defaultRootObject: 'index.html',\n      minimumProtocolVersion: cloudfront.SecurityPolicyProtocol.TLS_V1_2_2018,\n      errorResponses: [\n        {\n          httpStatus: 404,\n          responseHttpStatus: 200,\n          responsePagePath: '/index.html',\n          ttl: Duration.minutes(5),\n        },\n        {\n          httpStatus: 403,\n          responseHttpStatus: 200,\n          responsePagePath: '/index.html',\n          ttl: Duration.minutes(5),\n        },\n      ],\n    });\n\n    new route53.RecordSet(this, 'RecordSet', {\n      recordName: subDomain,\n      target: route53.RecordTarget.fromAlias(new targets.CloudFrontTarget(distribution)),\n      zone: hostedZone,\n      recordType: route53.RecordType.A,\n    });\n  }\n}\n"]}
@@ -1,6 +1,6 @@
1
1
  import * as constructs from 'constructs';
2
2
  import { DatadogOrganization } from '../../../datadog';
3
- import { DatadogAlertType } from '../../../datadog/datadogMonitor';
3
+ import { DatadogAlertType } from '../../../datadog/datadog-monitor';
4
4
  export interface DatadogUsageMonitoringProps {
5
5
  readonly serviceName: string;
6
6
  readonly alertType: DatadogAlertType;
@@ -7,7 +7,7 @@ const path = require("path");
7
7
  const ssm = require("aws-cdk-lib/aws-ssm");
8
8
  const constructs = require("constructs");
9
9
  const datadog_1 = require("../../../datadog");
10
- const datadogMonitor_1 = require("../../../datadog/datadogMonitor");
10
+ const datadog_monitor_1 = require("../../../datadog/datadog-monitor");
11
11
  const rio_claidometer_1 = require("../../../rio-claidometer");
12
12
  /**
13
13
  * This is a beta construct and is likely to have breaking changes.
@@ -115,7 +115,7 @@ class DatadogUsageMonitoring extends constructs.Construct {
115
115
  }
116
116
  createAnomalyMonitorForSpikes(serviceName, teamName, alertType, usageSpikeMonitor) {
117
117
  var _b;
118
- return new datadogMonitor_1.DatadogMonitor(this, usageSpikeMonitor.humanReadableName, {
118
+ return new datadog_monitor_1.DatadogMonitor(this, usageSpikeMonitor.humanReadableName, {
119
119
  serviceName,
120
120
  alertTypes: [alertType],
121
121
  monitor: {
@@ -151,4 +151,4 @@ function addClaidometer(scope) {
151
151
  version: '0.0.7',
152
152
  });
153
153
  }
154
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"datadog-usage-monitoring.js","sourceRoot":"","sources":["../../../../src/contributions/team-claid/datadog-usage-monitoring/datadog-usage-monitoring.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport * as ssm from 'aws-cdk-lib/aws-ssm';\nimport * as constructs from 'constructs';\nimport { DatadogOrganization, getOrganizationUrlPrefix } from '../../../datadog';\nimport { DatadogAlertType, DatadogMonitor } from '../../../datadog/datadogMonitor';\nimport { Claidometer, ClaidometerProduct } from '../../../rio-claidometer';\n\ninterface UsageSpikeMonitor {\n  humanReadableName: string;\n  query: string;\n  additionalMessage?: string;\n}\n\nexport interface DatadogUsageMonitoringProps {\n  readonly serviceName: string;\n  readonly alertType: DatadogAlertType;\n  readonly organization: DatadogOrganization;\n}\n\n/**\n * This is a beta construct and is likely to have breaking changes.\n * The construct creates monitoring (based on anomaly detection) for the usage of the most relevant Datadog resources\n */\nexport class DatadogUsageMonitoring extends constructs.Construct {\n  constructor(scope: constructs.Construct, id: string, props: DatadogUsageMonitoringProps) {\n    super(scope, id);\n\n    const teamName = ssm.StringParameter.fromStringParameterName(this, 'TeamName', '/config/team/name').stringValue;\n\n    const infrastructureUsageSpikeMonitors: UsageSpikeMonitor[] = [\n      {\n        humanReadableName: 'infrastructure hosts',\n        query: 'sum:datadog.estimated_usage.hosts{*}',\n      },\n      {\n        humanReadableName: \"infrastructure hosts for '{{child_org_name.name}}'\",\n        query: 'sum:datadog.estimated_usage.hosts{*} by {child_org_name}',\n      },\n      { humanReadableName: 'Fargate tasks', query: 'sum:datadog.apm.fargate_task_instance{*}' },\n      { humanReadableName: 'profiled Fargate tasks', query: 'sum:datadog.profiling.fargate{*}' },\n      {\n        humanReadableName: 'custom metrics',\n        query: 'sum:datadog.estimated_usage.metrics.custom{*}',\n      },\n      {\n        humanReadableName: \"custom metrics for '{{child_org_name.name}}'\",\n        query: 'sum:datadog.estimated_usage.metrics.custom{*} by {child_org_name}',\n      },\n    ];\n\n    const additionalMessageLogManagementForGloballyScopedMonitors = `Take a look at https://${getOrganizationUrlPrefix(\n      props.organization,\n    )}.datadoghq.eu/dash/integration/465/log-management-estimated-usage for more details.`;\n    const additionalMessageLogManagementForLogIndexScopedMonitors = `Take a look at https://${getOrganizationUrlPrefix(\n      props.organization,\n    )}.datadoghq.eu/dash/integration/465/log-management-estimated-usage?tpl_var_datadog_index={{datadog_index.name}} for more details.`;\n    const logManagementUsageSpikeMonitors: UsageSpikeMonitor[] = [\n      {\n        humanReadableName: 'log ingestion',\n        query: 'sum:datadog.estimated_usage.logs.ingested_bytes{*}.as_count()',\n        additionalMessage: additionalMessageLogManagementForGloballyScopedMonitors,\n      },\n      {\n        humanReadableName: \"log ingestion for '{{datadog_index.name}}'\",\n        query: 'sum:datadog.estimated_usage.logs.ingested_bytes{*} by {datadog_index}.as_count()',\n        additionalMessage: additionalMessageLogManagementForLogIndexScopedMonitors,\n      },\n      {\n        humanReadableName: 'log indexing',\n        query: 'sum:datadog.estimated_usage.logs.ingested_events{datadog_is_excluded:false}.as_count()',\n        additionalMessage: additionalMessageLogManagementForGloballyScopedMonitors,\n      },\n      {\n        humanReadableName: \"log indexing for '{{datadog_index.name}}'\",\n        query: 'sum:datadog.estimated_usage.logs.ingested_events{datadog_is_excluded:false} by {datadog_index}.as_count()',\n        additionalMessage: additionalMessageLogManagementForLogIndexScopedMonitors,\n      },\n    ];\n\n    const additionalMessageApmForGloballyScopedMonitors = `Take a look at https://${getOrganizationUrlPrefix(\n      props.organization,\n    )}.datadoghq.eu/dash/integration/489/apm-traces-estimated-usage for more details.`;\n    const additionalMessageApmForEnvScopedMonitors = `Take a look at https://${getOrganizationUrlPrefix(\n      props.organization,\n    )}.datadoghq.eu/dash/integration/489/apm-traces-estimated-usage?tpl_var_env={{env.name}} for more details.`;\n    const apmUsageSpikeMonitors: UsageSpikeMonitor[] = [\n      { humanReadableName: 'APM hosts', query: 'sum:datadog.estimated_usage.apm_hosts{*}' },\n      {\n        humanReadableName: \"APM hosts for '{{child_org_name.name}}'\",\n        query: 'sum:datadog.estimated_usage.apm_hosts{*} by {child_org_name}',\n        additionalMessage: additionalMessageApmForGloballyScopedMonitors,\n      },\n      {\n        humanReadableName: 'span ingestion',\n        query: 'sum:datadog.estimated_usage.apm.ingested_bytes{*}.as_count()',\n      },\n      {\n        humanReadableName: \"span ingestion for '{{env.name}}'\",\n        query: 'sum:datadog.estimated_usage.apm.ingested_bytes{*} by {env}.as_count()',\n        additionalMessage: additionalMessageApmForEnvScopedMonitors,\n      },\n      {\n        humanReadableName: 'span indexing',\n        query: 'sum:datadog.estimated_usage.apm.indexed_spans{*}.as_count()',\n        additionalMessage: additionalMessageApmForGloballyScopedMonitors,\n      },\n      {\n        humanReadableName: \"span indexing for '{{env.name}}'\",\n        query: 'sum:datadog.estimated_usage.apm.indexed_spans{*} by {env}.as_count()',\n        additionalMessage: additionalMessageApmForEnvScopedMonitors,\n      },\n    ];\n\n    const syntheticUsageSpikeMonitors: UsageSpikeMonitor[] = [\n      {\n        humanReadableName: 'Synthetic API tests',\n        query: 'sum:datadog.estimated_usage.synthetics.api_test_runs{*}.as_count()',\n      },\n    ];\n\n    const serverlessUsageSpikeMonitors: UsageSpikeMonitor[] = [\n      {\n        humanReadableName: 'serverless functions',\n        query: 'sum:datadog.estimated_usage.serverless.aws_lambda_functions{*}',\n      },\n      {\n        humanReadableName: \"serverless functions for '{{child_org_name.name}}'\",\n        query: 'sum:datadog.estimated_usage.serverless.aws_lambda_functions{*} by {child_org_name}',\n      },\n    ];\n\n    for (const usageSpikeMonitor of ([] as UsageSpikeMonitor[]).concat(\n      infrastructureUsageSpikeMonitors,\n      logManagementUsageSpikeMonitors,\n      apmUsageSpikeMonitors,\n      serverlessUsageSpikeMonitors,\n      syntheticUsageSpikeMonitors,\n    )) {\n      this.createAnomalyMonitorForSpikes(props.serviceName, teamName, props.alertType, usageSpikeMonitor);\n    }\n\n    if (props.organization === 'EU') {\n      addClaidometer(this);\n    }\n  }\n\n  private createAnomalyMonitorForSpikes(\n    serviceName: string,\n    teamName: string,\n    alertType: DatadogAlertType,\n    usageSpikeMonitor: UsageSpikeMonitor,\n  ): DatadogMonitor {\n    return new DatadogMonitor(this, usageSpikeMonitor.humanReadableName, {\n      serviceName,\n      alertTypes: [alertType],\n      monitor: {\n        name: `${usageSpikeMonitor.humanReadableName} spike`,\n        type: 'query alert',\n        query: `avg(last_12h):anomalies(${usageSpikeMonitor.query}, 'agile', 3, direction='above', interval=120, alert_window='last_30m', seasonality='weekly', timezone='europe/berlin', count_default_zero='true') >= 1`,\n        message: `[P4] ${usageSpikeMonitor.humanReadableName} are getting out-of-control. ${\n          usageSpikeMonitor.additionalMessage ?? ''\n        }`,\n        options: {\n          include_tags: false,\n          notify_no_data: false,\n          threshold_windows: {\n            trigger_window: 'last_30m',\n            recovery_window: 'last_15m',\n          },\n        },\n        tags: [\n          `service:${serviceName}`,\n          `team:${teamName}`,\n        ],\n      },\n    });\n  }\n}\n\nfunction addClaidometer(scope: constructs.Construct) {\n  const [feature, packageName] = __dirname.split(path.sep).reverse();\n  new Claidometer(scope, 'Claidometer', {\n    product: ClaidometerProduct.CDK_CONTRIBUTION,\n    packageName,\n    feature,\n    version: '0.0.7',\n  });\n}\n"]}
154
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"datadog-usage-monitoring.js","sourceRoot":"","sources":["../../../../src/contributions/team-claid/datadog-usage-monitoring/datadog-usage-monitoring.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport * as ssm from 'aws-cdk-lib/aws-ssm';\nimport * as constructs from 'constructs';\nimport { DatadogOrganization, getOrganizationUrlPrefix } from '../../../datadog';\nimport { DatadogAlertType, DatadogMonitor } from '../../../datadog/datadog-monitor';\nimport { Claidometer, ClaidometerProduct } from '../../../rio-claidometer';\n\ninterface UsageSpikeMonitor {\n  humanReadableName: string;\n  query: string;\n  additionalMessage?: string;\n}\n\nexport interface DatadogUsageMonitoringProps {\n  readonly serviceName: string;\n  readonly alertType: DatadogAlertType;\n  readonly organization: DatadogOrganization;\n}\n\n/**\n * This is a beta construct and is likely to have breaking changes.\n * The construct creates monitoring (based on anomaly detection) for the usage of the most relevant Datadog resources\n */\nexport class DatadogUsageMonitoring extends constructs.Construct {\n  constructor(scope: constructs.Construct, id: string, props: DatadogUsageMonitoringProps) {\n    super(scope, id);\n\n    const teamName = ssm.StringParameter.fromStringParameterName(this, 'TeamName', '/config/team/name').stringValue;\n\n    const infrastructureUsageSpikeMonitors: UsageSpikeMonitor[] = [\n      {\n        humanReadableName: 'infrastructure hosts',\n        query: 'sum:datadog.estimated_usage.hosts{*}',\n      },\n      {\n        humanReadableName: \"infrastructure hosts for '{{child_org_name.name}}'\",\n        query: 'sum:datadog.estimated_usage.hosts{*} by {child_org_name}',\n      },\n      { humanReadableName: 'Fargate tasks', query: 'sum:datadog.apm.fargate_task_instance{*}' },\n      { humanReadableName: 'profiled Fargate tasks', query: 'sum:datadog.profiling.fargate{*}' },\n      {\n        humanReadableName: 'custom metrics',\n        query: 'sum:datadog.estimated_usage.metrics.custom{*}',\n      },\n      {\n        humanReadableName: \"custom metrics for '{{child_org_name.name}}'\",\n        query: 'sum:datadog.estimated_usage.metrics.custom{*} by {child_org_name}',\n      },\n    ];\n\n    const additionalMessageLogManagementForGloballyScopedMonitors = `Take a look at https://${getOrganizationUrlPrefix(\n      props.organization,\n    )}.datadoghq.eu/dash/integration/465/log-management-estimated-usage for more details.`;\n    const additionalMessageLogManagementForLogIndexScopedMonitors = `Take a look at https://${getOrganizationUrlPrefix(\n      props.organization,\n    )}.datadoghq.eu/dash/integration/465/log-management-estimated-usage?tpl_var_datadog_index={{datadog_index.name}} for more details.`;\n    const logManagementUsageSpikeMonitors: UsageSpikeMonitor[] = [\n      {\n        humanReadableName: 'log ingestion',\n        query: 'sum:datadog.estimated_usage.logs.ingested_bytes{*}.as_count()',\n        additionalMessage: additionalMessageLogManagementForGloballyScopedMonitors,\n      },\n      {\n        humanReadableName: \"log ingestion for '{{datadog_index.name}}'\",\n        query: 'sum:datadog.estimated_usage.logs.ingested_bytes{*} by {datadog_index}.as_count()',\n        additionalMessage: additionalMessageLogManagementForLogIndexScopedMonitors,\n      },\n      {\n        humanReadableName: 'log indexing',\n        query: 'sum:datadog.estimated_usage.logs.ingested_events{datadog_is_excluded:false}.as_count()',\n        additionalMessage: additionalMessageLogManagementForGloballyScopedMonitors,\n      },\n      {\n        humanReadableName: \"log indexing for '{{datadog_index.name}}'\",\n        query: 'sum:datadog.estimated_usage.logs.ingested_events{datadog_is_excluded:false} by {datadog_index}.as_count()',\n        additionalMessage: additionalMessageLogManagementForLogIndexScopedMonitors,\n      },\n    ];\n\n    const additionalMessageApmForGloballyScopedMonitors = `Take a look at https://${getOrganizationUrlPrefix(\n      props.organization,\n    )}.datadoghq.eu/dash/integration/489/apm-traces-estimated-usage for more details.`;\n    const additionalMessageApmForEnvScopedMonitors = `Take a look at https://${getOrganizationUrlPrefix(\n      props.organization,\n    )}.datadoghq.eu/dash/integration/489/apm-traces-estimated-usage?tpl_var_env={{env.name}} for more details.`;\n    const apmUsageSpikeMonitors: UsageSpikeMonitor[] = [\n      { humanReadableName: 'APM hosts', query: 'sum:datadog.estimated_usage.apm_hosts{*}' },\n      {\n        humanReadableName: \"APM hosts for '{{child_org_name.name}}'\",\n        query: 'sum:datadog.estimated_usage.apm_hosts{*} by {child_org_name}',\n        additionalMessage: additionalMessageApmForGloballyScopedMonitors,\n      },\n      {\n        humanReadableName: 'span ingestion',\n        query: 'sum:datadog.estimated_usage.apm.ingested_bytes{*}.as_count()',\n      },\n      {\n        humanReadableName: \"span ingestion for '{{env.name}}'\",\n        query: 'sum:datadog.estimated_usage.apm.ingested_bytes{*} by {env}.as_count()',\n        additionalMessage: additionalMessageApmForEnvScopedMonitors,\n      },\n      {\n        humanReadableName: 'span indexing',\n        query: 'sum:datadog.estimated_usage.apm.indexed_spans{*}.as_count()',\n        additionalMessage: additionalMessageApmForGloballyScopedMonitors,\n      },\n      {\n        humanReadableName: \"span indexing for '{{env.name}}'\",\n        query: 'sum:datadog.estimated_usage.apm.indexed_spans{*} by {env}.as_count()',\n        additionalMessage: additionalMessageApmForEnvScopedMonitors,\n      },\n    ];\n\n    const syntheticUsageSpikeMonitors: UsageSpikeMonitor[] = [\n      {\n        humanReadableName: 'Synthetic API tests',\n        query: 'sum:datadog.estimated_usage.synthetics.api_test_runs{*}.as_count()',\n      },\n    ];\n\n    const serverlessUsageSpikeMonitors: UsageSpikeMonitor[] = [\n      {\n        humanReadableName: 'serverless functions',\n        query: 'sum:datadog.estimated_usage.serverless.aws_lambda_functions{*}',\n      },\n      {\n        humanReadableName: \"serverless functions for '{{child_org_name.name}}'\",\n        query: 'sum:datadog.estimated_usage.serverless.aws_lambda_functions{*} by {child_org_name}',\n      },\n    ];\n\n    for (const usageSpikeMonitor of ([] as UsageSpikeMonitor[]).concat(\n      infrastructureUsageSpikeMonitors,\n      logManagementUsageSpikeMonitors,\n      apmUsageSpikeMonitors,\n      serverlessUsageSpikeMonitors,\n      syntheticUsageSpikeMonitors,\n    )) {\n      this.createAnomalyMonitorForSpikes(props.serviceName, teamName, props.alertType, usageSpikeMonitor);\n    }\n\n    if (props.organization === 'EU') {\n      addClaidometer(this);\n    }\n  }\n\n  private createAnomalyMonitorForSpikes(\n    serviceName: string,\n    teamName: string,\n    alertType: DatadogAlertType,\n    usageSpikeMonitor: UsageSpikeMonitor,\n  ): DatadogMonitor {\n    return new DatadogMonitor(this, usageSpikeMonitor.humanReadableName, {\n      serviceName,\n      alertTypes: [alertType],\n      monitor: {\n        name: `${usageSpikeMonitor.humanReadableName} spike`,\n        type: 'query alert',\n        query: `avg(last_12h):anomalies(${usageSpikeMonitor.query}, 'agile', 3, direction='above', interval=120, alert_window='last_30m', seasonality='weekly', timezone='europe/berlin', count_default_zero='true') >= 1`,\n        message: `[P4] ${usageSpikeMonitor.humanReadableName} are getting out-of-control. ${\n          usageSpikeMonitor.additionalMessage ?? ''\n        }`,\n        options: {\n          include_tags: false,\n          notify_no_data: false,\n          threshold_windows: {\n            trigger_window: 'last_30m',\n            recovery_window: 'last_15m',\n          },\n        },\n        tags: [\n          `service:${serviceName}`,\n          `team:${teamName}`,\n        ],\n      },\n    });\n  }\n}\n\nfunction addClaidometer(scope: constructs.Construct) {\n  const [feature, packageName] = __dirname.split(path.sep).reverse();\n  new Claidometer(scope, 'Claidometer', {\n    product: ClaidometerProduct.CDK_CONTRIBUTION,\n    packageName,\n    feature,\n    version: '0.0.7',\n  });\n}\n"]}
@@ -1,6 +1,6 @@
1
1
  import { Reference } from 'aws-cdk-lib';
2
2
  import { Construct } from 'constructs';
3
- import { DatadogMonitor } from '../../../datadog/datadogMonitor';
3
+ import { DatadogMonitor } from '../../../datadog/datadog-monitor';
4
4
  interface DatadogPipelineMonitorProps {
5
5
  pipelineName: string | Reference;
6
6
  serviceName: string;
@@ -1,8 +1,8 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
3
  exports.DataDogPipelineErrorAlertForVulnerabilityChecks = exports.DatadogPipelineErrorAlert = void 0;
4
- const datadogMonitor_1 = require("../../../datadog/datadogMonitor");
5
- class DatadogPipelineErrorAlert extends datadogMonitor_1.DatadogMonitor {
4
+ const datadog_monitor_1 = require("../../../datadog/datadog-monitor");
5
+ class DatadogPipelineErrorAlert extends datadog_monitor_1.DatadogMonitor {
6
6
  constructor(scope, id, props) {
7
7
  super(scope, id, {
8
8
  serviceName: props.serviceName,
@@ -20,7 +20,7 @@ class DatadogPipelineErrorAlert extends datadogMonitor_1.DatadogMonitor {
20
20
  }
21
21
  }
22
22
  exports.DatadogPipelineErrorAlert = DatadogPipelineErrorAlert;
23
- class DataDogPipelineErrorAlertForVulnerabilityChecks extends datadogMonitor_1.DatadogMonitor {
23
+ class DataDogPipelineErrorAlertForVulnerabilityChecks extends datadog_monitor_1.DatadogMonitor {
24
24
  constructor(scope, id, props) {
25
25
  super(scope, id, {
26
26
  serviceName: props.serviceName,
@@ -39,4 +39,4 @@ class DataDogPipelineErrorAlertForVulnerabilityChecks extends datadogMonitor_1.D
39
39
  }
40
40
  }
41
41
  exports.DataDogPipelineErrorAlertForVulnerabilityChecks = DataDogPipelineErrorAlertForVulnerabilityChecks;
42
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGF0YWRvZy1tb25pdG9ycy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9jb250cmlidXRpb25zL3RlYW0tdHJhbnNwb3J0LXR3by9waXBlbGluZS9kYXRhZG9nLW1vbml0b3JzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUVBLG9FQUFpRTtBQVFqRSxNQUFhLHlCQUEwQixTQUFRLCtCQUFjO0lBQzNELFlBQVksS0FBZ0IsRUFBRSxFQUFVLEVBQUUsS0FBa0M7UUFDMUUsS0FBSyxDQUFDLEtBQUssRUFBRSxFQUFFLEVBQUU7WUFDZixXQUFXLEVBQUUsS0FBSyxDQUFDLFdBQVc7WUFDOUIsT0FBTyxFQUFFO2dCQUNQLElBQUksRUFBRSxhQUFhO2dCQUNuQixJQUFJLEVBQUUsQ0FBQyxNQUFNLENBQUM7Z0JBQ2QsS0FBSyxFQUFFLHFFQUFxRSxLQUFLLENBQUMsWUFBWSxlQUFlLEtBQUssQ0FBQyxTQUFTLE9BQU87Z0JBQ25JLElBQUksRUFBRSxvQkFBb0I7Z0JBQzFCLE9BQU8sRUFBRSwrSUFBK0k7Z0JBQ3hKLE9BQU8sRUFBRTtvQkFDUCxtQkFBbUIsRUFBRSxLQUFLO2lCQUMzQjthQUNGO1NBQ0YsQ0FBQyxDQUFDO0lBQ0wsQ0FBQztDQUNGO0FBaEJELDhEQWdCQztBQUVELE1BQWEsK0NBQWdELFNBQVEsK0JBQWM7SUFDakYsWUFBWSxLQUFnQixFQUFFLEVBQVUsRUFBRSxLQUFrQztRQUMxRSxLQUFLLENBQUMsS0FBSyxFQUFFLEVBQUUsRUFBRTtZQUNmLFdBQVcsRUFBRSxLQUFLLENBQUMsV0FBVztZQUM5QixVQUFVLEVBQUUsQ0FBQyxVQUFVLENBQUM7WUFDeEIsT0FBTyxFQUFFO2dCQUNQLElBQUksRUFBRSxhQUFhO2dCQUNuQixJQUFJLEVBQUUsQ0FBQyxNQUFNLENBQUM7Z0JBQ2QsS0FBSyxFQUFFLHFFQUFxRSxLQUFLLENBQUMsWUFBWSxlQUFlLEtBQUssQ0FBQyxTQUFTLE9BQU87Z0JBQ25JLElBQUksRUFBRSxzQkFBc0I7Z0JBQzVCLE9BQU8sRUFBRSwrQkFBK0I7Z0JBQ3hDLE9BQU8sRUFBRTtvQkFDUCxtQkFBbUIsRUFBRSxLQUFLO2lCQUMzQjthQUNGO1NBQ0YsQ0FBQyxDQUFDO0lBQ0wsQ0FBQztDQUNGO0FBakJELDBHQWlCQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IFJlZmVyZW5jZSB9IGZyb20gJ2F3cy1jZGstbGliJztcbmltcG9ydCB7IENvbnN0cnVjdCB9IGZyb20gJ2NvbnN0cnVjdHMnO1xuaW1wb3J0IHsgRGF0YWRvZ01vbml0b3IgfSBmcm9tICcuLi8uLi8uLi9kYXRhZG9nL2RhdGFkb2dNb25pdG9yJztcblxuaW50ZXJmYWNlIERhdGFkb2dQaXBlbGluZU1vbml0b3JQcm9wcyB7XG4gIHBpcGVsaW5lTmFtZTogc3RyaW5nIHwgUmVmZXJlbmNlO1xuICBzZXJ2aWNlTmFtZTogc3RyaW5nO1xuICBhY2NvdW50SWQ6IHN0cmluZztcbn1cblxuZXhwb3J0IGNsYXNzIERhdGFkb2dQaXBlbGluZUVycm9yQWxlcnQgZXh0ZW5kcyBEYXRhZG9nTW9uaXRvciB7XG4gIGNvbnN0cnVjdG9yKHNjb3BlOiBDb25zdHJ1Y3QsIGlkOiBzdHJpbmcsIHByb3BzOiBEYXRhZG9nUGlwZWxpbmVNb25pdG9yUHJvcHMpIHtcbiAgICBzdXBlcihzY29wZSwgaWQsIHtcbiAgICAgIHNlcnZpY2VOYW1lOiBwcm9wcy5zZXJ2aWNlTmFtZSxcbiAgICAgIG1vbml0b3I6IHtcbiAgICAgICAgdHlwZTogJ3F1ZXJ5IGFsZXJ0JyxcbiAgICAgICAgdGFnczogWydjaWNkJ10sXG4gICAgICAgIHF1ZXJ5OiBgbWluKGxhc3RfNW0pOm1pbjpkZXBsb3ltZW50LmNvZGVwaXBlbGluZS5leGVjdXRpb25ze3BpcGVsaW5lX25hbWU6JHtwcm9wcy5waXBlbGluZU5hbWV9LGFjY291bnRfaWQ6JHtwcm9wcy5hY2NvdW50SWR9fSA8IDFgLFxuICAgICAgICBuYW1lOiAnQ0kgcGlwZWxpbmUgc3RhdHVzJyxcbiAgICAgICAgbWVzc2FnZTogJ1tQNV0ge3sjaXNfYWxlcnR9fVxcXFxuVGhlIENJIGJ1aWxkIGlzIGJyb2tlblxcXFxue3svaXNfYWxlcnR9fSBcXFxcblxcXFxue3sjaXNfYWxlcnRfcmVjb3Zlcnl9fVxcXFxuVGhlIENJIGJ1aWxkIHdvcmtzIGFnYWluLlxcXFxue3svaXNfYWxlcnRfcmVjb3Zlcnl9fScsXG4gICAgICAgIG9wdGlvbnM6IHtcbiAgICAgICAgICByZXF1aXJlX2Z1bGxfd2luZG93OiBmYWxzZSxcbiAgICAgICAgfSxcbiAgICAgIH0sXG4gICAgfSk7XG4gIH1cbn1cblxuZXhwb3J0IGNsYXNzIERhdGFEb2dQaXBlbGluZUVycm9yQWxlcnRGb3JWdWxuZXJhYmlsaXR5Q2hlY2tzIGV4dGVuZHMgRGF0YWRvZ01vbml0b3Ige1xuICBjb25zdHJ1Y3RvcihzY29wZTogQ29uc3RydWN0LCBpZDogc3RyaW5nLCBwcm9wczogRGF0YWRvZ1BpcGVsaW5lTW9uaXRvclByb3BzKSB7XG4gICAgc3VwZXIoc2NvcGUsIGlkLCB7XG4gICAgICBzZXJ2aWNlTmFtZTogcHJvcHMuc2VydmljZU5hbWUsXG4gICAgICBhbGVydFR5cGVzOiBbJ29wc2dlbmllJ10sXG4gICAgICBtb25pdG9yOiB7XG4gICAgICAgIHR5cGU6ICdxdWVyeSBhbGVydCcsXG4gICAgICAgIHRhZ3M6IFsnY2ljZCddLFxuICAgICAgICBxdWVyeTogYG1pbihsYXN0XzVtKTptaW46ZGVwbG95bWVudC5jb2RlcGlwZWxpbmUuZXhlY3V0aW9uc3twaXBlbGluZV9uYW1lOiR7cHJvcHMucGlwZWxpbmVOYW1lfSxhY2NvdW50X2lkOiR7cHJvcHMuYWNjb3VudElkfX0gPCAxYCxcbiAgICAgICAgbmFtZTogJ1Z1bG5lcmFiaWxpdHkgY2hlY2tzJyxcbiAgICAgICAgbWVzc2FnZTogJ1tQM10gVnVsbmVyYWJpbGl0aWVzIGRldGVjdGVkJyxcbiAgICAgICAgb3B0aW9uczoge1xuICAgICAgICAgIHJlcXVpcmVfZnVsbF93aW5kb3c6IGZhbHNlLFxuICAgICAgICB9LFxuICAgICAgfSxcbiAgICB9KTtcbiAgfVxufVxuIl19
42
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGF0YWRvZy1tb25pdG9ycy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9jb250cmlidXRpb25zL3RlYW0tdHJhbnNwb3J0LXR3by9waXBlbGluZS9kYXRhZG9nLW1vbml0b3JzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUVBLHNFQUFrRTtBQVFsRSxNQUFhLHlCQUEwQixTQUFRLGdDQUFjO0lBQzNELFlBQVksS0FBZ0IsRUFBRSxFQUFVLEVBQUUsS0FBa0M7UUFDMUUsS0FBSyxDQUFDLEtBQUssRUFBRSxFQUFFLEVBQUU7WUFDZixXQUFXLEVBQUUsS0FBSyxDQUFDLFdBQVc7WUFDOUIsT0FBTyxFQUFFO2dCQUNQLElBQUksRUFBRSxhQUFhO2dCQUNuQixJQUFJLEVBQUUsQ0FBQyxNQUFNLENBQUM7Z0JBQ2QsS0FBSyxFQUFFLHFFQUFxRSxLQUFLLENBQUMsWUFBWSxlQUFlLEtBQUssQ0FBQyxTQUFTLE9BQU87Z0JBQ25JLElBQUksRUFBRSxvQkFBb0I7Z0JBQzFCLE9BQU8sRUFBRSwrSUFBK0k7Z0JBQ3hKLE9BQU8sRUFBRTtvQkFDUCxtQkFBbUIsRUFBRSxLQUFLO2lCQUMzQjthQUNGO1NBQ0YsQ0FBQyxDQUFDO0lBQ0wsQ0FBQztDQUNGO0FBaEJELDhEQWdCQztBQUVELE1BQWEsK0NBQWdELFNBQVEsZ0NBQWM7SUFDakYsWUFBWSxLQUFnQixFQUFFLEVBQVUsRUFBRSxLQUFrQztRQUMxRSxLQUFLLENBQUMsS0FBSyxFQUFFLEVBQUUsRUFBRTtZQUNmLFdBQVcsRUFBRSxLQUFLLENBQUMsV0FBVztZQUM5QixVQUFVLEVBQUUsQ0FBQyxVQUFVLENBQUM7WUFDeEIsT0FBTyxFQUFFO2dCQUNQLElBQUksRUFBRSxhQUFhO2dCQUNuQixJQUFJLEVBQUUsQ0FBQyxNQUFNLENBQUM7Z0JBQ2QsS0FBSyxFQUFFLHFFQUFxRSxLQUFLLENBQUMsWUFBWSxlQUFlLEtBQUssQ0FBQyxTQUFTLE9BQU87Z0JBQ25JLElBQUksRUFBRSxzQkFBc0I7Z0JBQzVCLE9BQU8sRUFBRSwrQkFBK0I7Z0JBQ3hDLE9BQU8sRUFBRTtvQkFDUCxtQkFBbUIsRUFBRSxLQUFLO2lCQUMzQjthQUNGO1NBQ0YsQ0FBQyxDQUFDO0lBQ0wsQ0FBQztDQUNGO0FBakJELDBHQWlCQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IFJlZmVyZW5jZSB9IGZyb20gJ2F3cy1jZGstbGliJztcbmltcG9ydCB7IENvbnN0cnVjdCB9IGZyb20gJ2NvbnN0cnVjdHMnO1xuaW1wb3J0IHsgRGF0YWRvZ01vbml0b3IgfSBmcm9tICcuLi8uLi8uLi9kYXRhZG9nL2RhdGFkb2ctbW9uaXRvcic7XG5cbmludGVyZmFjZSBEYXRhZG9nUGlwZWxpbmVNb25pdG9yUHJvcHMge1xuICBwaXBlbGluZU5hbWU6IHN0cmluZyB8IFJlZmVyZW5jZTtcbiAgc2VydmljZU5hbWU6IHN0cmluZztcbiAgYWNjb3VudElkOiBzdHJpbmc7XG59XG5cbmV4cG9ydCBjbGFzcyBEYXRhZG9nUGlwZWxpbmVFcnJvckFsZXJ0IGV4dGVuZHMgRGF0YWRvZ01vbml0b3Ige1xuICBjb25zdHJ1Y3RvcihzY29wZTogQ29uc3RydWN0LCBpZDogc3RyaW5nLCBwcm9wczogRGF0YWRvZ1BpcGVsaW5lTW9uaXRvclByb3BzKSB7XG4gICAgc3VwZXIoc2NvcGUsIGlkLCB7XG4gICAgICBzZXJ2aWNlTmFtZTogcHJvcHMuc2VydmljZU5hbWUsXG4gICAgICBtb25pdG9yOiB7XG4gICAgICAgIHR5cGU6ICdxdWVyeSBhbGVydCcsXG4gICAgICAgIHRhZ3M6IFsnY2ljZCddLFxuICAgICAgICBxdWVyeTogYG1pbihsYXN0XzVtKTptaW46ZGVwbG95bWVudC5jb2RlcGlwZWxpbmUuZXhlY3V0aW9uc3twaXBlbGluZV9uYW1lOiR7cHJvcHMucGlwZWxpbmVOYW1lfSxhY2NvdW50X2lkOiR7cHJvcHMuYWNjb3VudElkfX0gPCAxYCxcbiAgICAgICAgbmFtZTogJ0NJIHBpcGVsaW5lIHN0YXR1cycsXG4gICAgICAgIG1lc3NhZ2U6ICdbUDVdIHt7I2lzX2FsZXJ0fX1cXFxcblRoZSBDSSBidWlsZCBpcyBicm9rZW5cXFxcbnt7L2lzX2FsZXJ0fX0gXFxcXG5cXFxcbnt7I2lzX2FsZXJ0X3JlY292ZXJ5fX1cXFxcblRoZSBDSSBidWlsZCB3b3JrcyBhZ2Fpbi5cXFxcbnt7L2lzX2FsZXJ0X3JlY292ZXJ5fX0nLFxuICAgICAgICBvcHRpb25zOiB7XG4gICAgICAgICAgcmVxdWlyZV9mdWxsX3dpbmRvdzogZmFsc2UsXG4gICAgICAgIH0sXG4gICAgICB9LFxuICAgIH0pO1xuICB9XG59XG5cbmV4cG9ydCBjbGFzcyBEYXRhRG9nUGlwZWxpbmVFcnJvckFsZXJ0Rm9yVnVsbmVyYWJpbGl0eUNoZWNrcyBleHRlbmRzIERhdGFkb2dNb25pdG9yIHtcbiAgY29uc3RydWN0b3Ioc2NvcGU6IENvbnN0cnVjdCwgaWQ6IHN0cmluZywgcHJvcHM6IERhdGFkb2dQaXBlbGluZU1vbml0b3JQcm9wcykge1xuICAgIHN1cGVyKHNjb3BlLCBpZCwge1xuICAgICAgc2VydmljZU5hbWU6IHByb3BzLnNlcnZpY2VOYW1lLFxuICAgICAgYWxlcnRUeXBlczogWydvcHNnZW5pZSddLFxuICAgICAgbW9uaXRvcjoge1xuICAgICAgICB0eXBlOiAncXVlcnkgYWxlcnQnLFxuICAgICAgICB0YWdzOiBbJ2NpY2QnXSxcbiAgICAgICAgcXVlcnk6IGBtaW4obGFzdF81bSk6bWluOmRlcGxveW1lbnQuY29kZXBpcGVsaW5lLmV4ZWN1dGlvbnN7cGlwZWxpbmVfbmFtZToke3Byb3BzLnBpcGVsaW5lTmFtZX0sYWNjb3VudF9pZDoke3Byb3BzLmFjY291bnRJZH19IDwgMWAsXG4gICAgICAgIG5hbWU6ICdWdWxuZXJhYmlsaXR5IGNoZWNrcycsXG4gICAgICAgIG1lc3NhZ2U6ICdbUDNdIFZ1bG5lcmFiaWxpdGllcyBkZXRlY3RlZCcsXG4gICAgICAgIG9wdGlvbnM6IHtcbiAgICAgICAgICByZXF1aXJlX2Z1bGxfd2luZG93OiBmYWxzZSxcbiAgICAgICAgfSxcbiAgICAgIH0sXG4gICAgfSk7XG4gIH1cbn1cbiJdfQ==
@@ -1,5 +1,5 @@
1
1
  import { Construct } from 'constructs';
2
- import { DatadogAlertType } from './datadogMonitor';
2
+ import { DatadogAlertType } from './datadog-monitor';
3
3
  export interface DatadogLogIndexMonitoringProps {
4
4
  /**
5
5
  * The name of your service in Datadog. Can be used to query the monitors.
@@ -0,0 +1,140 @@
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
+ new DatadogLogDailyQuotaMonitor(this, 'DatadogLogDailyQuotaMonitor', props);
22
+ new DatadogLogsAnomalyMonitor(this, 'DatadogLogAlertAnomalyMonitor', props);
23
+ new DatadogLogDailyQuotaReachedMonitor(this, 'DatadogLogDailyQuotaReachedV2Monitor', props);
24
+ this.node.addValidation({
25
+ validate: () => {
26
+ const result = [];
27
+ if (props.dailyLogQuota.warningThresholdInPercent < 0 || props.dailyLogQuota.warningThresholdInPercent > 100) {
28
+ result.push('Invalid [warningThresholdInPercent]: expecting a number between 0 and 100');
29
+ }
30
+ if (props.dailyLogQuota.alertThresholdInPercent < 0 || props.dailyLogQuota.alertThresholdInPercent > 100) {
31
+ result.push('Invalid [alertThresholdInPercent]: expecting a number between 0 and 100');
32
+ }
33
+ if (props.dailyLogQuota.alertThresholdInPercent < props.dailyLogQuota.warningThresholdInPercent) {
34
+ result.push('Invalid [alertThresholdInPercent]: must be greater than or equal to [warningThresholdInPercent]');
35
+ }
36
+ return result;
37
+ },
38
+ });
39
+ }
40
+ }
41
+ exports.DatadogLogIndexMonitoring = DatadogLogIndexMonitoring;
42
+ _a = JSII_RTTI_SYMBOL_1;
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}`];
46
+ class DatadogLogDailyQuotaMonitor extends datadog_monitor_1.DatadogMonitor {
47
+ constructor(scope, id, props) {
48
+ const logQuotaInEvents = Math.floor(props.dailyLogQuota.valueInMillionEvents * 1000000);
49
+ const warningThreshold = Math.floor(logQuotaInEvents * (props.dailyLogQuota.warningThresholdInPercent / 100));
50
+ const alertThreshold = Math.floor(logQuotaInEvents * (props.dailyLogQuota.alertThresholdInPercent / 100));
51
+ super(scope, id, {
52
+ serviceName: props.serviceName,
53
+ alertTypes: [props.alertType],
54
+ monitor: {
55
+ name: `${commonMonitorNamePrefix(props.indexName)}: Log quota almost reached`,
56
+ type: 'metric alert',
57
+ query: `sum(last_1d):sum:datadog.estimated_usage.logs.ingested_events{datadog_is_excluded:false,datadog_index:${props.indexName}}.as_count() >= ${alertThreshold}`,
58
+ message: `[P3] Inspect your log volume at ${datadogLogsSearchWithIndexFilterDeepLink(props.indexName, props.organization)} or adapt the log quota at ${datadogLogsIndexesConfigurationDeepLink(props.organization)}`,
59
+ options: {
60
+ thresholds: {
61
+ critical: alertThreshold,
62
+ warning: warningThreshold,
63
+ },
64
+ include_tags: false,
65
+ notify_no_data: !props.sparseLogging,
66
+ no_data_timeframe: 24 * 60,
67
+ },
68
+ tags: [
69
+ ...commonTags(props.indexName),
70
+ ],
71
+ },
72
+ });
73
+ }
74
+ }
75
+ class DatadogLogsAnomalyMonitor extends datadog_monitor_1.DatadogMonitor {
76
+ constructor(scope, id, props) {
77
+ super(scope, id, {
78
+ serviceName: props.serviceName,
79
+ alertTypes: [props.alertType],
80
+ monitor: {
81
+ name: `${commonMonitorNamePrefix(props.indexName)}: Log amount is rising to unexpected levels`,
82
+ type: 'metric alert',
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`,
84
+ message: `[P4] Logs are getting out of control: ${datadogLogsSearchWithIndexFilterDeepLink(props.indexName, props.organization)}`,
85
+ options: {
86
+ include_tags: false,
87
+ notify_no_data: !props.sparseLogging,
88
+ threshold_windows: {
89
+ trigger_window: 'last_15m',
90
+ recovery_window: 'last_15m',
91
+ },
92
+ },
93
+ tags: [
94
+ ...commonTags(props.indexName),
95
+ ],
96
+ },
97
+ });
98
+ }
99
+ }
100
+ class DatadogLogDailyQuotaReachedMonitor extends datadog_monitor_1.DatadogMonitor {
101
+ constructor(scope, id, props) {
102
+ super(scope, id, {
103
+ serviceName: props.serviceName,
104
+ alertTypes: [props.alertType],
105
+ monitor: {
106
+ name: `${commonMonitorNamePrefix(props.indexName)}: Log quota reached`,
107
+ type: 'event-v2 alert',
108
+ query: `events("source:datadog datadog_index:${props.indexName} Daily quota reached").rollup("count").last("12h") > 0`,
109
+ 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.`,
110
+ options: {
111
+ timeout_h: 24,
112
+ thresholds: {
113
+ critical: 0,
114
+ },
115
+ },
116
+ tags: [
117
+ ...commonTags(props.indexName),
118
+ ],
119
+ },
120
+ });
121
+ }
122
+ }
123
+ function getOrganizationUrlPrefix(organization) {
124
+ switch (organization !== null && organization !== void 0 ? organization : 'EU') {
125
+ case 'EU':
126
+ return 'app';
127
+ case 'LATAM':
128
+ return 'rio-brazil';
129
+ case 'Landing Zone':
130
+ return 'rio-landingzone';
131
+ }
132
+ }
133
+ exports.getOrganizationUrlPrefix = getOrganizationUrlPrefix;
134
+ function datadogLogsSearchWithIndexFilterDeepLink(indexName, organization) {
135
+ return `https://${getOrganizationUrlPrefix(organization)}.datadoghq.eu/logs?index=${indexName}`;
136
+ }
137
+ function datadogLogsIndexesConfigurationDeepLink(organization) {
138
+ return `https://${getOrganizationUrlPrefix(organization)}.datadoghq.eu/logs/pipelines/indexes`;
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"]}
@@ -0,0 +1,19 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.DatadogMonitor = void 0;
4
+ const aws_cdk_lib_1 = require("aws-cdk-lib");
5
+ class DatadogMonitor {
6
+ constructor(stack, id, props) {
7
+ const monitor = new aws_cdk_lib_1.aws_cloudformation.CfnCustomResource(stack, id, {
8
+ serviceToken: aws_cdk_lib_1.Fn.importValue('custom-resource-datadog-monitor-function-arn'),
9
+ });
10
+ monitor.addPropertyOverride('ServiceName', props.serviceName);
11
+ monitor.addPropertyOverride('AlertTypes', props.alertTypes);
12
+ if (props.autoCloseOpsGenieAlerts !== undefined) {
13
+ monitor.addPropertyOverride('AutoCloseOpsGenieAlerts', props.autoCloseOpsGenieAlerts);
14
+ }
15
+ monitor.addPropertyOverride('Monitor', props.monitor);
16
+ }
17
+ }
18
+ exports.DatadogMonitor = DatadogMonitor;
19
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGF0YWRvZy1tb25pdG9yLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2RhdGFkb2cvZGF0YWRvZy1tb25pdG9yLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBLDZDQUFxRDtBQVlyRCxNQUFhLGNBQWM7SUFDekIsWUFBWSxLQUFnQixFQUFFLEVBQVUsRUFBRSxLQUEwQjtRQUNsRSxNQUFNLE9BQU8sR0FBRyxJQUFJLGdDQUFrQixDQUFDLGlCQUFpQixDQUFDLEtBQUssRUFBRSxFQUFFLEVBQUU7WUFDbEUsWUFBWSxFQUFFLGdCQUFFLENBQUMsV0FBVyxDQUFDLDhDQUE4QyxDQUFDO1NBQzdFLENBQUMsQ0FBQztRQUNILE9BQU8sQ0FBQyxtQkFBbUIsQ0FBQyxhQUFhLEVBQUUsS0FBSyxDQUFDLFdBQVcsQ0FBQyxDQUFDO1FBQzlELE9BQU8sQ0FBQyxtQkFBbUIsQ0FBQyxZQUFZLEVBQUUsS0FBSyxDQUFDLFVBQVUsQ0FBQyxDQUFDO1FBQzVELElBQUksS0FBSyxDQUFDLHVCQUF1QixLQUFLLFNBQVMsRUFBRTtZQUMvQyxPQUFPLENBQUMsbUJBQW1CLENBQUMseUJBQXlCLEVBQUUsS0FBSyxDQUFDLHVCQUF1QixDQUFDLENBQUM7U0FDdkY7UUFDRCxPQUFPLENBQUMsbUJBQW1CLENBQUMsU0FBUyxFQUFFLEtBQUssQ0FBQyxPQUFPLENBQUMsQ0FBQztJQUN4RCxDQUFDO0NBQ0Y7QUFaRCx3Q0FZQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGF3c19jbG91ZGZvcm1hdGlvbiwgRm4gfSBmcm9tICdhd3MtY2RrLWxpYic7XG5pbXBvcnQgeyBDb25zdHJ1Y3QgfSBmcm9tICdjb25zdHJ1Y3RzJztcblxuZXhwb3J0IHR5cGUgRGF0YWRvZ0FsZXJ0VHlwZSA9ICdvcHNnZW5pZScgfCAnc2xhY2snO1xuXG5leHBvcnQgaW50ZXJmYWNlIERhdGFkb2dNb25pdG9yUHJvcHMge1xuICBzZXJ2aWNlTmFtZTogc3RyaW5nO1xuICBhbGVydFR5cGVzPzogRGF0YWRvZ0FsZXJ0VHlwZVtdO1xuICBhdXRvQ2xvc2VPcHNHZW5pZUFsZXJ0cz86IGJvb2xlYW47XG4gIG1vbml0b3I6IHsgW2tleTogc3RyaW5nXTogYW55IH07XG59XG5cbmV4cG9ydCBjbGFzcyBEYXRhZG9nTW9uaXRvciB7XG4gIGNvbnN0cnVjdG9yKHN0YWNrOiBDb25zdHJ1Y3QsIGlkOiBzdHJpbmcsIHByb3BzOiBEYXRhZG9nTW9uaXRvclByb3BzKSB7XG4gICAgY29uc3QgbW9uaXRvciA9IG5ldyBhd3NfY2xvdWRmb3JtYXRpb24uQ2ZuQ3VzdG9tUmVzb3VyY2Uoc3RhY2ssIGlkLCB7XG4gICAgICBzZXJ2aWNlVG9rZW46IEZuLmltcG9ydFZhbHVlKCdjdXN0b20tcmVzb3VyY2UtZGF0YWRvZy1tb25pdG9yLWZ1bmN0aW9uLWFybicpLFxuICAgIH0pO1xuICAgIG1vbml0b3IuYWRkUHJvcGVydHlPdmVycmlkZSgnU2VydmljZU5hbWUnLCBwcm9wcy5zZXJ2aWNlTmFtZSk7XG4gICAgbW9uaXRvci5hZGRQcm9wZXJ0eU92ZXJyaWRlKCdBbGVydFR5cGVzJywgcHJvcHMuYWxlcnRUeXBlcyk7XG4gICAgaWYgKHByb3BzLmF1dG9DbG9zZU9wc0dlbmllQWxlcnRzICE9PSB1bmRlZmluZWQpIHtcbiAgICAgIG1vbml0b3IuYWRkUHJvcGVydHlPdmVycmlkZSgnQXV0b0Nsb3NlT3BzR2VuaWVBbGVydHMnLCBwcm9wcy5hdXRvQ2xvc2VPcHNHZW5pZUFsZXJ0cyk7XG4gICAgfVxuICAgIG1vbml0b3IuYWRkUHJvcGVydHlPdmVycmlkZSgnTW9uaXRvcicsIHByb3BzLm1vbml0b3IpO1xuICB9XG59XG4iXX0=
@@ -1 +1 @@
1
- export * from './datadogLogIndexMonitoring';
1
+ export * from './datadog-log-index-monitoring';
@@ -1,5 +1,5 @@
1
1
  "use strict";
2
- // We do not want to export anything from `./datadogMonitor` for the following reasons.
2
+ // We do not want to export anything from `./datadog-monitor` for the following reasons.
3
3
  // - There are already similar abstraction available, e.g., in `datadog-log-alarm.ts` and `datadog-metric-alarm.ts`.
4
4
  // - We should discuss a suitable abstraction first before making the API publicly available. At the moment, it is a
5
5
  // simple wrapper around the CloudFormation Custom Resource. We usually strive for higher-level constructs or
@@ -15,5 +15,5 @@ var __exportStar = (this && this.__exportStar) || function(m, exports) {
15
15
  for (var p in m) if (p !== "default" && !exports.hasOwnProperty(p)) __createBinding(exports, m, p);
16
16
  };
17
17
  Object.defineProperty(exports, "__esModule", { value: true });
18
- __exportStar(require("./datadogLogIndexMonitoring"), exports);
19
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvZGF0YWRvZy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUEsdUZBQXVGO0FBQ3ZGLHNIQUFzSDtBQUN0SCxzSEFBc0g7QUFDdEgsaUhBQWlIO0FBQ2pILDRDQUE0Qzs7Ozs7Ozs7Ozs7O0FBRTVDLDhEQUE0QyIsInNvdXJjZXNDb250ZW50IjpbIi8vIFdlIGRvIG5vdCB3YW50IHRvIGV4cG9ydCBhbnl0aGluZyBmcm9tIGAuL2RhdGFkb2dNb25pdG9yYCBmb3IgdGhlIGZvbGxvd2luZyByZWFzb25zLlxuLy8gICAtIFRoZXJlIGFyZSBhbHJlYWR5IHNpbWlsYXIgYWJzdHJhY3Rpb24gYXZhaWxhYmxlLCBlLmcuLCBpbiBgZGF0YWRvZy1sb2ctYWxhcm0udHNgIGFuZCBgZGF0YWRvZy1tZXRyaWMtYWxhcm0udHNgLlxuLy8gICAtIFdlIHNob3VsZCBkaXNjdXNzIGEgc3VpdGFibGUgYWJzdHJhY3Rpb24gZmlyc3QgYmVmb3JlIG1ha2luZyB0aGUgQVBJIHB1YmxpY2x5IGF2YWlsYWJsZS4gQXQgdGhlIG1vbWVudCwgaXQgaXMgYVxuLy8gICAgIHNpbXBsZSB3cmFwcGVyIGFyb3VuZCB0aGUgQ2xvdWRGb3JtYXRpb24gQ3VzdG9tIFJlc291cmNlLiBXZSB1c3VhbGx5IHN0cml2ZSBmb3IgaGlnaGVyLWxldmVsIGNvbnN0cnVjdHMgb3Jcbi8vICAgICBkaXJlY3RseSB1c2UgdGhlIGxvdy1sZXZlbCBjb25zdHJ1Y3QuXG5cbmV4cG9ydCAqIGZyb20gJy4vZGF0YWRvZ0xvZ0luZGV4TW9uaXRvcmluZyc7XG4iXX0=
18
+ __exportStar(require("./datadog-log-index-monitoring"), exports);
19
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvZGF0YWRvZy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUEsd0ZBQXdGO0FBQ3hGLHNIQUFzSDtBQUN0SCxzSEFBc0g7QUFDdEgsaUhBQWlIO0FBQ2pILDRDQUE0Qzs7Ozs7Ozs7Ozs7O0FBRTVDLGlFQUErQyIsInNvdXJjZXNDb250ZW50IjpbIi8vIFdlIGRvIG5vdCB3YW50IHRvIGV4cG9ydCBhbnl0aGluZyBmcm9tIGAuL2RhdGFkb2ctbW9uaXRvcmAgZm9yIHRoZSBmb2xsb3dpbmcgcmVhc29ucy5cbi8vICAgLSBUaGVyZSBhcmUgYWxyZWFkeSBzaW1pbGFyIGFic3RyYWN0aW9uIGF2YWlsYWJsZSwgZS5nLiwgaW4gYGRhdGFkb2ctbG9nLWFsYXJtLnRzYCBhbmQgYGRhdGFkb2ctbWV0cmljLWFsYXJtLnRzYC5cbi8vICAgLSBXZSBzaG91bGQgZGlzY3VzcyBhIHN1aXRhYmxlIGFic3RyYWN0aW9uIGZpcnN0IGJlZm9yZSBtYWtpbmcgdGhlIEFQSSBwdWJsaWNseSBhdmFpbGFibGUuIEF0IHRoZSBtb21lbnQsIGl0IGlzIGFcbi8vICAgICBzaW1wbGUgd3JhcHBlciBhcm91bmQgdGhlIENsb3VkRm9ybWF0aW9uIEN1c3RvbSBSZXNvdXJjZS4gV2UgdXN1YWxseSBzdHJpdmUgZm9yIGhpZ2hlci1sZXZlbCBjb25zdHJ1Y3RzIG9yXG4vLyAgICAgZGlyZWN0bHkgdXNlIHRoZSBsb3ctbGV2ZWwgY29uc3RydWN0LlxuXG5leHBvcnQgKiBmcm9tICcuL2RhdGFkb2ctbG9nLWluZGV4LW1vbml0b3JpbmcnO1xuIl19
package/lib/index.d.ts CHANGED
@@ -19,6 +19,7 @@ export * from './pipeline/rio-bitbucket-source-action';
19
19
  export * from './pipeline/rio-backup-secrets-restore-stage';
20
20
  export * from './kafka';
21
21
  export * from './datadog';
22
+ export * from './contributions/cop-frontend/cloudfront';
22
23
  export * from './contributions/team-transport-two/pipeline';
23
24
  export * from './contributions/team-claid/aws-backup-monitoring';
24
25
  export * from './contributions/team-claid/datadog-usage-monitoring';
package/lib/index.js CHANGED
@@ -31,7 +31,8 @@ __exportStar(require("./pipeline/rio-bitbucket-source-action"), exports);
31
31
  __exportStar(require("./pipeline/rio-backup-secrets-restore-stage"), exports);
32
32
  __exportStar(require("./kafka"), exports);
33
33
  __exportStar(require("./datadog"), exports);
34
+ __exportStar(require("./contributions/cop-frontend/cloudfront"), exports);
34
35
  __exportStar(require("./contributions/team-transport-two/pipeline"), exports);
35
36
  __exportStar(require("./contributions/team-claid/aws-backup-monitoring"), exports);
36
37
  __exportStar(require("./contributions/team-claid/datadog-usage-monitoring"), exports);
37
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7O0FBQUEsaURBQStCO0FBQy9CLG9EQUFrQztBQUNsQyxpREFBK0I7QUFDL0IsbURBQWlDO0FBQ2pDLHNEQUFvQztBQUNwQyxvREFBa0M7QUFDbEMseURBQXVDO0FBQ3ZDLGtFQUFnRDtBQUNoRCwrREFBNkM7QUFDN0Msc0RBQW9DO0FBQ3BDLDBEQUF3QztBQUN4QyxpREFBK0I7QUFDL0Isd0RBQXNDO0FBRXRDLHFEQUFtQztBQUNuQyxzREFBb0M7QUFDcEMsb0RBQWtDO0FBRWxDLGdFQUE4QztBQUU5Qyx5RUFBdUQ7QUFDdkQsOEVBQTREO0FBRTVELDBDQUF3QjtBQUV4Qiw0Q0FBMEI7QUFFMUIsOEVBQTREO0FBRTVELG1GQUFpRTtBQUNqRSxzRkFBb0UiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL3dhdGNoZnVsL2FsYic7XG5leHBvcnQgKiBmcm9tICcuL3dhdGNoZnVsL2FzcGVjdCc7XG5leHBvcnQgKiBmcm9tICcuL3dhdGNoZnVsL2Vjcyc7XG5leHBvcnQgKiBmcm9tICcuL3dhdGNoZnVsL2RvY2RiJztcbmV4cG9ydCAqIGZyb20gJy4vd2F0Y2hmdWwvZHluYW1vZGInO1xuZXhwb3J0ICogZnJvbSAnLi93YXRjaGZ1bC9sYW1iZGEnO1xuZXhwb3J0ICogZnJvbSAnLi93YXRjaGZ1bC90YXJnZXRncm91cCc7XG5leHBvcnQgKiBmcm9tICcuL3dhdGNoZnVsL2RhdGFkb2ctbWV0cmljLWFsYXJtJztcbmV4cG9ydCAqIGZyb20gJy4vd2F0Y2hmdWwvZGF0YWRvZy1sb2ctYWxhcm0nO1xuZXhwb3J0ICogZnJvbSAnLi93YXRjaGZ1bC93YXRjaGZ1bCc7XG5leHBvcnQgKiBmcm9tICcuL3dhdGNoZnVsL21ldHJpYy1hbGFybSc7XG5leHBvcnQgKiBmcm9tICcuL3dhdGNoZnVsL3Jkcyc7XG5leHBvcnQgKiBmcm9tICcuL3dhdGNoZnVsL2Nsb3VkZnJvbnQnO1xuXG5leHBvcnQgKiBmcm9tICcuL3Jpby1sYW5kaW5nLXpvbmUnO1xuZXhwb3J0ICogZnJvbSAnLi9yaW8tbG9hZC1iYWxhbmNlcic7XG5leHBvcnQgKiBmcm9tICcuL3Jpby1jbGFpZG9tZXRlcic7XG5cbmV4cG9ydCAqIGZyb20gJy4vZmFyZ2F0ZS9yaW8tZmFyZ2F0ZS1zZXJ2aWNlJztcblxuZXhwb3J0ICogZnJvbSAnLi9waXBlbGluZS9yaW8tYml0YnVja2V0LXNvdXJjZS1hY3Rpb24nO1xuZXhwb3J0ICogZnJvbSAnLi9waXBlbGluZS9yaW8tYmFja3VwLXNlY3JldHMtcmVzdG9yZS1zdGFnZSc7XG5cbmV4cG9ydCAqIGZyb20gJy4va2Fma2EnO1xuXG5leHBvcnQgKiBmcm9tICcuL2RhdGFkb2cnO1xuXG5leHBvcnQgKiBmcm9tICcuL2NvbnRyaWJ1dGlvbnMvdGVhbS10cmFuc3BvcnQtdHdvL3BpcGVsaW5lJztcblxuZXhwb3J0ICogZnJvbSAnLi9jb250cmlidXRpb25zL3RlYW0tY2xhaWQvYXdzLWJhY2t1cC1tb25pdG9yaW5nJztcbmV4cG9ydCAqIGZyb20gJy4vY29udHJpYnV0aW9ucy90ZWFtLWNsYWlkL2RhdGFkb2ctdXNhZ2UtbW9uaXRvcmluZyc7XG4iXX0=
38
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7O0FBQUEsaURBQStCO0FBQy9CLG9EQUFrQztBQUNsQyxpREFBK0I7QUFDL0IsbURBQWlDO0FBQ2pDLHNEQUFvQztBQUNwQyxvREFBa0M7QUFDbEMseURBQXVDO0FBQ3ZDLGtFQUFnRDtBQUNoRCwrREFBNkM7QUFDN0Msc0RBQW9DO0FBQ3BDLDBEQUF3QztBQUN4QyxpREFBK0I7QUFDL0Isd0RBQXNDO0FBRXRDLHFEQUFtQztBQUNuQyxzREFBb0M7QUFDcEMsb0RBQWtDO0FBRWxDLGdFQUE4QztBQUU5Qyx5RUFBdUQ7QUFDdkQsOEVBQTREO0FBRTVELDBDQUF3QjtBQUV4Qiw0Q0FBMEI7QUFFMUIsMEVBQXdEO0FBRXhELDhFQUE0RDtBQUU1RCxtRkFBaUU7QUFDakUsc0ZBQW9FIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi93YXRjaGZ1bC9hbGInO1xuZXhwb3J0ICogZnJvbSAnLi93YXRjaGZ1bC9hc3BlY3QnO1xuZXhwb3J0ICogZnJvbSAnLi93YXRjaGZ1bC9lY3MnO1xuZXhwb3J0ICogZnJvbSAnLi93YXRjaGZ1bC9kb2NkYic7XG5leHBvcnQgKiBmcm9tICcuL3dhdGNoZnVsL2R5bmFtb2RiJztcbmV4cG9ydCAqIGZyb20gJy4vd2F0Y2hmdWwvbGFtYmRhJztcbmV4cG9ydCAqIGZyb20gJy4vd2F0Y2hmdWwvdGFyZ2V0Z3JvdXAnO1xuZXhwb3J0ICogZnJvbSAnLi93YXRjaGZ1bC9kYXRhZG9nLW1ldHJpYy1hbGFybSc7XG5leHBvcnQgKiBmcm9tICcuL3dhdGNoZnVsL2RhdGFkb2ctbG9nLWFsYXJtJztcbmV4cG9ydCAqIGZyb20gJy4vd2F0Y2hmdWwvd2F0Y2hmdWwnO1xuZXhwb3J0ICogZnJvbSAnLi93YXRjaGZ1bC9tZXRyaWMtYWxhcm0nO1xuZXhwb3J0ICogZnJvbSAnLi93YXRjaGZ1bC9yZHMnO1xuZXhwb3J0ICogZnJvbSAnLi93YXRjaGZ1bC9jbG91ZGZyb250JztcblxuZXhwb3J0ICogZnJvbSAnLi9yaW8tbGFuZGluZy16b25lJztcbmV4cG9ydCAqIGZyb20gJy4vcmlvLWxvYWQtYmFsYW5jZXInO1xuZXhwb3J0ICogZnJvbSAnLi9yaW8tY2xhaWRvbWV0ZXInO1xuXG5leHBvcnQgKiBmcm9tICcuL2ZhcmdhdGUvcmlvLWZhcmdhdGUtc2VydmljZSc7XG5cbmV4cG9ydCAqIGZyb20gJy4vcGlwZWxpbmUvcmlvLWJpdGJ1Y2tldC1zb3VyY2UtYWN0aW9uJztcbmV4cG9ydCAqIGZyb20gJy4vcGlwZWxpbmUvcmlvLWJhY2t1cC1zZWNyZXRzLXJlc3RvcmUtc3RhZ2UnO1xuXG5leHBvcnQgKiBmcm9tICcuL2thZmthJztcblxuZXhwb3J0ICogZnJvbSAnLi9kYXRhZG9nJztcblxuZXhwb3J0ICogZnJvbSAnLi9jb250cmlidXRpb25zL2NvcC1mcm9udGVuZC9jbG91ZGZyb250JztcblxuZXhwb3J0ICogZnJvbSAnLi9jb250cmlidXRpb25zL3RlYW0tdHJhbnNwb3J0LXR3by9waXBlbGluZSc7XG5cbmV4cG9ydCAqIGZyb20gJy4vY29udHJpYnV0aW9ucy90ZWFtLWNsYWlkL2F3cy1iYWNrdXAtbW9uaXRvcmluZyc7XG5leHBvcnQgKiBmcm9tICcuL2NvbnRyaWJ1dGlvbnMvdGVhbS1jbGFpZC9kYXRhZG9nLXVzYWdlLW1vbml0b3JpbmcnO1xuIl19
package/package.json CHANGED
@@ -15,7 +15,7 @@
15
15
  ],
16
16
  "main": "lib/index.js",
17
17
  "license": "Apache-2.0",
18
- "version": "2.6.2",
18
+ "version": "2.7.0",
19
19
  "types": "lib/index.d.ts",
20
20
  "stability": "stable",
21
21
  "jsii": {
package/version.json CHANGED
@@ -1,3 +1,3 @@
1
1
  {
2
- "version": "2.6.2"
2
+ "version": "2.7.0"
3
3
  }
@@ -1,140 +0,0 @@
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 datadogMonitor_1 = require("./datadogMonitor");
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
- new DatadogLogDailyQuotaMonitor(this, 'DatadogLogDailyQuotaMonitor', props);
22
- new DatadogLogsAnomalyMonitor(this, 'DatadogLogAlertAnomalyMonitor', props);
23
- new DatadogLogDailyQuotaReachedMonitor(this, 'DatadogLogDailyQuotaReachedV2Monitor', props);
24
- this.node.addValidation({
25
- validate: () => {
26
- const result = [];
27
- if (props.dailyLogQuota.warningThresholdInPercent < 0 || props.dailyLogQuota.warningThresholdInPercent > 100) {
28
- result.push('Invalid [warningThresholdInPercent]: expecting a number between 0 and 100');
29
- }
30
- if (props.dailyLogQuota.alertThresholdInPercent < 0 || props.dailyLogQuota.alertThresholdInPercent > 100) {
31
- result.push('Invalid [alertThresholdInPercent]: expecting a number between 0 and 100');
32
- }
33
- if (props.dailyLogQuota.alertThresholdInPercent < props.dailyLogQuota.warningThresholdInPercent) {
34
- result.push('Invalid [alertThresholdInPercent]: must be greater than or equal to [warningThresholdInPercent]');
35
- }
36
- return result;
37
- },
38
- });
39
- }
40
- }
41
- exports.DatadogLogIndexMonitoring = DatadogLogIndexMonitoring;
42
- _a = JSII_RTTI_SYMBOL_1;
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}`];
46
- class DatadogLogDailyQuotaMonitor extends datadogMonitor_1.DatadogMonitor {
47
- constructor(scope, id, props) {
48
- const logQuotaInEvents = Math.floor(props.dailyLogQuota.valueInMillionEvents * 1000000);
49
- const warningThreshold = Math.floor(logQuotaInEvents * (props.dailyLogQuota.warningThresholdInPercent / 100));
50
- const alertThreshold = Math.floor(logQuotaInEvents * (props.dailyLogQuota.alertThresholdInPercent / 100));
51
- super(scope, id, {
52
- serviceName: props.serviceName,
53
- alertTypes: [props.alertType],
54
- monitor: {
55
- name: `${commonMonitorNamePrefix(props.indexName)}: Log quota almost reached`,
56
- type: 'metric alert',
57
- query: `sum(last_1d):sum:datadog.estimated_usage.logs.ingested_events{datadog_is_excluded:false,datadog_index:${props.indexName}}.as_count() >= ${alertThreshold}`,
58
- message: `[P3] Inspect your log volume at ${datadogLogsSearchWithIndexFilterDeepLink(props.indexName, props.organization)} or adapt the log quota at ${datadogLogsIndexesConfigurationDeepLink(props.organization)}`,
59
- options: {
60
- thresholds: {
61
- critical: alertThreshold,
62
- warning: warningThreshold,
63
- },
64
- include_tags: false,
65
- notify_no_data: !props.sparseLogging,
66
- no_data_timeframe: 24 * 60,
67
- },
68
- tags: [
69
- ...commonTags(props.indexName),
70
- ],
71
- },
72
- });
73
- }
74
- }
75
- class DatadogLogsAnomalyMonitor extends datadogMonitor_1.DatadogMonitor {
76
- constructor(scope, id, props) {
77
- super(scope, id, {
78
- serviceName: props.serviceName,
79
- alertTypes: [props.alertType],
80
- monitor: {
81
- name: `${commonMonitorNamePrefix(props.indexName)}: Log amount is rising to unexpected levels`,
82
- type: 'metric alert',
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`,
84
- message: `[P4] Logs are getting out of control: ${datadogLogsSearchWithIndexFilterDeepLink(props.indexName, props.organization)}`,
85
- options: {
86
- include_tags: false,
87
- notify_no_data: !props.sparseLogging,
88
- threshold_windows: {
89
- trigger_window: 'last_15m',
90
- recovery_window: 'last_15m',
91
- },
92
- },
93
- tags: [
94
- ...commonTags(props.indexName),
95
- ],
96
- },
97
- });
98
- }
99
- }
100
- class DatadogLogDailyQuotaReachedMonitor extends datadogMonitor_1.DatadogMonitor {
101
- constructor(scope, id, props) {
102
- super(scope, id, {
103
- serviceName: props.serviceName,
104
- alertTypes: [props.alertType],
105
- monitor: {
106
- name: `${commonMonitorNamePrefix(props.indexName)}: Log quota reached`,
107
- type: 'event-v2 alert',
108
- query: `events("source:datadog datadog_index:${props.indexName} Daily quota reached").rollup("count").last("12h") > 0`,
109
- 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.`,
110
- options: {
111
- timeout_h: 24,
112
- thresholds: {
113
- critical: 0,
114
- },
115
- },
116
- tags: [
117
- ...commonTags(props.indexName),
118
- ],
119
- },
120
- });
121
- }
122
- }
123
- function getOrganizationUrlPrefix(organization) {
124
- switch (organization !== null && organization !== void 0 ? organization : 'EU') {
125
- case 'EU':
126
- return 'app';
127
- case 'LATAM':
128
- return 'rio-brazil';
129
- case 'Landing Zone':
130
- return 'rio-landingzone';
131
- }
132
- }
133
- exports.getOrganizationUrlPrefix = getOrganizationUrlPrefix;
134
- function datadogLogsSearchWithIndexFilterDeepLink(indexName, organization) {
135
- return `https://${getOrganizationUrlPrefix(organization)}.datadoghq.eu/logs?index=${indexName}`;
136
- }
137
- function datadogLogsIndexesConfigurationDeepLink(organization) {
138
- return `https://${getOrganizationUrlPrefix(organization)}.datadoghq.eu/logs/pipelines/indexes`;
139
- }
140
- //# 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  /**\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"]}
@@ -1,19 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.DatadogMonitor = void 0;
4
- const aws_cdk_lib_1 = require("aws-cdk-lib");
5
- class DatadogMonitor {
6
- constructor(stack, id, props) {
7
- const monitor = new aws_cdk_lib_1.aws_cloudformation.CfnCustomResource(stack, id, {
8
- serviceToken: aws_cdk_lib_1.Fn.importValue('custom-resource-datadog-monitor-function-arn'),
9
- });
10
- monitor.addPropertyOverride('ServiceName', props.serviceName);
11
- monitor.addPropertyOverride('AlertTypes', props.alertTypes);
12
- if (props.autoCloseOpsGenieAlerts !== undefined) {
13
- monitor.addPropertyOverride('AutoCloseOpsGenieAlerts', props.autoCloseOpsGenieAlerts);
14
- }
15
- monitor.addPropertyOverride('Monitor', props.monitor);
16
- }
17
- }
18
- exports.DatadogMonitor = DatadogMonitor;
19
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGF0YWRvZ01vbml0b3IuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvZGF0YWRvZy9kYXRhZG9nTW9uaXRvci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSw2Q0FBcUQ7QUFZckQsTUFBYSxjQUFjO0lBQ3pCLFlBQVksS0FBZ0IsRUFBRSxFQUFVLEVBQUUsS0FBMEI7UUFDbEUsTUFBTSxPQUFPLEdBQUcsSUFBSSxnQ0FBa0IsQ0FBQyxpQkFBaUIsQ0FBQyxLQUFLLEVBQUUsRUFBRSxFQUFFO1lBQ2xFLFlBQVksRUFBRSxnQkFBRSxDQUFDLFdBQVcsQ0FBQyw4Q0FBOEMsQ0FBQztTQUM3RSxDQUFDLENBQUM7UUFDSCxPQUFPLENBQUMsbUJBQW1CLENBQUMsYUFBYSxFQUFFLEtBQUssQ0FBQyxXQUFXLENBQUMsQ0FBQztRQUM5RCxPQUFPLENBQUMsbUJBQW1CLENBQUMsWUFBWSxFQUFFLEtBQUssQ0FBQyxVQUFVLENBQUMsQ0FBQztRQUM1RCxJQUFJLEtBQUssQ0FBQyx1QkFBdUIsS0FBSyxTQUFTLEVBQUU7WUFDL0MsT0FBTyxDQUFDLG1CQUFtQixDQUFDLHlCQUF5QixFQUFFLEtBQUssQ0FBQyx1QkFBdUIsQ0FBQyxDQUFDO1NBQ3ZGO1FBQ0QsT0FBTyxDQUFDLG1CQUFtQixDQUFDLFNBQVMsRUFBRSxLQUFLLENBQUMsT0FBTyxDQUFDLENBQUM7SUFDeEQsQ0FBQztDQUNGO0FBWkQsd0NBWUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBhd3NfY2xvdWRmb3JtYXRpb24sIEZuIH0gZnJvbSAnYXdzLWNkay1saWInO1xuaW1wb3J0IHsgQ29uc3RydWN0IH0gZnJvbSAnY29uc3RydWN0cyc7XG5cbmV4cG9ydCB0eXBlIERhdGFkb2dBbGVydFR5cGUgPSAnb3BzZ2VuaWUnIHwgJ3NsYWNrJztcblxuZXhwb3J0IGludGVyZmFjZSBEYXRhZG9nTW9uaXRvclByb3BzIHtcbiAgc2VydmljZU5hbWU6IHN0cmluZztcbiAgYWxlcnRUeXBlcz86IERhdGFkb2dBbGVydFR5cGVbXTtcbiAgYXV0b0Nsb3NlT3BzR2VuaWVBbGVydHM/OiBib29sZWFuO1xuICBtb25pdG9yOiB7IFtrZXk6IHN0cmluZ106IGFueSB9O1xufVxuXG5leHBvcnQgY2xhc3MgRGF0YWRvZ01vbml0b3Ige1xuICBjb25zdHJ1Y3RvcihzdGFjazogQ29uc3RydWN0LCBpZDogc3RyaW5nLCBwcm9wczogRGF0YWRvZ01vbml0b3JQcm9wcykge1xuICAgIGNvbnN0IG1vbml0b3IgPSBuZXcgYXdzX2Nsb3VkZm9ybWF0aW9uLkNmbkN1c3RvbVJlc291cmNlKHN0YWNrLCBpZCwge1xuICAgICAgc2VydmljZVRva2VuOiBGbi5pbXBvcnRWYWx1ZSgnY3VzdG9tLXJlc291cmNlLWRhdGFkb2ctbW9uaXRvci1mdW5jdGlvbi1hcm4nKSxcbiAgICB9KTtcbiAgICBtb25pdG9yLmFkZFByb3BlcnR5T3ZlcnJpZGUoJ1NlcnZpY2VOYW1lJywgcHJvcHMuc2VydmljZU5hbWUpO1xuICAgIG1vbml0b3IuYWRkUHJvcGVydHlPdmVycmlkZSgnQWxlcnRUeXBlcycsIHByb3BzLmFsZXJ0VHlwZXMpO1xuICAgIGlmIChwcm9wcy5hdXRvQ2xvc2VPcHNHZW5pZUFsZXJ0cyAhPT0gdW5kZWZpbmVkKSB7XG4gICAgICBtb25pdG9yLmFkZFByb3BlcnR5T3ZlcnJpZGUoJ0F1dG9DbG9zZU9wc0dlbmllQWxlcnRzJywgcHJvcHMuYXV0b0Nsb3NlT3BzR2VuaWVBbGVydHMpO1xuICAgIH1cbiAgICBtb25pdG9yLmFkZFByb3BlcnR5T3ZlcnJpZGUoJ01vbml0b3InLCBwcm9wcy5tb25pdG9yKTtcbiAgfVxufVxuIl19