@rio-cloud/cdk-v2-constructs 7.13.12 → 7.14.0-alpha.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
@@ -23622,6 +23622,271 @@
23622
23622
  ],
23623
23623
  "symbolId": "src/datadogv2/datadog-shared-secrets:DatadogSharedSecret"
23624
23624
  },
23625
+ "@rio-cloud/cdk-v2-constructs.datadogv2.DatadogSlo": {
23626
+ "assembly": "@rio-cloud/cdk-v2-constructs",
23627
+ "base": "constructs.Construct",
23628
+ "docs": {
23629
+ "remarks": "Currently only SLOs of Type Monitor are implemented.",
23630
+ "stability": "stable",
23631
+ "summary": "Wrapper around Datadog's CloudFormation `Datadog::SLOs::SLO` with some RIO defaults applied."
23632
+ },
23633
+ "fqn": "@rio-cloud/cdk-v2-constructs.datadogv2.DatadogSlo",
23634
+ "initializer": {
23635
+ "docs": {
23636
+ "stability": "stable"
23637
+ },
23638
+ "locationInModule": {
23639
+ "filename": "src/datadogv2/datadog-slo.ts",
23640
+ "line": 100
23641
+ },
23642
+ "parameters": [
23643
+ {
23644
+ "name": "scope",
23645
+ "type": {
23646
+ "fqn": "constructs.Construct"
23647
+ }
23648
+ },
23649
+ {
23650
+ "name": "id",
23651
+ "type": {
23652
+ "primitive": "string"
23653
+ }
23654
+ },
23655
+ {
23656
+ "name": "props",
23657
+ "type": {
23658
+ "fqn": "@rio-cloud/cdk-v2-constructs.datadogv2.DatadogSloProps"
23659
+ }
23660
+ }
23661
+ ]
23662
+ },
23663
+ "kind": "class",
23664
+ "locationInModule": {
23665
+ "filename": "src/datadogv2/datadog-slo.ts",
23666
+ "line": 97
23667
+ },
23668
+ "name": "DatadogSlo",
23669
+ "namespace": "datadogv2",
23670
+ "properties": [
23671
+ {
23672
+ "docs": {
23673
+ "stability": "stable"
23674
+ },
23675
+ "immutable": true,
23676
+ "locationInModule": {
23677
+ "filename": "src/datadogv2/datadog-slo.ts",
23678
+ "line": 147
23679
+ },
23680
+ "name": "sloId",
23681
+ "type": {
23682
+ "primitive": "string"
23683
+ }
23684
+ }
23685
+ ],
23686
+ "symbolId": "src/datadogv2/datadog-slo:DatadogSlo"
23687
+ },
23688
+ "@rio-cloud/cdk-v2-constructs.datadogv2.DatadogSloProps": {
23689
+ "assembly": "@rio-cloud/cdk-v2-constructs",
23690
+ "datatype": true,
23691
+ "docs": {
23692
+ "stability": "stable",
23693
+ "summary": "Properties of {@link DatadogSlo}."
23694
+ },
23695
+ "fqn": "@rio-cloud/cdk-v2-constructs.datadogv2.DatadogSloProps",
23696
+ "kind": "interface",
23697
+ "locationInModule": {
23698
+ "filename": "src/datadogv2/datadog-slo.ts",
23699
+ "line": 13
23700
+ },
23701
+ "name": "DatadogSloProps",
23702
+ "namespace": "datadogv2",
23703
+ "properties": [
23704
+ {
23705
+ "abstract": true,
23706
+ "docs": {
23707
+ "stability": "stable",
23708
+ "summary": "The criticality level for the SLO as defined at [Capabilities - RIO Tech Platform Docs - MAN Confluence PRD](https://confluence.collaboration-man.com/x/nLUSEw)."
23709
+ },
23710
+ "immutable": true,
23711
+ "locationInModule": {
23712
+ "filename": "src/datadogv2/datadog-slo.ts",
23713
+ "line": 35
23714
+ },
23715
+ "name": "criticality",
23716
+ "type": {
23717
+ "primitive": "string"
23718
+ }
23719
+ },
23720
+ {
23721
+ "abstract": true,
23722
+ "docs": {
23723
+ "remarks": "Each entry must be a numeric string (for example \"1234567\").",
23724
+ "stability": "stable",
23725
+ "summary": "The Datadog monitor IDs used to calculate the SLO."
23726
+ },
23727
+ "immutable": true,
23728
+ "locationInModule": {
23729
+ "filename": "src/datadogv2/datadog-slo.ts",
23730
+ "line": 25
23731
+ },
23732
+ "name": "monitorsIds",
23733
+ "type": {
23734
+ "collection": {
23735
+ "elementtype": {
23736
+ "primitive": "string"
23737
+ },
23738
+ "kind": "array"
23739
+ }
23740
+ }
23741
+ },
23742
+ {
23743
+ "abstract": true,
23744
+ "docs": {
23745
+ "stability": "stable",
23746
+ "summary": "The name of the SLO in Datadog."
23747
+ },
23748
+ "immutable": true,
23749
+ "locationInModule": {
23750
+ "filename": "src/datadogv2/datadog-slo.ts",
23751
+ "line": 18
23752
+ },
23753
+ "name": "name",
23754
+ "type": {
23755
+ "primitive": "string"
23756
+ }
23757
+ },
23758
+ {
23759
+ "abstract": true,
23760
+ "docs": {
23761
+ "remarks": "Only \"monitor\" is supported for now.\nStill this property is required to make future extensions more seamless.",
23762
+ "stability": "stable",
23763
+ "summary": "The SLO type."
23764
+ },
23765
+ "immutable": true,
23766
+ "locationInModule": {
23767
+ "filename": "src/datadogv2/datadog-slo.ts",
23768
+ "line": 48
23769
+ },
23770
+ "name": "sloType",
23771
+ "type": {
23772
+ "primitive": "string"
23773
+ }
23774
+ },
23775
+ {
23776
+ "abstract": true,
23777
+ "docs": {
23778
+ "remarks": "This is used for filtering SLOs in dashboards.",
23779
+ "stability": "stable",
23780
+ "summary": "The product stream that this SLO belongs to."
23781
+ },
23782
+ "immutable": true,
23783
+ "locationInModule": {
23784
+ "filename": "src/datadogv2/datadog-slo.ts",
23785
+ "line": 30
23786
+ },
23787
+ "name": "stream",
23788
+ "type": {
23789
+ "primitive": "string"
23790
+ }
23791
+ },
23792
+ {
23793
+ "abstract": true,
23794
+ "docs": {
23795
+ "remarks": "Unless you have a specific use case it is recommended to use defaultThresholds(...).",
23796
+ "stability": "stable",
23797
+ "summary": "Threshold definitions applied to the SLO."
23798
+ },
23799
+ "immutable": true,
23800
+ "locationInModule": {
23801
+ "filename": "src/datadogv2/datadog-slo.ts",
23802
+ "line": 42
23803
+ },
23804
+ "name": "thresholds",
23805
+ "type": {
23806
+ "collection": {
23807
+ "elementtype": {
23808
+ "fqn": "@rio-cloud/cdk-v2-constructs.datadogv2.DatadogSloThreshold"
23809
+ },
23810
+ "kind": "array"
23811
+ }
23812
+ }
23813
+ },
23814
+ {
23815
+ "abstract": true,
23816
+ "docs": {
23817
+ "remarks": "Each entry must follow the \"key:value\" format.",
23818
+ "stability": "stable",
23819
+ "summary": "Additional tags to apply to the SLO."
23820
+ },
23821
+ "immutable": true,
23822
+ "locationInModule": {
23823
+ "filename": "src/datadogv2/datadog-slo.ts",
23824
+ "line": 55
23825
+ },
23826
+ "name": "extraTags",
23827
+ "optional": true,
23828
+ "type": {
23829
+ "collection": {
23830
+ "elementtype": {
23831
+ "primitive": "string"
23832
+ },
23833
+ "kind": "array"
23834
+ }
23835
+ }
23836
+ }
23837
+ ],
23838
+ "symbolId": "src/datadogv2/datadog-slo:DatadogSloProps"
23839
+ },
23840
+ "@rio-cloud/cdk-v2-constructs.datadogv2.DatadogSloThreshold": {
23841
+ "assembly": "@rio-cloud/cdk-v2-constructs",
23842
+ "datatype": true,
23843
+ "docs": {
23844
+ "stability": "stable"
23845
+ },
23846
+ "fqn": "@rio-cloud/cdk-v2-constructs.datadogv2.DatadogSloThreshold",
23847
+ "kind": "interface",
23848
+ "locationInModule": {
23849
+ "filename": "src/datadogv2/datadog-slo.ts",
23850
+ "line": 58
23851
+ },
23852
+ "name": "DatadogSloThreshold",
23853
+ "namespace": "datadogv2",
23854
+ "properties": [
23855
+ {
23856
+ "abstract": true,
23857
+ "docs": {
23858
+ "stability": "stable",
23859
+ "summary": "The target percentage for the SLO (for example 99.5)."
23860
+ },
23861
+ "immutable": true,
23862
+ "locationInModule": {
23863
+ "filename": "src/datadogv2/datadog-slo.ts",
23864
+ "line": 63
23865
+ },
23866
+ "name": "targetPercentage",
23867
+ "type": {
23868
+ "primitive": "number"
23869
+ }
23870
+ },
23871
+ {
23872
+ "abstract": true,
23873
+ "docs": {
23874
+ "stability": "stable",
23875
+ "summary": "The timeframe for the target (for example \"7d\", \"30d\", \"90d\")."
23876
+ },
23877
+ "immutable": true,
23878
+ "locationInModule": {
23879
+ "filename": "src/datadogv2/datadog-slo.ts",
23880
+ "line": 68
23881
+ },
23882
+ "name": "timeframe",
23883
+ "type": {
23884
+ "primitive": "string"
23885
+ }
23886
+ }
23887
+ ],
23888
+ "symbolId": "src/datadogv2/datadog-slo:DatadogSloThreshold"
23889
+ },
23625
23890
  "@rio-cloud/cdk-v2-constructs.datadogv2.DefaultSlackNotification": {
23626
23891
  "assembly": "@rio-cloud/cdk-v2-constructs",
23627
23892
  "docs": {
@@ -27528,5 +27793,5 @@
27528
27793
  }
27529
27794
  },
