@cdklabs/multi-az-observability 0.0.1-alpha.6 → 0.0.1-alpha.60
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 +2826 -1008
- package/.jsii.tabl.json +1 -0
- package/API.md +5406 -1585
- package/README.md +176 -152
- package/lib/alarmsandrules/AvailabilityAndLatencyAlarmsAndRules.d.ts +18 -14
- package/lib/alarmsandrules/AvailabilityAndLatencyAlarmsAndRules.js +98 -60
- package/lib/alarmsandrules/BaseOperationRegionalAlarmsAndRules.d.ts +3 -3
- package/lib/alarmsandrules/BaseOperationRegionalAlarmsAndRules.js +2 -2
- package/lib/alarmsandrules/BaseOperationZonalAlarmsAndRules.d.ts +6 -15
- package/lib/alarmsandrules/BaseOperationZonalAlarmsAndRules.js +2 -10
- package/lib/alarmsandrules/CanaryOperationZonalAlarmsAndRules.d.ts +11 -3
- package/lib/alarmsandrules/CanaryOperationZonalAlarmsAndRules.js +29 -13
- package/lib/alarmsandrules/IBaseOperationZonalAlarmsAndRules.d.ts +0 -8
- package/lib/alarmsandrules/IBaseOperationZonalAlarmsAndRules.js +1 -1
- package/lib/alarmsandrules/IOperationAlarmsAndRules.d.ts +26 -23
- package/lib/alarmsandrules/IOperationAlarmsAndRules.js +1 -1
- package/lib/alarmsandrules/IServerSideOperationRegionalAlarmsAndRules.js +1 -1
- package/lib/alarmsandrules/IServiceAlarmsAndRules.d.ts +19 -15
- package/lib/alarmsandrules/IServiceAlarmsAndRules.js +1 -1
- package/lib/alarmsandrules/OperationAlarmsAndRules.d.ts +21 -18
- package/lib/alarmsandrules/OperationAlarmsAndRules.js +60 -72
- package/lib/alarmsandrules/ServerSideOperationRegionalAlarmsAndRules.d.ts +5 -5
- package/lib/alarmsandrules/ServerSideOperationRegionalAlarmsAndRules.js +2 -3
- package/lib/alarmsandrules/ServerSideOperationZonalAlarmsAndRules.d.ts +21 -13
- package/lib/alarmsandrules/ServerSideOperationZonalAlarmsAndRules.js +48 -29
- package/lib/alarmsandrules/ServiceAlarmsAndRules.d.ts +19 -15
- package/lib/alarmsandrules/ServiceAlarmsAndRules.js +34 -136
- package/lib/alarmsandrules/props/BaseOperationRegionalAlarmsAndRulesProps.d.ts +4 -3
- package/lib/alarmsandrules/props/BaseOperationRegionalAlarmsAndRulesProps.js +1 -1
- package/lib/alarmsandrules/props/BaseOperationZonalAlarmsAndRulesProps.d.ts +6 -44
- package/lib/alarmsandrules/props/BaseOperationZonalAlarmsAndRulesProps.js +1 -1
- package/lib/alarmsandrules/props/OperationAlarmsAndRulesProps.d.ts +39 -7
- package/lib/alarmsandrules/props/OperationAlarmsAndRulesProps.js +1 -1
- package/lib/azmapper/AvailabilityZoneMapper.js +8 -9
- package/lib/basic_observability/BasicServiceDashboard.d.ts +0 -3
- package/lib/basic_observability/BasicServiceDashboard.js +24 -112
- package/lib/basic_observability/BasicServiceMultiAZObservability.d.ts +8 -9
- package/lib/basic_observability/BasicServiceMultiAZObservability.js +97 -349
- package/lib/basic_observability/props/ApplicationLoadBalancerDetectionProps.d.ts +66 -0
- package/lib/basic_observability/props/ApplicationLoadBalancerDetectionProps.js +3 -0
- package/lib/basic_observability/props/BasicServiceDashboardProps.d.ts +27 -13
- package/lib/basic_observability/props/BasicServiceDashboardProps.js +1 -1
- package/lib/basic_observability/props/BasicServiceMultiAZObservabilityProps.d.ts +10 -44
- package/lib/basic_observability/props/BasicServiceMultiAZObservabilityProps.js +1 -1
- package/lib/basic_observability/props/NatGatewayDetectionProps.d.ts +31 -0
- package/lib/basic_observability/props/NatGatewayDetectionProps.js +3 -0
- package/lib/canaries/CanaryFunction.js +14 -13
- package/lib/canaries/CanaryTest.js +4 -4
- package/lib/canaries/src/canary.zip +0 -0
- package/lib/dashboards/ContributorInsightsWidget.d.ts +1 -1
- package/lib/dashboards/ContributorInsightsWidget.js +13 -17
- package/lib/dashboards/OperationAvailabilityAndLatencyDashboard.d.ts +2 -6
- package/lib/dashboards/OperationAvailabilityAndLatencyDashboard.js +436 -510
- package/lib/dashboards/ServiceAvailabilityAndLatencyDashboard.d.ts +2 -10
- package/lib/dashboards/ServiceAvailabilityAndLatencyDashboard.js +420 -416
- package/lib/dashboards/props/OperationAvailabilityAndLatencyDashboardProps.d.ts +3 -66
- package/lib/dashboards/props/OperationAvailabilityAndLatencyDashboardProps.js +1 -1
- package/lib/dashboards/props/OperationAvailabilityWidgetProps.d.ts +7 -2
- package/lib/dashboards/props/OperationAvailabilityWidgetProps.js +1 -1
- package/lib/dashboards/props/OperationLatencyWidgetProps.d.ts +7 -2
- package/lib/dashboards/props/OperationLatencyWidgetProps.js +1 -1
- package/lib/dashboards/props/ServiceAvailabilityAndLatencyDashboardProps.d.ts +8 -8
- package/lib/dashboards/props/ServiceAvailabilityAndLatencyDashboardProps.js +1 -1
- package/lib/index.d.ts +56 -25
- package/lib/index.js +37 -19
- package/lib/metrics/ApplicationLoadBalancerMetrics.d.ts +79 -2
- package/lib/metrics/ApplicationLoadBalancerMetrics.js +883 -26
- package/lib/metrics/AvailabilityAndLatencyMetrics.d.ts +23 -0
- package/lib/metrics/AvailabilityAndLatencyMetrics.js +125 -31
- package/lib/metrics/NatGatewayMetrics.d.ts +113 -0
- package/lib/metrics/NatGatewayMetrics.js +357 -0
- package/lib/metrics/RegionalAvailabilityMetrics.js +9 -10
- package/lib/metrics/RegionalLatencyMetrics.d.ts +1 -1
- package/lib/metrics/RegionalLatencyMetrics.js +27 -20
- package/lib/metrics/ZonalAvailabilityMetrics.d.ts +2 -8
- package/lib/metrics/ZonalAvailabilityMetrics.js +13 -28
- package/lib/metrics/ZonalLatencyMetrics.d.ts +2 -1
- package/lib/metrics/ZonalLatencyMetrics.js +33 -23
- package/lib/metrics/props/AvailabilityAndLatencyMetricProps.d.ts +4 -0
- package/lib/metrics/props/AvailabilityAndLatencyMetricProps.js +1 -1
- package/lib/metrics/props/LatencyMetricProps.d.ts +6 -0
- package/lib/metrics/props/LatencyMetricProps.js +1 -1
- package/lib/metrics/props/ZonalAvailabilityMetricProps.d.ts +4 -0
- package/lib/metrics/props/ZonalAvailabilityMetricProps.js +1 -1
- package/lib/metrics/props/ZonalLatencyMetricProps.d.ts +4 -0
- package/lib/metrics/props/ZonalLatencyMetricProps.js +1 -1
- package/lib/monitoring/src/monitoring-layer.zip +0 -0
- package/lib/outlier-detection/ApplicationLoadBalancerAvailabilityOutlierAlgorithm.d.ts +10 -0
- package/lib/outlier-detection/ApplicationLoadBalancerAvailabilityOutlierAlgorithm.js +15 -0
- package/lib/{basic_observability/props/ApplicationLoadBalancerLatencyOutlierCalculation.d.ts → outlier-detection/ApplicationLoadBalancerLatencyOutlierAlgorithm.d.ts} +1 -1
- package/lib/outlier-detection/ApplicationLoadBalancerLatencyOutlierAlgorithm.js +23 -0
- package/lib/outlier-detection/LatencyOutlierMetricAggregation.d.ts +22 -0
- package/lib/outlier-detection/LatencyOutlierMetricAggregation.js +27 -0
- package/lib/outlier-detection/OutlierDetectionFunction.js +7 -6
- package/lib/outlier-detection/PacketLossOutlierAlgorithm.d.ts +10 -0
- package/lib/outlier-detection/PacketLossOutlierAlgorithm.js +15 -0
- package/lib/outlier-detection/src/outlier-detection.zip +0 -0
- package/lib/outlier-detection/src/scipy-layer.zip +0 -0
- package/lib/services/CanaryMetrics.d.ts +4 -3
- package/lib/services/CanaryMetrics.js +3 -4
- package/lib/services/CanaryTestAvailabilityMetricsOverride.d.ts +21 -0
- package/lib/services/CanaryTestAvailabilityMetricsOverride.js +23 -0
- package/lib/services/CanaryTestLatencyMetricsOverride.d.ts +15 -0
- package/lib/services/CanaryTestLatencyMetricsOverride.js +20 -0
- package/lib/services/CanaryTestMetricsOverride.d.ts +1 -13
- package/lib/services/CanaryTestMetricsOverride.js +2 -4
- package/lib/services/ContributorInsightRuleDetails.js +1 -1
- package/lib/services/ICanaryMetrics.d.ts +4 -3
- package/lib/services/ICanaryMetrics.js +1 -1
- package/lib/services/ICanaryTestAvailabilityMetricsOverride.d.ts +23 -0
- package/lib/services/ICanaryTestAvailabilityMetricsOverride.js +3 -0
- package/lib/services/ICanaryTestLatencyMetricsOverride.d.ts +13 -0
- package/lib/services/ICanaryTestLatencyMetricsOverride.js +3 -0
- package/lib/services/ICanaryTestMetricsOverride.d.ts +0 -12
- package/lib/services/ICanaryTestMetricsOverride.js +1 -1
- package/lib/services/IInstrumentedServiceMultiAZObservability.d.ts +13 -3
- package/lib/services/IInstrumentedServiceMultiAZObservability.js +1 -1
- package/lib/services/IOperation.d.ts +8 -6
- package/lib/services/IOperation.js +1 -1
- package/lib/services/IOperationAvailabilityMetricDetails.d.ts +18 -0
- package/lib/services/IOperationAvailabilityMetricDetails.js +3 -0
- package/lib/services/IOperationLatencyMetricDetails.d.ts +12 -0
- package/lib/services/IOperationLatencyMetricDetails.js +3 -0
- package/lib/services/IOperationMetricDetails.d.ts +0 -12
- package/lib/services/IOperationMetricDetails.js +1 -1
- package/lib/services/IService.d.ts +18 -4
- package/lib/services/IService.js +1 -1
- package/lib/services/IServiceAvailabilityMetricDetails.d.ts +18 -0
- package/lib/services/IServiceAvailabilityMetricDetails.js +3 -0
- package/lib/services/IServiceLatencyMetricDetails.d.ts +12 -0
- package/lib/services/IServiceLatencyMetricDetails.js +3 -0
- package/lib/services/IServiceMetricDetails.d.ts +0 -12
- package/lib/services/IServiceMetricDetails.js +1 -1
- package/lib/services/InstrumentedServiceMultiAZObservability.d.ts +15 -3
- package/lib/services/InstrumentedServiceMultiAZObservability.js +252 -228
- package/lib/services/Operation.d.ts +8 -6
- package/lib/services/Operation.js +4 -3
- package/lib/services/OperationAvailabilityMetricDetails.d.ts +22 -0
- package/lib/services/OperationAvailabilityMetricDetails.js +24 -0
- package/lib/services/OperationLatencyMetricDetails.d.ts +16 -0
- package/lib/services/OperationLatencyMetricDetails.js +21 -0
- package/lib/services/OperationMetricDetails.d.ts +1 -13
- package/lib/services/OperationMetricDetails.js +2 -8
- package/lib/services/Service.d.ts +18 -4
- package/lib/services/Service.js +4 -2
- package/lib/services/ServiceAvailabilityMetricDetails.d.ts +21 -0
- package/lib/services/ServiceAvailabilityMetricDetails.js +20 -0
- package/lib/services/ServiceLatencyMetricDetails.d.ts +15 -0
- package/lib/services/ServiceLatencyMetricDetails.js +19 -0
- package/lib/services/ServiceMetricDetails.d.ts +1 -13
- package/lib/services/ServiceMetricDetails.js +2 -4
- package/lib/services/props/CanaryMetricProps.d.ts +4 -3
- package/lib/services/props/CanaryMetricProps.js +1 -1
- package/lib/services/props/CanaryTestAvailabilityMetricsOverrideProps.d.ts +22 -0
- package/lib/services/props/CanaryTestAvailabilityMetricsOverrideProps.js +3 -0
- package/lib/services/props/CanaryTestLatencyMetricsOverrideProps.d.ts +14 -0
- package/lib/services/props/CanaryTestLatencyMetricsOverrideProps.js +3 -0
- package/lib/services/props/CanaryTestMetricsOverrideProps.d.ts +0 -16
- package/lib/services/props/CanaryTestMetricsOverrideProps.js +1 -1
- package/lib/services/props/InstrumentedServiceMultiAZObservabilityProps.d.ts +50 -5
- package/lib/services/props/InstrumentedServiceMultiAZObservabilityProps.js +1 -1
- package/lib/services/props/MetricDimensions.js +1 -1
- package/lib/services/props/OperationAvailabilityMetricDetailsProps.d.ts +22 -0
- package/lib/services/props/OperationAvailabilityMetricDetailsProps.js +3 -0
- package/lib/services/props/OperationLatencyMetricDetailsProps.d.ts +14 -0
- package/lib/services/props/OperationLatencyMetricDetailsProps.js +3 -0
- package/lib/services/props/OperationMetricDetailsProps.d.ts +0 -16
- package/lib/services/props/OperationMetricDetailsProps.js +1 -1
- package/lib/services/props/OperationProps.d.ts +4 -3
- package/lib/services/props/OperationProps.js +1 -1
- package/lib/services/props/ServiceAvailabilityMetricDetailsProps.d.ts +18 -0
- package/lib/services/props/ServiceAvailabilityMetricDetailsProps.js +3 -0
- package/lib/services/props/ServiceLatencyMetricDetailsProps.d.ts +12 -0
- package/lib/services/props/ServiceLatencyMetricDetailsProps.js +3 -0
- package/lib/services/props/ServiceMetricDetailsProps.d.ts +0 -12
- package/lib/services/props/ServiceMetricDetailsProps.js +1 -1
- package/lib/services/props/ServiceProps.d.ts +18 -4
- package/lib/services/props/ServiceProps.js +1 -1
- package/lib/utilities/MetricsHelper.d.ts +17 -9
- package/lib/utilities/MetricsHelper.js +34 -10
- package/lib/utilities/MinimumUnhealthyTargets.d.ts +25 -0
- package/lib/utilities/MinimumUnhealthyTargets.js +3 -0
- package/package.json +14 -14
- package/rosetta/default.ts-fixture +72 -0
- package/rosetta/service.ts-fixture +144 -0
- package/lib/basic_observability/props/ApplicationLoadBalancerLatencyOutlierCalculation.js +0 -23
- package/lib/services/IBasicServiceMultiAZObservability.d.ts +0 -45
- package/lib/services/IBasicServiceMultiAZObservability.js +0 -3
|
@@ -11,9 +11,11 @@ const constructs_1 = require("constructs");
|
|
|
11
11
|
const AvailabilityZoneMapper_1 = require("../azmapper/AvailabilityZoneMapper");
|
|
12
12
|
const BasicServiceDashboard_1 = require("./BasicServiceDashboard");
|
|
13
13
|
const ApplicationLoadBalancerMetrics_1 = require("../metrics/ApplicationLoadBalancerMetrics");
|
|
14
|
-
const AvailabilityMetricType_1 = require("../utilities/AvailabilityMetricType");
|
|
15
14
|
const MetricsHelper_1 = require("../utilities/MetricsHelper");
|
|
16
|
-
const
|
|
15
|
+
const ApplicationLoadBalancerLatencyOutlierAlgorithm_1 = require("../outlier-detection/ApplicationLoadBalancerLatencyOutlierAlgorithm");
|
|
16
|
+
const NatGatewayMetrics_1 = require("../metrics/NatGatewayMetrics");
|
|
17
|
+
const ApplicationLoadBalancerAvailabilityOutlierAlgorithm_1 = require("../outlier-detection/ApplicationLoadBalancerAvailabilityOutlierAlgorithm");
|
|
18
|
+
const PacketLossOutlierAlgorithm_1 = require("../outlier-detection/PacketLossOutlierAlgorithm");
|
|
17
19
|
/**
|
|
18
20
|
* Basic observability for a service using metrics from
|
|
19
21
|
* ALBs and NAT Gateways
|
|
@@ -21,10 +23,13 @@ const ApplicationLoadBalancerLatencyOutlierCalculation_1 = require("./props/Appl
|
|
|
21
23
|
class BasicServiceMultiAZObservability extends constructs_1.Construct {
|
|
22
24
|
constructor(scope, id, props) {
|
|
23
25
|
super(scope, id);
|
|
26
|
+
if (!(props.applicationLoadBalancerProps) && !(props.natGatewayProps)) {
|
|
27
|
+
throw new Error("You must define either ALBs or NAT Gateways for this service in order to create a dashboard.");
|
|
28
|
+
}
|
|
24
29
|
// Initialize class properties
|
|
25
30
|
this.serviceName = props.serviceName;
|
|
26
|
-
this.applicationLoadBalancers = props.
|
|
27
|
-
this.natGateways = props.natGateways;
|
|
31
|
+
this.applicationLoadBalancers = props.applicationLoadBalancerProps?.albTargetGroupMap.map(entry => entry.applicationLoadBalancer);
|
|
32
|
+
this.natGateways = props.natGatewayProps?.natGateways;
|
|
28
33
|
this.aggregateZonalIsolatedImpactAlarms = {};
|
|
29
34
|
this.albZonalIsolatedImpactAlarms = {};
|
|
30
35
|
this.natGWZonalIsolatedImpactAlarms = {};
|
|
@@ -32,26 +37,25 @@ class BasicServiceMultiAZObservability extends constructs_1.Construct {
|
|
|
32
37
|
this._azMapper = new AvailabilityZoneMapper_1.AvailabilityZoneMapper(this, 'availability-zone-mapper');
|
|
33
38
|
// Create ALB metrics and alarms per AZ
|
|
34
39
|
if (this.applicationLoadBalancers) {
|
|
35
|
-
this.albZonalIsolatedImpactAlarms = this.
|
|
40
|
+
this.albZonalIsolatedImpactAlarms = this.createAlbZonalImpactAlarms(props);
|
|
36
41
|
}
|
|
37
42
|
// Create NAT Gateway metrics and alarms per AZ
|
|
38
43
|
if (this.natGateways) {
|
|
39
|
-
this.natGWZonalIsolatedImpactAlarms = this.
|
|
44
|
+
this.natGWZonalIsolatedImpactAlarms = this.createNatGatewayZonalImpactAlarms(props);
|
|
40
45
|
}
|
|
41
46
|
// Look through all of the per AZ ALB alarms, if there's also a NAT GW alarm
|
|
42
47
|
// create a composite alarm if either of them trigger
|
|
43
|
-
Object.keys(this.albZonalIsolatedImpactAlarms).forEach((
|
|
44
|
-
let azLetter = az.substring(az.length - 1);
|
|
48
|
+
Object.keys(this.albZonalIsolatedImpactAlarms).forEach((azLetter) => {
|
|
45
49
|
let availabilityZoneId = this._azMapper.availabilityZoneIdFromAvailabilityZoneLetter(azLetter);
|
|
46
|
-
if (
|
|
47
|
-
this.aggregateZonalIsolatedImpactAlarms[
|
|
48
|
-
alarmRule: aws_cloudwatch_1.AlarmRule.anyOf(this.albZonalIsolatedImpactAlarms[
|
|
50
|
+
if (azLetter in this.natGWZonalIsolatedImpactAlarms) {
|
|
51
|
+
this.aggregateZonalIsolatedImpactAlarms[azLetter] = new aws_cloudwatch_1.CompositeAlarm(this, azLetter + "-isolated-impact-alarm", {
|
|
52
|
+
alarmRule: aws_cloudwatch_1.AlarmRule.anyOf(this.albZonalIsolatedImpactAlarms[azLetter], this.natGWZonalIsolatedImpactAlarms[azLetter]),
|
|
49
53
|
compositeAlarmName: availabilityZoneId + "-isolated-impact-alarm"
|
|
50
54
|
});
|
|
51
55
|
}
|
|
52
56
|
else {
|
|
53
|
-
this.aggregateZonalIsolatedImpactAlarms[
|
|
54
|
-
alarmRule: aws_cloudwatch_1.AlarmRule.anyOf(this.albZonalIsolatedImpactAlarms[
|
|
57
|
+
this.aggregateZonalIsolatedImpactAlarms[azLetter] = new aws_cloudwatch_1.CompositeAlarm(this, azLetter + "-isolated-impact-alarm", {
|
|
58
|
+
alarmRule: aws_cloudwatch_1.AlarmRule.anyOf(this.albZonalIsolatedImpactAlarms[azLetter]),
|
|
55
59
|
compositeAlarmName: availabilityZoneId + "-isolated-impact-alarm"
|
|
56
60
|
});
|
|
57
61
|
}
|
|
@@ -59,12 +63,11 @@ class BasicServiceMultiAZObservability extends constructs_1.Construct {
|
|
|
59
63
|
// Look through all of the per AZ NAT GW alarms. If there's an AZ we haven't seen in the ALB
|
|
60
64
|
// alarms yet, then it will just be a NAT GW alarm that we'll turn into the same kind of
|
|
61
65
|
// composite alarm
|
|
62
|
-
Object.keys(this.natGWZonalIsolatedImpactAlarms).forEach((
|
|
63
|
-
if (!(
|
|
64
|
-
let azLetter = az.substring(az.length - 1);
|
|
66
|
+
Object.keys(this.natGWZonalIsolatedImpactAlarms).forEach((azLetter) => {
|
|
67
|
+
if (!(azLetter in this.aggregateZonalIsolatedImpactAlarms)) {
|
|
65
68
|
let availabilityZoneId = this._azMapper.availabilityZoneIdFromAvailabilityZoneLetter(azLetter);
|
|
66
|
-
this.aggregateZonalIsolatedImpactAlarms[
|
|
67
|
-
alarmRule: aws_cloudwatch_1.AlarmRule.anyOf(this.natGWZonalIsolatedImpactAlarms[
|
|
69
|
+
this.aggregateZonalIsolatedImpactAlarms[azLetter] = new aws_cloudwatch_1.CompositeAlarm(this, azLetter + "-isolated-impact-alarm", {
|
|
70
|
+
alarmRule: aws_cloudwatch_1.AlarmRule.anyOf(this.natGWZonalIsolatedImpactAlarms[azLetter]),
|
|
68
71
|
compositeAlarmName: availabilityZoneId + "-isolated-impact-alarm"
|
|
69
72
|
});
|
|
70
73
|
}
|
|
@@ -74,167 +77,59 @@ class BasicServiceMultiAZObservability extends constructs_1.Construct {
|
|
|
74
77
|
this.dashboard = new BasicServiceDashboard_1.BasicServiceDashboard(this, 'BasicServiceDashboard', {
|
|
75
78
|
serviceName: props.serviceName.toLowerCase(),
|
|
76
79
|
zonalAggregateIsolatedImpactAlarms: this.aggregateZonalIsolatedImpactAlarms,
|
|
77
|
-
zonalLoadBalancerIsolatedImpactAlarms: this.albZonalIsolatedImpactAlarms,
|
|
78
|
-
zonalNatGatewayIsolatedImpactAlarms: this.natGWZonalIsolatedImpactAlarms,
|
|
79
80
|
interval: props.interval,
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
81
|
+
natgws: props.natGatewayProps,
|
|
82
|
+
albs: props.applicationLoadBalancerProps,
|
|
83
|
+
period: props.period ? props.period : aws_cdk_lib_1.Duration.minutes(1),
|
|
84
|
+
azMapper: this._azMapper
|
|
83
85
|
}).dashboard;
|
|
84
86
|
}
|
|
85
87
|
}
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
label: availabilityZoneId + '-' + alb.loadBalancerArn + '-fault-rate',
|
|
94
|
-
availabilityZone: availabilityZone,
|
|
95
|
-
availabilityZoneId: availabilityZoneId,
|
|
96
|
-
metricType: AvailabilityMetricType_1.AvailabilityMetricType.FAULT_RATE
|
|
97
|
-
}),
|
|
98
|
-
evaluationPeriods: evaluationPeriods,
|
|
99
|
-
datapointsToAlarm: datapointsToAlarm,
|
|
100
|
-
threshold: threshold,
|
|
101
|
-
comparisonOperator: aws_cloudwatch_1.ComparisonOperator.GREATER_THAN_THRESHOLD,
|
|
102
|
-
treatMissingData: aws_cloudwatch_1.TreatMissingData.IGNORE
|
|
103
|
-
});
|
|
104
|
-
}
|
|
105
|
-
static isThereAZLatencyImpactAlb(scope, alb, availabilityZoneId, availabilityZone, threshold, statistic, keyprefix, period, evaluationPeriods, datapointsToAlarm) {
|
|
106
|
-
// Create a fault rate alarm for the ALB in the specified AZ
|
|
107
|
-
return new aws_cloudwatch_1.Alarm(scope, keyprefix + '-latency-alarm', {
|
|
108
|
-
alarmName: availabilityZoneId + '-' + alb.loadBalancerArn + '-latency',
|
|
109
|
-
actionsEnabled: false,
|
|
110
|
-
metric: ApplicationLoadBalancerMetrics_1.ApplicationLoadBalancerMetrics.getPerAZLatencyMetric({
|
|
111
|
-
alb: alb,
|
|
112
|
-
availabilityZone: availabilityZone,
|
|
113
|
-
availabilityZoneId: availabilityZoneId,
|
|
114
|
-
label: availabilityZoneId + "-" + alb.loadBalancerArn + "-target-latency",
|
|
115
|
-
period: period,
|
|
116
|
-
statistic: statistic
|
|
117
|
-
}),
|
|
118
|
-
evaluationPeriods: evaluationPeriods,
|
|
119
|
-
datapointsToAlarm: datapointsToAlarm,
|
|
120
|
-
threshold: threshold,
|
|
121
|
-
comparisonOperator: aws_cloudwatch_1.ComparisonOperator.GREATER_THAN_THRESHOLD,
|
|
122
|
-
treatMissingData: aws_cloudwatch_1.TreatMissingData.IGNORE
|
|
123
|
-
});
|
|
124
|
-
}
|
|
125
|
-
static isAZAnOutlierForAvailabilityAlb(scope, alb, availabilityZoneId, availabilityZone, threshold, keyprefix, period, evaluationPeriods, datapointsToAlarm) {
|
|
126
|
-
let usingMetrics = {};
|
|
127
|
-
let azMetricId = "";
|
|
128
|
-
alb.vpc.availabilityZones.forEach((az) => {
|
|
129
|
-
let azFaultCount = ApplicationLoadBalancerMetrics_1.ApplicationLoadBalancerMetrics.getPerAZAvailabilityMetric(alb, {
|
|
130
|
-
metricType: AvailabilityMetricType_1.AvailabilityMetricType.FAULT_COUNT,
|
|
131
|
-
availabilityZone: availabilityZone,
|
|
132
|
-
availabilityZoneId: availabilityZoneId,
|
|
133
|
-
period: period,
|
|
134
|
-
label: availabilityZoneId + "-" + alb.loadBalancerArn + "-fault-count",
|
|
135
|
-
keyprefix: keyprefix
|
|
136
|
-
});
|
|
137
|
-
keyprefix = MetricsHelper_1.MetricsHelper.nextChar(keyprefix);
|
|
138
|
-
usingMetrics[`${keyprefix}1`] = azFaultCount;
|
|
139
|
-
if (az == availabilityZone) {
|
|
140
|
-
azMetricId = `${keyprefix}1`;
|
|
141
|
-
}
|
|
142
|
-
keyprefix = MetricsHelper_1.MetricsHelper.nextChar(keyprefix);
|
|
143
|
-
});
|
|
144
|
-
return new aws_cloudwatch_1.Alarm(scope, keyprefix + '-availability-outlier-alarm', {
|
|
145
|
-
alarmName: availabilityZoneId + '-' + alb.loadBalancerArn + '-availability-impact-outlier',
|
|
146
|
-
actionsEnabled: false,
|
|
147
|
-
metric: new aws_cloudwatch_1.MathExpression({
|
|
148
|
-
expression: `${azMetricId}/(${Object.keys(usingMetrics).join("+")})`,
|
|
149
|
-
usingMetrics: usingMetrics,
|
|
150
|
-
label: availabilityZoneId + '-' + alb.loadBalancerArn + '-percent-of-faults',
|
|
151
|
-
period: period,
|
|
152
|
-
}),
|
|
153
|
-
evaluationPeriods: evaluationPeriods,
|
|
154
|
-
datapointsToAlarm: datapointsToAlarm,
|
|
155
|
-
threshold: threshold,
|
|
156
|
-
comparisonOperator: aws_cloudwatch_1.ComparisonOperator.GREATER_THAN_THRESHOLD,
|
|
157
|
-
treatMissingData: aws_cloudwatch_1.TreatMissingData.IGNORE
|
|
158
|
-
});
|
|
159
|
-
}
|
|
160
|
-
static isAZAnOutlierForLatencyAlb(scope, alb, algorithm, availabilityZoneId, availabilityZone, statistic, threshold, period, evaluationPeriods, datapointsToAlarm, keyprefix) {
|
|
161
|
-
let usingMetrics = {};
|
|
162
|
-
let azMetricId = "";
|
|
163
|
-
switch (algorithm) {
|
|
164
|
-
case ApplicationLoadBalancerLatencyOutlierCalculation_1.ApplicationLoadBalancerLatencyOutlierCalculation.Z_SCORE:
|
|
165
|
-
default:
|
|
166
|
-
alb.vpc.availabilityZones.forEach((az, index) => {
|
|
167
|
-
// Target response time
|
|
168
|
-
let targetResponseTime = ApplicationLoadBalancerMetrics_1.ApplicationLoadBalancerMetrics.getPerAZLatencyMetric({
|
|
169
|
-
alb: alb,
|
|
170
|
-
availabilityZone: az,
|
|
171
|
-
label: az + "-target-response-time",
|
|
172
|
-
statistic: statistic,
|
|
173
|
-
period: period
|
|
174
|
-
});
|
|
175
|
-
if (az == availabilityZone) {
|
|
176
|
-
azMetricId = `a${index}`;
|
|
177
|
-
usingMetrics[`a${index}`] = targetResponseTime;
|
|
178
|
-
}
|
|
179
|
-
else {
|
|
180
|
-
usingMetrics[`b${index}`] = targetResponseTime;
|
|
181
|
-
}
|
|
182
|
-
});
|
|
183
|
-
return new aws_cloudwatch_1.Alarm(scope, keyprefix + "-latency-outlier-alarm", {
|
|
184
|
-
alarmName: availabilityZoneId + '-' + alb.loadBalancerArn + '-latency-impact-outlier',
|
|
185
|
-
actionsEnabled: false,
|
|
186
|
-
metric: new aws_cloudwatch_1.MathExpression({
|
|
187
|
-
expression: `(${azMetricId} - AVG(METRICS("b"))) / AVG(STDDEV(METRICS("b")))`,
|
|
188
|
-
usingMetrics: usingMetrics,
|
|
189
|
-
label: availabilityZoneId + '-' + alb.loadBalancerArn + '-latency-z-score',
|
|
190
|
-
period: period,
|
|
191
|
-
}),
|
|
192
|
-
evaluationPeriods: evaluationPeriods,
|
|
193
|
-
datapointsToAlarm: datapointsToAlarm,
|
|
194
|
-
threshold: threshold,
|
|
195
|
-
comparisonOperator: aws_cloudwatch_1.ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,
|
|
196
|
-
treatMissingData: aws_cloudwatch_1.TreatMissingData.IGNORE
|
|
197
|
-
});
|
|
198
|
-
case ApplicationLoadBalancerLatencyOutlierCalculation_1.ApplicationLoadBalancerLatencyOutlierCalculation.STATIC:
|
|
199
|
-
alb.vpc.availabilityZones.forEach((az, index) => {
|
|
200
|
-
// Target response time
|
|
201
|
-
let targetResponseTime = ApplicationLoadBalancerMetrics_1.ApplicationLoadBalancerMetrics.getPerAZLatencyMetric({
|
|
202
|
-
alb: alb,
|
|
203
|
-
availabilityZone: az,
|
|
204
|
-
label: az + "-target-response-time",
|
|
205
|
-
statistic: `TC(${threshold}:)`,
|
|
206
|
-
period: period
|
|
207
|
-
});
|
|
208
|
-
if (az == availabilityZone) {
|
|
209
|
-
azMetricId = `a${index}`;
|
|
210
|
-
usingMetrics[`a${index}`] = targetResponseTime;
|
|
211
|
-
}
|
|
212
|
-
else {
|
|
213
|
-
usingMetrics[`b${index}`] = targetResponseTime;
|
|
214
|
-
}
|
|
215
|
-
});
|
|
216
|
-
return new aws_cloudwatch_1.Alarm(scope, keyprefix + "-latency-outlier-alarm", {
|
|
217
|
-
alarmName: availabilityZoneId + '-' + alb.loadBalancerArn + '-latency-impact-outlier',
|
|
218
|
-
actionsEnabled: false,
|
|
219
|
-
metric: new aws_cloudwatch_1.MathExpression({
|
|
220
|
-
expression: `${azMetricId}/(${Object.keys(usingMetrics).join("+")})`,
|
|
221
|
-
usingMetrics: usingMetrics,
|
|
222
|
-
label: availabilityZoneId + '-' + alb.loadBalancerArn + '-latency-static',
|
|
223
|
-
period: period,
|
|
224
|
-
}),
|
|
225
|
-
evaluationPeriods: evaluationPeriods,
|
|
226
|
-
datapointsToAlarm: datapointsToAlarm,
|
|
227
|
-
threshold: .66,
|
|
228
|
-
comparisonOperator: aws_cloudwatch_1.ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,
|
|
229
|
-
treatMissingData: aws_cloudwatch_1.TreatMissingData.IGNORE
|
|
230
|
-
});
|
|
231
|
-
}
|
|
232
|
-
}
|
|
233
|
-
doAlbMetrics(props) {
|
|
88
|
+
/**
|
|
89
|
+
*
|
|
90
|
+
* @param props
|
|
91
|
+
* @returns A composite alarm per AZ to indicate isolated zonal impact. The dictionary
|
|
92
|
+
* is keyed by the az letter, like "a", "b", "c".
|
|
93
|
+
*/
|
|
94
|
+
createAlbZonalImpactAlarms(props) {
|
|
234
95
|
// Create impact alarms per AZ, with each ALB providing
|
|
235
96
|
// an alarm for its AZs
|
|
236
97
|
let perAZImpactAlarms = {};
|
|
237
98
|
let keyPrefix = MetricsHelper_1.MetricsHelper.nextChar();
|
|
99
|
+
let availabilityOutlierDetectionAlgorithm = props.applicationLoadBalancerProps?.availabilityOutlierAlgorithm ?
|
|
100
|
+
props.applicationLoadBalancerProps.availabilityOutlierAlgorithm :
|
|
101
|
+
ApplicationLoadBalancerAvailabilityOutlierAlgorithm_1.ApplicationLoadBalancerAvailabilityOutlierAlgorithm.STATIC;
|
|
102
|
+
let latencyOutlierDetectionAlgorithm = props.applicationLoadBalancerProps?.latencyOutlierAlgorithm ?
|
|
103
|
+
props.applicationLoadBalancerProps.latencyOutlierAlgorithm :
|
|
104
|
+
ApplicationLoadBalancerLatencyOutlierAlgorithm_1.ApplicationLoadBalancerLatencyOutlierAlgorithm.Z_SCORE;
|
|
105
|
+
let availabilityOutlierThreshold;
|
|
106
|
+
if (props.applicationLoadBalancerProps?.availabilityOutlierThreshold) {
|
|
107
|
+
availabilityOutlierThreshold = props.applicationLoadBalancerProps.availabilityOutlierThreshold;
|
|
108
|
+
}
|
|
109
|
+
else {
|
|
110
|
+
switch (availabilityOutlierDetectionAlgorithm) {
|
|
111
|
+
case ApplicationLoadBalancerAvailabilityOutlierAlgorithm_1.ApplicationLoadBalancerAvailabilityOutlierAlgorithm.STATIC:
|
|
112
|
+
default:
|
|
113
|
+
availabilityOutlierThreshold = 66;
|
|
114
|
+
break;
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
let latencyOutlierThreshold;
|
|
118
|
+
if (props.applicationLoadBalancerProps?.latencyOutlierThreshold) {
|
|
119
|
+
latencyOutlierThreshold = props.applicationLoadBalancerProps.latencyOutlierThreshold;
|
|
120
|
+
}
|
|
121
|
+
else {
|
|
122
|
+
switch (latencyOutlierDetectionAlgorithm) {
|
|
123
|
+
case ApplicationLoadBalancerLatencyOutlierAlgorithm_1.ApplicationLoadBalancerLatencyOutlierAlgorithm.Z_SCORE:
|
|
124
|
+
default:
|
|
125
|
+
latencyOutlierThreshold = 3;
|
|
126
|
+
break;
|
|
127
|
+
case ApplicationLoadBalancerLatencyOutlierAlgorithm_1.ApplicationLoadBalancerLatencyOutlierAlgorithm.STATIC:
|
|
128
|
+
latencyOutlierThreshold = 66;
|
|
129
|
+
break;
|
|
130
|
+
}
|
|
131
|
+
}
|
|
132
|
+
let period = props.period ? props.period : aws_cdk_lib_1.Duration.minutes(1);
|
|
238
133
|
// Iterate each ALB
|
|
239
134
|
this.applicationLoadBalancers.forEach((alb) => {
|
|
240
135
|
// Iterate each AZ in the VPC
|
|
@@ -247,15 +142,15 @@ class BasicServiceMultiAZObservability extends constructs_1.Construct {
|
|
|
247
142
|
// Map letter to AZ ID
|
|
248
143
|
let availabilityZoneId = this._azMapper.availabilityZoneIdFromAvailabilityZoneLetter(azLetter);
|
|
249
144
|
// Is there availability impact in this AZ?
|
|
250
|
-
let availabilityImpact =
|
|
145
|
+
let availabilityImpact = ApplicationLoadBalancerMetrics_1.ApplicationLoadBalancerMetrics.createAZAvailabilityImpactAlarm(this, alb, availabilityZoneId, az, props.applicationLoadBalancerProps.faultCountPercentThreshold, keyPrefix, period, props.evaluationPeriods, props.datapointsToAlarm);
|
|
251
146
|
// Is there latency impact in this AZ?
|
|
252
|
-
let latencyImpact =
|
|
147
|
+
let latencyImpact = ApplicationLoadBalancerMetrics_1.ApplicationLoadBalancerMetrics.createAZLatencyImpactAlarm(this, alb, availabilityZoneId, az, props.applicationLoadBalancerProps.latencyThreshold, props.applicationLoadBalancerProps.latencyStatistic, keyPrefix, period, props.evaluationPeriods, props.datapointsToAlarm);
|
|
253
148
|
// Is the AZ an outlier for faults
|
|
254
|
-
let availabilityOutlier =
|
|
149
|
+
let availabilityOutlier = ApplicationLoadBalancerMetrics_1.ApplicationLoadBalancerMetrics.createAZAvailabilityOutlierAlarm(this, alb, availabilityOutlierDetectionAlgorithm, availabilityZoneId, az, availabilityOutlierThreshold, keyPrefix, period, props.evaluationPeriods, props.datapointsToAlarm);
|
|
255
150
|
// Is the AZ an outlier for latency
|
|
256
|
-
let latencyOutlier =
|
|
151
|
+
let latencyOutlier = ApplicationLoadBalancerMetrics_1.ApplicationLoadBalancerMetrics.createAZLatencyOutlierAlarm(this, alb, latencyOutlierDetectionAlgorithm, az, props.applicationLoadBalancerProps.latencyStatistic, props.applicationLoadBalancerProps.latencyThreshold, latencyOutlierThreshold, period, props.evaluationPeriods, props.datapointsToAlarm, this._azMapper);
|
|
257
152
|
// Alarm if the AZ shows impact and is an outlier
|
|
258
|
-
let azImpactAlarm = new aws_cloudwatch_1.CompositeAlarm(this,
|
|
153
|
+
let azImpactAlarm = new aws_cloudwatch_1.CompositeAlarm(this, azLetter + "-composite-impact-alarm", {
|
|
259
154
|
alarmRule: aws_cloudwatch_1.AlarmRule.anyOf(aws_cloudwatch_1.AlarmRule.allOf(availabilityImpact, availabilityOutlier), aws_cloudwatch_1.AlarmRule.allOf(latencyImpact, latencyOutlier)),
|
|
260
155
|
compositeAlarmName: availabilityZoneId + "-" +
|
|
261
156
|
alb.loadBalancerName +
|
|
@@ -274,199 +169,52 @@ class BasicServiceMultiAZObservability extends constructs_1.Construct {
|
|
|
274
169
|
Object.keys(perAZImpactAlarms).forEach((az) => {
|
|
275
170
|
let azLetter = az.substring(az.length - 1);
|
|
276
171
|
let availabilityZoneId = this._azMapper.availabilityZoneIdFromAvailabilityZoneLetter(azLetter);
|
|
277
|
-
azCompositeAlarms[
|
|
172
|
+
azCompositeAlarms[azLetter] = new aws_cloudwatch_1.CompositeAlarm(this, azLetter + "-alb-impact-composite-alarm", {
|
|
278
173
|
alarmRule: aws_cloudwatch_1.AlarmRule.anyOf(...perAZImpactAlarms[az]),
|
|
279
174
|
compositeAlarmName: availabilityZoneId + "-alb-impact-composite-alarm"
|
|
280
175
|
});
|
|
281
176
|
});
|
|
282
177
|
return azCompositeAlarms;
|
|
283
178
|
}
|
|
284
|
-
|
|
285
|
-
let keyprefix = MetricsHelper_1.MetricsHelper.nextChar();
|
|
286
|
-
let packetDropCountMetrics = {};
|
|
287
|
-
let packetsInFromSourceMetrics = {};
|
|
288
|
-
let packetsInFromDestinationMetrics = {};
|
|
289
|
-
natgws.forEach((natgw) => {
|
|
290
|
-
packetDropCountMetrics[`${keyprefix}1`] = new aws_cloudwatch_1.Metric({
|
|
291
|
-
metricName: 'PacketsDropCount',
|
|
292
|
-
namespace: 'AWS/NATGateway',
|
|
293
|
-
statistic: aws_cloudwatch_1.Stats.SUM,
|
|
294
|
-
unit: aws_cloudwatch_1.Unit.COUNT,
|
|
295
|
-
label: availabilityZoneId + ' packet drops',
|
|
296
|
-
dimensionsMap: {
|
|
297
|
-
NatGatewayId: natgw.attrNatGatewayId,
|
|
298
|
-
},
|
|
299
|
-
period: period,
|
|
300
|
-
});
|
|
301
|
-
// Calculate packets in from source
|
|
302
|
-
packetsInFromSourceMetrics[`${keyprefix}2`] = new aws_cloudwatch_1.Metric({
|
|
303
|
-
metricName: 'PacketsInFromSource',
|
|
304
|
-
namespace: 'AWS/NATGateway',
|
|
305
|
-
statistic: aws_cloudwatch_1.Stats.SUM,
|
|
306
|
-
unit: aws_cloudwatch_1.Unit.COUNT,
|
|
307
|
-
label: availabilityZoneId + ' packets in from source',
|
|
308
|
-
dimensionsMap: {
|
|
309
|
-
NatGatewayId: natgw.attrNatGatewayId,
|
|
310
|
-
},
|
|
311
|
-
period: period,
|
|
312
|
-
});
|
|
313
|
-
// Calculate packets in from destination
|
|
314
|
-
packetsInFromDestinationMetrics[`${keyprefix}3`] = new aws_cloudwatch_1.Metric({
|
|
315
|
-
metricName: 'PacketsInFromDestination',
|
|
316
|
-
namespace: 'AWS/NATGateway',
|
|
317
|
-
statistic: aws_cloudwatch_1.Stats.SUM,
|
|
318
|
-
unit: aws_cloudwatch_1.Unit.COUNT,
|
|
319
|
-
label: availabilityZoneId + ' packets in from destination',
|
|
320
|
-
dimensionsMap: {
|
|
321
|
-
NatGatewayId: natgw.attrNatGatewayId,
|
|
322
|
-
},
|
|
323
|
-
period: period,
|
|
324
|
-
});
|
|
325
|
-
keyprefix = MetricsHelper_1.MetricsHelper.nextChar(keyprefix);
|
|
326
|
-
});
|
|
327
|
-
let packetDropTotal = new aws_cloudwatch_1.MathExpression({
|
|
328
|
-
expression: Object.keys(packetDropCountMetrics).join("+"),
|
|
329
|
-
usingMetrics: packetDropCountMetrics,
|
|
330
|
-
period: period
|
|
331
|
-
});
|
|
332
|
-
let packetsInFromSourceTotal = new aws_cloudwatch_1.MathExpression({
|
|
333
|
-
expression: Object.keys(packetsInFromSourceMetrics).join("+"),
|
|
334
|
-
usingMetrics: packetsInFromSourceMetrics,
|
|
335
|
-
period: period
|
|
336
|
-
});
|
|
337
|
-
let packetsInFromDestinationTotal = new aws_cloudwatch_1.MathExpression({
|
|
338
|
-
expression: Object.keys(packetsInFromDestinationMetrics).join("+"),
|
|
339
|
-
usingMetrics: packetsInFromDestinationMetrics,
|
|
340
|
-
period: period
|
|
341
|
-
});
|
|
342
|
-
let usingMetrics = {};
|
|
343
|
-
usingMetrics[`${keyprefix}1`] = packetDropTotal;
|
|
344
|
-
usingMetrics[`${keyprefix}2`] = packetsInFromSourceTotal;
|
|
345
|
-
usingMetrics[`${keyprefix}3`] = packetsInFromDestinationTotal;
|
|
346
|
-
// Calculate a percentage of dropped packets for the NAT GW
|
|
347
|
-
let packetDropPercentage = new aws_cloudwatch_1.MathExpression({
|
|
348
|
-
expression: `(${keyprefix}1 / (${keyprefix}2 + ${keyprefix}3))`,
|
|
349
|
-
usingMetrics: usingMetrics,
|
|
350
|
-
label: availabilityZoneId + ' packet drop percentage',
|
|
351
|
-
period: period,
|
|
352
|
-
});
|
|
353
|
-
// Create an alarm for this NAT GW if packet drops exceed the specified threshold
|
|
354
|
-
return new aws_cloudwatch_1.Alarm(scope, availabilityZone.substring(availabilityZone.length - 1) + "-packet-drop-impact-alarm", {
|
|
355
|
-
alarmName: availabilityZoneId +
|
|
356
|
-
'-packet-drop-impact',
|
|
357
|
-
actionsEnabled: false,
|
|
358
|
-
metric: packetDropPercentage,
|
|
359
|
-
threshold: threshold,
|
|
360
|
-
comparisonOperator: aws_cloudwatch_1.ComparisonOperator.GREATER_THAN_THRESHOLD,
|
|
361
|
-
evaluationPeriods: evaluationPeriods,
|
|
362
|
-
datapointsToAlarm: datapointsToAlarm,
|
|
363
|
-
treatMissingData: aws_cloudwatch_1.TreatMissingData.IGNORE
|
|
364
|
-
});
|
|
365
|
-
}
|
|
366
|
-
static isAZAnOutlierForPacketLossNATGW(scope, natgws, availabilityZoneId, availabilityZone, threshold, period, evaluationPeriods, datapointsToAlarm) {
|
|
367
|
-
let keyprefix = MetricsHelper_1.MetricsHelper.nextChar();
|
|
368
|
-
let azPacketDropCountMetrics = {};
|
|
369
|
-
let azKey = "";
|
|
370
|
-
Object.keys(natgws).forEach((az) => {
|
|
371
|
-
let packetDropCountMetrics = {};
|
|
372
|
-
natgws[az].forEach((natgw, index) => {
|
|
373
|
-
packetDropCountMetrics[`${keyprefix}${index}`] = new aws_cloudwatch_1.Metric({
|
|
374
|
-
metricName: 'PacketsDropCount',
|
|
375
|
-
namespace: 'AWS/NATGateway',
|
|
376
|
-
statistic: aws_cloudwatch_1.Stats.SUM,
|
|
377
|
-
unit: aws_cloudwatch_1.Unit.COUNT,
|
|
378
|
-
label: availabilityZoneId + ' packet drops',
|
|
379
|
-
dimensionsMap: {
|
|
380
|
-
NatGatewayId: natgw.attrNatGatewayId,
|
|
381
|
-
},
|
|
382
|
-
period: period,
|
|
383
|
-
});
|
|
384
|
-
keyprefix = MetricsHelper_1.MetricsHelper.nextChar(keyprefix);
|
|
385
|
-
});
|
|
386
|
-
azPacketDropCountMetrics[`${keyprefix}${natgws[az].length}`] = new aws_cloudwatch_1.MathExpression({
|
|
387
|
-
expression: Object.keys(packetDropCountMetrics).join("+"),
|
|
388
|
-
usingMetrics: packetDropCountMetrics,
|
|
389
|
-
period: period
|
|
390
|
-
});
|
|
391
|
-
if (az == availabilityZone) {
|
|
392
|
-
azKey = `${keyprefix}${natgws[az].length}`;
|
|
393
|
-
}
|
|
394
|
-
keyprefix = MetricsHelper_1.MetricsHelper.nextChar(keyprefix);
|
|
395
|
-
});
|
|
396
|
-
return new aws_cloudwatch_1.Alarm(scope, availabilityZone.substring(availabilityZone.length - 1) + "-packet-loss-outlier", {
|
|
397
|
-
metric: new aws_cloudwatch_1.MathExpression({
|
|
398
|
-
expression: `${azKey} / (${Object.keys(azPacketDropCountMetrics).join("+")})`,
|
|
399
|
-
usingMetrics: azPacketDropCountMetrics,
|
|
400
|
-
period: period
|
|
401
|
-
}),
|
|
402
|
-
threshold: threshold,
|
|
403
|
-
evaluationPeriods: evaluationPeriods,
|
|
404
|
-
datapointsToAlarm: datapointsToAlarm,
|
|
405
|
-
comparisonOperator: aws_cloudwatch_1.ComparisonOperator.GREATER_THAN_THRESHOLD,
|
|
406
|
-
treatMissingData: aws_cloudwatch_1.TreatMissingData.IGNORE
|
|
407
|
-
});
|
|
408
|
-
}
|
|
409
|
-
doNatGatewayMetrics(props) {
|
|
179
|
+
createNatGatewayZonalImpactAlarms(props) {
|
|
410
180
|
// Collect alarms for packet drops exceeding a threshold per NAT GW
|
|
411
181
|
let packetLossPerAZAlarms = {};
|
|
182
|
+
let packetLossOutlierAlgorithm = props.natGatewayProps?.packetLossOutlierAlgorithm ?
|
|
183
|
+
props.natGatewayProps?.packetLossOutlierAlgorithm :
|
|
184
|
+
PacketLossOutlierAlgorithm_1.PacketLossOutlierAlgorithm.STATIC;
|
|
185
|
+
let packetLossThreshold = props.natGatewayProps?.packetLossPercentThreshold ?
|
|
186
|
+
props.natGatewayProps.packetLossPercentThreshold :
|
|
187
|
+
0.01;
|
|
188
|
+
let outlierThreshold;
|
|
189
|
+
if (props.natGatewayProps?.packetLossOutlierThreshold) {
|
|
190
|
+
outlierThreshold = props.natGatewayProps.packetLossOutlierThreshold;
|
|
191
|
+
}
|
|
192
|
+
else {
|
|
193
|
+
switch (packetLossOutlierAlgorithm) {
|
|
194
|
+
case PacketLossOutlierAlgorithm_1.PacketLossOutlierAlgorithm.STATIC:
|
|
195
|
+
default:
|
|
196
|
+
outlierThreshold = 66;
|
|
197
|
+
break;
|
|
198
|
+
}
|
|
199
|
+
}
|
|
200
|
+
let period = props.period ? props.period : aws_cdk_lib_1.Duration.minutes(1);
|
|
412
201
|
// For each AZ, create metrics for each NAT GW
|
|
413
202
|
Object.keys(this.natGateways).forEach((az) => {
|
|
414
203
|
let azLetter = az.substring(az.length - 1);
|
|
415
204
|
let availabilityZoneId = this._azMapper.availabilityZoneIdFromAvailabilityZoneLetter(azLetter);
|
|
416
205
|
// Is there packet loss impact?
|
|
417
|
-
let packetLossImpact =
|
|
206
|
+
let packetLossImpact = NatGatewayMetrics_1.NatGatewayMetrics.isThereAnAZPacketLossImpactNATGW(this, this.natGateways[az], availabilityZoneId, az, packetLossThreshold, period, props.evaluationPeriods, props.datapointsToAlarm);
|
|
418
207
|
// Is this AZ an outlier for this NATGW?
|
|
419
|
-
let packetLossOutlier =
|
|
420
|
-
packetLossPerAZAlarms[
|
|
208
|
+
let packetLossOutlier = NatGatewayMetrics_1.NatGatewayMetrics.isAZAnOutlierForPacketLossNATGW(this, this.natGateways, packetLossOutlierAlgorithm, az, this._azMapper, outlierThreshold, props.period ? props.period : aws_cdk_lib_1.Duration.minutes(1), props.evaluationPeriods, props.datapointsToAlarm);
|
|
209
|
+
packetLossPerAZAlarms[azLetter] = new aws_cloudwatch_1.CompositeAlarm(this, az.substring(az.length - 1) + "-packet-loss-composite-alarm", {
|
|
421
210
|
alarmRule: aws_cloudwatch_1.AlarmRule.allOf(packetLossImpact, packetLossOutlier),
|
|
422
211
|
compositeAlarmName: availabilityZoneId + "-packet-loss-composite-alarm"
|
|
423
212
|
});
|
|
424
213
|
});
|
|
425
214
|
return packetLossPerAZAlarms;
|
|
426
215
|
}
|
|
427
|
-
getTotalPacketDropsPerZone(natgws, period) {
|
|
428
|
-
let dropsPerZone = {};
|
|
429
|
-
let metricsPerAZ = {};
|
|
430
|
-
let keyprefix = MetricsHelper_1.MetricsHelper.nextChar();
|
|
431
|
-
Object.keys(natgws).forEach((availabilityZone) => {
|
|
432
|
-
let azLetter = availabilityZone.substring(availabilityZone.length - 1);
|
|
433
|
-
let availabilityZoneId = this._azMapper.availabilityZoneIdFromAvailabilityZoneLetter(azLetter);
|
|
434
|
-
if (!(availabilityZone in metricsPerAZ)) {
|
|
435
|
-
metricsPerAZ[availabilityZone] = [];
|
|
436
|
-
}
|
|
437
|
-
natgws[availabilityZone].forEach((natgw) => {
|
|
438
|
-
metricsPerAZ[availabilityZone].push(new aws_cloudwatch_1.Metric({
|
|
439
|
-
metricName: 'PacketsDropCount',
|
|
440
|
-
namespace: 'AWS/NATGateway',
|
|
441
|
-
statistic: aws_cloudwatch_1.Stats.SUM,
|
|
442
|
-
unit: aws_cloudwatch_1.Unit.COUNT,
|
|
443
|
-
label: availabilityZoneId + ' packet drops',
|
|
444
|
-
dimensionsMap: {
|
|
445
|
-
NatGatewayId: natgw.attrNatGatewayId,
|
|
446
|
-
},
|
|
447
|
-
period: period,
|
|
448
|
-
}));
|
|
449
|
-
});
|
|
450
|
-
});
|
|
451
|
-
Object.keys(metricsPerAZ).forEach((availabilityZone) => {
|
|
452
|
-
let azLetter = availabilityZone.substring(availabilityZone.length - 1);
|
|
453
|
-
let availabilityZoneId = this._azMapper.availabilityZoneIdFromAvailabilityZoneLetter(azLetter);
|
|
454
|
-
let usingMetrics = {};
|
|
455
|
-
metricsPerAZ[availabilityZone].forEach((metric) => {
|
|
456
|
-
usingMetrics[`${keyprefix}1`] = metric;
|
|
457
|
-
keyprefix = MetricsHelper_1.MetricsHelper.nextChar(keyprefix);
|
|
458
|
-
});
|
|
459
|
-
dropsPerZone[availabilityZone] = new aws_cloudwatch_1.MathExpression({
|
|
460
|
-
expression: Object.keys(usingMetrics).join("+"),
|
|
461
|
-
usingMetrics: usingMetrics,
|
|
462
|
-
label: availabilityZoneId + " total packet drops",
|
|
463
|
-
period: period
|
|
464
|
-
});
|
|
465
|
-
});
|
|
466
|
-
return dropsPerZone;
|
|
467
|
-
}
|
|
468
216
|
}
|
|
469
217
|
exports.BasicServiceMultiAZObservability = BasicServiceMultiAZObservability;
|
|
470
218
|
_a = JSII_RTTI_SYMBOL_1;
|
|
471
|
-
BasicServiceMultiAZObservability[_a] = { fqn: "@cdklabs/multi-az-observability.BasicServiceMultiAZObservability", version: "0.0.1-alpha.
|
|
472
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"BasicServiceMultiAZObservability.js","sourceRoot":"","sources":["../../src/basic_observability/BasicServiceMultiAZObservability.ts"],"names":[],"mappings":";;;;;AAAA,qEAAqE;AACrE,sCAAsC;AAEtC,6CAAuC;AACvC,+DAaoC;AAOpC,2CAAmD;AAGnD,+EAA4E;AAE5E,mEAAgE;AAChE,8FAA2F;AAC3F,gFAA6E;AAC7E,8DAA2D;AAC3D,+HAA4H;AAE5H;;;GAGG;AACH,MAAa,gCACX,SAAQ,sBAAS;IA8CjB,YACE,KAAgB,EAChB,EAAU,EACV,KAA4C;QAE5C,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEjB,8BAA8B;QAC9B,IAAI,CAAC,WAAW,GAAG,KAAK,CAAC,WAAW,CAAC;QACrC,IAAI,CAAC,wBAAwB,GAAG,KAAK,CAAC,wBAAwB,CAAC;QAC/D,IAAI,CAAC,WAAW,GAAG,KAAK,CAAC,WAAW,CAAC;QAErC,IAAI,CAAC,kCAAkC,GAAG,EAAE,CAAC;QAC7C,IAAI,CAAC,4BAA4B,GAAG,EAAE,CAAC;QACvC,IAAI,CAAC,8BAA8B,GAAG,EAAE,CAAC;QAEzC,6DAA6D;QAC7D,IAAI,CAAC,SAAS,GAAG,IAAI,+CAAsB,CAAC,IAAI,EAAE,0BAA0B,CAAC,CAAC;QAE9E,uCAAuC;QACvC,IAAI,IAAI,CAAC,wBAAwB,EAAE,CAAC;YAClC,IAAI,CAAC,4BAA4B,GAAG,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,CAAC;QAC/D,CAAC;QAED,+CAA+C;QAC/C,IAAI,IAAI,CAAC,WAAW,EAAE,CAAC;YACrB,IAAI,CAAC,8BAA8B,GAAG,IAAI,CAAC,mBAAmB,CAAC,KAAK,CAAC,CAAC;QACxE,CAAC;QAED,4EAA4E;QAC5E,qDAAqD;QACrD,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,4BAA4B,CAAC,CAAC,OAAO,CAAC,CAAC,EAAU,EAAE,EAAE;YACpE,IAAI,QAAQ,GAAG,EAAE,CAAC,SAAS,CAAC,EAAE,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;YAC3C,IAAI,kBAAkB,GAAG,IAAI,CAAC,SAAS,CAAC,4CAA4C,CAAC,QAAQ,CAAC,CAAC;YAC/F,IAAI,EAAE,IAAI,IAAI,CAAC,8BAA+B,EAAE,CAAC;gBAC/C,IAAI,CAAC,kCAAkC,CAAC,EAAE,CAAC,GAAG,IAAI,+BAAc,CAAC,IAAI,EAAE,EAAE,CAAC,SAAS,CAAC,EAAE,CAAC,MAAM,GAAG,CAAC,CAAC,GAAG,wBAAwB,EAAE;oBAC7H,SAAS,EAAE,0BAAS,CAAC,KAAK,CAAC,IAAI,CAAC,4BAA6B,CAAC,EAAE,CAAC,EAAE,IAAI,CAAC,8BAA+B,CAAC,EAAE,CAAC,CAAC;oBAC5G,kBAAkB,EAAE,kBAAkB,GAAG,wBAAwB;iBAClE,CAAC,CAAC;YACL,CAAC;iBACI,CAAC;gBACJ,IAAI,CAAC,kCAAkC,CAAC,EAAE,CAAC,GAAG,IAAI,+BAAc,CAAC,IAAI,EAAE,EAAE,CAAC,SAAS,CAAC,EAAE,CAAC,MAAM,GAAG,CAAC,CAAC,GAAG,wBAAwB,EAAE;oBAC7H,SAAS,EAAE,0BAAS,CAAC,KAAK,CAAC,IAAI,CAAC,4BAA6B,CAAC,EAAE,CAAC,CAAC;oBAClE,kBAAkB,EAAE,kBAAkB,GAAG,wBAAwB;iBAClE,CAAC,CAAC;YACL,CAAC;QACH,CAAC,CAAC,CAAC;QAEH,4FAA4F;QAC5F,wFAAwF;QACxF,kBAAkB;QAClB,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,8BAA8B,CAAC,CAAC,OAAO,CAAC,CAAC,EAAU,EAAE,EAAE;YACtE,IAAI,CAAC,CAAC,EAAE,IAAI,IAAI,CAAC,kCAAkC,CAAC,EAAE,CAAC;gBACrD,IAAI,QAAQ,GAAG,EAAE,CAAC,SAAS,CAAC,EAAE,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;gBAC3C,IAAI,kBAAkB,GAAG,IAAI,CAAC,SAAS,CAAC,4CAA4C,CAAC,QAAQ,CAAC,CAAC;gBAE/F,IAAI,CAAC,kCAAkC,CAAC,EAAE,CAAC,GAAG,IAAI,+BAAc,CAAC,IAAI,EAAE,EAAE,CAAC,SAAS,CAAC,EAAE,CAAC,MAAM,GAAG,CAAC,CAAC,GAAG,wBAAwB,EAAE;oBAC7H,SAAS,EAAE,0BAAS,CAAC,KAAK,CAAC,IAAI,CAAC,8BAA+B,CAAC,EAAE,CAAC,CAAC;oBACpE,kBAAkB,EAAE,kBAAkB,GAAG,wBAAwB;iBAClE,CAAC,CAAC;YACL,CAAC;QACH,CAAC,CAAC,CAAC;QAEH,iCAAiC;QACjC,IAAI,KAAK,CAAC,eAAe,IAAI,IAAI,EAAE,CAAC;YAClC,IAAI,CAAC,SAAS,GAAG,IAAI,6CAAqB,CACxC,IAAI,EACJ,uBAAuB,EACvB;gBACE,WAAW,EAAE,KAAK,CAAC,WAAW,CAAC,WAAW,EAAE;gBAC5C,kCAAkC,EAChC,IAAI,CAAC,kCAAkC;gBACzC,qCAAqC,EACnC,IAAI,CAAC,4BAA4B;gBACnC,mCAAmC,EACjC,IAAI,CAAC,8BAA8B;gBACrC,QAAQ,EAAE,KAAK,CAAC,QAAQ;gBACxB,iCAAiC,EAAE,+DAA8B,CAAC,4BAA4B,CAC5F,KAAK,CAAC,wBAAwB,CAAC,CAAC,CAAC,KAAK,CAAC,wBAAwB,CAAC,CAAC,CAAC,EAAE,EACpE,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,sBAAQ,CAAC,OAAO,CAAC,CAAC,CAAC,EACjD,IAAI,CAAC,SAAS,CACf;gBACD,gCAAgC,EAAE,IAAI,CAAC,0BAA0B,CAC/D,KAAK,CAAC,WAAW,CAAC,CAAC,CAAC,KAAK,CAAC,WAAW,CAAC,CAAC,CAAC,EAAE,EAC1C,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,sBAAQ,CAAC,OAAO,CAAC,CAAC,CAAC,CAClD;gBACD,QAAQ,EAAE,IAAI,CAAC,SAAS;aACzB,CACF,CAAC,SAAS,CAAC;QACd,CAAC;IACH,CAAC;IAEO,MAAM,CAAC,gCAAgC,CAC7C,KAAiB,EACjB,GAA6B,EAC7B,kBAA0B,EAC1B,gBAAwB,EACxB,SAAiB,EACjB,SAAiB,EACjB,MAAgB,EAChB,iBAAyB,EACzB,iBAAyB;QAGzB,4DAA4D;QAC5D,OAAO,IAAI,sBAAK,CACd,KAAK,EACL,SAAS,GAAG,mBAAmB,EAC/B;YACE,SAAS,EACP,kBAAkB,GAAG,GAAG,GAAG,GAAG,CAAC,eAAe,GAAG,aAAa;YAChE,cAAc,EAAE,KAAK;YACrB,MAAM,EAAE,+DAA8B,CAAC,0BAA0B,CAAC,GAAG,EAAE;gBACrE,MAAM,EAAE,MAAM;gBACd,KAAK,EAAE,kBAAkB,GAAG,GAAG,GAAG,GAAG,CAAC,eAAe,GAAG,aAAa;gBACrE,gBAAgB,EAAE,gBAAgB;gBAClC,kBAAkB,EAAE,kBAAkB;gBACtC,UAAU,EAAE,+CAAsB,CAAC,UAAU;aAC9C,CAAC;YACF,iBAAiB,EAAE,iBAAiB;YACpC,iBAAiB,EAAE,iBAAiB;YACpC,SAAS,EAAE,SAAS;YACpB,kBAAkB,EAAE,mCAAkB,CAAC,sBAAsB;YAC7D,gBAAgB,EAAE,iCAAgB,CAAC,MAAM;SAC1C,CACF,CAAC;IACJ,CAAC;IAEO,MAAM,CAAC,yBAAyB,CACtC,KAAiB,EACjB,GAA6B,EAC7B,kBAA0B,EAC1B,gBAAwB,EACxB,SAAiB,EACjB,SAAiB,EACjB,SAAiB,EACjB,MAAgB,EAChB,iBAAyB,EACzB,iBAAyB;QAGzB,4DAA4D;QAC5D,OAAO,IAAI,sBAAK,CACd,KAAK,EACL,SAAS,GAAG,gBAAgB,EAC5B;YACE,SAAS,EACP,kBAAkB,GAAG,GAAG,GAAG,GAAG,CAAC,eAAe,GAAG,UAAU;YAC7D,cAAc,EAAE,KAAK;YACrB,MAAM,EAAE,+DAA8B,CAAC,qBAAqB,CAAC;gBAC3D,GAAG,EAAE,GAAG;gBACR,gBAAgB,EAAE,gBAAgB;gBAClC,kBAAkB,EAAE,kBAAkB;gBACtC,KAAK,EAAE,kBAAkB,GAAG,GAAG,GAAG,GAAG,CAAC,eAAe,GAAG,iBAAiB;gBACzE,MAAM,EAAE,MAAM;gBACd,SAAS,EAAE,SAAS;aACrB,CAAC;YACF,iBAAiB,EAAE,iBAAiB;YACpC,iBAAiB,EAAE,iBAAiB;YACpC,SAAS,EAAE,SAAS;YACpB,kBAAkB,EAAE,mCAAkB,CAAC,sBAAsB;YAC7D,gBAAgB,EAAE,iCAAgB,CAAC,MAAM;SAC1C,CACF,CAAC;IACJ,CAAC;IAEO,MAAM,CAAC,+BAA+B,CAC5C,KAAiB,EACjB,GAA6B,EAC7B,kBAA0B,EAC1B,gBAAwB,EACxB,SAAiB,EACjB,SAAiB,EACjB,MAAgB,EAChB,iBAAyB,EACzB,iBAAyB;QAGzB,IAAI,YAAY,GAA+B,EAAE,CAAC;QAClD,IAAI,UAAU,GAAW,EAAE,CAAC;QAE5B,GAAG,CAAC,GAAI,CAAC,iBAAiB,CAAC,OAAO,CAAC,CAAC,EAAU,EAAE,EAAE;YAEhD,IAAI,YAAY,GAAG,+DAA8B,CAAC,0BAA0B,CAC1E,GAAG,EACH;gBACE,UAAU,EAAE,+CAAsB,CAAC,WAAW;gBAC9C,gBAAgB,EAAE,gBAAgB;gBAClC,kBAAkB,EAAE,kBAAkB;gBACtC,MAAM,EAAE,MAAM;gBACd,KAAK,EAAE,kBAAkB,GAAG,GAAG,GAAG,GAAG,CAAC,eAAe,GAAG,cAAc;gBACtE,SAAS,EAAE,SAAS;aACrB,CACF,CAAC;YAEF,SAAS,GAAG,6BAAa,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAC;YAE9C,YAAY,CAAC,GAAG,SAAS,GAAG,CAAC,GAAG,YAAY,CAAC;YAE7C,IAAI,EAAE,IAAI,gBAAgB,EAAE,CAAC;gBAC3B,UAAU,GAAG,GAAG,SAAS,GAAG,CAAC;YAC/B,CAAC;YAED,SAAS,GAAG,6BAAa,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAC;QAChD,CAAC,CAAC,CAAC;QAEH,OAAO,IAAI,sBAAK,CACd,KAAK,EACL,SAAS,GAAG,6BAA6B,EACzC;YACE,SAAS,EACP,kBAAkB,GAAG,GAAG,GAAG,GAAG,CAAC,eAAe,GAAG,8BAA8B;YACjF,cAAc,EAAE,KAAK;YACrB,MAAM,EAAE,IAAI,+BAAc,CAAC;gBACzB,UAAU,EAAE,GAAG,UAAW,KAAK,MAAM,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG;gBACrE,YAAY,EAAE,YAAY;gBAC1B,KAAK,EAAE,kBAAkB,GAAG,GAAG,GAAG,GAAG,CAAC,eAAe,GAAG,oBAAoB;gBAC5E,MAAM,EAAE,MAAM;aACf,CAAC;YACF,iBAAiB,EAAE,iBAAiB;YACpC,iBAAiB,EAAE,iBAAiB;YACpC,SAAS,EAAE,SAAS;YACpB,kBAAkB,EAAE,mCAAkB,CAAC,sBAAsB;YAC7D,gBAAgB,EAAE,iCAAgB,CAAC,MAAM;SAC1C,CACF,CAAC;IACJ,CAAC;IAEO,MAAM,CAAC,0BAA0B,CACvC,KAAiB,EACjB,GAA6B,EAC7B,SAA2D,EAC3D,kBAA0B,EAC1B,gBAAwB,EACxB,SAAiB,EACjB,SAAiB,EACjB,MAAgB,EAChB,iBAAyB,EACzB,iBAAyB,EACzB,SAAiB;QAGjB,IAAI,YAAY,GAA+B,EAAE,CAAC;QAClD,IAAI,UAAU,GAAW,EAAE,CAAC;QAE5B,QAAQ,SAAS,EACjB,CAAC;YACC,KAAK,mGAAgD,CAAC,OAAO,CAAC;YAC9D;gBAEE,GAAG,CAAC,GAAI,CAAC,iBAAiB,CAAC,OAAO,CAAC,CAAC,EAAU,EAAE,KAAa,EAAE,EAAE;oBAE/D,uBAAuB;oBACvB,IAAI,kBAAkB,GAAY,+DAA8B,CAAC,qBAAqB,CAAC;wBACrF,GAAG,EAAE,GAAG;wBACR,gBAAgB,EAAE,EAAE;wBACpB,KAAK,EAAE,EAAE,GAAG,uBAAuB;wBACnC,SAAS,EAAE,SAAS;wBACpB,MAAM,EAAE,MAAM;qBACf,CAAC,CAAC;oBAEH,IAAI,EAAE,IAAI,gBAAgB,EAAE,CAAC;wBAC3B,UAAU,GAAG,IAAI,KAAK,EAAE,CAAA;wBACxB,YAAY,CAAC,IAAI,KAAK,EAAE,CAAC,GAAG,kBAAkB,CAAC;oBACjD,CAAC;yBACI,CAAC;wBACJ,YAAY,CAAC,IAAI,KAAK,EAAE,CAAC,GAAG,kBAAkB,CAAC;oBACjD,CAAC;gBACH,CAAC,CAAC,CAAC;gBAEH,OAAO,IAAI,sBAAK,CACd,KAAK,EACL,SAAS,GAAG,wBAAwB,EACpC;oBACE,SAAS,EACP,kBAAkB,GAAG,GAAG,GAAG,GAAG,CAAC,eAAe,GAAG,yBAAyB;oBAC5E,cAAc,EAAE,KAAK;oBACrB,MAAM,EAAE,IAAI,+BAAc,CAAC;wBACzB,UAAU,EAAE,IAAI,UAAW,mDAAmD;wBAC9E,YAAY,EAAE,YAAY;wBAC1B,KAAK,EAAE,kBAAkB,GAAG,GAAG,GAAG,GAAG,CAAC,eAAe,GAAG,kBAAkB;wBAC1E,MAAM,EAAE,MAAM;qBACf,CAAC;oBACF,iBAAiB,EAAE,iBAAiB;oBACpC,iBAAiB,EAAE,iBAAiB;oBACpC,SAAS,EAAE,SAAS;oBACpB,kBAAkB,EAAE,mCAAkB,CAAC,kCAAkC;oBACzE,gBAAgB,EAAE,iCAAgB,CAAC,MAAM;iBAC1C,CACF,CAAC;YACJ,KAAK,mGAAgD,CAAC,MAAM;gBAE1D,GAAG,CAAC,GAAI,CAAC,iBAAiB,CAAC,OAAO,CAAC,CAAC,EAAU,EAAE,KAAa,EAAE,EAAE;oBAC/D,uBAAuB;oBACvB,IAAI,kBAAkB,GAAY,+DAA8B,CAAC,qBAAqB,CAAC;wBACrF,GAAG,EAAE,GAAG;wBACR,gBAAgB,EAAE,EAAE;wBACpB,KAAK,EAAE,EAAE,GAAG,uBAAuB;wBACnC,SAAS,EAAE,MAAM,SAAS,IAAI;wBAC9B,MAAM,EAAE,MAAM;qBACf,CAAC,CAAC;oBAEH,IAAI,EAAE,IAAI,gBAAgB,EAAE,CAAC;wBAC3B,UAAU,GAAG,IAAI,KAAK,EAAE,CAAA;wBACxB,YAAY,CAAC,IAAI,KAAK,EAAE,CAAC,GAAG,kBAAkB,CAAC;oBACjD,CAAC;yBACI,CAAC;wBACJ,YAAY,CAAC,IAAI,KAAK,EAAE,CAAC,GAAG,kBAAkB,CAAC;oBACjD,CAAC;gBAEH,CAAC,CAAC,CAAC;gBAEH,OAAO,IAAI,sBAAK,CACd,KAAK,EACL,SAAS,GAAG,wBAAwB,EACpC;oBACE,SAAS,EACP,kBAAkB,GAAG,GAAG,GAAG,GAAG,CAAC,eAAe,GAAG,yBAAyB;oBAC5E,cAAc,EAAE,KAAK;oBACrB,MAAM,EAAE,IAAI,+BAAc,CAAC;wBACzB,UAAU,EAAE,GAAG,UAAU,KAAK,MAAM,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG;wBACpE,YAAY,EAAE,YAAY;wBAC1B,KAAK,EAAE,kBAAkB,GAAG,GAAG,GAAG,GAAG,CAAC,eAAe,GAAG,iBAAiB;wBACzE,MAAM,EAAE,MAAM;qBACf,CAAC;oBACF,iBAAiB,EAAE,iBAAiB;oBACpC,iBAAiB,EAAE,iBAAiB;oBACpC,SAAS,EAAE,GAAG;oBACd,kBAAkB,EAAE,mCAAkB,CAAC,kCAAkC;oBACzE,gBAAgB,EAAE,iCAAgB,CAAC,MAAM;iBAC1C,CACF,CAAC;QACN,CAAC;IACH,CAAC;IAEO,YAAY,CAClB,KAA4C;QAG5C,uDAAuD;QACvD,uBAAuB;QACvB,IAAI,iBAAiB,GAAgC,EAAE,CAAC;QAExD,IAAI,SAAS,GAAW,6BAAa,CAAC,QAAQ,EAAE,CAAC;QAEjD,mBAAmB;QACnB,IAAI,CAAC,wBAAyB,CAAC,OAAO,CAAC,CAAC,GAAG,EAAE,EAAE;YAE7C,6BAA6B;YAC7B,GAAG,CAAC,GAAG,EAAE,iBAAiB,CAAC,OAAO,CAAC,CAAC,EAAE,EAAE,EAAE;gBAExC,IAAI,CAAC,CAAC,EAAE,IAAI,iBAAiB,CAAC,EAAE,CAAC;oBAC/B,iBAAiB,CAAC,EAAE,CAAC,GAAG,EAAE,CAAC;gBAC7B,CAAC;gBAED,gBAAgB;gBAChB,IAAI,QAAQ,GAAG,EAAE,CAAC,SAAS,CAAC,EAAE,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;gBAE3C,sBAAsB;gBACtB,IAAI,kBAAkB,GACpB,IAAI,CAAC,SAAS,CAAC,4CAA4C,CAAC,QAAQ,CAAC,CAAC;gBAExE,2CAA2C;gBAC3C,IAAI,kBAAkB,GAAW,gCAAgC,CAAC,gCAAgC,CAChG,IAAI,EACJ,GAAG,EACH,kBAAkB,EAClB,EAAE,EACF,KAAK,CAAC,6BAA6B,CAAC,CAAC,CAAC,KAAK,CAAC,6BAA6B,CAAC,CAAC,CAAC,IAAI,EAChF,SAAS,EACT,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,sBAAQ,CAAC,OAAO,CAAC,CAAC,CAAC,EACjD,KAAK,CAAC,iBAAiB,EACvB,KAAK,CAAC,iBAAiB,CACxB,CAAC;gBAEF,sCAAsC;gBACtC,IAAI,aAAa,GAAW,gCAAgC,CAAC,yBAAyB,CACpF,IAAI,EACJ,GAAG,EACH,kBAAkB,EAClB,EAAE,EACF,KAAK,CAAC,gBAAgB,EACtB,KAAK,CAAC,gBAAgB,EACtB,SAAS,EACT,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,sBAAQ,CAAC,OAAO,CAAC,CAAC,CAAC,EACjD,KAAK,CAAC,iBAAiB,EACvB,KAAK,CAAC,iBAAiB,CACxB,CAAC;gBAEF,kCAAkC;gBAClC,IAAI,mBAAmB,GAAG,gCAAgC,CAAC,+BAA+B,CACxF,IAAI,EACJ,GAAG,EACH,kBAAkB,EAClB,EAAE,EACF,KAAK,CAAC,6BAA6B,CAAC,CAAC,CAAC,KAAK,CAAC,6BAA6B,CAAC,CAAC,CAAC,IAAI,EAChF,SAAS,EACT,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,sBAAQ,CAAC,OAAO,CAAC,CAAC,CAAC,EACjD,KAAK,CAAC,iBAAiB,EACvB,KAAK,CAAC,iBAAiB,CACxB,CAAC;gBAEF,mCAAmC;gBACnC,IAAI,cAAc,GAAG,gCAAgC,CAAC,0BAA0B,CAC9E,IAAI,EACJ,GAAG,EACH,KAAK,CAAC,yBAAyB,CAAC,CAAC,CAAC,KAAK,CAAC,yBAAyB,CAAC,CAAC,CAAC,mGAAgD,CAAC,OAAO,EAC5H,kBAAkB,EAClB,EAAE,EACF,KAAK,CAAC,gBAAgB,EACtB,KAAK,CAAC,yBAAyB,IAAI,mGAAgD,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,gBAAgB,EACxH,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,sBAAQ,CAAC,OAAO,CAAC,CAAC,CAAC,EACjD,KAAK,CAAC,iBAAiB,EACvB,KAAK,CAAC,iBAAiB,EACvB,QAAQ,CACT,CAAC;gBAEF,iDAAiD;gBACjD,IAAI,aAAa,GAAW,IAAI,+BAAc,CAAC,IAAI,EACjD,EAAE,CAAC,SAAS,CAAC,EAAE,CAAC,MAAM,GAAG,CAAC,CAAC,GAAI,yBAAyB,EACxD;oBACE,SAAS,EAAE,0BAAS,CAAC,KAAK,CACxB,0BAAS,CAAC,KAAK,CAAC,kBAAkB,EAAE,mBAAmB,CAAC,EACxD,0BAAS,CAAC,KAAK,CAAC,aAAa,EAAE,cAAc,CAAC,CAC/C;oBACD,kBAAkB,EAChB,kBAAkB,GAAG,GAAG;wBACvB,GAA2C,CAAC,gBAAgB;wBAC7D,iCAAiC;oBACnC,cAAc,EAAE,KAAK;iBACtB,CACF,CAAC;gBAEF,kCAAkC;gBAClC,iBAAiB,CAAC,EAAE,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;gBAE1C,sBAAsB;gBACtB,SAAS,GAAG,6BAAa,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAC;YAChD,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,IAAI,iBAAiB,GAA4B,EAAE,CAAC;QAEpD,4DAA4D;QAC5D,4CAA4C;QAC5C,MAAM,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC,OAAO,CAAC,CAAC,EAAE,EAAE,EAAE;YAC5C,IAAI,QAAQ,GAAG,EAAE,CAAC,SAAS,CAAC,EAAE,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;YAC3C,IAAI,kBAAkB,GAAW,IAAI,CAAC,SAAS,CAAC,4CAA4C,CAAC,QAAQ,CAAC,CAAC;YAEvG,iBAAiB,CAAC,EAAE,CAAC,GAAG,IAAI,+BAAc,CAAC,IAAI,EAAE,QAAQ,GAAG,6BAA6B,EAAE;gBACzF,SAAS,EAAE,0BAAS,CAAC,KAAK,CAAC,GAAG,iBAAiB,CAAC,EAAE,CAAC,CAAC;gBACpD,kBAAkB,EAAE,kBAAkB,GAAG,6BAA6B;aACvE,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,OAAO,iBAAiB,CAAC;IAC3B,CAAC;IAEO,MAAM,CAAC,gCAAgC,CAC7C,KAAiB,EACjB,MAAuB,EACvB,kBAA0B,EAC1B,gBAAwB,EACxB,SAAiB,EACjB,MAAgB,EAChB,iBAAyB,EACzB,iBAAyB;QAGzB,IAAI,SAAS,GAAG,6BAAa,CAAC,QAAQ,EAAE,CAAC;QAEzC,IAAI,sBAAsB,GAA6B,EAAE,CAAC;QAC1D,IAAI,0BAA0B,GAA6B,EAAE,CAAC;QAC9D,IAAI,+BAA+B,GAA6B,EAAE,CAAC;QAEnE,MAAM,CAAC,OAAO,CAAC,CAAC,KAAoB,EAAC,EAAE;YAErC,sBAAsB,CAAC,GAAG,SAAS,GAAG,CAAC,GAAG,IAAI,uBAAM,CAAC;gBACnD,UAAU,EAAE,kBAAkB;gBAC9B,SAAS,EAAE,gBAAgB;gBAC3B,SAAS,EAAE,sBAAK,CAAC,GAAG;gBACpB,IAAI,EAAE,qBAAI,CAAC,KAAK;gBAChB,KAAK,EAAE,kBAAkB,GAAG,eAAe;gBAC3C,aAAa,EAAE;oBACb,YAAY,EAAE,KAAK,CAAC,gBAAgB;iBACrC;gBACD,MAAM,EAAE,MAAM;aACf,CAAC,CAAC;YAEH,mCAAmC;YACnC,0BAA0B,CAAC,GAAG,SAAS,GAAG,CAAC,GAAG,IAAI,uBAAM,CAAC;gBACvD,UAAU,EAAE,qBAAqB;gBACjC,SAAS,EAAE,gBAAgB;gBAC3B,SAAS,EAAE,sBAAK,CAAC,GAAG;gBACpB,IAAI,EAAE,qBAAI,CAAC,KAAK;gBAChB,KAAK,EAAE,kBAAkB,GAAG,yBAAyB;gBACrD,aAAa,EAAE;oBACb,YAAY,EAAE,KAAK,CAAC,gBAAgB;iBACrC;gBACD,MAAM,EAAE,MAAM;aACf,CAAC,CAAC;YAEH,wCAAwC;YACxC,+BAA+B,CAAC,GAAG,SAAS,GAAG,CAAC,GAAG,IAAI,uBAAM,CAAC;gBAC5D,UAAU,EAAE,0BAA0B;gBACtC,SAAS,EAAE,gBAAgB;gBAC3B,SAAS,EAAE,sBAAK,CAAC,GAAG;gBACpB,IAAI,EAAE,qBAAI,CAAC,KAAK;gBAChB,KAAK,EAAE,kBAAkB,GAAG,8BAA8B;gBAC1D,aAAa,EAAE;oBACb,YAAY,EAAE,KAAK,CAAC,gBAAgB;iBACrC;gBACD,MAAM,EAAE,MAAM;aACf,CAAC,CAAC;YAEH,SAAS,GAAG,6BAAa,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAC;QAChD,CAAC,CAAC,CAAC;QAEH,IAAI,eAAe,GAAY,IAAI,+BAAc,CAAC;YAChD,UAAU,EAAE,MAAM,CAAC,IAAI,CAAC,sBAAsB,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC;YACzD,YAAY,EAAE,sBAAsB;YACpC,MAAM,EAAE,MAAM;SACf,CAAC,CAAC;QAEH,IAAI,wBAAwB,GAAY,IAAI,+BAAc,CAAC;YACzD,UAAU,EAAE,MAAM,CAAC,IAAI,CAAC,0BAA0B,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC;YAC7D,YAAY,EAAE,0BAA0B;YACxC,MAAM,EAAE,MAAM;SACf,CAAC,CAAC;QAEH,IAAI,6BAA6B,GAAY,IAAI,+BAAc,CAAC;YAC9D,UAAU,EAAE,MAAM,CAAC,IAAI,CAAC,+BAA+B,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC;YAClE,YAAY,EAAE,+BAA+B;YAC7C,MAAM,EAAE,MAAM;SACf,CAAC,CAAC;QAEH,IAAI,YAAY,GAA+B,EAAE,CAAC;QAClD,YAAY,CAAC,GAAG,SAAS,GAAG,CAAC,GAAG,eAAe,CAAC;QAChD,YAAY,CAAC,GAAG,SAAS,GAAG,CAAC,GAAG,wBAAwB,CAAC;QACzD,YAAY,CAAC,GAAG,SAAS,GAAG,CAAC,GAAG,6BAA6B,CAAC;QAE9D,2DAA2D;QAC3D,IAAI,oBAAoB,GAAY,IAAI,+BAAc,CAAC;YACrD,UAAU,EAAE,IAAI,SAAS,QAAQ,SAAS,OAAO,SAAS,KAAK;YAC/D,YAAY,EAAE,YAAY;YAC1B,KAAK,EAAE,kBAAkB,GAAG,yBAAyB;YACrD,MAAM,EAAE,MAAM;SACf,CAAC,CAAC;QAEH,iFAAiF;QACjF,OAAO,IAAI,sBAAK,CACd,KAAK,EACL,gBAAgB,CAAC,SAAS,CAAC,gBAAgB,CAAC,MAAM,GAAG,CAAC,CAAC,GAAG,2BAA2B,EACrF;YACE,SAAS,EACP,kBAAkB;gBAClB,qBAAqB;YACvB,cAAc,EAAE,KAAK;YACrB,MAAM,EAAE,oBAAoB;YAC5B,SAAS,EAAE,SAAS;YACpB,kBAAkB,EAAE,mCAAkB,CAAC,sBAAsB;YAC7D,iBAAiB,EAAE,iBAAiB;YACpC,iBAAiB,EAAE,iBAAiB;YACpC,gBAAgB,EAAE,iCAAgB,CAAC,MAAM;SAC1C,CACF,CAAC;IACJ,CAAC;IAEO,MAAM,CAAC,+BAA+B,CAC5C,KAAiB,EACjB,MAAwC,EACxC,kBAA0B,EAC1B,gBAAwB,EACxB,SAAiB,EACjB,MAAgB,EAChB,iBAAyB,EACzB,iBAAyB;QAGzB,IAAI,SAAS,GAAG,6BAAa,CAAC,QAAQ,EAAE,CAAC;QACzC,IAAI,wBAAwB,GAA6B,EAAE,CAAC;QAC5D,IAAI,KAAK,GAAW,EAAE,CAAC;QAEvB,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC,CAAC,EAAU,EAAE,EAAE;YAEzC,IAAI,sBAAsB,GAA6B,EAAE,CAAC;YAI1D,MAAM,CAAC,EAAE,CAAC,CAAC,OAAO,CAAC,CAAC,KAAoB,EAAE,KAAa,EAAE,EAAE;gBACzD,sBAAsB,CAAC,GAAG,SAAS,GAAG,KAAK,EAAE,CAAC,GAAG,IAAI,uBAAM,CAAC;oBAC1D,UAAU,EAAE,kBAAkB;oBAC9B,SAAS,EAAE,gBAAgB;oBAC3B,SAAS,EAAE,sBAAK,CAAC,GAAG;oBACpB,IAAI,EAAE,qBAAI,CAAC,KAAK;oBAChB,KAAK,EAAE,kBAAkB,GAAG,eAAe;oBAC3C,aAAa,EAAE;wBACb,YAAY,EAAE,KAAK,CAAC,gBAAgB;qBACrC;oBACD,MAAM,EAAE,MAAM;iBACf,CAAC,CAAC;gBACH,SAAS,GAAG,6BAAa,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAC;YAChD,CAAC,CAAC,CAAC;YAEH,wBAAwB,CAAC,GAAG,SAAS,GAAG,MAAM,CAAC,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC,GAAG,IAAI,+BAAc,CAAC;gBAChF,UAAU,EAAE,MAAM,CAAC,IAAI,CAAC,sBAAsB,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC;gBACzD,YAAY,EAAE,sBAAsB;gBACpC,MAAM,EAAE,MAAM;aACf,CAAC,CAAC;YAEH,IAAI,EAAE,IAAI,gBAAgB,EAAE,CAAC;gBAC3B,KAAK,GAAG,GAAG,SAAS,GAAG,MAAM,CAAC,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC;YAC7C,CAAC;YAED,SAAS,GAAG,6BAAa,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAC;QAChD,CAAC,CAAC,CAAC;QAEH,OAAO,IAAI,sBAAK,CACd,KAAK,EACL,gBAAgB,CAAC,SAAS,CAAC,gBAAgB,CAAC,MAAM,GAAG,CAAC,CAAC,GAAG,sBAAsB,EAChF;YACG,MAAM,EAAE,IAAI,+BAAc,CAAC;gBACxB,UAAU,EAAE,GAAG,KAAK,OAAO,MAAM,CAAC,IAAI,CAAC,wBAAwB,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG;gBAC7E,YAAY,EAAE,wBAAwB;gBACtC,MAAM,EAAE,MAAM;aAChB,CAAC;YACF,SAAS,EAAE,SAAS;YACpB,iBAAiB,EAAE,iBAAiB;YACpC,iBAAiB,EAAE,iBAAiB;YACpC,kBAAkB,EAAE,mCAAkB,CAAC,sBAAsB;YAC7D,gBAAgB,EAAE,iCAAgB,CAAC,MAAM;SAC3C,CACF,CAAC;IACJ,CAAC;IAEO,mBAAmB,CACzB,KAA4C;QAG5C,mEAAmE;QACnE,IAAI,qBAAqB,GAA8B,EAAE,CAAC;QAE1D,8CAA8C;QAC9C,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,WAAY,CAAC,CAAC,OAAO,CAAC,CAAC,EAAU,EAAE,EAAE;YAEpD,IAAI,QAAQ,GAAW,EAAE,CAAC,SAAS,CAAC,EAAE,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;YACnD,IAAI,kBAAkB,GACpB,IAAI,CAAC,SAAS,CAAC,4CAA4C,CAAC,QAAQ,CAAC,CAAC;YAExE,+BAA+B;YAC/B,IAAI,gBAAgB,GAAW,gCAAgC,CAAC,gCAAgC,CAC9F,IAAI,EACJ,IAAI,CAAC,WAAY,CAAC,EAAE,CAAC,EACrB,kBAAkB,EAClB,EAAE,EACF,KAAK,CAAC,mCAAmC,CAAC,CAAC,CAAC,KAAK,CAAC,mCAAmC,CAAC,CAAC,CAAC,IAAI,EAC5F,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,sBAAQ,CAAC,OAAO,CAAC,CAAC,CAAC,EACjD,KAAK,CAAC,iBAAiB,EACvB,KAAK,CAAC,iBAAiB,CACxB,CAAC;YACF,wCAAwC;YACxC,IAAI,iBAAiB,GAAW,gCAAgC,CAAC,+BAA+B,CAC9F,IAAI,EACJ,IAAI,CAAC,WAAY,EACjB,kBAAkB,EAClB,EAAE,EACF,IAAI,EACJ,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,sBAAQ,CAAC,OAAO,CAAC,CAAC,CAAC,EACjD,KAAK,CAAC,iBAAiB,EACvB,KAAK,CAAC,iBAAiB,CACxB,CAAC;YACF,qBAAqB,CAAC,EAAE,CAAC,GAAG,IAAI,+BAAc,CAAC,IAAI,EAAE,EAAE,CAAC,SAAS,CAAC,EAAE,CAAC,MAAM,GAAG,CAAC,CAAC,GAAG,8BAA8B,EAAE;gBACjH,SAAS,EAAE,0BAAS,CAAC,KAAK,CAAC,gBAAgB,EAAE,iBAAiB,CAAC;gBAC/D,kBAAkB,EAAE,kBAAkB,GAAG,8BAA8B;aACxE,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,OAAO,qBAAqB,CAAC;IAC/B,CAAC;IAEO,0BAA0B,CAChC,MAAwC,EACxC,MAAgB;QAGhB,IAAI,YAAY,GAA6B,EAAE,CAAC;QAChD,IAAI,YAAY,GAA+B,EAAE,CAAC;QAClD,IAAI,SAAS,GAAW,6BAAa,CAAC,QAAQ,EAAE,CAAC;QAEjD,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC,CAAC,gBAAwB,EAAE,EAAE;YAEvD,IAAI,QAAQ,GAAG,gBAAgB,CAAC,SAAS,CAAC,gBAAgB,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;YACvE,IAAI,kBAAkB,GAAG,IAAI,CAAC,SAAS,CAAC,4CAA4C,CAAC,QAAQ,CAAC,CAAC;YAE/F,IAAI,CAAC,CAAC,gBAAgB,IAAI,YAAY,CAAC,EAAE,CAAC;gBACxC,YAAY,CAAC,gBAAgB,CAAC,GAAG,EAAE,CAAC;YACtC,CAAC;YAED,MAAM,CAAC,gBAAgB,CAAC,CAAC,OAAO,CAAC,CAAC,KAAoB,EAAE,EAAE;gBAExD,YAAY,CAAC,gBAAgB,CAAC,CAAC,IAAI,CAAC,IAAI,uBAAM,CAAC;oBAC7C,UAAU,EAAE,kBAAkB;oBAC9B,SAAS,EAAE,gBAAgB;oBAC3B,SAAS,EAAE,sBAAK,CAAC,GAAG;oBACpB,IAAI,EAAE,qBAAI,CAAC,KAAK;oBAChB,KAAK,EAAE,kBAAkB,GAAG,eAAe;oBAC3C,aAAa,EAAE;wBACb,YAAY,EAAE,KAAK,CAAC,gBAAgB;qBACrC;oBACD,MAAM,EAAE,MAAM;iBACf,CAAC,CAAC,CAAC;YACN,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,MAAM,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC,OAAO,CAAC,CAAC,gBAAwB,EAAE,EAAE;YAC7D,IAAI,QAAQ,GAAG,gBAAgB,CAAC,SAAS,CAAC,gBAAgB,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;YACvE,IAAI,kBAAkB,GAAG,IAAI,CAAC,SAAS,CAAC,4CAA4C,CAAC,QAAQ,CAAC,CAAC;YAE/F,IAAI,YAAY,GAA6B,EAAE,CAAC;YAEhD,YAAY,CAAC,gBAAgB,CAAC,CAAC,OAAO,CAAC,CAAC,MAAe,EAAE,EAAE;gBACzD,YAAY,CAAC,GAAG,SAAS,GAAG,CAAC,GAAG,MAAM,CAAC;gBACvC,SAAS,GAAG,6BAAa,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAC;YAChD,CAAC,CAAC,CAAC;YAEH,YAAY,CAAC,gBAAgB,CAAC,GAAG,IAAI,+BAAc,CAAC;gBAClD,UAAU,EAAE,MAAM,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC;gBAC/C,YAAY,EAAE,YAAY;gBAC1B,KAAK,EAAE,kBAAkB,GAAG,qBAAqB;gBACjD,MAAM,EAAE,MAAM;aACf,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,OAAO,YAAY,CAAC;IACtB,CAAC;;AA7wBH,4EA8wBC","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';\nimport { ApplicationLoadBalancerLatencyOutlierCalculation } from './props/ApplicationLoadBalancerLatencyOutlierCalculation';\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\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 = `${keyprefix}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    algorithm: ApplicationLoadBalancerLatencyOutlierCalculation,\n    availabilityZoneId: string,\n    availabilityZone: string,\n    statistic: string,\n    threshold: number,\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    switch (algorithm)\n    {\n      case ApplicationLoadBalancerLatencyOutlierCalculation.Z_SCORE:\n      default:\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: threshold,\n            comparisonOperator: ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,\n            treatMissingData: TreatMissingData.IGNORE\n          }\n        );\n      case ApplicationLoadBalancerLatencyOutlierCalculation.STATIC:\n        \n        alb.vpc!.availabilityZones.forEach((az: string, index: number) => {\n          // Target response time\n          let targetResponseTime: IMetric = ApplicationLoadBalancerMetrics.getPerAZLatencyMetric({\n            alb: alb,\n            availabilityZone: az,\n            label: az + \"-target-response-time\",\n            statistic: `TC(${threshold}:)`,\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\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}/(${Object.keys(usingMetrics).join(\"+\")})`,\n              usingMetrics: usingMetrics,\n              label: availabilityZoneId + '-' + alb.loadBalancerArn + '-latency-static',\n              period: period,\n            }),\n            evaluationPeriods: evaluationPeriods,\n            datapointsToAlarm: datapointsToAlarm,\n            threshold: .66,\n            comparisonOperator: ComparisonOperator.GREATER_THAN_OR_EQUAL_TO_THRESHOLD,\n            treatMissingData: TreatMissingData.IGNORE\n          }\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          props.latencyOutlierCalculation ? props.latencyOutlierCalculation : ApplicationLoadBalancerLatencyOutlierCalculation.Z_SCORE,\n          availabilityZoneId,\n          az,\n          props.latencyStatistic,\n          props.latencyOutlierCalculation == ApplicationLoadBalancerLatencyOutlierCalculation.Z_SCORE ? 3 : props.latencyThreshold,\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"]}
|
|
219
|
+
BasicServiceMultiAZObservability[_a] = { fqn: "@cdklabs/multi-az-observability.BasicServiceMultiAZObservability", version: "0.0.1-alpha.60" };
|
|
220
|
+
//# 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  AlarmRule,\n  CompositeAlarm,\n  Dashboard,\n  IAlarm\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 } 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 { MetricsHelper } from '../utilities/MetricsHelper';\nimport { ApplicationLoadBalancerLatencyOutlierAlgorithm } from '../outlier-detection/ApplicationLoadBalancerLatencyOutlierAlgorithm';\nimport { NatGatewayMetrics } from '../metrics/NatGatewayMetrics';\nimport { ApplicationLoadBalancerAvailabilityOutlierAlgorithm } from '../outlier-detection/ApplicationLoadBalancerAvailabilityOutlierAlgorithm';\nimport { PacketLossOutlierAlgorithm } from '../outlier-detection/PacketLossOutlierAlgorithm';\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    if (!(props.applicationLoadBalancerProps) && !(props.natGatewayProps)) {\n      throw new Error(\"You must define either ALBs or NAT Gateways for this service in order to create a dashboard.\");\n    }\n\n    // Initialize class properties\n    this.serviceName = props.serviceName;\n    this.applicationLoadBalancers = props.applicationLoadBalancerProps?.albTargetGroupMap.map(entry => entry.applicationLoadBalancer);\n    this.natGateways = props.natGatewayProps?.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.createAlbZonalImpactAlarms(props);\n    }\n\n    // Create NAT Gateway metrics and alarms per AZ\n    if (this.natGateways) {\n      this.natGWZonalIsolatedImpactAlarms = this.createNatGatewayZonalImpactAlarms(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((azLetter: string) => {\n      let availabilityZoneId = this._azMapper.availabilityZoneIdFromAvailabilityZoneLetter(azLetter);\n      if (azLetter in this.natGWZonalIsolatedImpactAlarms!) {\n        this.aggregateZonalIsolatedImpactAlarms[azLetter] = new CompositeAlarm(this, azLetter + \"-isolated-impact-alarm\", {\n          alarmRule: AlarmRule.anyOf(this.albZonalIsolatedImpactAlarms![azLetter], this.natGWZonalIsolatedImpactAlarms![azLetter]),\n          compositeAlarmName: availabilityZoneId + \"-isolated-impact-alarm\"\n        });\n      }\n      else {\n        this.aggregateZonalIsolatedImpactAlarms[azLetter] = new CompositeAlarm(this, azLetter + \"-isolated-impact-alarm\", {\n          alarmRule: AlarmRule.anyOf(this.albZonalIsolatedImpactAlarms![azLetter]),\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((azLetter: string) => {\n      if (!(azLetter in this.aggregateZonalIsolatedImpactAlarms)) {\n        let availabilityZoneId = this._azMapper.availabilityZoneIdFromAvailabilityZoneLetter(azLetter);\n\n        this.aggregateZonalIsolatedImpactAlarms[azLetter] = new CompositeAlarm(this, azLetter + \"-isolated-impact-alarm\", {\n          alarmRule: AlarmRule.anyOf(this.natGWZonalIsolatedImpactAlarms![azLetter]),\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          interval: props.interval,\n          natgws: props.natGatewayProps,\n          albs: props.applicationLoadBalancerProps,\n          period: props.period ? props.period : Duration.minutes(1),\n          azMapper: this._azMapper\n        },\n      ).dashboard;\n    }\n  }\n\n  /**\n   * \n   * @param props \n   * @returns A composite alarm per AZ to indicate isolated zonal impact. The dictionary\n   * is keyed by the az letter, like \"a\", \"b\", \"c\".\n   */\n  private createAlbZonalImpactAlarms(\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    let availabilityOutlierDetectionAlgorithm: ApplicationLoadBalancerAvailabilityOutlierAlgorithm = \n      props.applicationLoadBalancerProps?.availabilityOutlierAlgorithm ?\n      props.applicationLoadBalancerProps.availabilityOutlierAlgorithm :\n      ApplicationLoadBalancerAvailabilityOutlierAlgorithm.STATIC;\n\n    let latencyOutlierDetectionAlgorithm: ApplicationLoadBalancerLatencyOutlierAlgorithm =\n      props.applicationLoadBalancerProps?.latencyOutlierAlgorithm ? \n      props.applicationLoadBalancerProps.latencyOutlierAlgorithm :\n      ApplicationLoadBalancerLatencyOutlierAlgorithm.Z_SCORE;\n\n    let availabilityOutlierThreshold: number;\n    if (props.applicationLoadBalancerProps?.availabilityOutlierThreshold) {\n      availabilityOutlierThreshold = props.applicationLoadBalancerProps.availabilityOutlierThreshold;\n    }\n    else {\n      switch (availabilityOutlierDetectionAlgorithm) {\n        case ApplicationLoadBalancerAvailabilityOutlierAlgorithm.STATIC:\n        default:\n          availabilityOutlierThreshold = 66;\n          break;\n      }\n    }\n\n    let latencyOutlierThreshold: number;\n\n    if (props.applicationLoadBalancerProps?.latencyOutlierThreshold) {\n      latencyOutlierThreshold = props.applicationLoadBalancerProps.latencyOutlierThreshold;\n    }\n    else {\n      switch (latencyOutlierDetectionAlgorithm) {\n        case ApplicationLoadBalancerLatencyOutlierAlgorithm.Z_SCORE:\n        default:\n          latencyOutlierThreshold = 3;\n          break;\n        case ApplicationLoadBalancerLatencyOutlierAlgorithm.STATIC:\n          latencyOutlierThreshold = 66;\n          break;\n      }\n    }\n\n    let period: Duration = props.period ? props.period : Duration.minutes(1);\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 = ApplicationLoadBalancerMetrics.createAZAvailabilityImpactAlarm(\n          this,\n          alb,\n          availabilityZoneId,\n          az,\n          props.applicationLoadBalancerProps!.faultCountPercentThreshold,\n          keyPrefix,\n          period,\n          props.evaluationPeriods,\n          props.datapointsToAlarm\n        );\n\n        // Is there latency impact in this AZ?\n        let latencyImpact: IAlarm = ApplicationLoadBalancerMetrics.createAZLatencyImpactAlarm(\n          this,\n          alb,\n          availabilityZoneId,\n          az,\n          props.applicationLoadBalancerProps!.latencyThreshold,\n          props.applicationLoadBalancerProps!.latencyStatistic,\n          keyPrefix,\n          period,\n          props.evaluationPeriods,\n          props.datapointsToAlarm\n        );\n\n        // Is the AZ an outlier for faults\n        let availabilityOutlier = ApplicationLoadBalancerMetrics.createAZAvailabilityOutlierAlarm(\n          this,\n          alb,\n          availabilityOutlierDetectionAlgorithm,\n          availabilityZoneId,\n          az,\n          availabilityOutlierThreshold,\n          keyPrefix,\n          period,\n          props.evaluationPeriods,\n          props.datapointsToAlarm\n        );\n\n        // Is the AZ an outlier for latency\n        let latencyOutlier = ApplicationLoadBalancerMetrics.createAZLatencyOutlierAlarm(\n          this,\n          alb,\n          latencyOutlierDetectionAlgorithm,\n          az,\n          props.applicationLoadBalancerProps!.latencyStatistic,\n          props.applicationLoadBalancerProps!.latencyThreshold,\n          latencyOutlierThreshold,\n          period,\n          props.evaluationPeriods,\n          props.datapointsToAlarm,\n          this._azMapper\n        );\n\n        // Alarm if the AZ shows impact and is an outlier\n        let azImpactAlarm: IAlarm = new CompositeAlarm(this, \n          azLetter +  \"-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[azLetter] = 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 createNatGatewayZonalImpactAlarms(\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    let packetLossOutlierAlgorithm: PacketLossOutlierAlgorithm = props.natGatewayProps?.packetLossOutlierAlgorithm ?\n      props.natGatewayProps?.packetLossOutlierAlgorithm :\n      PacketLossOutlierAlgorithm.STATIC;\n\n    let packetLossThreshold: number = props.natGatewayProps?.packetLossPercentThreshold ? \n      props.natGatewayProps.packetLossPercentThreshold :\n      0.01;\n\n    let outlierThreshold: number;\n\n    if (props.natGatewayProps?.packetLossOutlierThreshold) {\n      outlierThreshold = props.natGatewayProps.packetLossOutlierThreshold;\n    }\n    else {\n      switch (packetLossOutlierAlgorithm) {\n        case PacketLossOutlierAlgorithm.STATIC:\n        default:\n          outlierThreshold = 66;\n          break;\n      }\n    }\n      \n\n    let period: Duration = props.period ? props.period : Duration.minutes(1);\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 = NatGatewayMetrics.isThereAnAZPacketLossImpactNATGW(\n        this,\n        this.natGateways![az],\n        availabilityZoneId,\n        az,\n        packetLossThreshold,\n        period,\n        props.evaluationPeriods,\n        props.datapointsToAlarm\n      );\n\n      // Is this AZ an outlier for this NATGW?\n      let packetLossOutlier: IAlarm = NatGatewayMetrics.isAZAnOutlierForPacketLossNATGW(\n        this,\n        this.natGateways!,\n        packetLossOutlierAlgorithm,\n        az,\n        this._azMapper,\n        outlierThreshold,\n        props.period ? props.period : Duration.minutes(1),\n        props.evaluationPeriods,\n        props.datapointsToAlarm\n      );\n\n      packetLossPerAZAlarms[azLetter] = 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"]}
|