@cdklabs/multi-az-observability 0.0.0-alpha.0 → 0.0.1-alpha.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.jsii +138 -98
- package/API.md +54 -64
- package/lib/alarmsandrules/AvailabilityAndLatencyAlarmsAndRules.d.ts +18 -2
- package/lib/alarmsandrules/AvailabilityAndLatencyAlarmsAndRules.js +136 -7
- package/lib/alarmsandrules/BaseOperationZonalAlarmsAndRules.d.ts +2 -2
- package/lib/alarmsandrules/BaseOperationZonalAlarmsAndRules.js +1 -18
- package/lib/alarmsandrules/CanaryOperationZonalAlarmsAndRules.d.ts +8 -0
- package/lib/alarmsandrules/CanaryOperationZonalAlarmsAndRules.js +18 -1
- package/lib/alarmsandrules/ServerSideOperationZonalAlarmsAndRules.d.ts +8 -0
- package/lib/alarmsandrules/ServerSideOperationZonalAlarmsAndRules.js +18 -1
- package/lib/alarmsandrules/ServiceAlarmsAndRules.js +3 -3
- package/lib/alarmsandrules/props/CanaryOperationZonalAlarmsAndRulesProps.js +1 -1
- package/lib/azmapper/AvailabilityZoneMapper.js +1 -1
- package/lib/basic_observability/BasicServiceDashboard.js +130 -0
- package/lib/{services → basic_observability}/BasicServiceMultiAZObservability.d.ts +9 -7
- package/lib/basic_observability/BasicServiceMultiAZObservability.js +434 -0
- package/lib/basic_observability/IBasicServiceMultiAZObservability.d.ts +51 -0
- package/lib/basic_observability/IBasicServiceMultiAZObservability.js +5 -0
- package/lib/{dashboards → basic_observability}/props/BasicServiceDashboardProps.d.ts +3 -0
- package/lib/basic_observability/props/BasicServiceDashboardProps.js +3 -0
- package/lib/{services → basic_observability}/props/BasicServiceMultiAZObservabilityProps.d.ts +29 -46
- package/lib/basic_observability/props/BasicServiceMultiAZObservabilityProps.js +5 -0
- package/lib/basic_observability/props/RegionalApplicationLoadBalancerAvailabilityMetricProps.d.ts +8 -0
- package/lib/basic_observability/props/RegionalApplicationLoadBalancerAvailabilityMetricProps.js +3 -0
- package/lib/basic_observability/props/RegionalApplicationLoadBalancerLatencyMetricProps.d.ts +8 -0
- package/lib/basic_observability/props/RegionalApplicationLoadBalancerLatencyMetricProps.js +3 -0
- package/lib/basic_observability/props/ZonalApplicationLoadBalancerAvailabilityMetricProps.d.ts +10 -0
- package/lib/basic_observability/props/ZonalApplicationLoadBalancerAvailabilityMetricProps.js +3 -0
- package/lib/basic_observability/props/ZonalApplicationLoadBalancerLatencyMetricProps.d.ts +10 -0
- package/lib/basic_observability/props/ZonalApplicationLoadBalancerLatencyMetricProps.js +3 -0
- package/lib/canaries/src/canary.zip +0 -0
- package/lib/dashboards/OperationAvailabilityAndLatencyDashboard.js +16 -18
- package/lib/dashboards/ServiceAvailabilityAndLatencyDashboard.js +19 -9
- package/lib/index.d.ts +2 -2
- package/lib/index.js +2 -2
- package/lib/metrics/ApplicationLoadBalancerMetrics.d.ts +55 -17
- package/lib/metrics/ApplicationLoadBalancerMetrics.js +459 -105
- package/lib/metrics/AvailabilityAndLatencyMetrics.d.ts +0 -11
- package/lib/metrics/AvailabilityAndLatencyMetrics.js +1 -24
- package/lib/metrics/RegionalLatencyMetrics.js +7 -6
- package/lib/metrics/ZonalAvailabilityMetrics.js +2 -2
- package/lib/metrics/ZonalLatencyMetrics.js +4 -3
- package/lib/monitoring/src/monitoring-layer.zip +0 -0
- package/lib/outlier-detection/OutlierDetectionFunction.js +5 -5
- package/lib/outlier-detection/src/outlier-detection.zip +0 -0
- package/lib/outlier-detection/src/scipy-layer.zip +0 -0
- package/lib/services/CanaryMetrics.js +1 -1
- package/lib/services/CanaryTestMetricsOverride.js +1 -1
- package/lib/services/ContributorInsightRuleDetails.js +1 -1
- package/lib/services/InstrumentedServiceMultiAZObservability.js +1 -1
- package/lib/services/Operation.js +1 -1
- package/lib/services/OperationMetricDetails.js +1 -1
- package/lib/services/Service.js +1 -1
- package/lib/services/ServiceMetricDetails.js +1 -1
- package/lib/services/props/MetricDimensions.js +1 -1
- package/lib/utilities/LatencyMetricType.d.ts +5 -1
- package/lib/utilities/LatencyMetricType.js +5 -1
- package/lib/utilities/MetricsHelper.d.ts +13 -0
- package/lib/utilities/MetricsHelper.js +30 -0
- package/package.json +8 -8
- package/lib/dashboards/BasicServiceDashboard.js +0 -130
- package/lib/dashboards/props/BasicServiceDashboardProps.js +0 -3
- package/lib/services/BasicServiceMultiAZObservability.js +0 -504
- package/lib/services/props/BasicServiceMultiAZObservabilityProps.js +0 -3
- /package/lib/{dashboards → basic_observability}/BasicServiceDashboard.d.ts +0 -0
|
@@ -50,15 +50,17 @@ export declare class BasicServiceMultiAZObservability extends Construct implemen
|
|
|
50
50
|
*/
|
|
51
51
|
dashboard?: Dashboard;
|
|
52
52
|
/**
|
|
53
|
-
* The
|
|
53
|
+
* The AZ mapper resource
|
|
54
54
|
*/
|
|
55
|
-
private
|
|
56
|
-
private azMapper;
|
|
57
|
-
private _natGWZonalIsolatedImpactAlarms;
|
|
58
|
-
private _albZonalIsolatedImpactAlarms;
|
|
59
|
-
private _packetDropsPerZone;
|
|
60
|
-
private _faultsPerZone;
|
|
55
|
+
private _azMapper;
|
|
61
56
|
constructor(scope: Construct, id: string, props: BasicServiceMultiAZObservabilityProps);
|
|
57
|
+
private static isThereAnAZAvailabilityImpactAlb;
|
|
58
|
+
private static isThereAZLatencyImpactAlb;
|
|
59
|
+
private static isAZAnOutlierForAvailabilityAlb;
|
|
60
|
+
private static isAZAnOutlierForLatencyAlb;
|
|
62
61
|
private doAlbMetrics;
|
|
62
|
+
private static isThereAnAZPacketLossImpactNATGW;
|
|
63
|
+
private static isAZAnOutlierForPacketLossNATGW;
|
|
63
64
|
private doNatGatewayMetrics;
|
|
65
|
+
private getTotalPacketDropsPerZone;
|
|
64
66
|
}
|
|
@@ -0,0 +1,434 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var _a;
|
|
3
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
4
|
+
exports.BasicServiceMultiAZObservability = void 0;
|
|
5
|
+
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
|
6
|
+
// Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
|
|
7
|
+
// SPDX-License-Identifier: Apache-2.0
|
|
8
|
+
const aws_cdk_lib_1 = require("aws-cdk-lib");
|
|
9
|
+
const aws_cloudwatch_1 = require("aws-cdk-lib/aws-cloudwatch");
|
|
10
|
+
const constructs_1 = require("constructs");
|
|
11
|
+
const AvailabilityZoneMapper_1 = require("../azmapper/AvailabilityZoneMapper");
|
|
12
|
+
const BasicServiceDashboard_1 = require("./BasicServiceDashboard");
|
|
13
|
+
const ApplicationLoadBalancerMetrics_1 = require("../metrics/ApplicationLoadBalancerMetrics");
|
|
14
|
+
const AvailabilityMetricType_1 = require("../utilities/AvailabilityMetricType");
|
|
15
|
+
const MetricsHelper_1 = require("../utilities/MetricsHelper");
|
|
16
|
+
/**
|
|
17
|
+
* Basic observability for a service using metrics from
|
|
18
|
+
* ALBs and NAT Gateways
|
|
19
|
+
*/
|
|
20
|
+
class BasicServiceMultiAZObservability extends constructs_1.Construct {
|
|
21
|
+
constructor(scope, id, props) {
|
|
22
|
+
super(scope, id);
|
|
23
|
+
// Initialize class properties
|
|
24
|
+
this.serviceName = props.serviceName;
|
|
25
|
+
this.applicationLoadBalancers = props.applicationLoadBalancers;
|
|
26
|
+
this.natGateways = props.natGateways;
|
|
27
|
+
this.aggregateZonalIsolatedImpactAlarms = {};
|
|
28
|
+
this.albZonalIsolatedImpactAlarms = {};
|
|
29
|
+
this.natGWZonalIsolatedImpactAlarms = {};
|
|
30
|
+
// Create the AZ mapper resource to translate AZ names to ids
|
|
31
|
+
this._azMapper = new AvailabilityZoneMapper_1.AvailabilityZoneMapper(this, 'availability-zone-mapper');
|
|
32
|
+
// Create ALB metrics and alarms per AZ
|
|
33
|
+
if (this.applicationLoadBalancers) {
|
|
34
|
+
this.albZonalIsolatedImpactAlarms = this.doAlbMetrics(props);
|
|
35
|
+
}
|
|
36
|
+
// Create NAT Gateway metrics and alarms per AZ
|
|
37
|
+
if (this.natGateways) {
|
|
38
|
+
this.natGWZonalIsolatedImpactAlarms = this.doNatGatewayMetrics(props);
|
|
39
|
+
}
|
|
40
|
+
// Look through all of the per AZ ALB alarms, if there's also a NAT GW alarm
|
|
41
|
+
// create a composite alarm if either of them trigger
|
|
42
|
+
Object.keys(this.albZonalIsolatedImpactAlarms).forEach((az) => {
|
|
43
|
+
let azLetter = az.substring(az.length - 1);
|
|
44
|
+
let availabilityZoneId = this._azMapper.availabilityZoneIdFromAvailabilityZoneLetter(azLetter);
|
|
45
|
+
if (az in this.natGWZonalIsolatedImpactAlarms) {
|
|
46
|
+
this.aggregateZonalIsolatedImpactAlarms[az] = new aws_cloudwatch_1.CompositeAlarm(this, az.substring(az.length - 1) + "-isolated-impact-alarm", {
|
|
47
|
+
alarmRule: aws_cloudwatch_1.AlarmRule.anyOf(this.albZonalIsolatedImpactAlarms[az], this.natGWZonalIsolatedImpactAlarms[az]),
|
|
48
|
+
compositeAlarmName: availabilityZoneId + "-isolated-impact-alarm"
|
|
49
|
+
});
|
|
50
|
+
}
|
|
51
|
+
else {
|
|
52
|
+
this.aggregateZonalIsolatedImpactAlarms[az] = new aws_cloudwatch_1.CompositeAlarm(this, az.substring(az.length - 1) + "-isolated-impact-alarm", {
|
|
53
|
+
alarmRule: aws_cloudwatch_1.AlarmRule.anyOf(this.albZonalIsolatedImpactAlarms[az]),
|
|
54
|
+
compositeAlarmName: availabilityZoneId + "-isolated-impact-alarm"
|
|
55
|
+
});
|
|
56
|
+
}
|
|
57
|
+
});
|
|
58
|
+
// Look through all of the per AZ NAT GW alarms. If there's an AZ we haven't seen in the ALB
|
|
59
|
+
// alarms yet, then it will just be a NAT GW alarm that we'll turn into the same kind of
|
|
60
|
+
// composite alarm
|
|
61
|
+
Object.keys(this.natGWZonalIsolatedImpactAlarms).forEach((az) => {
|
|
62
|
+
if (!(az in this.aggregateZonalIsolatedImpactAlarms)) {
|
|
63
|
+
let azLetter = az.substring(az.length - 1);
|
|
64
|
+
let availabilityZoneId = this._azMapper.availabilityZoneIdFromAvailabilityZoneLetter(azLetter);
|
|
65
|
+
this.aggregateZonalIsolatedImpactAlarms[az] = new aws_cloudwatch_1.CompositeAlarm(this, az.substring(az.length - 1) + "-isolated-impact-alarm", {
|
|
66
|
+
alarmRule: aws_cloudwatch_1.AlarmRule.anyOf(this.natGWZonalIsolatedImpactAlarms[az]),
|
|
67
|
+
compositeAlarmName: availabilityZoneId + "-isolated-impact-alarm"
|
|
68
|
+
});
|
|
69
|
+
}
|
|
70
|
+
});
|
|
71
|
+
// Should we create the dashboard
|
|
72
|
+
if (props.createDashboard == true) {
|
|
73
|
+
this.dashboard = new BasicServiceDashboard_1.BasicServiceDashboard(this, 'BasicServiceDashboard', {
|
|
74
|
+
serviceName: props.serviceName.toLowerCase(),
|
|
75
|
+
zonalAggregateIsolatedImpactAlarms: this.aggregateZonalIsolatedImpactAlarms,
|
|
76
|
+
zonalLoadBalancerIsolatedImpactAlarms: this.albZonalIsolatedImpactAlarms,
|
|
77
|
+
zonalNatGatewayIsolatedImpactAlarms: this.natGWZonalIsolatedImpactAlarms,
|
|
78
|
+
interval: props.interval,
|
|
79
|
+
zonalLoadBalancerFaultRateMetrics: ApplicationLoadBalancerMetrics_1.ApplicationLoadBalancerMetrics.getTotalAlbFaultCountPerZone(props.applicationLoadBalancers ? props.applicationLoadBalancers : [], props.period ? props.period : aws_cdk_lib_1.Duration.minutes(1), this._azMapper),
|
|
80
|
+
zonalNatGatewayPacketDropMetrics: this.getTotalPacketDropsPerZone(props.natGateways ? props.natGateways : {}, props.period ? props.period : aws_cdk_lib_1.Duration.minutes(1)),
|
|
81
|
+
azMapper: this._azMapper,
|
|
82
|
+
}).dashboard;
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
static isThereAnAZAvailabilityImpactAlb(scope, alb, availabilityZoneId, availabilityZone, threshold, keyprefix, period, evaluationPeriods, datapointsToAlarm) {
|
|
86
|
+
// Create a fault rate alarm for the ALB in the specified AZ
|
|
87
|
+
return new aws_cloudwatch_1.Alarm(scope, keyprefix + '-fault-rate-alarm', {
|
|
88
|
+
alarmName: availabilityZoneId + '-' + alb.loadBalancerArn + '-fault-rate',
|
|
89
|
+
actionsEnabled: false,
|
|
90
|
+
metric: ApplicationLoadBalancerMetrics_1.ApplicationLoadBalancerMetrics.getPerAZAvailabilityMetric(alb, {
|
|
91
|
+
period: period,
|
|
92
|
+
label: availabilityZoneId + '-' + alb.loadBalancerArn + '-fault-rate',
|
|
93
|
+
availabilityZone: availabilityZone,
|
|
94
|
+
availabilityZoneId: availabilityZoneId,
|
|
95
|
+
metricType: AvailabilityMetricType_1.AvailabilityMetricType.FAULT_RATE
|
|
96
|
+
}),
|
|
97
|
+
evaluationPeriods: evaluationPeriods,
|
|
98
|
+
datapointsToAlarm: datapointsToAlarm,
|
|
99
|
+
threshold: threshold,
|
|
100
|
+
comparisonOperator: aws_cloudwatch_1.ComparisonOperator.GREATER_THAN_THRESHOLD,
|
|
101
|
+
treatMissingData: aws_cloudwatch_1.TreatMissingData.IGNORE
|
|
102
|
+
});
|
|
103
|
+
}
|
|
104
|
+
static isThereAZLatencyImpactAlb(scope, alb, availabilityZoneId, availabilityZone, threshold, statistic, keyprefix, period, evaluationPeriods, datapointsToAlarm) {
|
|
105
|
+
// Create a fault rate alarm for the ALB in the specified AZ
|
|
106
|
+
return new aws_cloudwatch_1.Alarm(scope, keyprefix + '-latency-alarm', {
|
|
107
|
+
alarmName: availabilityZoneId + '-' + alb.loadBalancerArn + '-latency',
|
|
108
|
+
actionsEnabled: false,
|
|
109
|
+
metric: ApplicationLoadBalancerMetrics_1.ApplicationLoadBalancerMetrics.getPerAZLatencyMetric({
|
|
110
|
+
alb: alb,
|
|
111
|
+
availabilityZone: availabilityZone,
|
|
112
|
+
availabilityZoneId: availabilityZoneId,
|
|
113
|
+
label: availabilityZoneId + "-" + alb.loadBalancerArn + "-target-latency",
|
|
114
|
+
period: period,
|
|
115
|
+
statistic: statistic
|
|
116
|
+
}),
|
|
117
|
+
evaluationPeriods: evaluationPeriods,
|
|
118
|
+
datapointsToAlarm: datapointsToAlarm,
|
|
119
|
+
threshold: threshold,
|
|
120
|
+
comparisonOperator: aws_cloudwatch_1.ComparisonOperator.GREATER_THAN_THRESHOLD,
|
|
121
|
+
treatMissingData: aws_cloudwatch_1.TreatMissingData.IGNORE
|
|
122
|
+
});
|
|
123
|
+
}
|
|
124
|
+
static isAZAnOutlierForAvailabilityAlb(scope, alb, availabilityZoneId, availabilityZone, threshold, keyprefix, period, evaluationPeriods, datapointsToAlarm) {
|
|
125
|
+
let usingMetrics = {};
|
|
126
|
+
let azMetricId;
|
|
127
|
+
alb.vpc.availabilityZones.forEach((az) => {
|
|
128
|
+
let azFaultCount = ApplicationLoadBalancerMetrics_1.ApplicationLoadBalancerMetrics.getPerAZAvailabilityMetric(alb, {
|
|
129
|
+
metricType: AvailabilityMetricType_1.AvailabilityMetricType.FAULT_COUNT,
|
|
130
|
+
availabilityZone: availabilityZone,
|
|
131
|
+
availabilityZoneId: availabilityZoneId,
|
|
132
|
+
period: period,
|
|
133
|
+
label: availabilityZoneId + "-" + alb.loadBalancerArn + "-fault-count",
|
|
134
|
+
keyprefix: keyprefix
|
|
135
|
+
});
|
|
136
|
+
keyprefix = MetricsHelper_1.MetricsHelper.nextChar(keyprefix);
|
|
137
|
+
usingMetrics[`${keyprefix}1`] = azFaultCount;
|
|
138
|
+
if (az == availabilityZone) {
|
|
139
|
+
azMetricId = Object.keys(usingMetrics)[-1];
|
|
140
|
+
}
|
|
141
|
+
keyprefix = MetricsHelper_1.MetricsHelper.nextChar(keyprefix);
|
|
142
|
+
});
|
|
143
|
+
return new aws_cloudwatch_1.Alarm(scope, keyprefix + '-availability-outlier-alarm', {
|
|
144
|
+
alarmName: availabilityZoneId + '-' + alb.loadBalancerArn + '-availability-impact-outlier',
|
|
145
|
+
actionsEnabled: false,
|
|
146
|
+
metric: new aws_cloudwatch_1.MathExpression({
|
|
147
|
+
expression: `${azMetricId}/(${Object.keys(usingMetrics).join("+")})`,
|
|
148
|
+
usingMetrics: usingMetrics,
|
|
149
|
+
label: availabilityZoneId + '-' + alb.loadBalancerArn + '-percent-of-faults',
|
|
150
|
+
period: period,
|
|
151
|
+
}),
|
|
152
|
+
evaluationPeriods: evaluationPeriods,
|
|
153
|
+
datapointsToAlarm: datapointsToAlarm,
|
|
154
|
+
threshold: threshold,
|
|
155
|
+
comparisonOperator: aws_cloudwatch_1.ComparisonOperator.GREATER_THAN_THRESHOLD,
|
|
156
|
+
treatMissingData: aws_cloudwatch_1.TreatMissingData.IGNORE
|
|
157
|
+
});
|
|
158
|
+
}
|
|
159
|
+
static isAZAnOutlierForLatencyAlb(scope, alb, availabilityZoneId, availabilityZone, statistic, period, evaluationPeriods, datapointsToAlarm, keyprefix) {
|
|
160
|
+
let usingMetrics = {};
|
|
161
|
+
let azMetricId;
|
|
162
|
+
alb.vpc.availabilityZones.forEach((az, index) => {
|
|
163
|
+
// Target response time
|
|
164
|
+
let targetResponseTime = ApplicationLoadBalancerMetrics_1.ApplicationLoadBalancerMetrics.getPerAZLatencyMetric({
|
|
165
|
+
alb: alb,
|
|
166
|
+
availabilityZone: az,
|
|
167
|
+
label: az + "-target-response-time",
|
|
168
|
+
statistic: statistic,
|
|
169
|
+
period: period
|
|
170
|
+
});
|
|
171
|
+
if (az == availabilityZone) {
|
|
172
|
+
azMetricId = `a${index}`;
|
|
173
|
+
usingMetrics[`a${index}`] = targetResponseTime;
|
|
174
|
+
}
|
|
175
|
+
else {
|
|
176
|
+
usingMetrics[`b${index}`] = targetResponseTime;
|
|
177
|
+
}
|
|
178
|
+
});
|
|
179
|
+
return new aws_cloudwatch_1.Alarm(scope, keyprefix + "-latency-outlier-alarm", {
|
|
180
|
+
alarmName: availabilityZoneId + '-' + alb.loadBalancerArn + '-latency-impact-outlier',
|
|
181
|
+
actionsEnabled: false,
|
|
182
|
+
metric: new aws_cloudwatch_1.MathExpression({
|
|
183
|
+
expression: `(${azMetricId} - AVG(METRICS("b"))) / AVG(STDDEV(METRICS("b")))`,
|
|
184
|
+
usingMetrics: usingMetrics,
|
|
185
|
+
label: availabilityZoneId + '-' + alb.loadBalancerArn + '-latency-z-score',
|
|
186
|
+
period: period,
|
|
187
|
+
}),
|
|
188
|
+
evaluationPeriods: evaluationPeriods,
|
|
189
|
+
datapointsToAlarm: datapointsToAlarm,
|
|
190
|
+
threshold: 3,
|
|
191
|
+
comparisonOperator: aws_cloudwatch_1.ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,
|
|
192
|
+
treatMissingData: aws_cloudwatch_1.TreatMissingData.IGNORE
|
|
193
|
+
});
|
|
194
|
+
}
|
|
195
|
+
doAlbMetrics(props) {
|
|
196
|
+
// Create impact alarms per AZ, with each ALB providing
|
|
197
|
+
// an alarm for its AZs
|
|
198
|
+
let perAZImpactAlarms = {};
|
|
199
|
+
let keyPrefix = MetricsHelper_1.MetricsHelper.nextChar('');
|
|
200
|
+
// Iterate each ALB
|
|
201
|
+
this.applicationLoadBalancers.forEach((alb) => {
|
|
202
|
+
// Iterate each AZ in the VPC
|
|
203
|
+
alb.vpc?.availabilityZones.forEach((az) => {
|
|
204
|
+
if (!(az in perAZImpactAlarms)) {
|
|
205
|
+
perAZImpactAlarms[az] = [];
|
|
206
|
+
}
|
|
207
|
+
// Get AZ letter
|
|
208
|
+
let azLetter = az.substring(az.length - 1);
|
|
209
|
+
// Map letter to AZ ID
|
|
210
|
+
let availabilityZoneId = this._azMapper.availabilityZoneIdFromAvailabilityZoneLetter(azLetter);
|
|
211
|
+
// Is there availability impact in this AZ?
|
|
212
|
+
let availabilityImpact = BasicServiceMultiAZObservability.isThereAnAZAvailabilityImpactAlb(this, alb, availabilityZoneId, az, props.faultCountPercentageThreshold ? props.faultCountPercentageThreshold : 0.05, keyPrefix, props.period ? props.period : aws_cdk_lib_1.Duration.minutes(1), props.evaluationPeriods, props.datapointsToAlarm);
|
|
213
|
+
// Is there latency impact in this AZ?
|
|
214
|
+
let latencyImpact = BasicServiceMultiAZObservability.isThereAZLatencyImpactAlb(this, alb, availabilityZoneId, az, props.latencyThreshold, props.latencyStatistic, keyPrefix, props.period ? props.period : aws_cdk_lib_1.Duration.minutes(1), props.evaluationPeriods, props.datapointsToAlarm);
|
|
215
|
+
// Is the AZ an outlier for faults
|
|
216
|
+
let availabilityOutlier = BasicServiceMultiAZObservability.isAZAnOutlierForAvailabilityAlb(this, alb, availabilityZoneId, az, props.faultCountPercentageThreshold ? props.faultCountPercentageThreshold : 0.05, keyPrefix, props.period ? props.period : aws_cdk_lib_1.Duration.minutes(1), props.evaluationPeriods, props.datapointsToAlarm);
|
|
217
|
+
// Is the AZ an outlier for latency
|
|
218
|
+
let latencyOutlier = BasicServiceMultiAZObservability.isAZAnOutlierForLatencyAlb(this, alb, availabilityZoneId, az, props.latencyStatistic, props.period ? props.period : aws_cdk_lib_1.Duration.minutes(1), props.evaluationPeriods, props.datapointsToAlarm, azLetter);
|
|
219
|
+
// Alarm if the AZ shows impact and is an outlier
|
|
220
|
+
let azImpactAlarm = new aws_cloudwatch_1.CompositeAlarm(this, az.substring(az.length - 1) + "-composite-impact-alarm", {
|
|
221
|
+
alarmRule: aws_cloudwatch_1.AlarmRule.anyOf(aws_cloudwatch_1.AlarmRule.allOf(availabilityImpact, availabilityOutlier), aws_cloudwatch_1.AlarmRule.allOf(latencyImpact, latencyOutlier)),
|
|
222
|
+
compositeAlarmName: availabilityZoneId + "-" +
|
|
223
|
+
alb.loadBalancerName +
|
|
224
|
+
"-latency-or-availability-impact",
|
|
225
|
+
actionsEnabled: false
|
|
226
|
+
});
|
|
227
|
+
// Add this ALB's fault rate alarm
|
|
228
|
+
perAZImpactAlarms[az].push(azImpactAlarm);
|
|
229
|
+
// Get next unique key
|
|
230
|
+
keyPrefix = MetricsHelper_1.MetricsHelper.nextChar(keyPrefix);
|
|
231
|
+
});
|
|
232
|
+
});
|
|
233
|
+
let azCompositeAlarms = {};
|
|
234
|
+
// Iterate AZs for the ALB impact alarms so we can join them
|
|
235
|
+
// into a single composite alarm for each AZ
|
|
236
|
+
Object.keys(perAZImpactAlarms).forEach((az) => {
|
|
237
|
+
let azLetter = az.substring(az.length - 1);
|
|
238
|
+
let availabilityZoneId = this._azMapper.availabilityZoneIdFromAvailabilityZoneLetter(azLetter);
|
|
239
|
+
azCompositeAlarms[az] = new aws_cloudwatch_1.CompositeAlarm(this, azLetter + "-alb-impact-composite-alarm", {
|
|
240
|
+
alarmRule: aws_cloudwatch_1.AlarmRule.anyOf(...perAZImpactAlarms[az]),
|
|
241
|
+
compositeAlarmName: availabilityZoneId + "-alb-impact-composite-alarm"
|
|
242
|
+
});
|
|
243
|
+
});
|
|
244
|
+
return azCompositeAlarms;
|
|
245
|
+
}
|
|
246
|
+
static isThereAnAZPacketLossImpactNATGW(scope, natgws, availabilityZoneId, availabilityZone, threshold, period, evaluationPeriods, datapointsToAlarm) {
|
|
247
|
+
let keyprefix = MetricsHelper_1.MetricsHelper.nextChar('');
|
|
248
|
+
let packetDropCountMetrics = {};
|
|
249
|
+
let packetsInFromSourceMetrics = {};
|
|
250
|
+
let packetsInFromDestinationMetrics = {};
|
|
251
|
+
natgws.forEach((natgw) => {
|
|
252
|
+
packetDropCountMetrics[`${keyprefix}1`] = new aws_cloudwatch_1.Metric({
|
|
253
|
+
metricName: 'PacketsDropCount',
|
|
254
|
+
namespace: 'AWS/NATGateway',
|
|
255
|
+
statistic: aws_cloudwatch_1.Stats.SUM,
|
|
256
|
+
unit: aws_cloudwatch_1.Unit.COUNT,
|
|
257
|
+
label: availabilityZoneId + ' packet drops',
|
|
258
|
+
dimensionsMap: {
|
|
259
|
+
NatGatewayId: natgw.attrNatGatewayId,
|
|
260
|
+
},
|
|
261
|
+
period: period,
|
|
262
|
+
});
|
|
263
|
+
// Calculate packets in from source
|
|
264
|
+
packetsInFromSourceMetrics[`${keyprefix}2`] = new aws_cloudwatch_1.Metric({
|
|
265
|
+
metricName: 'PacketsInFromSource',
|
|
266
|
+
namespace: 'AWS/NATGateway',
|
|
267
|
+
statistic: aws_cloudwatch_1.Stats.SUM,
|
|
268
|
+
unit: aws_cloudwatch_1.Unit.COUNT,
|
|
269
|
+
label: availabilityZoneId + ' packets in from source',
|
|
270
|
+
dimensionsMap: {
|
|
271
|
+
NatGatewayId: natgw.attrNatGatewayId,
|
|
272
|
+
},
|
|
273
|
+
period: period,
|
|
274
|
+
});
|
|
275
|
+
// Calculate packets in from destination
|
|
276
|
+
packetsInFromDestinationMetrics[`${keyprefix}3`] = new aws_cloudwatch_1.Metric({
|
|
277
|
+
metricName: 'PacketsInFromDestination',
|
|
278
|
+
namespace: 'AWS/NATGateway',
|
|
279
|
+
statistic: aws_cloudwatch_1.Stats.SUM,
|
|
280
|
+
unit: aws_cloudwatch_1.Unit.COUNT,
|
|
281
|
+
label: availabilityZoneId + ' packets in from destination',
|
|
282
|
+
dimensionsMap: {
|
|
283
|
+
NatGatewayId: natgw.attrNatGatewayId,
|
|
284
|
+
},
|
|
285
|
+
period: period,
|
|
286
|
+
});
|
|
287
|
+
keyprefix = MetricsHelper_1.MetricsHelper.nextChar(keyprefix);
|
|
288
|
+
});
|
|
289
|
+
let packetDropTotal = new aws_cloudwatch_1.MathExpression({
|
|
290
|
+
expression: Object.keys(packetDropCountMetrics).join("+"),
|
|
291
|
+
usingMetrics: packetDropCountMetrics,
|
|
292
|
+
period: period
|
|
293
|
+
});
|
|
294
|
+
let packetsInFromSourceTotal = new aws_cloudwatch_1.MathExpression({
|
|
295
|
+
expression: Object.keys(packetsInFromSourceMetrics).join("+"),
|
|
296
|
+
usingMetrics: packetsInFromSourceMetrics,
|
|
297
|
+
period: period
|
|
298
|
+
});
|
|
299
|
+
let packetsInFromDestinationTotal = new aws_cloudwatch_1.MathExpression({
|
|
300
|
+
expression: Object.keys(packetsInFromDestinationMetrics).join("+"),
|
|
301
|
+
usingMetrics: packetsInFromDestinationMetrics,
|
|
302
|
+
period: period
|
|
303
|
+
});
|
|
304
|
+
let usingMetrics = {};
|
|
305
|
+
usingMetrics[`${keyprefix}1`] = packetDropTotal;
|
|
306
|
+
usingMetrics[`${keyprefix}2`] = packetsInFromSourceTotal;
|
|
307
|
+
usingMetrics[`${keyprefix}3`] = packetsInFromDestinationTotal;
|
|
308
|
+
// Calculate a percentage of dropped packets for the NAT GW
|
|
309
|
+
let packetDropPercentage = new aws_cloudwatch_1.MathExpression({
|
|
310
|
+
expression: `(${keyprefix}1 / (${keyprefix}2 + ${keyprefix}3))`,
|
|
311
|
+
usingMetrics: usingMetrics,
|
|
312
|
+
label: availabilityZoneId + ' packet drop percentage',
|
|
313
|
+
period: period,
|
|
314
|
+
});
|
|
315
|
+
// Create an alarm for this NAT GW if packet drops exceed the specified threshold
|
|
316
|
+
return new aws_cloudwatch_1.Alarm(scope, availabilityZone.substring(availabilityZone.length - 1) + "-packet-drop-impact-alarm", {
|
|
317
|
+
alarmName: availabilityZoneId +
|
|
318
|
+
'-packet-drop-impact',
|
|
319
|
+
actionsEnabled: false,
|
|
320
|
+
metric: packetDropPercentage,
|
|
321
|
+
threshold: threshold,
|
|
322
|
+
comparisonOperator: aws_cloudwatch_1.ComparisonOperator.GREATER_THAN_THRESHOLD,
|
|
323
|
+
evaluationPeriods: evaluationPeriods,
|
|
324
|
+
datapointsToAlarm: datapointsToAlarm,
|
|
325
|
+
treatMissingData: aws_cloudwatch_1.TreatMissingData.IGNORE
|
|
326
|
+
});
|
|
327
|
+
}
|
|
328
|
+
static isAZAnOutlierForPacketLossNATGW(scope, natgws, availabilityZoneId, availabilityZone, threshold, period, evaluationPeriods, datapointsToAlarm) {
|
|
329
|
+
let keyprefix = MetricsHelper_1.MetricsHelper.nextChar('');
|
|
330
|
+
let azPacketDropCountMetrics = {};
|
|
331
|
+
let azKey = "";
|
|
332
|
+
Object.keys(natgws).forEach((az) => {
|
|
333
|
+
let packetDropCountMetrics = {};
|
|
334
|
+
natgws[az].forEach((natgw, index) => {
|
|
335
|
+
packetDropCountMetrics[`${keyprefix}${index}`] = new aws_cloudwatch_1.Metric({
|
|
336
|
+
metricName: 'PacketsDropCount',
|
|
337
|
+
namespace: 'AWS/NATGateway',
|
|
338
|
+
statistic: aws_cloudwatch_1.Stats.SUM,
|
|
339
|
+
unit: aws_cloudwatch_1.Unit.COUNT,
|
|
340
|
+
label: availabilityZoneId + ' packet drops',
|
|
341
|
+
dimensionsMap: {
|
|
342
|
+
NatGatewayId: natgw.attrNatGatewayId,
|
|
343
|
+
},
|
|
344
|
+
period: period,
|
|
345
|
+
});
|
|
346
|
+
keyprefix = MetricsHelper_1.MetricsHelper.nextChar(keyprefix);
|
|
347
|
+
});
|
|
348
|
+
azPacketDropCountMetrics[`${keyprefix}${natgws[az].length}`] = new aws_cloudwatch_1.MathExpression({
|
|
349
|
+
expression: Object.keys(packetDropCountMetrics).join("+"),
|
|
350
|
+
usingMetrics: packetDropCountMetrics,
|
|
351
|
+
period: period
|
|
352
|
+
});
|
|
353
|
+
if (az == availabilityZone) {
|
|
354
|
+
azKey = `${keyprefix}${natgws[az].length}`;
|
|
355
|
+
}
|
|
356
|
+
keyprefix = MetricsHelper_1.MetricsHelper.nextChar(keyprefix);
|
|
357
|
+
});
|
|
358
|
+
return new aws_cloudwatch_1.Alarm(scope, availabilityZone.substring(availabilityZone.length - 1) + "-packet-loss-outlier", {
|
|
359
|
+
metric: new aws_cloudwatch_1.MathExpression({
|
|
360
|
+
expression: `${azKey} / (${Object.keys(azPacketDropCountMetrics).join("+")})`,
|
|
361
|
+
usingMetrics: azPacketDropCountMetrics,
|
|
362
|
+
period: period
|
|
363
|
+
}),
|
|
364
|
+
threshold: threshold,
|
|
365
|
+
evaluationPeriods: evaluationPeriods,
|
|
366
|
+
datapointsToAlarm: datapointsToAlarm,
|
|
367
|
+
comparisonOperator: aws_cloudwatch_1.ComparisonOperator.GREATER_THAN_THRESHOLD,
|
|
368
|
+
treatMissingData: aws_cloudwatch_1.TreatMissingData.IGNORE
|
|
369
|
+
});
|
|
370
|
+
}
|
|
371
|
+
doNatGatewayMetrics(props) {
|
|
372
|
+
// Collect alarms for packet drops exceeding a threshold per NAT GW
|
|
373
|
+
let packetLossPerAZAlarms = {};
|
|
374
|
+
// For each AZ, create metrics for each NAT GW
|
|
375
|
+
Object.keys(this.natGateways).forEach((az) => {
|
|
376
|
+
let azLetter = az.substring(az.length - 1);
|
|
377
|
+
let availabilityZoneId = this._azMapper.availabilityZoneIdFromAvailabilityZoneLetter(azLetter);
|
|
378
|
+
// Is there packet loss impact?
|
|
379
|
+
let packetLossImpact = BasicServiceMultiAZObservability.isThereAnAZPacketLossImpactNATGW(this, this.natGateways[az], availabilityZoneId, az, props.packetLossImpactPercentageThreshold ? props.packetLossImpactPercentageThreshold : 0.01, props.period ? props.period : aws_cdk_lib_1.Duration.minutes(1), props.evaluationPeriods, props.datapointsToAlarm);
|
|
380
|
+
// Is this AZ an outlier for this NATGW?
|
|
381
|
+
let packetLossOutlier = BasicServiceMultiAZObservability.isAZAnOutlierForPacketLossNATGW(this, this.natGateways, availabilityZoneId, az, 0.66, props.period ? props.period : aws_cdk_lib_1.Duration.minutes(1), props.evaluationPeriods, props.datapointsToAlarm);
|
|
382
|
+
packetLossPerAZAlarms[az] = new aws_cloudwatch_1.CompositeAlarm(this, az.substring(az.length - 1) + "-packet-loss-composite-alarm", {
|
|
383
|
+
alarmRule: aws_cloudwatch_1.AlarmRule.allOf(packetLossImpact, packetLossOutlier),
|
|
384
|
+
compositeAlarmName: availabilityZoneId + "-packet-loss-composite-alarm"
|
|
385
|
+
});
|
|
386
|
+
});
|
|
387
|
+
return packetLossPerAZAlarms;
|
|
388
|
+
}
|
|
389
|
+
getTotalPacketDropsPerZone(natgws, period) {
|
|
390
|
+
let dropsPerZone = {};
|
|
391
|
+
let metricsPerAZ = {};
|
|
392
|
+
let keyprefix = MetricsHelper_1.MetricsHelper.nextChar('');
|
|
393
|
+
Object.keys(natgws).forEach((availabilityZone) => {
|
|
394
|
+
let azLetter = availabilityZone.substring(availabilityZone.length - 1);
|
|
395
|
+
let availabilityZoneId = this._azMapper.availabilityZoneIdFromAvailabilityZoneLetter(azLetter);
|
|
396
|
+
if (!(availabilityZone in metricsPerAZ)) {
|
|
397
|
+
metricsPerAZ[availabilityZone] = [];
|
|
398
|
+
}
|
|
399
|
+
natgws[availabilityZone].forEach((natgw) => {
|
|
400
|
+
metricsPerAZ[availabilityZone].push(new aws_cloudwatch_1.Metric({
|
|
401
|
+
metricName: 'PacketsDropCount',
|
|
402
|
+
namespace: 'AWS/NATGateway',
|
|
403
|
+
statistic: aws_cloudwatch_1.Stats.SUM,
|
|
404
|
+
unit: aws_cloudwatch_1.Unit.COUNT,
|
|
405
|
+
label: availabilityZoneId + ' packet drops',
|
|
406
|
+
dimensionsMap: {
|
|
407
|
+
NatGatewayId: natgw.attrNatGatewayId,
|
|
408
|
+
},
|
|
409
|
+
period: period,
|
|
410
|
+
}));
|
|
411
|
+
});
|
|
412
|
+
});
|
|
413
|
+
Object.keys(metricsPerAZ).forEach((availabilityZone) => {
|
|
414
|
+
let azLetter = availabilityZone.substring(availabilityZone.length - 1);
|
|
415
|
+
let availabilityZoneId = this._azMapper.availabilityZoneIdFromAvailabilityZoneLetter(azLetter);
|
|
416
|
+
let usingMetrics = {};
|
|
417
|
+
metricsPerAZ[availabilityZone].forEach((metric) => {
|
|
418
|
+
usingMetrics[`${keyprefix}1`] = metric;
|
|
419
|
+
keyprefix = MetricsHelper_1.MetricsHelper.nextChar(keyprefix);
|
|
420
|
+
});
|
|
421
|
+
dropsPerZone[availabilityZone] = new aws_cloudwatch_1.MathExpression({
|
|
422
|
+
expression: Object.keys(usingMetrics).join("+"),
|
|
423
|
+
usingMetrics: usingMetrics,
|
|
424
|
+
label: availabilityZoneId + " total packet drops",
|
|
425
|
+
period: period
|
|
426
|
+
});
|
|
427
|
+
});
|
|
428
|
+
return dropsPerZone;
|
|
429
|
+
}
|
|
430
|
+
}
|
|
431
|
+
exports.BasicServiceMultiAZObservability = BasicServiceMultiAZObservability;
|
|
432
|
+
_a = JSII_RTTI_SYMBOL_1;
|
|
433
|
+
BasicServiceMultiAZObservability[_a] = { fqn: "@cdklabs/multi-az-observability.BasicServiceMultiAZObservability", version: "0.0.1-alpha.1" };
|
|
434
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"BasicServiceMultiAZObservability.js","sourceRoot":"","sources":["../../src/basic_observability/BasicServiceMultiAZObservability.ts"],"names":[],"mappings":"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","sourcesContent":["// Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.\n// SPDX-License-Identifier: Apache-2.0\n\nimport { Duration } from 'aws-cdk-lib';\nimport {\n  Alarm,\n  AlarmRule,\n  ComparisonOperator,\n  CompositeAlarm,\n  Dashboard,\n  IAlarm,\n  IMetric,\n  MathExpression,\n  Metric,\n  Stats,\n  TreatMissingData,\n  Unit,\n} from 'aws-cdk-lib/aws-cloudwatch';\nimport { CfnNatGateway } from 'aws-cdk-lib/aws-ec2';\nimport {\n  BaseLoadBalancer,\n  IApplicationLoadBalancer,\n  ILoadBalancerV2,\n} from 'aws-cdk-lib/aws-elasticloadbalancingv2';\nimport { Construct, IConstruct } from 'constructs';\nimport { IBasicServiceMultiAZObservability } from './IBasicServiceMultiAZObservability';\nimport { BasicServiceMultiAZObservabilityProps } from './props/BasicServiceMultiAZObservabilityProps';\nimport { AvailabilityZoneMapper } from '../azmapper/AvailabilityZoneMapper';\nimport { IAvailabilityZoneMapper } from '../azmapper/IAvailabilityZoneMapper';\nimport { BasicServiceDashboard } from './BasicServiceDashboard';\nimport { ApplicationLoadBalancerMetrics } from '../metrics/ApplicationLoadBalancerMetrics';\nimport { AvailabilityMetricType } from '../utilities/AvailabilityMetricType';\nimport { MetricsHelper } from '../utilities/MetricsHelper';\n\n/**\n * Basic observability for a service using metrics from\n * ALBs and NAT Gateways\n */\nexport class BasicServiceMultiAZObservability\n  extends Construct\n  implements IBasicServiceMultiAZObservability {\n  /**\n   * The NAT Gateways being used in the service, each set of NAT Gateways\n   * are keyed by their Availability Zone Id\n   */\n  natGateways?: { [key: string]: CfnNatGateway[] };\n\n  /**\n   * The application load balancers being used by the service\n   */\n  applicationLoadBalancers?: IApplicationLoadBalancer[];\n\n  /**\n   * The name of the service\n   */\n  serviceName: string;\n\n  /**\n   * The alarms indicating if an AZ is an outlier for NAT GW\n   * packet loss and has isolated impact\n   */\n  natGWZonalIsolatedImpactAlarms?: { [key: string]: IAlarm };\n\n  /**\n   * The alarms indicating if an AZ is an outlier for ALB\n   * faults and has isolated impact\n   */\n  albZonalIsolatedImpactAlarms?: { [key: string]: IAlarm };\n\n  /**\n   * The alarms indicating if an AZ has isolated impact\n   * from either ALB or NAT GW metrics\n   */\n  aggregateZonalIsolatedImpactAlarms: { [key: string]: IAlarm };\n\n  /**\n   * The dashboard that is optionally created\n   */\n  dashboard?: Dashboard;\n\n  /**\n   * The AZ mapper resource\n   */\n  private _azMapper: IAvailabilityZoneMapper;\n\n  constructor(\n    scope: Construct,\n    id: string,\n    props: BasicServiceMultiAZObservabilityProps,\n  ) {\n    super(scope, id);\n\n    // Initialize class properties\n    this.serviceName = props.serviceName;\n    this.applicationLoadBalancers = props.applicationLoadBalancers;\n    this.natGateways = props.natGateways;\n\n    this.aggregateZonalIsolatedImpactAlarms = {};\n    this.albZonalIsolatedImpactAlarms = {};\n    this.natGWZonalIsolatedImpactAlarms = {};\n\n    // Create the AZ mapper resource to translate AZ names to ids\n    this._azMapper = new AvailabilityZoneMapper(this, 'availability-zone-mapper');\n\n    // Create ALB metrics and alarms per AZ\n    if (this.applicationLoadBalancers) {\n      this.albZonalIsolatedImpactAlarms = this.doAlbMetrics(props);\n    }\n\n    // Create NAT Gateway metrics and alarms per AZ\n    if (this.natGateways) {\n      this.natGWZonalIsolatedImpactAlarms = this.doNatGatewayMetrics(props);\n    }\n\n    // Look through all of the per AZ ALB alarms, if there's also a NAT GW alarm\n    // create a composite alarm if either of them trigger\n    Object.keys(this.albZonalIsolatedImpactAlarms).forEach((az: string) => {\n      let azLetter = az.substring(az.length - 1);\n      let availabilityZoneId = this._azMapper.availabilityZoneIdFromAvailabilityZoneLetter(azLetter);\n      if (az in this.natGWZonalIsolatedImpactAlarms!) {\n        this.aggregateZonalIsolatedImpactAlarms[az] = new CompositeAlarm(this, az.substring(az.length - 1) + \"-isolated-impact-alarm\", {\n          alarmRule: AlarmRule.anyOf(this.albZonalIsolatedImpactAlarms![az], this.natGWZonalIsolatedImpactAlarms![az]),\n          compositeAlarmName: availabilityZoneId + \"-isolated-impact-alarm\"\n        });\n      }\n      else {\n        this.aggregateZonalIsolatedImpactAlarms[az] = new CompositeAlarm(this, az.substring(az.length - 1) + \"-isolated-impact-alarm\", {\n          alarmRule: AlarmRule.anyOf(this.albZonalIsolatedImpactAlarms![az]),\n          compositeAlarmName: availabilityZoneId + \"-isolated-impact-alarm\"\n        });\n      }\n    });\n\n    // Look through all of the per AZ NAT GW alarms. If there's an AZ we haven't seen in the ALB\n    // alarms yet, then it will just be a NAT GW alarm that we'll turn into the same kind of\n    // composite alarm\n    Object.keys(this.natGWZonalIsolatedImpactAlarms).forEach((az: string) => {\n      if (!(az in this.aggregateZonalIsolatedImpactAlarms)) {\n        let azLetter = az.substring(az.length - 1);\n        let availabilityZoneId = this._azMapper.availabilityZoneIdFromAvailabilityZoneLetter(azLetter);\n\n        this.aggregateZonalIsolatedImpactAlarms[az] = new CompositeAlarm(this, az.substring(az.length - 1) + \"-isolated-impact-alarm\", {\n          alarmRule: AlarmRule.anyOf(this.natGWZonalIsolatedImpactAlarms![az]),\n          compositeAlarmName: availabilityZoneId + \"-isolated-impact-alarm\"\n        });\n      }\n    });\n    \n    // Should we create the dashboard\n    if (props.createDashboard == true) {\n      this.dashboard = new BasicServiceDashboard(\n        this,\n        'BasicServiceDashboard',\n        {\n          serviceName: props.serviceName.toLowerCase(),\n          zonalAggregateIsolatedImpactAlarms:\n            this.aggregateZonalIsolatedImpactAlarms,\n          zonalLoadBalancerIsolatedImpactAlarms:\n            this.albZonalIsolatedImpactAlarms,\n          zonalNatGatewayIsolatedImpactAlarms:\n            this.natGWZonalIsolatedImpactAlarms,\n          interval: props.interval,\n          zonalLoadBalancerFaultRateMetrics: ApplicationLoadBalancerMetrics.getTotalAlbFaultCountPerZone(\n            props.applicationLoadBalancers ? props.applicationLoadBalancers : [], \n            props.period ? props.period : Duration.minutes(1),\n            this._azMapper\n          ),\n          zonalNatGatewayPacketDropMetrics: this.getTotalPacketDropsPerZone(\n            props.natGateways ? props.natGateways : {},\n            props.period ? props.period : Duration.minutes(1)\n          ),\n          azMapper: this._azMapper,\n        },\n      ).dashboard;\n    }\n  }\n\n  private static isThereAnAZAvailabilityImpactAlb(\n    scope: IConstruct,\n    alb: IApplicationLoadBalancer, \n    availabilityZoneId: string,\n    availabilityZone: string,\n    threshold: number,\n    keyprefix: string,\n    period: Duration,\n    evaluationPeriods: number,\n    datapointsToAlarm: number\n  ) : IAlarm {\n   \n    // Create a fault rate alarm for the ALB in the specified AZ\n    return new Alarm(\n      scope,\n      keyprefix + '-fault-rate-alarm',\n      {\n        alarmName:\n          availabilityZoneId + '-' + alb.loadBalancerArn + '-fault-rate',\n        actionsEnabled: false,\n        metric: ApplicationLoadBalancerMetrics.getPerAZAvailabilityMetric(alb, {\n          period: period,\n          label: availabilityZoneId + '-' + alb.loadBalancerArn + '-fault-rate',\n          availabilityZone: availabilityZone,\n          availabilityZoneId: availabilityZoneId,\n          metricType: AvailabilityMetricType.FAULT_RATE\n        }),\n        evaluationPeriods: evaluationPeriods,\n        datapointsToAlarm: datapointsToAlarm,\n        threshold: threshold,\n        comparisonOperator: ComparisonOperator.GREATER_THAN_THRESHOLD,\n        treatMissingData: TreatMissingData.IGNORE\n      }\n    );\n  }\n\n  private static isThereAZLatencyImpactAlb(\n    scope: IConstruct,\n    alb: IApplicationLoadBalancer, \n    availabilityZoneId: string,\n    availabilityZone: string,\n    threshold: number,\n    statistic: string,\n    keyprefix: string,\n    period: Duration,\n    evaluationPeriods: number,\n    datapointsToAlarm: number\n  ): IAlarm {\n    \n    // Create a fault rate alarm for the ALB in the specified AZ\n    return new Alarm(\n      scope,\n      keyprefix + '-latency-alarm',\n      {\n        alarmName:\n          availabilityZoneId + '-' + alb.loadBalancerArn + '-latency',\n        actionsEnabled: false,\n        metric: ApplicationLoadBalancerMetrics.getPerAZLatencyMetric({\n          alb: alb,\n          availabilityZone: availabilityZone,\n          availabilityZoneId: availabilityZoneId,\n          label: availabilityZoneId + \"-\" + alb.loadBalancerArn + \"-target-latency\",\n          period: period,\n          statistic: statistic\n        }),\n        evaluationPeriods: evaluationPeriods,\n        datapointsToAlarm: datapointsToAlarm,\n        threshold: threshold,\n        comparisonOperator: ComparisonOperator.GREATER_THAN_THRESHOLD,\n        treatMissingData: TreatMissingData.IGNORE\n      }\n    );\n  }\n\n  private static isAZAnOutlierForAvailabilityAlb(\n    scope: IConstruct,\n    alb: IApplicationLoadBalancer, \n    availabilityZoneId: string,\n    availabilityZone: string,\n    threshold: number,\n    keyprefix: string,\n    period: Duration,\n    evaluationPeriods: number,\n    datapointsToAlarm: number\n  ) : IAlarm {\n\n    let usingMetrics: { [key: string]: IMetric } = {};\n    let azMetricId: string;\n\n    alb.vpc!.availabilityZones.forEach((az: string) => {\n      let azFaultCount = ApplicationLoadBalancerMetrics.getPerAZAvailabilityMetric(\n        alb,\n        {\n          metricType: AvailabilityMetricType.FAULT_COUNT,\n          availabilityZone: availabilityZone,\n          availabilityZoneId: availabilityZoneId,\n          period: period,\n          label: availabilityZoneId + \"-\" + alb.loadBalancerArn + \"-fault-count\",\n          keyprefix: keyprefix\n        }\n      );\n\n      keyprefix = MetricsHelper.nextChar(keyprefix);\n\n      usingMetrics[`${keyprefix}1`] = azFaultCount;\n\n      if (az == availabilityZone) {\n        azMetricId = Object.keys(usingMetrics)[-1];\n      }\n\n      keyprefix = MetricsHelper.nextChar(keyprefix);\n    });\n\n    return new Alarm(\n      scope,\n      keyprefix + '-availability-outlier-alarm',\n      {\n        alarmName:\n          availabilityZoneId + '-' + alb.loadBalancerArn + '-availability-impact-outlier',\n        actionsEnabled: false,\n        metric: new MathExpression({\n          expression: `${azMetricId!}/(${Object.keys(usingMetrics).join(\"+\")})`,\n          usingMetrics: usingMetrics,\n          label: availabilityZoneId + '-' + alb.loadBalancerArn + '-percent-of-faults',\n          period: period,\n        }),\n        evaluationPeriods: evaluationPeriods,\n        datapointsToAlarm: datapointsToAlarm,\n        threshold: threshold,\n        comparisonOperator: ComparisonOperator.GREATER_THAN_THRESHOLD,\n        treatMissingData: TreatMissingData.IGNORE\n      }\n    );\n  }\n\n  private static isAZAnOutlierForLatencyAlb(\n    scope: IConstruct,\n    alb: IApplicationLoadBalancer, \n    availabilityZoneId: string,\n    availabilityZone: string,\n    statistic: string,\n    period: Duration,\n    evaluationPeriods: number,\n    datapointsToAlarm: number,\n    keyprefix: string\n  ) : IAlarm {\n\n    let usingMetrics: { [key: string]: IMetric } = {};\n    let azMetricId: string;\n\n    alb.vpc!.availabilityZones.forEach((az: string, index: number) => {\n\n      // Target response time\n      let targetResponseTime: IMetric = ApplicationLoadBalancerMetrics.getPerAZLatencyMetric({\n        alb: alb,\n        availabilityZone: az,\n        label: az + \"-target-response-time\",\n        statistic: statistic,\n        period: period\n      });\n\n      if (az == availabilityZone) {       \n        azMetricId = `a${index}`\n        usingMetrics[`a${index}`] = targetResponseTime;\n      }\n      else {\n        usingMetrics[`b${index}`] = targetResponseTime;\n      }\n    });\n\n    return new Alarm(\n      scope,\n      keyprefix + \"-latency-outlier-alarm\",\n      {\n        alarmName:\n          availabilityZoneId + '-' + alb.loadBalancerArn + '-latency-impact-outlier',\n        actionsEnabled: false,\n        metric: new MathExpression({\n          expression: `(${azMetricId!} - AVG(METRICS(\"b\"))) / AVG(STDDEV(METRICS(\"b\")))`,\n          usingMetrics: usingMetrics,\n          label: availabilityZoneId + '-' + alb.loadBalancerArn + '-latency-z-score',\n          period: period,\n        }),\n        evaluationPeriods: evaluationPeriods,\n        datapointsToAlarm: datapointsToAlarm,\n        threshold: 3,\n        comparisonOperator: ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,\n        treatMissingData: TreatMissingData.IGNORE\n      }\n    );\n  }\n\n  private doAlbMetrics(\n    props: BasicServiceMultiAZObservabilityProps\n  ) : { [key: string]: IAlarm } {\n\n    // Create impact alarms per AZ, with each ALB providing\n    // an alarm for its AZs\n    let perAZImpactAlarms: { [key: string]: IAlarm[] } = {};\n\n    let keyPrefix: string = MetricsHelper.nextChar('');\n\n    // Iterate each ALB\n    this.applicationLoadBalancers!.forEach((alb) => {\n      \n      // Iterate each AZ in the VPC\n      alb.vpc?.availabilityZones.forEach((az) => {\n\n        if (!(az in perAZImpactAlarms)) {\n          perAZImpactAlarms[az] = [];\n        }\n     \n        // Get AZ letter\n        let azLetter = az.substring(az.length - 1);\n\n        // Map letter to AZ ID\n        let availabilityZoneId: string =\n          this._azMapper.availabilityZoneIdFromAvailabilityZoneLetter(azLetter);\n\n        // Is there availability impact in this AZ?\n        let availabilityImpact: IAlarm = BasicServiceMultiAZObservability.isThereAnAZAvailabilityImpactAlb(\n          this,\n          alb,\n          availabilityZoneId,\n          az,\n          props.faultCountPercentageThreshold ? props.faultCountPercentageThreshold : 0.05,\n          keyPrefix,\n          props.period ? props.period : Duration.minutes(1),\n          props.evaluationPeriods,\n          props.datapointsToAlarm\n        );\n\n        // Is there latency impact in this AZ?\n        let latencyImpact: IAlarm = BasicServiceMultiAZObservability.isThereAZLatencyImpactAlb(\n          this,\n          alb,\n          availabilityZoneId,\n          az,\n          props.latencyThreshold,\n          props.latencyStatistic,\n          keyPrefix,\n          props.period ? props.period : Duration.minutes(1),\n          props.evaluationPeriods,\n          props.datapointsToAlarm\n        );\n\n        // Is the AZ an outlier for faults\n        let availabilityOutlier = BasicServiceMultiAZObservability.isAZAnOutlierForAvailabilityAlb(\n          this,\n          alb,\n          availabilityZoneId,\n          az,\n          props.faultCountPercentageThreshold ? props.faultCountPercentageThreshold : 0.05,\n          keyPrefix,\n          props.period ? props.period : Duration.minutes(1),\n          props.evaluationPeriods,\n          props.datapointsToAlarm\n        );\n\n        // Is the AZ an outlier for latency\n        let latencyOutlier = BasicServiceMultiAZObservability.isAZAnOutlierForLatencyAlb(\n          this,\n          alb,\n          availabilityZoneId,\n          az,\n          props.latencyStatistic,\n          props.period ? props.period : Duration.minutes(1),\n          props.evaluationPeriods,\n          props.datapointsToAlarm,\n          azLetter\n        );\n\n        // Alarm if the AZ shows impact and is an outlier\n        let azImpactAlarm: IAlarm = new CompositeAlarm(this, \n          az.substring(az.length - 1) +  \"-composite-impact-alarm\", \n          {\n            alarmRule: AlarmRule.anyOf(\n              AlarmRule.allOf(availabilityImpact, availabilityOutlier), \n              AlarmRule.allOf(latencyImpact, latencyOutlier)\n            ),\n            compositeAlarmName: \n              availabilityZoneId + \"-\" + \n              (alb as ILoadBalancerV2 as BaseLoadBalancer).loadBalancerName + \n              \"-latency-or-availability-impact\",\n            actionsEnabled: false\n          }\n        );\n\n        // Add this ALB's fault rate alarm\n        perAZImpactAlarms[az].push(azImpactAlarm);\n\n        // Get next unique key\n        keyPrefix = MetricsHelper.nextChar(keyPrefix);\n      });\n    });\n\n    let azCompositeAlarms: {[key: string]: IAlarm} = {};\n\n    // Iterate AZs for the ALB impact alarms so we can join them\n    // into a single composite alarm for each AZ\n    Object.keys(perAZImpactAlarms).forEach((az) => {\n      let azLetter = az.substring(az.length - 1);\n      let availabilityZoneId: string = this._azMapper.availabilityZoneIdFromAvailabilityZoneLetter(azLetter);\n\n      azCompositeAlarms[az] = new CompositeAlarm(this, azLetter + \"-alb-impact-composite-alarm\", {\n        alarmRule: AlarmRule.anyOf(...perAZImpactAlarms[az]),\n        compositeAlarmName: availabilityZoneId + \"-alb-impact-composite-alarm\"\n      });\n    });\n\n    return azCompositeAlarms;\n  }\n\n  private static isThereAnAZPacketLossImpactNATGW(\n    scope: IConstruct,\n    natgws: CfnNatGateway[], \n    availabilityZoneId: string,\n    availabilityZone: string,\n    threshold: number,\n    period: Duration,\n    evaluationPeriods: number,\n    datapointsToAlarm: number\n  ) : IAlarm {\n    \n    let keyprefix = MetricsHelper.nextChar('');\n\n    let packetDropCountMetrics: {[key: string]: IMetric} = {};\n    let packetsInFromSourceMetrics: {[key: string]: IMetric} = {};\n    let packetsInFromDestinationMetrics: {[key: string]: IMetric} = {};\n   \n    natgws.forEach((natgw: CfnNatGateway)=> {\n\n      packetDropCountMetrics[`${keyprefix}1`] = new Metric({\n        metricName: 'PacketsDropCount',\n        namespace: 'AWS/NATGateway',\n        statistic: Stats.SUM,\n        unit: Unit.COUNT,\n        label: availabilityZoneId + ' packet drops',\n        dimensionsMap: {\n          NatGatewayId: natgw.attrNatGatewayId,\n        },\n        period: period,\n      });\n  \n      // Calculate packets in from source\n      packetsInFromSourceMetrics[`${keyprefix}2`] = new Metric({\n        metricName: 'PacketsInFromSource',\n        namespace: 'AWS/NATGateway',\n        statistic: Stats.SUM,\n        unit: Unit.COUNT,\n        label: availabilityZoneId + ' packets in from source',\n        dimensionsMap: {\n          NatGatewayId: natgw.attrNatGatewayId,\n        },\n        period: period,\n      });\n  \n      // Calculate packets in from destination\n      packetsInFromDestinationMetrics[`${keyprefix}3`] = new Metric({\n        metricName: 'PacketsInFromDestination',\n        namespace: 'AWS/NATGateway',\n        statistic: Stats.SUM,\n        unit: Unit.COUNT,\n        label: availabilityZoneId + ' packets in from destination',\n        dimensionsMap: {\n          NatGatewayId: natgw.attrNatGatewayId,\n        },\n        period: period,\n      });\n\n      keyprefix = MetricsHelper.nextChar(keyprefix);\n    });\n\n    let packetDropTotal: IMetric = new MathExpression({\n      expression: Object.keys(packetDropCountMetrics).join(\"+\"),\n      usingMetrics: packetDropCountMetrics,\n      period: period\n    });\n\n    let packetsInFromSourceTotal: IMetric = new MathExpression({\n      expression: Object.keys(packetsInFromSourceMetrics).join(\"+\"),\n      usingMetrics: packetsInFromSourceMetrics,\n      period: period\n    });\n\n    let packetsInFromDestinationTotal: IMetric = new MathExpression({\n      expression: Object.keys(packetsInFromDestinationMetrics).join(\"+\"),\n      usingMetrics: packetsInFromDestinationMetrics,\n      period: period\n    });\n    \n    let usingMetrics: { [key: string]: IMetric } = {};\n    usingMetrics[`${keyprefix}1`] = packetDropTotal;\n    usingMetrics[`${keyprefix}2`] = packetsInFromSourceTotal;\n    usingMetrics[`${keyprefix}3`] = packetsInFromDestinationTotal;\n\n    // Calculate a percentage of dropped packets for the NAT GW\n    let packetDropPercentage: IMetric = new MathExpression({\n      expression: `(${keyprefix}1 / (${keyprefix}2 + ${keyprefix}3))`,\n      usingMetrics: usingMetrics,\n      label: availabilityZoneId + ' packet drop percentage',\n      period: period,\n    });\n\n    // Create an alarm for this NAT GW if packet drops exceed the specified threshold\n    return new Alarm(\n      scope,\n      availabilityZone.substring(availabilityZone.length - 1) + \"-packet-drop-impact-alarm\",\n      {\n        alarmName:\n          availabilityZoneId +\n          '-packet-drop-impact',\n        actionsEnabled: false,\n        metric: packetDropPercentage,\n        threshold: threshold,\n        comparisonOperator: ComparisonOperator.GREATER_THAN_THRESHOLD,\n        evaluationPeriods: evaluationPeriods,\n        datapointsToAlarm: datapointsToAlarm,\n        treatMissingData: TreatMissingData.IGNORE\n      }\n    );\n  }\n\n  private static isAZAnOutlierForPacketLossNATGW(\n    scope: IConstruct,\n    natgws: {[key: string]: CfnNatGateway[]}, \n    availabilityZoneId: string,\n    availabilityZone: string,\n    threshold: number,\n    period: Duration,\n    evaluationPeriods: number,\n    datapointsToAlarm: number\n  ) : IAlarm {\n\n    let keyprefix = MetricsHelper.nextChar('');\n    let azPacketDropCountMetrics: {[key: string]: IMetric} = {};\n    let azKey: string = \"\";\n\n    Object.keys(natgws).forEach((az: string) => {\n      \n      let packetDropCountMetrics: {[key: string]: IMetric} = {};\n\n      \n\n      natgws[az].forEach((natgw: CfnNatGateway, index: number) => {\n        packetDropCountMetrics[`${keyprefix}${index}`] = new Metric({\n          metricName: 'PacketsDropCount',\n          namespace: 'AWS/NATGateway',\n          statistic: Stats.SUM,\n          unit: Unit.COUNT,\n          label: availabilityZoneId + ' packet drops',\n          dimensionsMap: {\n            NatGatewayId: natgw.attrNatGatewayId,\n          },\n          period: period,\n        });\n        keyprefix = MetricsHelper.nextChar(keyprefix);\n      });\n\n      azPacketDropCountMetrics[`${keyprefix}${natgws[az].length}`] = new MathExpression({\n        expression: Object.keys(packetDropCountMetrics).join(\"+\"),\n        usingMetrics: packetDropCountMetrics,\n        period: period\n      });\n\n      if (az == availabilityZone) {\n        azKey = `${keyprefix}${natgws[az].length}`;\n      }\n\n      keyprefix = MetricsHelper.nextChar(keyprefix);\n    });\n\n    return new Alarm(\n      scope, \n      availabilityZone.substring(availabilityZone.length - 1) + \"-packet-loss-outlier\", \n      {\n         metric: new MathExpression({\n            expression: `${azKey} / (${Object.keys(azPacketDropCountMetrics).join(\"+\")})`,\n            usingMetrics: azPacketDropCountMetrics,\n            period: period\n         }),\n         threshold: threshold,\n         evaluationPeriods: evaluationPeriods,\n         datapointsToAlarm: datapointsToAlarm,\n         comparisonOperator: ComparisonOperator.GREATER_THAN_THRESHOLD,\n         treatMissingData: TreatMissingData.IGNORE\n      }\n    );\n  }\n\n  private doNatGatewayMetrics(\n    props: BasicServiceMultiAZObservabilityProps\n  ): {[key: string]: IAlarm} {\n\n    // Collect alarms for packet drops exceeding a threshold per NAT GW\n    let packetLossPerAZAlarms: { [key: string]: IAlarm } = {};\n\n    // For each AZ, create metrics for each NAT GW\n    Object.keys(this.natGateways!).forEach((az: string) => {\n\n      let azLetter: string = az.substring(az.length - 1);\n      let availabilityZoneId =\n        this._azMapper.availabilityZoneIdFromAvailabilityZoneLetter(azLetter);\n\n      // Is there packet loss impact?\n      let packetLossImpact: IAlarm = BasicServiceMultiAZObservability.isThereAnAZPacketLossImpactNATGW(\n        this,\n        this.natGateways![az],\n        availabilityZoneId,\n        az,\n        props.packetLossImpactPercentageThreshold ? props.packetLossImpactPercentageThreshold : 0.01,\n        props.period ? props.period : Duration.minutes(1),\n        props.evaluationPeriods,\n        props.datapointsToAlarm\n      );\n      // Is this AZ an outlier for this NATGW?\n      let packetLossOutlier: IAlarm = BasicServiceMultiAZObservability.isAZAnOutlierForPacketLossNATGW(\n        this,\n        this.natGateways!,\n        availabilityZoneId,\n        az,\n        0.66,\n        props.period ? props.period : Duration.minutes(1),\n        props.evaluationPeriods,\n        props.datapointsToAlarm\n      );\n      packetLossPerAZAlarms[az] = new CompositeAlarm(this, az.substring(az.length - 1) + \"-packet-loss-composite-alarm\", {\n        alarmRule: AlarmRule.allOf(packetLossImpact, packetLossOutlier),\n        compositeAlarmName: availabilityZoneId + \"-packet-loss-composite-alarm\"\n      });\n    });\n\n    return packetLossPerAZAlarms;\n  }\n\n  private getTotalPacketDropsPerZone(\n    natgws: {[key: string]: CfnNatGateway[]},\n    period: Duration\n  ) : {[key: string]: IMetric}\n  {\n    let dropsPerZone: {[key: string]: IMetric} = {};\n    let metricsPerAZ: {[key: string]: IMetric[]} = {};\n    let keyprefix: string = MetricsHelper.nextChar('');\n\n    Object.keys(natgws).forEach((availabilityZone: string) => {\n\n      let azLetter = availabilityZone.substring(availabilityZone.length - 1);\n      let availabilityZoneId = this._azMapper.availabilityZoneIdFromAvailabilityZoneLetter(azLetter);\n\n      if (!(availabilityZone in metricsPerAZ)) {\n        metricsPerAZ[availabilityZone] = [];\n      }\n\n      natgws[availabilityZone].forEach((natgw: CfnNatGateway) => {\n\n        metricsPerAZ[availabilityZone].push(new Metric({\n          metricName: 'PacketsDropCount',\n          namespace: 'AWS/NATGateway',\n          statistic: Stats.SUM,\n          unit: Unit.COUNT,\n          label: availabilityZoneId + ' packet drops',\n          dimensionsMap: {\n            NatGatewayId: natgw.attrNatGatewayId,\n          },\n          period: period,\n        }));\n      });   \n    });\n\n    Object.keys(metricsPerAZ).forEach((availabilityZone: string) => {\n      let azLetter = availabilityZone.substring(availabilityZone.length - 1);\n      let availabilityZoneId = this._azMapper.availabilityZoneIdFromAvailabilityZoneLetter(azLetter);\n\n      let usingMetrics: {[key: string]: IMetric} = {};\n\n      metricsPerAZ[availabilityZone].forEach((metric: IMetric) => {\n        usingMetrics[`${keyprefix}1`] = metric;\n        keyprefix = MetricsHelper.nextChar(keyprefix);\n      });\n\n      dropsPerZone[availabilityZone] = new MathExpression({\n        expression: Object.keys(usingMetrics).join(\"+\"),\n        usingMetrics: usingMetrics,\n        label: availabilityZoneId + \" total packet drops\",\n        period: period\n      });\n    });\n\n    return dropsPerZone;\n  }\n}\n"]}
|
|
@@ -0,0 +1,51 @@
|
|
|
1
|
+
import { Dashboard, IAlarm } from 'aws-cdk-lib/aws-cloudwatch';
|
|
2
|
+
import { CfnNatGateway } from 'aws-cdk-lib/aws-ec2';
|
|
3
|
+
import { IApplicationLoadBalancer } from 'aws-cdk-lib/aws-elasticloadbalancingv2';
|
|
4
|
+
import { IConstruct } from 'constructs';
|
|
5
|
+
/**
|
|
6
|
+
* Properties of a basic service
|
|
7
|
+
*/
|
|
8
|
+
export interface IBasicServiceMultiAZObservability extends IConstruct {
|
|
9
|
+
/**
|
|
10
|
+
* The NAT Gateways being used in the service, each set of NAT Gateways
|
|
11
|
+
* are keyed by their Availability Zone Id
|
|
12
|
+
*/
|
|
13
|
+
natGateways?: {
|
|
14
|
+
[key: string]: CfnNatGateway[];
|
|
15
|
+
};
|
|
16
|
+
/**
|
|
17
|
+
* The application load balancers being used by the service
|
|
18
|
+
*/
|
|
19
|
+
applicationLoadBalancers?: IApplicationLoadBalancer[];
|
|
20
|
+
/**
|
|
21
|
+
* The name of the service
|
|
22
|
+
*/
|
|
23
|
+
serviceName: string;
|
|
24
|
+
/**
|
|
25
|
+
* The alarms indicating if an AZ is an outlier for NAT GW
|
|
26
|
+
* packet loss and has isolated impact. This will be 1 composite alarm
|
|
27
|
+
* per AZ that triggers if any NAT GW in that AZ sees outlier impact.
|
|
28
|
+
*/
|
|
29
|
+
natGWZonalIsolatedImpactAlarms?: {
|
|
30
|
+
[key: string]: IAlarm;
|
|
31
|
+
};
|
|
32
|
+
/**
|
|
33
|
+
* The alarms indicating if an AZ is an outlier for ALB
|
|
34
|
+
* faults and has isolated impact. This will be 1 composite alarm
|
|
35
|
+
* per AZ that triggers if any ALB in that AZ sees outlier impact.
|
|
36
|
+
*/
|
|
37
|
+
albZonalIsolatedImpactAlarms?: {
|
|
38
|
+
[key: string]: IAlarm;
|
|
39
|
+
};
|
|
40
|
+
/**
|
|
41
|
+
* The alarms indicating if an AZ has isolated impact
|
|
42
|
+
* from either ALB or NAT GW metrics
|
|
43
|
+
*/
|
|
44
|
+
aggregateZonalIsolatedImpactAlarms: {
|
|
45
|
+
[key: string]: IAlarm;
|
|
46
|
+
};
|
|
47
|
+
/**
|
|
48
|
+
* The optional dashboard created for observability
|
|
49
|
+
*/
|
|
50
|
+
dashboard?: Dashboard;
|
|
51
|
+
}
|