27530
27795
  "version": "0.0.0",
27531
- "fingerprint": "S9i4nxRYZl2Wdeg2ZiM0vbksAXbTDRedEeNXRIGJs60="
27796
+ "fingerprint": "KOBrD6MG1FTa5/Mcw2GQt+DYEYv9cb94xntV51wt0H8="
27532
27797
  }
@@ -0,0 +1,71 @@
1
+ import { Construct } from 'constructs';
2
+ /**
3
+ * SLO criticality levels used to select default threshold targets.
4
+ */
5
+ export type CriticalityLevel = 'high' | 'medium' | 'low';
6
+ /**
7
+ * Properties of {@link DatadogSlo}
8
+ */
9
+ export interface DatadogSloProps {
10
+ /**
11
+ * The name of the SLO in Datadog.
12
+ */
13
+ readonly name: string;
14
+ /**
15
+ * The Datadog monitor IDs used to calculate the SLO.
16
+ *
17
+ * Each entry must be a numeric string (for example "1234567").
18
+ */
19
+ readonly monitorsIds: string[];
20
+ /**
21
+ * The product stream that this SLO belongs to. This is used for filtering SLOs in dashboards.
22
+ */
23
+ readonly stream: 'platform' | 'shipper' | 'transport';
24
+ /**
25
+ * The criticality level for the SLO as defined at [Capabilities - RIO Tech Platform Docs - MAN Confluence PRD](https://confluence.collaboration-man.com/x/nLUSEw).
26
+ */
27
+ readonly criticality: CriticalityLevel;
28
+ /**
29
+ * Threshold definitions applied to the SLO.
30
+ *
31
+ * Unless you have a specific use case it is recommended to use defaultThresholds(...).
32
+ */
33
+ readonly thresholds: DatadogSloThreshold[];
34
+ /**
35
+ * The SLO type. Only "monitor" is supported for now.
36
+ * Still this property is required to make future extensions more seamless.
37
+ */
38
+ readonly sloType: 'monitor';
39
+ /**
40
+ * Additional tags to apply to the SLO.
41
+ *
42
+ * Each entry must follow the "key:value" format.
43
+ */
44
+ readonly extraTags?: string[];
45
+ }
46
+ export interface DatadogSloThreshold {
47
+ /**
48
+ * The target percentage for the SLO (for example 99.5).
49
+ */
50
+ readonly targetPercentage: number;
51
+ /**
52
+ * The timeframe for the target (for example "7d", "30d", "90d").
53
+ */
54
+ readonly timeframe: DatadogSloTimeframe;
55
+ }
56
+ export declare const DATADOG_SLO_TIMEFRAMES: readonly ["7d", "30d", "90d"];
57
+ export type DatadogSloTimeframe = typeof DATADOG_SLO_TIMEFRAMES[number];
58
+ /**
59
+ * Generate default SLO thresholds for a given criticality level.
60
+ */
61
+ export declare function defaultThresholds(level: CriticalityLevel): DatadogSloThreshold[];
62
+ /**
63
+ * Wrapper around Datadog's CloudFormation `Datadog::SLOs::SLO` with some RIO defaults applied.
64
+ *
65
+ * Currently only SLOs of Type Monitor are implemented.
66
+ */
67
+ export declare class DatadogSlo extends Construct {
68
+ private readonly slo;
69
+ constructor(scope: Construct, id: string, props: DatadogSloProps);
70
+ get sloId(): string;
71
+ }
@@ -0,0 +1,118 @@
1
+ "use strict";
2
+ var _a;
3
+ Object.defineProperty(exports, "__esModule", { value: true });
4
+ exports.DatadogSlo = exports.DATADOG_SLO_TIMEFRAMES = void 0;
5
+ exports.defaultThresholds = defaultThresholds;
6
+ const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
7
+ const cdk = require("aws-cdk-lib");
8
+ const constructs_1 = require("constructs");
9
+ const rio_landing_zone_1 = require("../rio-landing-zone");
10
+ exports.DATADOG_SLO_TIMEFRAMES = ['7d', '30d', '90d'];
11
+ /**
12
+ * Generate default SLO thresholds for a given criticality level.
13
+ */
14
+ function defaultThresholds(level) {
15
+ // [Capabilities - RIO Tech Platform Docs - MAN Confluence PRD](https://confluence.collaboration-man.com/x/nLUSEw)
16
+ const targets = {
17
+ high: 99.5,
18
+ medium: 98,
19
+ low: 95,
20
+ };
21
+ return exports.DATADOG_SLO_TIMEFRAMES.map((timeframe) => ({
22
+ targetPercentage: targets[level],
23
+ timeframe: timeframe,
24
+ }));
25
+ }
26
+ /**
27
+ * Wrapper around Datadog's CloudFormation `Datadog::SLOs::SLO` with some RIO defaults applied.
28
+ *
29
+ * Currently only SLOs of Type Monitor are implemented.
30
+ */
31
+ class DatadogSlo extends constructs_1.Construct {
32
+ constructor(scope, id, props) {
33
+ super(scope, id);
34
+ validateName(props.name);
35
+ validateStream(props.stream);
36
+ validateMonitorsIds(props.monitorsIds);
37
+ validateThresholds(props.thresholds);
38
+ validateType(props.sloType);
39
+ validateCustomTags(props.extraTags);
40
+ if (!matchesDefaultThresholdsForCriticality(props.criticality, props.thresholds)) {
41
+ cdk.Annotations.of(this).addWarning('DatadogSlo thresholds do not match the defaults for the specified criticality level.');
42
+ }
43
+ const teamName = rio_landing_zone_1.RioLandingZone.getTeamNameParameter(this);
44
+ const teamIdentifier = rio_landing_zone_1.RioLandingZone.getTeamIdentifierParameter(this);
45
+ const accountName = rio_landing_zone_1.RioLandingZone.getAccountNameParameter(this);
46
+ // Cloudformation expects uppercase property keys
47
+ const thresholdsForCfn = props.thresholds.map(({ targetPercentage, timeframe }) => ({
48
+ Target: targetPercentage,
49
+ Timeframe: timeframe,
50
+ }));
51
+ // https://github.com/DataDog/datadog-cloudformation-resources/tree/master/datadog-slos-slo-handler
52
+ this.slo = new cdk.CfnResource(this, 'SLO', {
53
+ type: 'Datadog::SLOs::SLO',
54
+ properties: {
55
+ Name: props.name,
56
+ Type: props.sloType,
57
+ MonitorIds: props.monitorsIds,
58
+ Thresholds: thresholdsForCfn,
59
+ Tags: [
60
+ ...(props.extraTags ?? []),
61
+ `criticality:${props.criticality}`,
62
+ `stream:${props.stream}`,
63
+ `account_id:${cdk.Stack.of(this).account}`,
64
+ `account_name:${accountName.stringValue}`,
65
+ `team:${teamName.stringValue}`,
66
+ `team_identifier:${teamIdentifier.stringValue}`,
67
+ ],
68
+ },
69
+ });
70
+ }
71
+ get sloId() {
72
+ return cdk.Fn.ref(this.slo.logicalId);
73
+ }
74
+ }
75
+ exports.DatadogSlo = DatadogSlo;
76
+ _a = JSII_RTTI_SYMBOL_1;
77
+ DatadogSlo[_a] = { fqn: "@rio-cloud/cdk-v2-constructs.datadogv2.DatadogSlo", version: "0.0.0" };
78
+ function validateName(name) {
79
+ if (!name.trim()) {
80
+ throw new Error('DatadogSlo name must be non-empty.');
81
+ }
82
+ }
83
+ function validateStream(stream) {
84
+ if (!stream.trim()) {
85
+ throw new Error('DatadogSlo stream must be non-empty.');
86
+ }
87
+ }
88
+ function validateMonitorsIds(monitorsIds) {
89
+ if (monitorsIds.length === 0) {
90
+ throw new Error('DatadogSlo monitorsIds must include at least one monitor ID.');
91
+ }
92
+ if (monitorsIds.some((id) => !/^\d+$/.test(id))) {
93
+ throw new Error('DatadogSlo monitorsIds must contain only numeric values.');
94
+ }
95
+ }
96
+ function validateThresholds(thresholds) {
97
+ if (thresholds.length === 0) {
98
+ throw new Error('DatadogSlo thresholds must include at least one threshold.');
99
+ }
100
+ }
101
+ function validateType(type) {
102
+ if (type !== undefined && type !== 'monitor') {
103
+ throw new Error('DatadogSlo type must be "monitor".');
104
+ }
105
+ }
106
+ function matchesDefaultThresholdsForCriticality(level, thresholds) {
107
+ const defaultValues = defaultThresholds(level);
108
+ return JSON.stringify(defaultValues) === JSON.stringify(thresholds);
109
+ }
110
+ function validateCustomTags(customTags) {
111
+ if (customTags?.some((tag) => !tag.trim())) {
112
+ throw new Error('DatadogSlo customTags must not contain empty values.');
113
+ }
114
+ if (customTags?.some((tag) => !/^[^:]+:.+$/.test(tag))) {
115
+ throw new Error('DatadogSlo customTags must follow the "key:value" format.');
116
+ }
117
+ }
118
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"datadog-slo.js","sourceRoot":"","sources":["../../src/datadogv2/datadog-slo.ts"],"names":[],"mappings":";;;;AA6EA,8CAYC;;AAzFD,mCAAmC;AACnC,2CAAuC;AACvC,0DAAqD;AAoExC,QAAA,sBAAsB,GAAG,CAAC,IAAI,EAAE,KAAK,EAAE,KAAK,CAAU,CAAC;AAIpE;;GAEG;AACH,SAAgB,iBAAiB,CAAC,KAAuB;IACvD,kHAAkH;IAClH,MAAM,OAAO,GAAqC;QAChD,IAAI,EAAE,IAAI;QACV,MAAM,EAAE,EAAE;QACV,GAAG,EAAE,EAAE;KACR,CAAC;IAEF,OAAO,8BAAsB,CAAC,GAAG,CAAC,CAAC,SAAS,EAAE,EAAE,CAAC,CAAC;QAChD,gBAAgB,EAAE,OAAO,CAAC,KAAK,CAAC;QAChC,SAAS,EAAE,SAAS;KACrB,CAAC,CAAC,CAAC;AACN,CAAC;AAED;;;;GAIG;AACH,MAAa,UAAW,SAAQ,sBAAS;IAGvC,YAAY,KAAgB,EAAE,EAAU,EAAE,KAAsB;QAC9D,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEjB,YAAY,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;QACzB,cAAc,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;QAC7B,mBAAmB,CAAC,KAAK,CAAC,WAAW,CAAC,CAAC;QACvC,kBAAkB,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC;QACrC,YAAY,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;QAC5B,kBAAkB,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;QAEpC,IAAI,CAAC,sCAAsC,CAAC,KAAK,CAAC,WAAW,EAAE,KAAK,CAAC,UAAU,CAAC,EAAE,CAAC;YACjF,GAAG,CAAC,WAAW,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,UAAU,CACjC,sFAAsF,CACvF,CAAC;QACJ,CAAC;QAED,MAAM,QAAQ,GAAG,iCAAc,CAAC,oBAAoB,CAAC,IAAI,CAAC,CAAC;QAC3D,MAAM,cAAc,GAAG,iCAAc,CAAC,0BAA0B,CAAC,IAAI,CAAC,CAAC;QACvE,MAAM,WAAW,GAAG,iCAAc,CAAC,uBAAuB,CAAC,IAAI,CAAC,CAAC;QAEjE,iDAAiD;QACjD,MAAM,gBAAgB,GAAG,KAAK,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC,EAAE,gBAAgB,EAAE,SAAS,EAAE,EAAE,EAAE,CAAC,CAAC;YAClF,MAAM,EAAE,gBAAgB;YACxB,SAAS,EAAE,SAAS;SACrB,CAAC,CAAC,CAAC;QAEJ,mGAAmG;QACnG,IAAI,CAAC,GAAG,GAAG,IAAI,GAAG,CAAC,WAAW,CAAC,IAAI,EAAE,KAAK,EAAE;YAC1C,IAAI,EAAE,oBAAoB;YAC1B,UAAU,EAAE;gBACV,IAAI,EAAE,KAAK,CAAC,IAAI;gBAChB,IAAI,EAAE,KAAK,CAAC,OAAO;gBACnB,UAAU,EAAE,KAAK,CAAC,WAAW;gBAC7B,UAAU,EAAE,gBAAgB;gBAC5B,IAAI,EAAE;oBACJ,GAAG,CAAC,KAAK,CAAC,SAAS,IAAI,EAAE,CAAC;oBAC1B,eAAe,KAAK,CAAC,WAAW,EAAE;oBAClC,UAAU,KAAK,CAAC,MAAM,EAAE;oBACxB,cAAc,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,OAAO,EAAE;oBAC1C,gBAAgB,WAAW,CAAC,WAAW,EAAE;oBACzC,QAAQ,QAAQ,CAAC,WAAW,EAAE;oBAC9B,mBAAmB,cAAc,CAAC,WAAW,EAAE;iBAChD;aACF;SACF,CAAC,CAAC;IACL,CAAC;IAED,IAAI,KAAK;QACP,OAAO,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,IAAI,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;IACxC,CAAC;;AApDH,gCAqDC;;;AAED,SAAS,YAAY,CAAC,IAAY;IAChC,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,EAAE,CAAC;QACjB,MAAM,IAAI,KAAK,CAAC,oCAAoC,CAAC,CAAC;IACxD,CAAC;AACH,CAAC;AAED,SAAS,cAAc,CAAC,MAAc;IACpC,IAAI,CAAC,MAAM,CAAC,IAAI,EAAE,EAAE,CAAC;QACnB,MAAM,IAAI,KAAK,CAAC,sCAAsC,CAAC,CAAC;IAC1D,CAAC;AACH,CAAC;AAED,SAAS,mBAAmB,CAAC,WAAqB;IAChD,IAAI,WAAW,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QAC7B,MAAM,IAAI,KAAK,CAAC,8DAA8D,CAAC,CAAC;IAClF,CAAC;IACD,IAAI,WAAW,CAAC,IAAI,CAAC,CAAC,EAAE,EAAE,EAAE,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC,EAAE,CAAC;QAChD,MAAM,IAAI,KAAK,CAAC,0DAA0D,CAAC,CAAC;IAC9E,CAAC;AACH,CAAC;AAED,SAAS,kBAAkB,CAAC,UAAiC;IAC3D,IAAI,UAAU,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QAC5B,MAAM,IAAI,KAAK,CAAC,4DAA4D,CAAC,CAAC;IAChF,CAAC;AACH,CAAC;AAED,SAAS,YAAY,CAAC,IAAgB;IACpC,IAAI,IAAI,KAAK,SAAS,IAAI,IAAI,KAAK,SAAS,EAAE,CAAC;QAC7C,MAAM,IAAI,KAAK,CAAC,oCAAoC,CAAC,CAAC;IACxD,CAAC;AACH,CAAC;AAED,SAAS,sCAAsC,CAAC,KAAuB,EAAE,UAAiC;IACxG,MAAM,aAAa,GAAG,iBAAiB,CAAC,KAAK,CAAC,CAAC;IAC/C,OAAO,IAAI,CAAC,SAAS,CAAC,aAAa,CAAC,KAAK,IAAI,CAAC,SAAS,CAAC,UAAU,CAAC,CAAC;AACtE,CAAC;AAED,SAAS,kBAAkB,CAAC,UAAqB;IAC/C,IAAI,UAAU,EAAE,IAAI,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC,GAAG,CAAC,IAAI,EAAE,CAAC,EAAE,CAAC;QAC3C,MAAM,IAAI,KAAK,CAAC,sDAAsD,CAAC,CAAC;IAC1E,CAAC;IACD,IAAI,UAAU,EAAE,IAAI,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC,YAAY,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,CAAC;QACvD,MAAM,IAAI,KAAK,CAAC,2DAA2D,CAAC,CAAC;IAC/E,CAAC;AACH,CAAC","sourcesContent":["import * as cdk from 'aws-cdk-lib';\nimport { Construct } from 'constructs';\nimport { RioLandingZone } from '../rio-landing-zone';\n\n/**\n * SLO criticality levels used to select default threshold targets.\n */\nexport type CriticalityLevel = 'high' | 'medium' | 'low';\n\n/**\n * Properties of {@link DatadogSlo}\n */\nexport interface DatadogSloProps {\n\n  /**\n   * The name of the SLO in Datadog.\n   */\n  readonly name: string;\n\n  /**\n   * The Datadog monitor IDs used to calculate the SLO.\n   *\n   * Each entry must be a numeric string (for example \"1234567\").\n   */\n  readonly monitorsIds: string[];\n\n  /**\n   * The product stream that this SLO belongs to. This is used for filtering SLOs in dashboards.\n   */\n  readonly stream: 'platform' | 'shipper' | 'transport';\n\n  /**\n   * The criticality level for the SLO as defined at [Capabilities - RIO Tech Platform Docs - MAN Confluence PRD](https://confluence.collaboration-man.com/x/nLUSEw).\n   */\n  readonly criticality: CriticalityLevel;\n\n  /**\n   * Threshold definitions applied to the SLO.\n   *\n   * Unless you have a specific use case it is recommended to use defaultThresholds(...).\n   */\n  readonly thresholds: DatadogSloThreshold[];\n\n  /**\n   * The SLO type. Only \"monitor\" is supported for now.\n   * Still this property is required to make future extensions more seamless.\n   */\n  readonly sloType: 'monitor';\n\n  /**\n   * Additional tags to apply to the SLO.\n   *\n   * Each entry must follow the \"key:value\" format.\n   */\n  readonly extraTags?: string[];\n}\n\nexport interface DatadogSloThreshold {\n\n  /**\n   * The target percentage for the SLO (for example 99.5).\n   */\n  readonly targetPercentage: number;\n\n  /**\n   * The timeframe for the target (for example \"7d\", \"30d\", \"90d\").\n   */\n  readonly timeframe: DatadogSloTimeframe;\n}\n\nexport const DATADOG_SLO_TIMEFRAMES = ['7d', '30d', '90d'] as const;\n\nexport type DatadogSloTimeframe = typeof DATADOG_SLO_TIMEFRAMES[number];\n\n/**\n * Generate default SLO thresholds for a given criticality level.\n */\nexport function defaultThresholds(level: CriticalityLevel): DatadogSloThreshold[] {\n  // [Capabilities - RIO Tech Platform Docs - MAN Confluence PRD](https://confluence.collaboration-man.com/x/nLUSEw)\n  const targets: Record<CriticalityLevel, number> = {\n    high: 99.5,\n    medium: 98,\n    low: 95,\n  };\n\n  return DATADOG_SLO_TIMEFRAMES.map((timeframe) => ({\n    targetPercentage: targets[level],\n    timeframe: timeframe,\n  }));\n}\n\n/**\n * Wrapper around Datadog's CloudFormation `Datadog::SLOs::SLO` with some RIO defaults applied.\n *\n * Currently only SLOs of Type Monitor are implemented.\n */\nexport class DatadogSlo extends Construct {\n  private readonly slo: cdk.CfnResource;\n\n  constructor(scope: Construct, id: string, props: DatadogSloProps) {\n    super(scope, id);\n\n    validateName(props.name);\n    validateStream(props.stream);\n    validateMonitorsIds(props.monitorsIds);\n    validateThresholds(props.thresholds);\n    validateType(props.sloType);\n    validateCustomTags(props.extraTags);\n\n    if (!matchesDefaultThresholdsForCriticality(props.criticality, props.thresholds)) {\n      cdk.Annotations.of(this).addWarning(\n        'DatadogSlo thresholds do not match the defaults for the specified criticality level.',\n      );\n    }\n\n    const teamName = RioLandingZone.getTeamNameParameter(this);\n    const teamIdentifier = RioLandingZone.getTeamIdentifierParameter(this);\n    const accountName = RioLandingZone.getAccountNameParameter(this);\n\n    // Cloudformation expects uppercase property keys\n    const thresholdsForCfn = props.thresholds.map(({ targetPercentage, timeframe }) => ({\n      Target: targetPercentage,\n      Timeframe: timeframe,\n    }));\n\n    // https://github.com/DataDog/datadog-cloudformation-resources/tree/master/datadog-slos-slo-handler\n    this.slo = new cdk.CfnResource(this, 'SLO', {\n      type: 'Datadog::SLOs::SLO',\n      properties: {\n        Name: props.name,\n        Type: props.sloType,\n        MonitorIds: props.monitorsIds,\n        Thresholds: thresholdsForCfn,\n        Tags: [\n          ...(props.extraTags ?? []),\n          `criticality:${props.criticality}`,\n          `stream:${props.stream}`,\n          `account_id:${cdk.Stack.of(this).account}`,\n          `account_name:${accountName.stringValue}`,\n          `team:${teamName.stringValue}`,\n          `team_identifier:${teamIdentifier.stringValue}`,\n        ],\n      },\n    });\n  }\n\n  get sloId() {\n    return cdk.Fn.ref(this.slo.logicalId);\n  }\n}\n\nfunction validateName(name: string) {\n  if (!name.trim()) {\n    throw new Error('DatadogSlo name must be non-empty.');\n  }\n}\n\nfunction validateStream(stream: string) {\n  if (!stream.trim()) {\n    throw new Error('DatadogSlo stream must be non-empty.');\n  }\n}\n\nfunction validateMonitorsIds(monitorsIds: string[]) {\n  if (monitorsIds.length === 0) {\n    throw new Error('DatadogSlo monitorsIds must include at least one monitor ID.');\n  }\n  if (monitorsIds.some((id) => !/^\\d+$/.test(id))) {\n    throw new Error('DatadogSlo monitorsIds must contain only numeric values.');\n  }\n}\n\nfunction validateThresholds(thresholds: DatadogSloThreshold[]) {\n  if (thresholds.length === 0) {\n    throw new Error('DatadogSlo thresholds must include at least one threshold.');\n  }\n}\n\nfunction validateType(type?: 'monitor') {\n  if (type !== undefined && type !== 'monitor') {\n    throw new Error('DatadogSlo type must be \"monitor\".');\n  }\n}\n\nfunction matchesDefaultThresholdsForCriticality(level: CriticalityLevel, thresholds: DatadogSloThreshold[]) {\n  const defaultValues = defaultThresholds(level);\n  return JSON.stringify(defaultValues) === JSON.stringify(thresholds);\n}\n\nfunction validateCustomTags(customTags?: string[]) {\n  if (customTags?.some((tag) => !tag.trim())) {\n    throw new Error('DatadogSlo customTags must not contain empty values.');\n  }\n  if (customTags?.some((tag) => !/^[^:]+:.+$/.test(tag))) {\n    throw new Error('DatadogSlo customTags must follow the \"key:value\" format.');\n  }\n}\n"]}
@@ -5,3 +5,4 @@ export * from './datadog-notification';
5
5
  export * from './datadog-lambda-instrumentation';
6
6
  export * from './datadog-shared-secrets';
7
7
  export * from './datadog-service-catalog-entry';
8
+ export * from './datadog-slo';
@@ -21,4 +21,5 @@ __exportStar(require("./datadog-notification"), exports);
21
21
  __exportStar(require("./datadog-lambda-instrumentation"), exports);
22
22
  __exportStar(require("./datadog-shared-secrets"), exports);
23
23
  __exportStar(require("./datadog-service-catalog-entry"), exports);
24
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvZGF0YWRvZ3YyL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7QUFBQSxzREFBb0M7QUFDcEMsaUVBQStDO0FBQy9DLG9EQUFrQztBQUNsQyx5REFBdUM7QUFDdkMsbUVBQWlEO0FBQ2pELDJEQUF5QztBQUN6QyxrRUFBZ0QiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL2RhdGFkb2ctZGFzaGJvYXJkJztcbmV4cG9ydCAqIGZyb20gJy4vZGF0YWRvZy1sb2ctaW5kZXgtbW9uaXRvcmluZyc7XG5leHBvcnQgKiBmcm9tICcuL2RhdGFkb2ctbW9uaXRvcic7XG5leHBvcnQgKiBmcm9tICcuL2RhdGFkb2ctbm90aWZpY2F0aW9uJztcbmV4cG9ydCAqIGZyb20gJy4vZGF0YWRvZy1sYW1iZGEtaW5zdHJ1bWVudGF0aW9uJztcbmV4cG9ydCAqIGZyb20gJy4vZGF0YWRvZy1zaGFyZWQtc2VjcmV0cyc7XG5leHBvcnQgKiBmcm9tICcuL2RhdGFkb2ctc2VydmljZS1jYXRhbG9nLWVudHJ5JzsiXX0=
24
+ __exportStar(require("./datadog-slo"), exports);
25
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvZGF0YWRvZ3YyL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7QUFBQSxzREFBb0M7QUFDcEMsaUVBQStDO0FBQy9DLG9EQUFrQztBQUNsQyx5REFBdUM7QUFDdkMsbUVBQWlEO0FBQ2pELDJEQUF5QztBQUN6QyxrRUFBZ0Q7QUFDaEQsZ0RBQThCIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9kYXRhZG9nLWRhc2hib2FyZCc7XG5leHBvcnQgKiBmcm9tICcuL2RhdGFkb2ctbG9nLWluZGV4LW1vbml0b3JpbmcnO1xuZXhwb3J0ICogZnJvbSAnLi9kYXRhZG9nLW1vbml0b3InO1xuZXhwb3J0ICogZnJvbSAnLi9kYXRhZG9nLW5vdGlmaWNhdGlvbic7XG5leHBvcnQgKiBmcm9tICcuL2RhdGFkb2ctbGFtYmRhLWluc3RydW1lbnRhdGlvbic7XG5leHBvcnQgKiBmcm9tICcuL2RhdGFkb2ctc2hhcmVkLXNlY3JldHMnO1xuZXhwb3J0ICogZnJvbSAnLi9kYXRhZG9nLXNlcnZpY2UtY2F0YWxvZy1lbnRyeSc7XG5leHBvcnQgKiBmcm9tICcuL2RhdGFkb2ctc2xvJztcbiJdfQ==
package/package.json CHANGED
@@ -15,7 +15,7 @@
15
15
  ],
16
16
  "main": "lib/index.js",
17
17
  "license": "Apache-2.0",
18
- "version": "7.13.12",
18
+ "version": "7.14.0-alpha.0",
19
19
  "types": "lib/index.d.ts",
20
20
  "stability": "stable",
21
21
  "exports": {
package/version.json CHANGED
@@ -1,3 +1,3 @@
1
1
  {
2
- "version": "7.13.12"
2
+ "version": "7.14.0-alpha.0"
3
3
  }